Merging r38765 through r38817 from trunk into vgroup_modifiers.
[blender.git] / source / blender / modifiers / intern / MOD_weightvgmix.c
1 /*
2 * $Id$
3 *
4 * ***** BEGIN GPL LICENSE BLOCK *****
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software  Foundation,
18 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 *
20 * The Original Code is Copyright (C) 2011 by Bastien Montagne.
21 * All rights reserved.
22 *
23 * Contributor(s): None yet.
24 *
25 * ***** END GPL LICENSE BLOCK *****
26 *
27 */
28
29 /*
30  * XXX I’d like to make modified weights visible in WeightPaint mode,
31  *     but couldn’t figure a way to do this…
32  *     Maybe this will need changes in mesh_calc_modifiers (DerivedMesh.c)?
33  *     Or the WeightPaint mode code itself?
34  */
35
36 #include "BLI_utildefines.h"
37 #include "BLI_math.h"
38 #include "BLI_string.h"
39
40 #include "DNA_mesh_types.h"
41 #include "DNA_meshdata_types.h"
42 #include "DNA_modifier_types.h"
43 #include "DNA_object_types.h"
44
45 #include "BKE_cdderivedmesh.h"
46 #include "BKE_deform.h"
47 #include "BKE_mesh.h"
48 #include "BKE_modifier.h"
49 #include "BKE_texture.h"          /* Texture masking. */
50
51 #include "depsgraph_private.h"
52 #include "MEM_guardedalloc.h"
53 #include "MOD_util.h"
54 #include "MOD_weightvg_util.h"
55
56
57 /**
58  * This mixes the old weight with the new weight factor.
59  */
60 static float mix_weight(float weight, float weight2, char mix_mode)
61 {
62 #if 0
63         /*
64          * XXX Don’t know why, but the switch version takes many CPU time,
65          *     and produces lag in realtime playback…
66          */
67         switch (mix_mode)
68         {
69         case MOD_WVG_MIX_ADD:
70                 return (weight + weight2);
71         case MOD_WVG_MIX_SUB:
72                 return (weight - weight2);
73         case MOD_WVG_MIX_MUL:
74                 return (weight * weight2);
75         case MOD_WVG_MIX_DIV:
76                 /* Avoid dividing by zero (or really small values). */
77                 if (0.0 <= weight2 < MOD_WVG_ZEROFLOOR)
78                         weight2 = MOD_WVG_ZEROFLOOR;
79                 else if (-MOD_WVG_ZEROFLOOR < weight2)
80                         weight2 = -MOD_WVG_ZEROFLOOR;
81                 return (weight / weight2);
82         case MOD_WVG_MIX_DIF:
83                 return (weight < weight2 ? weight2 - weight : weight - weight2);
84         case MOD_WVG_MIX_AVG:
85                 return (weight + weight2) / 2.0;
86         case MOD_WVG_MIX_SET:
87         default:
88                 return weight2;
89         }
90 #endif
91         if (mix_mode == MOD_WVG_MIX_SET)
92                 return weight2;
93         else if (mix_mode == MOD_WVG_MIX_ADD)
94                 return (weight + weight2);
95         else if (mix_mode == MOD_WVG_MIX_SUB)
96                 return (weight - weight2);
97         else if (mix_mode == MOD_WVG_MIX_MUL)
98                 return (weight * weight2);
99         else if (mix_mode == MOD_WVG_MIX_DIV) {
100                 /* Avoid dividing by zero (or really small values). */
101                 if (weight2 < 0.0 && weight2 > -MOD_WVG_ZEROFLOOR)
102                         weight2 = -MOD_WVG_ZEROFLOOR;
103                 else if (weight2 >= 0.0 && weight2 < MOD_WVG_ZEROFLOOR)
104                         weight2 = MOD_WVG_ZEROFLOOR;
105                 return (weight / weight2);
106         }
107         else if (mix_mode == MOD_WVG_MIX_DIF)
108                 return (weight < weight2 ? weight2 - weight : weight - weight2);
109         else if (mix_mode == MOD_WVG_MIX_AVG)
110                 return (weight + weight2) / 2.0;
111         else return weight2;
112 }
113
114 /**************************************
115  * Modifiers functions.               *
116  **************************************/
117 static void initData(ModifierData *md)
118 {
119         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
120
121         wmd->default_weight         = 0.0;
122         wmd->default_weight2        = 0.0;
123         wmd->mix_mode               = MOD_WVG_MIX_SET;
124         wmd->mix_set                = MOD_WVG_SET_INTER;
125
126         wmd->mask_constant          = 1.0f;
127         wmd->mask_tex_use_channel   = MOD_WVG_MASK_TEX_USE_INT; /* Use intensity by default. */
128         wmd->mask_tex_mapping       = MOD_DISP_MAP_LOCAL;
129 }
130
131 static void copyData(ModifierData *md, ModifierData *target)
132 {
133         WeightVGMixModifierData *wmd  = (WeightVGMixModifierData*) md;
134         WeightVGMixModifierData *twmd = (WeightVGMixModifierData*) target;
135
136         BLI_strncpy(twmd->defgrp_name, wmd->defgrp_name, sizeof(twmd->defgrp_name));
137         BLI_strncpy(twmd->defgrp_name2, wmd->defgrp_name2, sizeof(twmd->defgrp_name2));
138         twmd->default_weight         = wmd->default_weight;
139         twmd->default_weight2        = wmd->default_weight2;
140         twmd->mix_mode               = wmd->mix_mode;
141         twmd->mix_set                = wmd->mix_set;
142
143         twmd->mask_constant          = wmd->mask_constant;
144         BLI_strncpy(twmd->mask_defgrp_name, wmd->mask_defgrp_name, sizeof(twmd->mask_defgrp_name));
145         twmd->mask_texture           = wmd->mask_texture;
146         twmd->mask_tex_use_channel   = wmd->mask_tex_use_channel;
147         twmd->mask_tex_mapping       = wmd->mask_tex_mapping;
148         twmd->mask_tex_map_obj       = wmd->mask_tex_map_obj;
149         BLI_strncpy(twmd->mask_tex_uvlayer_name, wmd->mask_tex_uvlayer_name,
150                     sizeof(twmd->mask_tex_uvlayer_name));
151 }
152
153 static CustomDataMask requiredDataMask(Object *UNUSED(ob), ModifierData *md)
154 {
155         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
156         CustomDataMask dataMask = 0;
157
158         /* We need vertex groups! */
159         dataMask |= CD_MASK_MDEFORMVERT;
160
161         /* Ask for UV coordinates if we need them. */
162         if(wmd->mask_tex_mapping == MOD_DISP_MAP_UV)
163                 dataMask |= CD_MASK_MTFACE;
164
165         return dataMask;
166 }
167
168 static int dependsOnTime(ModifierData *md)
169 {
170         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
171
172         if(wmd->mask_texture)
173                 return BKE_texture_dependsOnTime(wmd->mask_texture);
174         return 0;
175 }
176
177 static void foreachObjectLink(ModifierData *md, Object *ob,
178                               void (*walk)(void *userData, Object *ob, Object **obpoin),
179                               void *userData)
180 {
181         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
182         walk(userData, ob, &wmd->mask_tex_map_obj);
183 }
184
185 static void foreachIDLink(ModifierData *md, Object *ob, IDWalkFunc walk, void *userData)
186 {
187         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
188
189         walk(userData, ob, (ID **)&wmd->mask_texture);
190
191         foreachObjectLink(md, ob, (ObjectWalkFunc)walk, userData);
192 }
193
194 static void updateDepgraph(ModifierData *md, DagForest *forest, struct Scene *UNUSED(scene),
195                            Object *UNUSED(ob), DagNode *obNode)
196 {
197         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
198         DagNode *curNode;
199
200         if(wmd->mask_tex_map_obj && wmd->mask_tex_mapping == MOD_DISP_MAP_OBJECT) {
201                 curNode = dag_get_node(forest, wmd->mask_tex_map_obj);
202
203                 dag_add_relation(forest, curNode, obNode, DAG_RL_DATA_DATA|DAG_RL_OB_DATA,
204                                  "WeightVGMix Modifier");
205         }
206
207         if(wmd->mask_tex_mapping == MOD_DISP_MAP_GLOBAL)
208                 dag_add_relation(forest, obNode, obNode, DAG_RL_DATA_DATA|DAG_RL_OB_DATA,
209                                  "WeightVGMix Modifier");
210 }
211
212 static int isDisabled(ModifierData *md, int UNUSED(useRenderParams))
213 {
214         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
215         /* If no vertex group, bypass. */
216         return (wmd->defgrp_name == NULL);
217 }
218
219 static DerivedMesh *applyModifier(ModifierData *md, Object *ob, DerivedMesh *derivedData,
220                                   int UNUSED(useRenderParams), int UNUSED(isFinalCalc))
221 {
222         WeightVGMixModifierData *wmd = (WeightVGMixModifierData*) md;
223         DerivedMesh *dm = derivedData, *ret = NULL;
224 #if 0
225         Mesh *ob_m = NULL;
226 #endif
227         MDeformVert *dvert = NULL;
228         int numVerts;
229         int defgrp_idx, defgrp_idx2 = -1;
230         float *org_w = NULL;
231         float *new_w = NULL;
232         int *tidx, *indices = NULL;
233         int numIdx = 0;
234         int i, j;
235         char rel_ret = 0; /* Boolean, whether we have to release ret dm or not, when not using it! */
236
237         /* Get number of verts. */
238         numVerts = dm->getNumVerts(dm);
239
240         /* Check if we can just return the original mesh.
241          * Must have verts and therefore verts assigned to vgroups to do anything useful!
242          */
243         if ((numVerts == 0) || (ob->defbase.first == NULL))
244                 return dm;
245
246         /* Get vgroup idx from its name. */
247         defgrp_idx = defgroup_name_index(ob, wmd->defgrp_name);
248         if (defgrp_idx < 0)
249                 return dm;
250         /* Get seconf vgroup idx from its name, if given. */
251         if (wmd->defgrp_name2[0] != (char)0) {
252                 defgrp_idx2 = defgroup_name_index(ob, wmd->defgrp_name2);
253                 if (defgrp_idx2 < 0)
254                         return dm;
255         }
256
257         /* XXX All this to avoid copying dm when not needed… However, it nearly doubles compute
258          *     time! See scene 5 of the WeighVG test file…
259          */
260 #if 0
261         /* Get actual dverts (ie vertex group data). */
262         dvert = dm->getVertDataArray(dm, CD_MDEFORMVERT);
263         /* If no dverts, return unmodified data… */
264         if (dvert == NULL)
265                 return dm;
266
267         /* Get org mesh, only to test whether affected cdata layer has already been copied
268          * somewhere up in the modifiers stack.
269          */
270         ob_m = get_mesh(ob);
271         if (ob_m == NULL)
272                 return dm;
273
274         /* Create a copy of our dmesh, only if our affected cdata layer is the same as org mesh. */
275         if (dvert == CustomData_get_layer(&ob_m->vdata, CD_MDEFORMVERT)) {
276                 /* XXX Seems to create problems with weightpaint mode???
277                  *     I’m missing something here, I guess…
278                  */
279 //              DM_set_only_copy(dm, CD_MASK_MDEFORMVERT); /* Only copy defgroup layer. */
280                 ret = CDDM_copy(dm);
281                 dvert = ret->getVertDataArray(ret, CD_MDEFORMVERT);
282                 if (dvert == NULL) {
283                         ret->release(ret);
284                         return dm;
285                 }
286                 rel_ret = 1;
287         }
288         else
289                 ret = dm;
290 #else
291         ret = CDDM_copy(dm);
292         rel_ret = 1;
293         dvert = ret->getVertDataArray(ret, CD_MDEFORMVERT);
294         if (dvert == NULL) {
295                 if (rel_ret)
296                         ret->release(ret);
297                 return dm;
298         }
299 #endif
300
301         /* Find out which vertices to work on. */
302         tidx = MEM_mallocN(sizeof(int) * numVerts, "WeightVGMix Modifier, tidx");
303         switch (wmd->mix_set) {
304         case MOD_WVG_SET_ORG:
305                 /* All vertices in first vgroup. */
306                 for (i = 0; i < numVerts; i++) {
307                         for (j = 0; j < dvert[i].totweight; j++) {
308                                 if(dvert[i].dw[j].def_nr == defgrp_idx) {
309                                         tidx[numIdx++] = i;
310                                         break;
311                                 }
312                         }
313                 }
314                 break;
315         case MOD_WVG_SET_NEW:
316                 /* All vertices in second vgroup. */
317                 for (i = 0; i < numVerts; i++) {
318                         for (j = 0; j < dvert[i].totweight; j++) {
319                                 if(dvert[i].dw[j].def_nr == defgrp_idx2) {
320                                         tidx[numIdx++] = i;
321                                         break;
322                                 }
323                         }
324                 }
325                 break;
326         case MOD_WVG_SET_UNION:
327                 /* All vertices in one vgroup or the other. */
328                 for (i = 0; i < numVerts; i++) {
329                         for (j = 0; j < dvert[i].totweight; j++) {
330                                 if(dvert[i].dw[j].def_nr == defgrp_idx || dvert[i].dw[j].def_nr == defgrp_idx2) {
331                                         tidx[numIdx++] = i;
332                                         break;
333                                 }
334                         }
335                 }
336                 break;
337         case MOD_WVG_SET_INTER:
338                 /* All vertices in both vgroups. */
339                 for (i = 0; i < numVerts; i++) {
340                         char idx1 = 0;
341                         char idx2 = 0;
342                         for (j = 0; j < dvert[i].totweight; j++) {
343                                 if(dvert[i].dw[j].def_nr == defgrp_idx) {
344                                         if (idx2) {
345                                                 tidx[numIdx++] = i;
346                                                 break;
347                                         }
348                                         else
349                                                 idx1 = 1;
350                                 }
351                                 else if(dvert[i].dw[j].def_nr == defgrp_idx2) {
352                                         if (idx1) {
353                                                 tidx[numIdx++] = i;
354                                                 break;
355                                         }
356                                         else
357                                                 idx2 = 1;
358                                 }
359                         }
360                 }
361                 break;
362         case MOD_WVG_SET_ALL:
363         default:
364                 /* Use all vertices, no need to do anything here. */
365                 break;
366         }
367         if (numIdx) {
368                 indices = MEM_mallocN(sizeof(int) * numIdx, "WeightVGMix Modifier, indices");
369                 memcpy(indices, tidx, sizeof(int) * numIdx);
370         }
371         else
372                 numIdx = numVerts;
373         MEM_freeN(tidx);
374
375         org_w = MEM_mallocN(sizeof(float) * numIdx, "WeightVGMix Modifier, org_w");
376         new_w = MEM_mallocN(sizeof(float) * numIdx, "WeightVGMix Modifier, new_w");
377
378         /* Mix weights. */
379         for (i = 0; i < numIdx; i++) {
380                 float weight2 = 0.0;
381                 char w1 = 0;
382                 char w2 = 0;
383                 int idx = indices ? indices[i] : i;
384                 for (j = 0; j < dvert[idx].totweight; j++) {
385                         if(dvert[idx].dw[j].def_nr == defgrp_idx) {
386                                 org_w[i] = dvert[idx].dw[j].weight;
387                                 w1 = 1;
388                                 if (w2)
389                                         break;
390                         }
391                         else if(dvert[idx].dw[j].def_nr == defgrp_idx2) {
392                                 weight2 = dvert[idx].dw[j].weight;
393                                 w2 = 1;
394                                 if (w1)
395                                         break;
396                         }
397                 }
398                 if (w1 == 0)
399                         org_w[i] = wmd->default_weight;
400                 if (w2 == 0)
401                         weight2 = wmd->default_weight2;
402                 new_w[i] = mix_weight(org_w[i], weight2, wmd->mix_mode);
403         }
404
405         /* Do masking. */
406         weightvg_do_mask(numIdx, indices, org_w, new_w, ob, ret, wmd->mask_constant,
407                          wmd->mask_defgrp_name, wmd->mask_texture, wmd->mask_tex_use_channel,
408                          wmd->mask_tex_mapping, wmd->mask_tex_map_obj, wmd->mask_tex_uvlayer_name);
409
410         /* Update (add to) vgroup.
411          * XXX Depending on the MOD_WVG_SET_xxx option chosen, we might have to add vertices to vgroup.
412          */
413         weightvg_update_vg(dvert, defgrp_idx, numIdx, indices, org_w, 1, -FLT_MAX, 0, 0.0f);
414
415         /* Freeing stuff. */
416         if (org_w)
417                 MEM_freeN(org_w);
418         if (new_w)
419                 MEM_freeN(new_w);
420         if (indices)
421                 MEM_freeN(indices);
422
423         /* Return the vgroup-modified mesh. */
424         return ret;
425 }
426
427 static DerivedMesh *applyModifierEM(ModifierData *md, Object *ob,
428                                     struct EditMesh *UNUSED(editData),
429                                     DerivedMesh *derivedData)
430 {
431         return applyModifier(md, ob, derivedData, 0, 1);
432 }
433
434
435 ModifierTypeInfo modifierType_WeightVGMix = {
436         /* name */              "WeightVGMix",
437         /* structName */        "WeightVGMixModifierData",
438         /* structSize */        sizeof(WeightVGMixModifierData),
439         /* type */              eModifierTypeType_Nonconstructive,
440         /* flags */             eModifierTypeFlag_AcceptsMesh
441 /*                             |eModifierTypeFlag_SupportsMapping*/
442                                |eModifierTypeFlag_SupportsEditmode,
443
444         /* copyData */          copyData,
445         /* deformVerts */       NULL,
446         /* deformMatrices */    NULL,
447         /* deformVertsEM */     NULL,
448         /* deformMatricesEM */  NULL,
449         /* applyModifier */     applyModifier,
450         /* applyModifierEM */   applyModifierEM,
451         /* initData */          initData,
452         /* requiredDataMask */  requiredDataMask,
453         /* freeData */          NULL,
454         /* isDisabled */        isDisabled,
455         /* updateDepgraph */    updateDepgraph,
456         /* dependsOnTime */     dependsOnTime,
457         /* dependsOnNormals */  NULL,
458         /* foreachObjectLink */ foreachObjectLink,
459         /* foreachIDLink */     foreachIDLink,
460 };
461