Blender V4.3
MOD_displace.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2005 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
9#include "BLI_utildefines.h"
10
11#include "BLI_math_matrix.h"
12#include "BLI_math_vector.h"
13#include "BLI_task.h"
14
15#include "BLT_translation.hh"
16
17#include "DNA_defaults.h"
18#include "DNA_mesh_types.h"
19#include "DNA_meshdata_types.h"
20#include "DNA_object_types.h"
21#include "DNA_screen_types.h"
22
23#include "BKE_customdata.hh"
24#include "BKE_deform.hh"
25#include "BKE_image.hh"
26#include "BKE_lib_query.hh"
27#include "BKE_mesh.hh"
28#include "BKE_modifier.hh"
29#include "BKE_texture.h"
30
31#include "UI_interface.hh"
32#include "UI_resources.hh"
33
34#include "RNA_access.hh"
35#include "RNA_prototypes.hh"
36
37#include "DEG_depsgraph.hh"
39
40#include "MEM_guardedalloc.h"
41
42#include "MOD_ui_common.hh"
43#include "MOD_util.hh"
44
45#include "RE_texture.h"
46
47/* Displace */
48
57
58static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
59{
61
62 /* Ask for vertex-groups if we need them. */
63 if (dmd->defgrp_name[0] != '\0') {
64 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
65 }
66
67 /* ask for UV coordinates if we need them */
68 if (dmd->texmapping == MOD_DISP_MAP_UV) {
69 r_cddata_masks->fmask |= CD_MASK_MTFACE;
70 }
71
72 if (dmd->direction == MOD_DISP_DIR_CLNOR) {
73 r_cddata_masks->lmask |= CD_MASK_CUSTOMLOOPNORMAL;
74 }
75}
76
77static bool depends_on_time(Scene * /*scene*/, ModifierData *md)
78{
80
81 if (dmd->texture) {
83 }
84
85 return false;
86}
87
88static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
89{
91
92 walk(user_data, ob, (ID **)&dmd->texture, IDWALK_CB_USER);
93 walk(user_data, ob, (ID **)&dmd->map_object, IDWALK_CB_NOP);
94}
95
96static void foreach_tex_link(ModifierData *md, Object *ob, TexWalkFunc walk, void *user_data)
97{
98 PointerRNA ptr = RNA_pointer_create(&ob->id, &RNA_Modifier, md);
99 PropertyRNA *prop = RNA_struct_find_property(&ptr, "texture");
100 walk(user_data, ob, md, &ptr, prop);
101}
102
103static bool is_disabled(const Scene * /*scene*/, ModifierData *md, bool /*use_render_params*/)
104{
106 return ((!dmd->texture && dmd->direction == MOD_DISP_DIR_RGB_XYZ) || dmd->strength == 0.0f);
107}
108
110{
112 bool need_transform_relation = false;
113
114 if (dmd->space == MOD_DISP_SPACE_GLOBAL &&
116 {
117 need_transform_relation = true;
118 }
119
120 if (dmd->texture != nullptr) {
121 DEG_add_generic_id_relation(ctx->node, &dmd->texture->id, "Displace Modifier");
122
123 if (dmd->map_object != nullptr && dmd->texmapping == MOD_DISP_MAP_OBJECT) {
125 ctx->node, dmd->map_object, dmd->map_bone, "Displace Modifier");
126 need_transform_relation = true;
127 }
128 if (dmd->texmapping == MOD_DISP_MAP_GLOBAL) {
129 need_transform_relation = true;
130 }
131 }
132
133 if (need_transform_relation) {
134 DEG_add_depends_on_transform_relation(ctx->node, "Displace Modifier");
135 }
136}
137
154
155static void displaceModifier_do_task(void *__restrict userdata,
156 const int iter,
157 const TaskParallelTLS *__restrict /*tls*/)
158{
159 DisplaceUserdata *data = (DisplaceUserdata *)userdata;
160 DisplaceModifierData *dmd = data->dmd;
161 const MDeformVert *dvert = data->dvert;
162 const bool invert_vgroup = (dmd->flag & MOD_DISP_INVERT_VGROUP) != 0;
163 float weight = data->weight;
164 int defgrp_index = data->defgrp_index;
165 int direction = data->direction;
166 bool use_global_direction = data->use_global_direction;
167 float(*tex_co)[3] = data->tex_co;
168 blender::MutableSpan<blender::float3> positions = data->positions;
169 float(*vert_clnors)[3] = data->vert_clnors;
170
171 /* When no texture is used, we fallback to white. */
172 const float delta_fixed = 1.0f - dmd->midlevel;
173
174 TexResult texres;
175 float strength = dmd->strength;
176 float delta;
177 float local_vec[3];
178
179 if (dvert) {
180 weight = invert_vgroup ? 1.0f - BKE_defvert_find_weight(dvert + iter, defgrp_index) :
181 BKE_defvert_find_weight(dvert + iter, defgrp_index);
182 if (weight == 0.0f) {
183 return;
184 }
185 }
186
187 if (data->tex_target) {
188 BKE_texture_get_value_ex(data->tex_target, tex_co[iter], &texres, data->pool, false);
189 delta = texres.tin - dmd->midlevel;
190 }
191 else {
192 delta = delta_fixed; /* (1.0f - dmd->midlevel) */ /* never changes */
193 }
194
195 if (dvert) {
196 strength *= weight;
197 }
198
199 delta *= strength;
200 CLAMP(delta, -10000, 10000);
201
202 switch (direction) {
203 case MOD_DISP_DIR_X:
204 if (use_global_direction) {
205 positions[iter][0] += delta * data->local_mat[0][0];
206 positions[iter][1] += delta * data->local_mat[1][0];
207 positions[iter][2] += delta * data->local_mat[2][0];
208 }
209 else {
210 positions[iter][0] += delta;
211 }
212 break;
213 case MOD_DISP_DIR_Y:
214 if (use_global_direction) {
215 positions[iter][0] += delta * data->local_mat[0][1];
216 positions[iter][1] += delta * data->local_mat[1][1];
217 positions[iter][2] += delta * data->local_mat[2][1];
218 }
219 else {
220 positions[iter][1] += delta;
221 }
222 break;
223 case MOD_DISP_DIR_Z:
224 if (use_global_direction) {
225 positions[iter][0] += delta * data->local_mat[0][2];
226 positions[iter][1] += delta * data->local_mat[1][2];
227 positions[iter][2] += delta * data->local_mat[2][2];
228 }
229 else {
230 positions[iter][2] += delta;
231 }
232 break;
234 local_vec[0] = texres.trgba[0] - dmd->midlevel;
235 local_vec[1] = texres.trgba[1] - dmd->midlevel;
236 local_vec[2] = texres.trgba[2] - dmd->midlevel;
237 if (use_global_direction) {
238 mul_transposed_mat3_m4_v3(data->local_mat, local_vec);
239 }
240 mul_v3_fl(local_vec, strength);
241 add_v3_v3(positions[iter], local_vec);
242 break;
243 case MOD_DISP_DIR_NOR:
244 madd_v3_v3fl(positions[iter], data->vert_normals[iter], delta);
245 break;
247 madd_v3_v3fl(positions[iter], vert_clnors[iter], delta);
248 break;
249 }
250}
251
253 const ModifierEvalContext *ctx,
254 Mesh *mesh,
256{
257 Object *ob = ctx->object;
258 const MDeformVert *dvert;
259 int direction = dmd->direction;
260 int defgrp_index;
261 float(*tex_co)[3];
262 float weight = 1.0f; /* init value unused but some compilers may complain */
263 float(*vert_clnors)[3] = nullptr;
264 float local_mat[4][4] = {{0}};
265 const bool use_global_direction = dmd->space == MOD_DISP_SPACE_GLOBAL;
266
267 if (dmd->texture == nullptr && dmd->direction == MOD_DISP_DIR_RGB_XYZ) {
268 return;
269 }
270 if (dmd->strength == 0.0f) {
271 return;
272 }
273
274 MOD_get_vgroup(ob, mesh, dmd->defgrp_name, &dvert, &defgrp_index);
275
276 if (defgrp_index >= 0 && dvert == nullptr) {
277 /* There is a vertex group, but it has no vertices. */
278 return;
279 }
280
281 Tex *tex_target = dmd->texture;
282 if (tex_target != nullptr) {
283 tex_co = static_cast<float(*)[3]>(MEM_calloc_arrayN(
284 size_t(positions.size()), sizeof(*tex_co), "displaceModifier_do tex_co"));
286 ctx,
287 ob,
288 mesh,
289 reinterpret_cast<float(*)[3]>(positions.data()),
290 tex_co);
291
293 }
294 else {
295 tex_co = nullptr;
296 }
297
298 if (direction == MOD_DISP_DIR_CLNOR) {
299 if (CustomData_has_layer(&mesh->corner_data, CD_CUSTOMLOOPNORMAL)) {
300 vert_clnors = static_cast<float(*)[3]>(
301 MEM_malloc_arrayN(positions.size(), sizeof(*vert_clnors), __func__));
303 positions.size(),
304 mesh->corner_verts().data(),
305 mesh->corners_num,
306 reinterpret_cast<const float(*)[3]>(mesh->corner_normals().data()),
307 vert_clnors);
308 }
309 else {
310 direction = MOD_DISP_DIR_NOR;
311 }
312 }
314 use_global_direction)
315 {
316 copy_m4_m4(local_mat, ob->object_to_world().ptr());
317 }
318
319 DisplaceUserdata data = {nullptr};
321 data.dmd = dmd;
322 data.dvert = dvert;
323 data.weight = weight;
324 data.defgrp_index = defgrp_index;
325 data.direction = direction;
326 data.use_global_direction = use_global_direction;
327 data.tex_target = tex_target;
328 data.tex_co = tex_co;
329 data.positions = positions;
330 copy_m4_m4(data.local_mat, local_mat);
331 if (direction == MOD_DISP_DIR_NOR) {
332 data.vert_normals = mesh->vert_normals();
333 }
334 data.vert_clnors = vert_clnors;
335 if (tex_target != nullptr) {
336 data.pool = BKE_image_pool_new();
337 BKE_texture_fetch_images_for_pool(tex_target, data.pool);
338 }
339 TaskParallelSettings settings;
341 settings.use_threading = (positions.size() > 512);
342 BLI_task_parallel_range(0, positions.size(), &data, displaceModifier_do_task, &settings);
343
344 if (data.pool != nullptr) {
345 BKE_image_pool_free(data.pool);
346 }
347
348 if (tex_co) {
349 MEM_freeN(tex_co);
350 }
351
352 if (vert_clnors) {
353 MEM_freeN(vert_clnors);
354 }
355}
356
358 const ModifierEvalContext *ctx,
359 Mesh *mesh,
361{
362 displaceModifier_do((DisplaceModifierData *)md, ctx, mesh, positions);
363}
364
365static void panel_draw(const bContext *C, Panel *panel)
366{
367 uiLayout *col;
368 uiLayout *layout = panel->layout;
369
370 PointerRNA ob_ptr;
372
373 PointerRNA obj_data_ptr = RNA_pointer_get(&ob_ptr, "data");
374
375 PointerRNA texture_ptr = RNA_pointer_get(ptr, "texture");
376 bool has_texture = !RNA_pointer_is_null(&texture_ptr);
377 int texture_coords = RNA_enum_get(ptr, "texture_coords");
378
379 uiLayoutSetPropSep(layout, true);
380
381 uiTemplateID(layout, C, ptr, "texture", "texture.new", nullptr, nullptr);
382
383 col = uiLayoutColumn(layout, false);
384 uiLayoutSetActive(col, has_texture);
385 uiItemR(col, ptr, "texture_coords", UI_ITEM_NONE, IFACE_("Coordinates"), ICON_NONE);
386 if (texture_coords == MOD_DISP_MAP_OBJECT) {
387 uiItemR(col, ptr, "texture_coords_object", UI_ITEM_NONE, IFACE_("Object"), ICON_NONE);
388 PointerRNA texture_coords_obj_ptr = RNA_pointer_get(ptr, "texture_coords_object");
389 if (!RNA_pointer_is_null(&texture_coords_obj_ptr) &&
390 (RNA_enum_get(&texture_coords_obj_ptr, "type") == OB_ARMATURE))
391 {
392 PointerRNA texture_coords_obj_data_ptr = RNA_pointer_get(&texture_coords_obj_ptr, "data");
394 ptr,
395 "texture_coords_bone",
396 &texture_coords_obj_data_ptr,
397 "bones",
398 IFACE_("Bone"),
399 ICON_NONE);
400 }
401 }
402 else if (texture_coords == MOD_DISP_MAP_UV && RNA_enum_get(&ob_ptr, "type") == OB_MESH) {
403 uiItemPointerR(col, ptr, "uv_layer", &obj_data_ptr, "uv_layers", nullptr, ICON_GROUP_UVS);
404 }
405
406 uiItemS(layout);
407
408 col = uiLayoutColumn(layout, false);
409 uiItemR(col, ptr, "direction", UI_ITEM_NONE, nullptr, ICON_NONE);
410 if (ELEM(RNA_enum_get(ptr, "direction"),
415 {
416 uiItemR(col, ptr, "space", UI_ITEM_NONE, nullptr, ICON_NONE);
417 }
418
419 uiItemS(layout);
420
421 col = uiLayoutColumn(layout, false);
422 uiItemR(col, ptr, "strength", UI_ITEM_NONE, nullptr, ICON_NONE);
423 uiItemR(col, ptr, "mid_level", UI_ITEM_NONE, nullptr, ICON_NONE);
424
425 modifier_vgroup_ui(col, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", nullptr);
426
427 modifier_panel_end(layout, ptr);
428}
429
430static void panel_register(ARegionType *region_type)
431{
433}
434
436 /*idname*/ "Displace",
437 /*name*/ N_("Displace"),
438 /*struct_name*/ "DisplaceModifierData",
439 /*struct_size*/ sizeof(DisplaceModifierData),
440 /*srna*/ &RNA_DisplaceModifier,
443 /*icon*/ ICON_MOD_DISPLACE,
444
445 /*copy_data*/ BKE_modifier_copydata_generic,
446
447 /*deform_verts*/ deform_verts,
448 /*deform_matrices*/ nullptr,
449 /*deform_verts_EM*/ nullptr,
450 /*deform_matrices_EM*/ nullptr,
451 /*modify_mesh*/ nullptr,
452 /*modify_geometry_set*/ nullptr,
453
454 /*init_data*/ init_data,
455 /*required_data_mask*/ required_data_mask,
456 /*free_data*/ nullptr,
457 /*is_disabled*/ is_disabled,
458 /*update_depsgraph*/ update_depsgraph,
459 /*depends_on_time*/ depends_on_time,
460 /*depends_on_normals*/ nullptr,
461 /*foreach_ID_link*/ foreach_ID_link,
462 /*foreach_tex_link*/ foreach_tex_link,
463 /*free_runtime_data*/ nullptr,
464 /*panel_register*/ panel_register,
465 /*blend_write*/ nullptr,
466 /*blend_read*/ nullptr,
467 /*foreach_cache*/ nullptr,
468};
CustomData interface, see also DNA_customdata_types.h.
bool CustomData_has_layer(const CustomData *data, eCustomDataType type)
support for deformation groups and hooks.
float BKE_defvert_find_weight(const MDeformVert *dvert, int defgroup)
Definition deform.cc:770
void BKE_image_pool_free(ImagePool *pool)
ImagePool * BKE_image_pool_new(void)
@ IDWALK_CB_USER
@ IDWALK_CB_NOP
void BKE_mesh_normals_loop_to_vertex(int numVerts, const int *corner_verts, int numLoops, const float(*clnors)[3], float(*r_vert_clnors)[3])
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
void(*)(void *user_data, Object *ob, ModifierData *md, const PointerRNA *ptr, PropertyRNA *texture_prop) TexWalkFunc
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
void(*)(void *user_data, Object *ob, ID **idpoin, int cb_flag) IDWalkFunc
bool BKE_texture_dependsOnTime(const struct Tex *texture)
void BKE_texture_fetch_images_for_pool(struct Tex *texture, struct ImagePool *pool)
Definition texture.cc:745
void BKE_texture_get_value_ex(struct Tex *texture, const float *tex_co, struct TexResult *texres, struct ImagePool *pool, bool use_color_management)
Definition texture.cc:700
#define BLI_assert(a)
Definition BLI_assert.h:50
void copy_m4_m4(float m1[4][4], const float m2[4][4])
void mul_transposed_mat3_m4_v3(const float M[4][4], float r[3])
MINLINE void madd_v3_v3fl(float r[3], const float a[3], float f)
MINLINE void mul_v3_fl(float r[3], float f)
MINLINE void add_v3_v3(float r[3], const float a[3])
void BLI_task_parallel_range(int start, int stop, void *userdata, TaskParallelRangeFunc func, const TaskParallelSettings *settings)
Definition task_range.cc:99
BLI_INLINE void BLI_parallel_range_settings_defaults(TaskParallelSettings *settings)
Definition BLI_task.h:230
#define CLAMP(a, b, c)
#define ELEM(...)
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#define IFACE_(msgid)
void DEG_add_generic_id_relation(DepsNodeHandle *node_handle, ID *id, const char *description)
void DEG_add_depends_on_transform_relation(DepsNodeHandle *node_handle, const char *description)
Scene * DEG_get_evaluated_scene(const Depsgraph *graph)
#define CD_MASK_MDEFORMVERT
#define CD_MASK_MTFACE
@ CD_CUSTOMLOOPNORMAL
#define CD_MASK_CUSTOMLOOPNORMAL
#define DNA_struct_default_get(struct_name)
@ MOD_DISP_MAP_OBJECT
@ MOD_DISP_MAP_GLOBAL
@ MOD_DISP_MAP_UV
@ MOD_DISP_SPACE_GLOBAL
@ eModifierType_Displace
@ MOD_DISP_DIR_RGB_XYZ
@ MOD_DISP_DIR_Z
@ MOD_DISP_DIR_Y
@ MOD_DISP_DIR_NOR
@ MOD_DISP_DIR_X
@ MOD_DISP_DIR_CLNOR
@ MOD_DISP_INVERT_VGROUP
struct DisplaceModifierData DisplaceModifierData
Object is a sort of wrapper for general info.
@ OB_ARMATURE
@ OB_MESH
static bool is_disabled
Read Guarded memory(de)allocation.
static void init_data(ModifierData *md)
static void displaceModifier_do(DisplaceModifierData *dmd, const ModifierEvalContext *ctx, Mesh *mesh, blender::MutableSpan< blender::float3 > positions)
static void deform_verts(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh, blender::MutableSpan< blender::float3 > positions)
static void panel_register(ARegionType *region_type)
static void displaceModifier_do_task(void *__restrict userdata, const int iter, const TaskParallelTLS *__restrict)
static bool depends_on_time(Scene *, ModifierData *md)
static void foreach_tex_link(ModifierData *md, Object *ob, TexWalkFunc walk, void *user_data)
static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
ModifierTypeInfo modifierType_Displace
static void panel_draw(const bContext *C, Panel *panel)
void modifier_panel_end(uiLayout *layout, PointerRNA *ptr)
PanelType * modifier_panel_register(ARegionType *region_type, ModifierType type, PanelDrawFn draw)
PointerRNA * modifier_panel_get_property_pointers(Panel *panel, PointerRNA *r_ob_ptr)
void modifier_vgroup_ui(uiLayout *layout, PointerRNA *ptr, PointerRNA *ob_ptr, const char *vgroup_prop, const char *invert_vgroup_prop, const char *text)
void MOD_init_texture(MappingInfoModifierData *dmd, const ModifierEvalContext *ctx)
Definition MOD_util.cc:39
void MOD_depsgraph_update_object_bone_relation(DepsNodeHandle *node, Object *object, const char *bonename, const char *description)
Definition MOD_util.cc:189
void MOD_get_vgroup(const Object *ob, const Mesh *mesh, const char *name, const MDeformVert **dvert, int *defgrp_index)
Definition MOD_util.cc:159
void MOD_get_texture_coords(MappingInfoModifierData *dmd, const ModifierEvalContext *, Object *ob, Mesh *mesh, float(*cos)[3], float(*r_texco)[3])
Definition MOD_util.cc:52
void uiLayoutSetActive(uiLayout *layout, bool active)
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
void uiItemS(uiLayout *layout)
#define UI_ITEM_NONE
void uiItemPointerR(uiLayout *layout, PointerRNA *ptr, const char *propname, PointerRNA *searchptr, const char *searchpropname, const char *name, int icon)
void uiTemplateID(uiLayout *layout, const bContext *C, PointerRNA *ptr, const char *propname, const char *newop, const char *openop, const char *unlinkop, int filter=UI_TEMPLATE_ID_FILTER_ALL, bool live_icon=false, const char *text=nullptr)
uiLayout * uiLayoutColumn(uiLayout *layout, bool align)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
draw_view in_light_buf[] float
uint col
void *(* MEM_malloc_arrayN)(size_t len, size_t size, const char *str)
Definition mallocn.cc:45
void *(* MEM_calloc_arrayN)(size_t len, size_t size, const char *str)
Definition mallocn.cc:43
void MEM_freeN(void *vmemh)
Definition mallocn.cc:105
PointerRNA RNA_pointer_get(PointerRNA *ptr, const char *name)
PropertyRNA * RNA_struct_find_property(PointerRNA *ptr, const char *identifier)
bool RNA_pointer_is_null(const PointerRNA *ptr)
PointerRNA RNA_pointer_create(ID *id, StructRNA *type, void *data)
int RNA_enum_get(PointerRNA *ptr, const char *name)
float(* vert_clnors)[3]
const MDeformVert * dvert
float(* tex_co)[3]
blender::MutableSpan< blender::float3 > positions
DisplaceModifierData * dmd
float local_mat[4][4]
blender::Span< blender::float3 > vert_normals
Definition DNA_ID.h:413
struct uiLayout * layout
float tin
Definition RE_texture.h:87
float trgba[4]
Definition RE_texture.h:88
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4126