170fbbed7f1a20c02efd7c3d6cc0f17d34b42b11
[blender-staging.git] / source / blender / modifiers / intern / MOD_weightvgmix.c
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software  Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * The Original Code is Copyright (C) 2011 by Bastien Montagne.
19  * All rights reserved.
20  *
21  * Contributor(s): None yet.
22  *
23  * ***** END GPL LICENSE BLOCK *****
24  *
25  */
26
27 /** \file blender/modifiers/intern/MOD_weightvgmix.c
28  *  \ingroup modifiers
29  */
30
31 #include "BLI_utildefines.h"
32 #include "BLI_math.h"
33 #include "BLI_listbase.h"
34
35 #include "DNA_meshdata_types.h"
36 #include "DNA_modifier_types.h"
37 #include "DNA_object_types.h"
38
39 #include "BKE_cdderivedmesh.h"
40 #include "BKE_deform.h"
41 #include "BKE_library.h"
42 #include "BKE_library_query.h"
43 #include "BKE_modifier.h"
44 #include "BKE_texture.h"          /* Texture masking. */
45
46 #include "depsgraph_private.h"
47 #include "DEG_depsgraph_build.h"
48
49 #include "MEM_guardedalloc.h"
50 #include "MOD_weightvg_util.h"
51
52
53 /**
54  * This mixes the old weight with the new weight factor.
55  */
56 static float mix_weight(float weight, float weight2, char mix_mode)
57 {
58 #if 0
59         /*
60          * XXX Don't know why, but the switch version takes many CPU time,
61          *     and produces lag in realtime playback...
62          */
63         switch (mix_mode)
64         {
65                 case MOD_WVG_MIX_ADD:
66                         return (weight + weight2);
67                 case MOD_WVG_MIX_SUB:
68                         return (weight - weight2);
69                 case MOD_WVG_MIX_MUL:
70                         return (weight * weight2);
71                 case MOD_WVG_MIX_DIV:
72                         /* Avoid dividing by zero (or really small values). */
73                         if (0.0 <= weight2 < MOD_WVG_ZEROFLOOR)
74                                 weight2 = MOD_WVG_ZEROFLOOR;
75                         else if (-MOD_WVG_ZEROFLOOR < weight2)
76                                 weight2 = -MOD_WVG_ZEROFLOOR;
77                         return (weight / weight2);
78                 case MOD_WVG_MIX_DIF:
79                         return (weight < weight2 ? weight2 - weight : weight - weight2);
80                 case MOD_WVG_MIX_AVG:
81                         return (weight + weight2) / 2.0;
82                 case MOD_WVG_MIX_SET:
83                 default:
84                         return weight2;
85         }
86 #endif
87         if (mix_mode == MOD_WVG_MIX_SET)
88                 return weight2;
89         else if (mix_mode == MOD_WVG_MIX_ADD)
90                 return (weight + weight2);
91         else if (mix_mode == MOD_WVG_MIX_SUB)
92                 return (weight - weight2);
93         else if (mix_mode == MOD_WVG_MIX_MUL)
94                 return (weight * weight2);
95         else if (mix_mode == MOD_WVG_MIX_DIV) {
96                 /* Avoid dividing by zero (or really small values). */
97                 if (weight2 < 0.0f && weight2 > -MOD_WVG_ZEROFLOOR)
98                         weight2 = -MOD_WVG_ZEROFLOOR;
99                 else if (weight2 >= 0.0f && weight2 < MOD_WVG_ZEROFLOOR)
100                         weight2 = MOD_WVG_ZEROFLOOR;
101                 return (weight / weight2);
102         }
103         else if (mix_mode == MOD_WVG_MIX_DIF)
104                 return (weight < weight2 ? weight2 - weight : weight - weight2);
105         else if (mix_mode == MOD_WVG_MIX_AVG)
106                 return (weight + weight2) * 0.5f;
107         else return weight2;
108 }
109
110 /**************************************
111  * Modifiers functions.               *
112  **************************************/
113 static void initData(ModifierData *md)
114 {
115         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
116
117         wmd->default_weight_a       = 0.0f;
118         wmd->default_weight_b       = 0.0f;
119         wmd->mix_mode               = MOD_WVG_MIX_SET;
120         wmd->mix_set                = MOD_WVG_SET_AND;
121
122         wmd->mask_constant          = 1.0f;
123         wmd->mask_tex_use_channel   = MOD_WVG_MASK_TEX_USE_INT; /* Use intensity by default. */
124         wmd->mask_tex_mapping       = MOD_DISP_MAP_LOCAL;
125 }
126
127 static void freeData(ModifierData *md)
128 {
129         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
130         if (wmd->mask_texture) {
131                 id_us_min(&wmd->mask_texture->id);
132         }
133 }
134
135 static void copyData(ModifierData *md, ModifierData *target)
136 {
137 #if 0
138         WeightVGMixModifierData *wmd  = (WeightVGMixModifierData *) md;
139 #endif
140         WeightVGMixModifierData *twmd = (WeightVGMixModifierData *) target;
141
142         modifier_copyData_generic(md, target);
143
144         if (twmd->mask_texture) {
145                 id_us_plus(&twmd->mask_texture->id);
146         }
147 }
148
149 static CustomDataMask requiredDataMask(Object *UNUSED(ob), ModifierData *md)
150 {
151         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
152         CustomDataMask dataMask = 0;
153
154         /* We need vertex groups! */
155         dataMask |= CD_MASK_MDEFORMVERT;
156
157         /* Ask for UV coordinates if we need them. */
158         if (wmd->mask_tex_mapping == MOD_DISP_MAP_UV)
159                 dataMask |= CD_MASK_MTFACE;
160
161         /* No need to ask for CD_PREVIEW_MLOOPCOL... */
162
163         return dataMask;
164 }
165
166 static bool dependsOnTime(ModifierData *md)
167 {
168         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
169
170         if (wmd->mask_texture)
171                 return BKE_texture_dependsOnTime(wmd->mask_texture);
172         return false;
173 }
174
175 static void foreachObjectLink(ModifierData *md, Object *ob, ObjectWalkFunc walk, void *userData)
176 {
177         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
178         walk(userData, ob, &wmd->mask_tex_map_obj, IDWALK_NOP);
179 }
180
181 static void foreachIDLink(ModifierData *md, Object *ob, IDWalkFunc walk, void *userData)
182 {
183         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
184
185         walk(userData, ob, (ID **)&wmd->mask_texture, IDWALK_USER);
186
187         foreachObjectLink(md, ob, (ObjectWalkFunc)walk, userData);
188 }
189
190 static void foreachTexLink(ModifierData *md, Object *ob, TexWalkFunc walk, void *userData)
191 {
192         walk(userData, ob, md, "mask_texture");
193 }
194
195 static void updateDepgraph(ModifierData *md, DagForest *forest,
196                            struct Main *UNUSED(bmain),
197                            struct Scene *UNUSED(scene),
198                            Object *UNUSED(ob), DagNode *obNode)
199 {
200         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
201         DagNode *curNode;
202
203         if (wmd->mask_tex_map_obj && wmd->mask_tex_mapping == MOD_DISP_MAP_OBJECT) {
204                 curNode = dag_get_node(forest, wmd->mask_tex_map_obj);
205
206                 dag_add_relation(forest, curNode, obNode, DAG_RL_DATA_DATA | DAG_RL_OB_DATA,
207                                  "WeightVGMix Modifier");
208         }
209
210         if (wmd->mask_tex_mapping == MOD_DISP_MAP_GLOBAL)
211                 dag_add_relation(forest, obNode, obNode, DAG_RL_DATA_DATA | DAG_RL_OB_DATA,
212                                  "WeightVGMix Modifier");
213 }
214
215 static void updateDepsgraph(ModifierData *md,
216                             struct Main *UNUSED(bmain),
217                             struct Scene *UNUSED(scene),
218                             Object *ob,
219                             struct DepsNodeHandle *node)
220 {
221         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
222         if (wmd->mask_tex_map_obj != NULL && wmd->mask_tex_mapping == MOD_DISP_MAP_OBJECT) {
223                 DEG_add_object_relation(node, wmd->mask_tex_map_obj, DEG_OB_COMP_TRANSFORM, "WeightVGMix Modifier");
224                 DEG_add_object_relation(node, wmd->mask_tex_map_obj, DEG_OB_COMP_GEOMETRY, "WeightVGMix Modifier");
225         }
226         if (wmd->mask_tex_mapping == MOD_DISP_MAP_GLOBAL) {
227                 DEG_add_object_relation(node, ob, DEG_OB_COMP_TRANSFORM, "WeightVGMix Modifier");
228                 DEG_add_object_relation(node, ob, DEG_OB_COMP_GEOMETRY, "WeightVGMix Modifier");
229         }
230 }
231
232 static bool isDisabled(ModifierData *md, int UNUSED(useRenderParams))
233 {
234         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
235         /* If no vertex group, bypass. */
236         return (wmd->defgrp_name_a[0] == '\0');
237 }
238
239 static DerivedMesh *applyModifier(ModifierData *md, Object *ob, DerivedMesh *derivedData,
240                                   ModifierApplyFlag UNUSED(flag))
241 {
242         WeightVGMixModifierData *wmd = (WeightVGMixModifierData *) md;
243         DerivedMesh *dm = derivedData;
244         MDeformVert *dvert = NULL;
245         MDeformWeight **dw1, **tdw1, **dw2, **tdw2;
246         int numVerts;
247         int defgrp_index, defgrp_index_other = -1;
248         float *org_w;
249         float *new_w;
250         int *tidx, *indices = NULL;
251         int numIdx = 0;
252         int i;
253         /* Flags. */
254 #if 0
255         const bool do_prev = (wmd->modifier.mode & eModifierMode_DoWeightPreview) != 0;
256 #endif
257
258         /* Get number of verts. */
259         numVerts = dm->getNumVerts(dm);
260
261         /* Check if we can just return the original mesh.
262          * Must have verts and therefore verts assigned to vgroups to do anything useful!
263          */
264         if ((numVerts == 0) || BLI_listbase_is_empty(&ob->defbase))
265                 return dm;
266
267         /* Get vgroup idx from its name. */
268         defgrp_index = defgroup_name_index(ob, wmd->defgrp_name_a);
269         if (defgrp_index == -1)
270                 return dm;
271         /* Get second vgroup idx from its name, if given. */
272         if (wmd->defgrp_name_b[0] != (char)0) {
273                 defgrp_index_other = defgroup_name_index(ob, wmd->defgrp_name_b);
274                 if (defgrp_index_other == -1)
275                         return dm;
276         }
277
278         dvert = CustomData_duplicate_referenced_layer(&dm->vertData, CD_MDEFORMVERT, numVerts);
279         /* If no vertices were ever added to an object's vgroup, dvert might be NULL. */
280         if (!dvert) {
281                 /* If not affecting all vertices, just return. */
282                 if (wmd->mix_set != MOD_WVG_SET_ALL)
283                         return dm;
284                 /* Else, add a valid data layer! */
285                 dvert = CustomData_add_layer_named(&dm->vertData, CD_MDEFORMVERT, CD_CALLOC,
286                                                    NULL, numVerts, wmd->defgrp_name_a);
287                 /* Ultimate security check. */
288                 if (!dvert)
289                         return dm;
290         }
291         /* Find out which vertices to work on. */
292         tidx = MEM_mallocN(sizeof(int) * numVerts, "WeightVGMix Modifier, tidx");
293         tdw1 = MEM_mallocN(sizeof(MDeformWeight *) * numVerts, "WeightVGMix Modifier, tdw1");
294         tdw2 = MEM_mallocN(sizeof(MDeformWeight *) * numVerts, "WeightVGMix Modifier, tdw2");
295         switch (wmd->mix_set) {
296                 case MOD_WVG_SET_A:
297                         /* All vertices in first vgroup. */
298                         for (i = 0; i < numVerts; i++) {
299                                 MDeformWeight *dw = defvert_find_index(&dvert[i], defgrp_index);
300                                 if (dw) {
301                                         tdw1[numIdx] = dw;
302                                         tdw2[numIdx] = (defgrp_index_other >= 0) ? defvert_find_index(&dvert[i], defgrp_index_other) : NULL;
303                                         tidx[numIdx++] = i;
304                                 }
305                         }
306                         break;
307                 case MOD_WVG_SET_B:
308                         /* All vertices in second vgroup. */
309                         for (i = 0; i < numVerts; i++) {
310                                 MDeformWeight *dw = (defgrp_index_other >= 0) ? defvert_find_index(&dvert[i], defgrp_index_other) : NULL;
311                                 if (dw) {
312                                         tdw1[numIdx] = defvert_find_index(&dvert[i], defgrp_index);
313                                         tdw2[numIdx] = dw;
314                                         tidx[numIdx++] = i;
315                                 }
316                         }
317                         break;
318                 case MOD_WVG_SET_OR:
319                         /* All vertices in one vgroup or the other. */
320                         for (i = 0; i < numVerts; i++) {
321                                 MDeformWeight *adw = defvert_find_index(&dvert[i], defgrp_index);
322                                 MDeformWeight *bdw = (defgrp_index_other >= 0) ? defvert_find_index(&dvert[i], defgrp_index_other) : NULL;
323                                 if (adw || bdw) {
324                                         tdw1[numIdx] = adw;
325                                         tdw2[numIdx] = bdw;
326                                         tidx[numIdx++] = i;
327                                 }
328                         }
329                         break;
330                 case MOD_WVG_SET_AND:
331                         /* All vertices in both vgroups. */
332                         for (i = 0; i < numVerts; i++) {
333                                 MDeformWeight *adw = defvert_find_index(&dvert[i], defgrp_index);
334                                 MDeformWeight *bdw = (defgrp_index_other >= 0) ? defvert_find_index(&dvert[i], defgrp_index_other) : NULL;
335                                 if (adw && bdw) {
336                                         tdw1[numIdx] = adw;
337                                         tdw2[numIdx] = bdw;
338                                         tidx[numIdx++] = i;
339                                 }
340                         }
341                         break;
342                 case MOD_WVG_SET_ALL:
343                 default:
344                         /* Use all vertices. */
345                         for (i = 0; i < numVerts; i++) {
346                                 tdw1[i] = defvert_find_index(&dvert[i], defgrp_index);
347                                 tdw2[i] = (defgrp_index_other >= 0) ? defvert_find_index(&dvert[i], defgrp_index_other) : NULL;
348                         }
349                         numIdx = -1;
350                         break;
351         }
352         if (numIdx == 0) {
353                 /* Use no vertices! Hence, return org data. */
354                 MEM_freeN(tdw1);
355                 MEM_freeN(tdw2);
356                 MEM_freeN(tidx);
357                 return dm;
358         }
359         if (numIdx != -1) {
360                 indices = MEM_mallocN(sizeof(int) * numIdx, "WeightVGMix Modifier, indices");
361                 memcpy(indices, tidx, sizeof(int) * numIdx);
362                 dw1 = MEM_mallocN(sizeof(MDeformWeight *) * numIdx, "WeightVGMix Modifier, dw1");
363                 memcpy(dw1, tdw1, sizeof(MDeformWeight *) * numIdx);
364                 MEM_freeN(tdw1);
365                 dw2 = MEM_mallocN(sizeof(MDeformWeight *) * numIdx, "WeightVGMix Modifier, dw2");
366                 memcpy(dw2, tdw2, sizeof(MDeformWeight *) * numIdx);
367                 MEM_freeN(tdw2);
368         }
369         else {
370                 /* Use all vertices. */
371                 numIdx = numVerts;
372                 /* Just copy MDeformWeight pointers arrays, they will be freed at the end. */
373                 dw1 = tdw1;
374                 dw2 = tdw2;
375         }
376         MEM_freeN(tidx);
377
378         org_w = MEM_mallocN(sizeof(float) * numIdx, "WeightVGMix Modifier, org_w");
379         new_w = MEM_mallocN(sizeof(float) * numIdx, "WeightVGMix Modifier, new_w");
380
381         /* Mix weights. */
382         for (i = 0; i < numIdx; i++) {
383                 float weight2;
384                 org_w[i] = dw1[i] ? dw1[i]->weight : wmd->default_weight_a;
385                 weight2  = dw2[i] ? dw2[i]->weight : wmd->default_weight_b;
386
387                 new_w[i] = mix_weight(org_w[i], weight2, wmd->mix_mode);
388         }
389
390         /* Do masking. */
391         weightvg_do_mask(numIdx, indices, org_w, new_w, ob, dm, wmd->mask_constant,
392                          wmd->mask_defgrp_name, wmd->modifier.scene, wmd->mask_texture,
393                          wmd->mask_tex_use_channel, wmd->mask_tex_mapping,
394                          wmd->mask_tex_map_obj, wmd->mask_tex_uvlayer_name);
395
396         /* Update (add to) vgroup.
397          * XXX Depending on the MOD_WVG_SET_xxx option chosen, we might have to add vertices to vgroup.
398          */
399         weightvg_update_vg(dvert, defgrp_index, dw1, numIdx, indices, org_w, true, -FLT_MAX, false, 0.0f);
400
401         /* If weight preview enabled... */
402 #if 0 /* XXX Currently done in mod stack :/ */
403         if (do_prev)
404                 DM_update_weight_mcol(ob, dm, 0, org_w, numIdx, indices);
405 #endif
406
407         /* Freeing stuff. */
408         MEM_freeN(org_w);
409         MEM_freeN(new_w);
410         MEM_freeN(dw1);
411         MEM_freeN(dw2);
412
413         if (indices)
414                 MEM_freeN(indices);
415
416         /* Return the vgroup-modified mesh. */
417         return dm;
418 }
419
420
421 ModifierTypeInfo modifierType_WeightVGMix = {
422         /* name */              "VertexWeightMix",
423         /* structName */        "WeightVGMixModifierData",
424         /* structSize */        sizeof(WeightVGMixModifierData),
425         /* type */              eModifierTypeType_NonGeometrical,
426         /* flags */             eModifierTypeFlag_AcceptsMesh |
427                                 eModifierTypeFlag_SupportsMapping |
428                                 eModifierTypeFlag_SupportsEditmode |
429                                 eModifierTypeFlag_UsesPreview,
430
431         /* copyData */          copyData,
432         /* deformVerts */       NULL,
433         /* deformMatrices */    NULL,
434         /* deformVertsEM */     NULL,
435         /* deformMatricesEM */  NULL,
436         /* applyModifier */     applyModifier,
437         /* applyModifierEM */   NULL,
438         /* initData */          initData,
439         /* requiredDataMask */  requiredDataMask,
440         /* freeData */          freeData,
441         /* isDisabled */        isDisabled,
442         /* updateDepgraph */    updateDepgraph,
443         /* updateDepsgraph */   updateDepsgraph,
444         /* dependsOnTime */     dependsOnTime,
445         /* dependsOnNormals */  NULL,
446         /* foreachObjectLink */ foreachObjectLink,
447         /* foreachIDLink */     foreachIDLink,
448         /* foreachTexLink */    foreachTexLink,
449 };