Workbench: Material colors + settings during sculpting
[blender.git] / source / blender / modifiers / intern / MOD_normal_edit.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  * Contributor(s): Bastien Montagne
19  *
20  * ***** END GPL LICENSE BLOCK *****
21  *
22  */
23
24 /** \file blender/modifiers/intern/MOD_normal_edit.c
25  *  \ingroup modifiers
26  */
27
28 #include <string.h>
29
30 #include "MEM_guardedalloc.h"
31
32 #include "DNA_object_types.h"
33 #include "DNA_meshdata_types.h"
34 #include "DNA_mesh_types.h"
35
36 #include "BLI_math.h"
37 #include "BLI_utildefines.h"
38 #include "BLI_bitmap.h"
39
40 #include "BKE_cdderivedmesh.h"
41 #include "BKE_library.h"
42 #include "BKE_library_query.h"
43 #include "BKE_mesh.h"
44 #include "BKE_deform.h"
45
46 #include "MOD_util.h"
47
48
49 static void generate_vert_coordinates(
50         Mesh *mesh, Object *ob, Object *ob_center, const float offset[3],
51         const int num_verts, float (*r_cos)[3], float r_size[3])
52 {
53         float min_co[3], max_co[3];
54         float diff[3];
55         bool do_diff = false;
56
57         INIT_MINMAX(min_co, max_co);
58
59         MVert *mv = mesh->mvert;
60         for (int i = 0; i < mesh->totvert; i++, mv++) {
61                 copy_v3_v3(r_cos[i], mv->co);
62                 if (r_size != NULL && ob_center == NULL) {
63                         minmax_v3v3_v3(min_co, max_co, r_cos[i]);
64                 }
65         }
66
67         /* Get size (i.e. deformation of the spheroid generating normals), either from target object, or own geometry. */
68         if (r_size != NULL) {
69                 if (ob_center != NULL) {
70                         /* Not we are not interested in signs here - they are even troublesome actually, due to security clamping! */
71                         abs_v3_v3(r_size, ob_center->size);
72                 }
73                 else {
74                         /* Set size. */
75                         sub_v3_v3v3(r_size, max_co, min_co);
76                 }
77
78                 /* Error checks - we do not want one or more of our sizes to be null! */
79                 if (is_zero_v3(r_size)) {
80                         r_size[0] = r_size[1] = r_size[2] = 1.0f;
81                 }
82                 else {
83                         CLAMP_MIN(r_size[0], FLT_EPSILON);
84                         CLAMP_MIN(r_size[1], FLT_EPSILON);
85                         CLAMP_MIN(r_size[2], FLT_EPSILON);
86                 }
87         }
88
89         if (ob_center != NULL) {
90                 float inv_obmat[4][4];
91
92                 /* Translate our coordinates so that center of ob_center is at (0, 0, 0). */
93                 /* Get ob_center (world) coordinates in ob local coordinates.
94                  * No need to take into account ob_center's space here, see T44027. */
95                 invert_m4_m4(inv_obmat, ob->obmat);
96                 mul_v3_m4v3(diff, inv_obmat, ob_center->obmat[3]);
97                 negate_v3(diff);
98
99                 do_diff = true;
100         }
101         else if (offset != NULL && !is_zero_v3(offset)) {
102                 negate_v3_v3(diff, offset);
103
104                 do_diff = true;
105         }
106         /* Else, no need to change coordinates! */
107
108         if (do_diff) {
109                 int i = num_verts;
110                 while (i--) {
111                         add_v3_v3(r_cos[i], diff);
112                 }
113         }
114 }
115
116 /* Note this modifies nos_new in-place. */
117 static void mix_normals(
118         const float mix_factor, MDeformVert *dvert, const int defgrp_index, const bool use_invert_vgroup,
119         const float mix_limit, const short mix_mode,
120         const int num_verts, MLoop *mloop, float (*nos_old)[3], float (*nos_new)[3], const int num_loops)
121 {
122         /* Mix with org normals... */
123         float *facs = NULL, *wfac;
124         float (*no_new)[3], (*no_old)[3];
125         int i;
126
127         if (dvert) {
128                 facs = MEM_malloc_arrayN((size_t)num_loops, sizeof(*facs), __func__);
129                 BKE_defvert_extract_vgroup_to_loopweights(
130                             dvert, defgrp_index, num_verts, mloop, num_loops, facs, use_invert_vgroup);
131         }
132
133         for (i = num_loops, no_new = nos_new, no_old = nos_old, wfac = facs; i--; no_new++, no_old++, wfac++) {
134                 const float fac = facs ? *wfac * mix_factor : mix_factor;
135
136                 switch (mix_mode) {
137                         case MOD_NORMALEDIT_MIX_ADD:
138                                 add_v3_v3(*no_new, *no_old);
139                                 normalize_v3(*no_new);
140                                 break;
141                         case MOD_NORMALEDIT_MIX_SUB:
142                                 sub_v3_v3(*no_new, *no_old);
143                                 normalize_v3(*no_new);
144                                 break;
145                         case MOD_NORMALEDIT_MIX_MUL:
146                                 mul_v3_v3(*no_new, *no_old);
147                                 normalize_v3(*no_new);
148                                 break;
149                         case MOD_NORMALEDIT_MIX_COPY:
150                                 break;
151                 }
152
153                 interp_v3_v3v3_slerp_safe(
154                         *no_new, *no_old, *no_new,
155                         (mix_limit < (float)M_PI) ? min_ff(fac, mix_limit / angle_v3v3(*no_new, *no_old)) : fac);
156         }
157
158         MEM_SAFE_FREE(facs);
159 }
160
161 /* Check poly normals and new loop normals are compatible, otherwise flip polygons
162  * (and invert matching poly normals). */
163 static bool polygons_check_flip(
164         MLoop *mloop, float (*nos)[3], CustomData *ldata,
165         MPoly *mpoly, float (*polynors)[3], const int num_polys)
166 {
167         MPoly *mp;
168         MDisps *mdisp = CustomData_get_layer(ldata, CD_MDISPS);
169         int i;
170         bool flipped = false;
171
172         for (i = 0, mp = mpoly; i < num_polys; i++, mp++) {
173                 float norsum[3] = {0.0f};
174                 float (*no)[3];
175                 int j;
176
177                 for (j = 0, no = &nos[mp->loopstart]; j < mp->totloop; j++, no++) {
178                         add_v3_v3(norsum, *no);
179                 }
180
181                 if (!normalize_v3(norsum)) {
182                         continue;
183                 }
184
185                 /* If average of new loop normals is opposed to polygon normal, flip polygon. */
186                 if (dot_v3v3(polynors[i], norsum) < 0.0f) {
187                         BKE_mesh_polygon_flip_ex(mp, mloop, ldata, nos, mdisp, true);
188                         negate_v3(polynors[i]);
189                         flipped = true;
190                 }
191         }
192
193         return flipped;
194 }
195
196 static void normalEditModifier_do_radial(
197         NormalEditModifierData *enmd, Object *ob, Mesh *mesh,
198         short (*clnors)[2], float (*loopnors)[3], float (*polynors)[3],
199         const short mix_mode, const float mix_factor, const float mix_limit,
200         MDeformVert *dvert, const int defgrp_index, const bool use_invert_vgroup,
201         MVert *mvert, const int num_verts, MEdge *medge, const int num_edges,
202         MLoop *mloop, const int num_loops, MPoly *mpoly, const int num_polys)
203 {
204         int i;
205
206         float (*cos)[3] = MEM_malloc_arrayN((size_t)num_verts, sizeof(*cos), __func__);
207         float (*nos)[3] = MEM_malloc_arrayN((size_t)num_loops, sizeof(*nos), __func__);
208         float size[3];
209
210         BLI_bitmap *done_verts = BLI_BITMAP_NEW((size_t)num_verts, __func__);
211
212         generate_vert_coordinates(mesh, ob, enmd->target, enmd->offset, num_verts, cos, size);
213
214         /**
215          * size gives us our spheroid coefficients ``(A, B, C)``.
216          * Then, we want to find out for each vert its (a, b, c) triple (proportional to (A, B, C) one).
217          *
218          * Ellipsoid basic equation: ``(x^2/a^2) + (y^2/b^2) + (z^2/c^2) = 1.``
219          * Since we want to find (a, b, c) matching this equation and proportional to (A, B, C), we can do:
220          * <pre>
221          *     m = B / A
222          *     n = C / A
223          * </pre>
224          *
225          * hence:
226          * <pre>
227          *     (x^2/a^2) + (y^2/b^2) + (z^2/c^2) = 1
228          *  -> b^2*c^2*x^2 + a^2*c^2*y^2 + a^2*b^2*z^2 = a^2*b^2*c^2
229          *     b = ma
230          *     c = na
231          *  -> m^2*a^2*n^2*a^2*x^2 + a^2*n^2*a^2*y^2 + a^2*m^2*a^2*z^2 = a^2*m^2*a^2*n^2*a^2
232          *  -> m^2*n^2*a^4*x^2 + n^2*a^4*y^2 + m^2*a^4*z^2 = m^2*n^2*a^6
233          *  -> a^2 = (m^2*n^2*x^2 + n^2y^2 + m^2z^2) / (m^2*n^2) = x^2 + (y^2 / m^2) + (z^2 / n^2)
234          *  -> b^2 = (m^2*n^2*x^2 + n^2y^2 + m^2z^2) / (n^2)     = (m^2 * x^2) + y^2 + (m^2 * z^2 / n^2)
235          *  -> c^2 = (m^2*n^2*x^2 + n^2y^2 + m^2z^2) / (m^2)     = (n^2 * x^2) + (n^2 * y^2 / m^2) + z^2
236          * </pre>
237          *
238          * All we have to do now is compute normal of the spheroid at that point:
239          * <pre>
240          *     n = (x / a^2, y / b^2, z / c^2)
241          * </pre>
242          * And we are done!
243          */
244         {
245                 const float a = size[0], b = size[1], c = size[2];
246                 const float m2 = (b * b) / (a * a);
247                 const float n2 = (c * c) / (a * a);
248
249                 MLoop *ml;
250                 float (*no)[3];
251
252                 /* We reuse cos to now store the ellipsoid-normal of the verts! */
253                 for (i = num_loops, ml = mloop, no = nos; i-- ; ml++, no++) {
254                         const int vidx = ml->v;
255                         float *co = cos[vidx];
256
257                         if (!BLI_BITMAP_TEST(done_verts, vidx)) {
258                                 const float x2 = co[0] * co[0];
259                                 const float y2 = co[1] * co[1];
260                                 const float z2 = co[2] * co[2];
261                                 const float a2 = x2 + (y2 / m2) + (z2 / n2);
262                                 const float b2 = (m2 * x2) + y2 + (m2 * z2 / n2);
263                                 const float c2 = (n2 * x2) + (n2 * y2 / m2) + z2;
264
265                                 co[0] /= a2;
266                                 co[1] /= b2;
267                                 co[2] /= c2;
268                                 normalize_v3(co);
269
270                                 BLI_BITMAP_ENABLE(done_verts, vidx);
271                         }
272                         copy_v3_v3(*no, co);
273                 }
274         }
275
276         if (loopnors) {
277                 mix_normals(mix_factor, dvert, defgrp_index, use_invert_vgroup,
278                             mix_limit, mix_mode, num_verts, mloop, loopnors, nos, num_loops);
279         }
280
281         if (polygons_check_flip(mloop, nos, &mesh->ldata, mpoly, polynors, num_polys)) {
282                 /* XXX TODO is this still needed? */
283                 // mesh->dirty |= DM_DIRTY_TESS_CDLAYERS;
284                 /* We need to recompute vertex normals! */
285                 BKE_mesh_calc_normals(mesh);
286         }
287
288         BKE_mesh_normals_loop_custom_set(mvert, num_verts, medge, num_edges, mloop, nos, num_loops,
289                                          mpoly, (const float(*)[3])polynors, num_polys, clnors);
290
291         MEM_freeN(cos);
292         MEM_freeN(nos);
293         MEM_freeN(done_verts);
294 }
295
296 static void normalEditModifier_do_directional(
297         NormalEditModifierData *enmd, Object *ob, Mesh *mesh,
298         short (*clnors)[2], float (*loopnors)[3], float (*polynors)[3],
299         const short mix_mode, const float mix_factor, const float mix_limit,
300         MDeformVert *dvert, const int defgrp_index, const bool use_invert_vgroup,
301         MVert *mvert, const int num_verts, MEdge *medge, const int num_edges,
302         MLoop *mloop, const int num_loops, MPoly *mpoly, const int num_polys)
303 {
304         const bool use_parallel_normals = (enmd->flag & MOD_NORMALEDIT_USE_DIRECTION_PARALLEL) != 0;
305
306         float (*nos)[3] = MEM_malloc_arrayN((size_t)num_loops, sizeof(*nos), __func__);
307
308         float target_co[3];
309         int i;
310
311         /* Get target's center coordinates in ob local coordinates. */
312         float mat[4][4];
313
314         invert_m4_m4(mat, ob->obmat);
315         mul_m4_m4m4(mat, mat, enmd->target->obmat);
316         copy_v3_v3(target_co, mat[3]);
317
318         if (use_parallel_normals) {
319                 float no[3];
320
321                 sub_v3_v3v3(no, target_co, enmd->offset);
322                 normalize_v3(no);
323
324                 for (i = num_loops; i--; ) {
325                         copy_v3_v3(nos[i], no);
326                 }
327         }
328         else {
329                 float (*cos)[3] = MEM_malloc_arrayN((size_t)num_verts, sizeof(*cos), __func__);
330                 generate_vert_coordinates(mesh, ob, enmd->target, NULL, num_verts, cos, NULL);
331
332                 BLI_bitmap *done_verts = BLI_BITMAP_NEW((size_t)num_verts, __func__);
333                 MLoop *ml;
334                 float (*no)[3];
335
336                 /* We reuse cos to now store the 'to target' normal of the verts! */
337                 for (i = num_loops, no = nos, ml = mloop; i--; no++, ml++) {
338                         const int vidx = ml->v;
339                         float *co = cos[vidx];
340
341                         if (!BLI_BITMAP_TEST(done_verts, vidx)) {
342                                 sub_v3_v3v3(co, target_co, co);
343                                 normalize_v3(co);
344
345                                 BLI_BITMAP_ENABLE(done_verts, vidx);
346                         }
347
348                         copy_v3_v3(*no, co);
349                 }
350
351                 MEM_freeN(done_verts);
352                 MEM_freeN(cos);
353         }
354
355         if (loopnors) {
356                 mix_normals(mix_factor, dvert, defgrp_index, use_invert_vgroup,
357                             mix_limit, mix_mode, num_verts, mloop, loopnors, nos, num_loops);
358         }
359
360         if (polygons_check_flip(mloop, nos, &mesh->ldata, mpoly, polynors, num_polys)) {
361                 mesh->runtime.cd_dirty_vert |= CD_MASK_NORMAL;
362         }
363
364         BKE_mesh_normals_loop_custom_set(mvert, num_verts, medge, num_edges, mloop, nos, num_loops,
365                                          mpoly, (const float(*)[3])polynors, num_polys, clnors);
366
367         MEM_freeN(nos);
368 }
369
370 static bool is_valid_target(NormalEditModifierData *enmd)
371 {
372         if (enmd->mode == MOD_NORMALEDIT_MODE_RADIAL) {
373                 return true;
374         }
375         else if ((enmd->mode == MOD_NORMALEDIT_MODE_DIRECTIONAL) && enmd->target) {
376                 return true;
377         }
378         modifier_setError((ModifierData *)enmd, "Invalid target settings");
379         return false;
380 }
381
382 static Mesh *normalEditModifier_do(NormalEditModifierData *enmd, Object *ob, Mesh *mesh)
383 {
384         const bool use_invert_vgroup = ((enmd->flag & MOD_NORMALEDIT_INVERT_VGROUP) != 0);
385         const bool use_current_clnors = !((enmd->mix_mode == MOD_NORMALEDIT_MIX_COPY) &&
386                                           (enmd->mix_factor == 1.0f) &&
387                                           (enmd->defgrp_name[0] == '\0') &&
388                                           (enmd->mix_limit == (float)M_PI));
389
390         /* Do not run that modifier at all if autosmooth is disabled! */
391         if (!is_valid_target(enmd) || mesh->totloop == 0) {
392                 return mesh;
393         }
394
395         /* XXX TODO ARG GRRR XYQWNMPRXTYY
396          * Once we fully switch to Mesh evaluation of modifiers, we can expect to get that flag from the COW copy.
397          * But for now, it is lost in the DM intermediate step, so we need to directly check orig object's data. */
398 #if 0
399         if (!(mesh->flag & ME_AUTOSMOOTH)) {
400 #else
401         if (!(((Mesh *)ob->data)->flag & ME_AUTOSMOOTH)) {
402 #endif
403                 modifier_setError((ModifierData *)enmd, "Enable 'Auto Smooth' option in mesh settings");
404                 return mesh;
405         }
406
407         Mesh *result;
408         if (mesh->medge == ((Mesh *)ob->data)->medge) {
409                 /* We need to duplicate data here, otherwise setting custom normals (which may also affect sharp edges) could
410                  * modify org mesh, see T43671. */
411                 BKE_id_copy_ex(
412                         NULL, &mesh->id, (ID **)&result,
413                         LIB_ID_CREATE_NO_MAIN |
414                         LIB_ID_CREATE_NO_USER_REFCOUNT |
415                         LIB_ID_CREATE_NO_DEG_TAG |
416                         LIB_ID_COPY_NO_PREVIEW,
417                         false);
418         }
419         else {
420                 result = mesh;
421         }
422
423         const int num_verts = result->totvert;
424         const int num_edges = result->totedge;
425         const int num_loops = result->totloop;
426         const int num_polys = result->totpoly;
427         MVert *mvert = result->mvert;
428         MEdge *medge = result->medge;
429         MLoop *mloop = result->mloop;
430         MPoly *mpoly = result->mpoly;
431
432         int defgrp_index;
433         MDeformVert *dvert;
434
435         float (*loopnors)[3] = NULL;
436         short (*clnors)[2] = NULL;
437
438         float (*polynors)[3];
439
440         CustomData *ldata = &result->ldata;
441         if (CustomData_has_layer(ldata, CD_NORMAL)) {
442                 loopnors = CustomData_get_layer(ldata, CD_NORMAL);
443         }
444         else {
445                 loopnors = CustomData_add_layer(ldata, CD_NORMAL, CD_CALLOC, NULL, num_loops);
446         }
447
448         /* Compute poly (always needed) and vert normals. */
449         CustomData *pdata = &result->pdata;
450         polynors = CustomData_get_layer(pdata, CD_NORMAL);
451         if (!polynors) {
452                 polynors = CustomData_add_layer(pdata, CD_NORMAL, CD_CALLOC, NULL, num_polys);
453         }
454         BKE_mesh_calc_normals_poly(mvert, NULL, num_verts, mloop, mpoly, num_loops, num_polys, polynors,
455                                    (result->runtime.cd_dirty_vert & CD_MASK_NORMAL) ? false : true);
456
457         result->runtime.cd_dirty_vert &= ~CD_MASK_NORMAL;
458
459         if (use_current_clnors) {
460                 clnors = CustomData_duplicate_referenced_layer(ldata, CD_CUSTOMLOOPNORMAL, num_loops);
461
462                 BKE_mesh_normals_loop_split(mvert, num_verts, medge, num_edges, mloop, loopnors, num_loops,
463                                             mpoly, (const float (*)[3])polynors, num_polys,
464                                             true, result->smoothresh,
465                                             NULL, clnors, NULL);
466         }
467
468         if (!clnors) {
469                 clnors = CustomData_add_layer(ldata, CD_CUSTOMLOOPNORMAL, CD_CALLOC, NULL, num_loops);
470         }
471
472         modifier_get_vgroup_mesh(ob, result, enmd->defgrp_name, &dvert, &defgrp_index);
473
474         if (enmd->mode == MOD_NORMALEDIT_MODE_RADIAL) {
475                 normalEditModifier_do_radial(
476                             enmd, ob, result, clnors, loopnors, polynors,
477                             enmd->mix_mode, enmd->mix_factor, enmd->mix_limit, dvert, defgrp_index, use_invert_vgroup,
478                             mvert, num_verts, medge, num_edges, mloop, num_loops, mpoly, num_polys);
479         }
480         else if (enmd->mode == MOD_NORMALEDIT_MODE_DIRECTIONAL) {
481                 normalEditModifier_do_directional(
482                             enmd, ob, result, clnors, loopnors, polynors,
483                             enmd->mix_mode, enmd->mix_factor, enmd->mix_limit, dvert, defgrp_index, use_invert_vgroup,
484                             mvert, num_verts, medge, num_edges, mloop, num_loops, mpoly, num_polys);
485         }
486
487         return result;
488 }
489
490 static void initData(ModifierData *md)
491 {
492         NormalEditModifierData *enmd = (NormalEditModifierData *)md;
493
494         enmd->mode = MOD_NORMALEDIT_MODE_RADIAL;
495
496         enmd->mix_mode = MOD_NORMALEDIT_MIX_COPY;
497         enmd->mix_factor = 1.0f;
498         enmd->mix_limit = M_PI;
499 }
500
501 static CustomDataMask requiredDataMask(Object *UNUSED(ob), ModifierData *md)
502 {
503         NormalEditModifierData *enmd = (NormalEditModifierData *)md;
504         CustomDataMask dataMask = CD_CUSTOMLOOPNORMAL;
505
506         /* Ask for vertexgroups if we need them. */
507         if (enmd->defgrp_name[0]) {
508                 dataMask |= (CD_MASK_MDEFORMVERT);
509         }
510
511         return dataMask;
512 }
513
514 static bool dependsOnNormals(ModifierData *UNUSED(md))
515 {
516         return true;
517 }
518
519 static void foreachObjectLink(ModifierData *md, Object *ob, ObjectWalkFunc walk, void *userData)
520 {
521         NormalEditModifierData *enmd = (NormalEditModifierData *) md;
522
523         walk(userData, ob, &enmd->target, IDWALK_CB_NOP);
524 }
525
526 static bool isDisabled(ModifierData *md, int UNUSED(useRenderParams))
527 {
528         NormalEditModifierData *enmd = (NormalEditModifierData *)md;
529
530         return !is_valid_target(enmd);
531 }
532
533 static void updateDepsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
534 {
535         NormalEditModifierData *enmd = (NormalEditModifierData *) md;
536         if (enmd->target) {
537                 DEG_add_object_relation(ctx->node, enmd->target, DEG_OB_COMP_TRANSFORM, "NormalEdit Modifier");
538         }
539 }
540
541 static Mesh *applyModifier(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
542 {
543         return normalEditModifier_do((NormalEditModifierData *)md, ctx->object, mesh);
544 }
545
546 ModifierTypeInfo modifierType_NormalEdit = {
547         /* name */              "Set Split Normals",
548         /* structName */        "NormalEditModifierData",
549         /* structSize */        sizeof(NormalEditModifierData),
550         /* type */              eModifierTypeType_Constructive,
551         /* flags */             eModifierTypeFlag_AcceptsMesh |
552                                 eModifierTypeFlag_SupportsMapping |
553                                 eModifierTypeFlag_SupportsEditmode |
554                                 eModifierTypeFlag_EnableInEditmode,
555
556         /* copyData */          modifier_copyData_generic,
557
558         /* deformVerts_DM */    NULL,
559         /* deformMatrices_DM */ NULL,
560         /* deformVertsEM_DM */  NULL,
561         /* deformMatricesEM_DM*/NULL,
562         /* applyModifier_DM */  NULL,
563         /* applyModifierEM_DM */NULL,
564
565         /* deformVerts */       NULL,
566         /* deformMatrices */    NULL,
567         /* deformVertsEM */     NULL,
568         /* deformMatricesEM */  NULL,
569         /* applyModifier */     applyModifier,
570         /* applyModifierEM */   NULL,
571
572         /* initData */          initData,
573         /* requiredDataMask */  requiredDataMask,
574         /* freeData */          NULL,
575         /* isDisabled */        isDisabled,
576         /* updateDepsgraph */   updateDepsgraph,
577         /* dependsOnTime */     NULL,
578         /* dependsOnNormals */  dependsOnNormals,
579         /* foreachObjectLink */ foreachObjectLink,
580         /* foreachIDLink */     NULL,
581         /* foreachTexLink */    NULL,
582 };