Blender V5.0
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
8
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#include "DNA_texture_types.h"
23
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"
33#include "UI_resources.hh"
34
35#include "RNA_access.hh"
36#include "RNA_prototypes.hh"
37
38#include "DEG_depsgraph.hh"
40
41#include "MEM_guardedalloc.h"
42
43#include "MOD_ui_common.hh"
44#include "MOD_util.hh"
45
46#include "RE_texture.h"
47
48/* Displace */
49
58
59static void required_data_mask(ModifierData *md, CustomData_MeshMasks *r_cddata_masks)
60{
62
63 /* Ask for vertex-groups if we need them. */
64 if (dmd->defgrp_name[0] != '\0') {
65 r_cddata_masks->vmask |= CD_MASK_MDEFORMVERT;
66 }
67
68 /* ask for UV coordinates if we need them */
69 if (dmd->texmapping == MOD_DISP_MAP_UV) {
70 r_cddata_masks->fmask |= CD_MASK_MTFACE;
71 }
72}
73
74static bool depends_on_time(Scene * /*scene*/, ModifierData *md)
75{
77
78 if (dmd->texture) {
80 }
81
82 return false;
83}
84
85static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
86{
88
89 walk(user_data, ob, (ID **)&dmd->texture, IDWALK_CB_USER);
90 walk(user_data, ob, (ID **)&dmd->map_object, IDWALK_CB_NOP);
91}
92
93static void foreach_tex_link(ModifierData *md, Object *ob, TexWalkFunc walk, void *user_data)
94{
95 PointerRNA ptr = RNA_pointer_create_discrete(&ob->id, &RNA_Modifier, md);
96 PropertyRNA *prop = RNA_struct_find_property(&ptr, "texture");
97 walk(user_data, ob, md, &ptr, prop);
98}
99
100static bool is_disabled(const Scene * /*scene*/, ModifierData *md, bool /*use_render_params*/)
101{
103 return ((!dmd->texture && dmd->direction == MOD_DISP_DIR_RGB_XYZ) || dmd->strength == 0.0f);
104}
105
107{
109 bool need_transform_relation = false;
110
111 if (dmd->space == MOD_DISP_SPACE_GLOBAL &&
113 {
114 need_transform_relation = true;
115 }
116
117 if (dmd->texture != nullptr) {
118 DEG_add_generic_id_relation(ctx->node, &dmd->texture->id, "Displace Modifier");
119
120 if (dmd->map_object != nullptr && dmd->texmapping == MOD_DISP_MAP_OBJECT) {
122 ctx->node, dmd->map_object, dmd->map_bone, "Displace Modifier");
123 need_transform_relation = true;
124 }
125 if (dmd->texmapping == MOD_DISP_MAP_GLOBAL) {
126 need_transform_relation = true;
127 }
128 }
129
130 if (need_transform_relation) {
131 DEG_add_depends_on_transform_relation(ctx->node, "Displace Modifier");
132 }
133}
134
150
151static void displaceModifier_do_task(void *__restrict userdata,
152 const int iter,
153 const TaskParallelTLS *__restrict /*tls*/)
154{
156 DisplaceModifierData *dmd = data->dmd;
157 const MDeformVert *dvert = data->dvert;
158 const bool invert_vgroup = (dmd->flag & MOD_DISP_INVERT_VGROUP) != 0;
159 float weight = data->weight;
160 int defgrp_index = data->defgrp_index;
161 int direction = data->direction;
162 bool use_global_direction = data->use_global_direction;
163 float (*tex_co)[3] = data->tex_co;
164 blender::MutableSpan<blender::float3> positions = data->positions;
165
166 /* When no texture is used, we fall back to white. */
167 const float delta_fixed = 1.0f - dmd->midlevel;
168
169 TexResult texres;
170 float strength = dmd->strength;
171 float delta;
172 float local_vec[3];
173
174 if (dvert) {
175 weight = invert_vgroup ? 1.0f - BKE_defvert_find_weight(dvert + iter, defgrp_index) :
176 BKE_defvert_find_weight(dvert + iter, defgrp_index);
177 if (weight == 0.0f) {
178 return;
179 }
180 }
181
182 if (data->tex_target) {
183 BKE_texture_get_value_ex(data->tex_target, tex_co[iter], &texres, data->pool, false);
184 delta = texres.tin - dmd->midlevel;
185 }
186 else {
187 delta = delta_fixed; /* (1.0f - dmd->midlevel) */ /* never changes */
188 }
189
190 if (dvert) {
191 strength *= weight;
192 }
193
194 delta *= strength;
195 CLAMP(delta, -10000, 10000);
196
197 switch (direction) {
198 case MOD_DISP_DIR_X:
199 if (use_global_direction) {
200 positions[iter][0] += delta * data->local_mat[0][0];
201 positions[iter][1] += delta * data->local_mat[1][0];
202 positions[iter][2] += delta * data->local_mat[2][0];
203 }
204 else {
205 positions[iter][0] += delta;
206 }
207 break;
208 case MOD_DISP_DIR_Y:
209 if (use_global_direction) {
210 positions[iter][0] += delta * data->local_mat[0][1];
211 positions[iter][1] += delta * data->local_mat[1][1];
212 positions[iter][2] += delta * data->local_mat[2][1];
213 }
214 else {
215 positions[iter][1] += delta;
216 }
217 break;
218 case MOD_DISP_DIR_Z:
219 if (use_global_direction) {
220 positions[iter][0] += delta * data->local_mat[0][2];
221 positions[iter][1] += delta * data->local_mat[1][2];
222 positions[iter][2] += delta * data->local_mat[2][2];
223 }
224 else {
225 positions[iter][2] += delta;
226 }
227 break;
229 local_vec[0] = texres.trgba[0] - dmd->midlevel;
230 local_vec[1] = texres.trgba[1] - dmd->midlevel;
231 local_vec[2] = texres.trgba[2] - dmd->midlevel;
232 if (use_global_direction) {
233 mul_transposed_mat3_m4_v3(data->local_mat, local_vec);
234 }
235 mul_v3_fl(local_vec, strength);
236 add_v3_v3(positions[iter], local_vec);
237 break;
238 case MOD_DISP_DIR_NOR:
240 madd_v3_v3fl(positions[iter], data->vert_normals[iter], delta);
241 break;
242 }
243}
244
246 const ModifierEvalContext *ctx,
247 Mesh *mesh,
249{
250 Object *ob = ctx->object;
251 const MDeformVert *dvert;
252 int direction = dmd->direction;
253 int defgrp_index;
254 float (*tex_co)[3];
255 float weight = 1.0f; /* init value unused but some compilers may complain */
256 const bool use_global_direction = dmd->space == MOD_DISP_SPACE_GLOBAL;
257
258 if (dmd->texture == nullptr && dmd->direction == MOD_DISP_DIR_RGB_XYZ) {
259 return;
260 }
261 if (dmd->strength == 0.0f) {
262 return;
263 }
264
265 MOD_get_vgroup(ob, mesh, dmd->defgrp_name, &dvert, &defgrp_index);
266
267 if (defgrp_index >= 0 && dvert == nullptr) {
268 /* There is a vertex group, but it has no vertices. */
269 return;
270 }
271
272 Tex *tex_target = dmd->texture;
273 if (tex_target != nullptr) {
274 tex_co = MEM_calloc_arrayN<float[3]>(positions.size(), "displaceModifier_do tex_co");
276 ctx,
277 ob,
278 mesh,
279 reinterpret_cast<float (*)[3]>(positions.data()),
280 tex_co);
281
283 }
284 else {
285 tex_co = nullptr;
286 }
287
288 DisplaceUserdata data = {nullptr};
290 data.dmd = dmd;
291 data.dvert = dvert;
292 data.weight = weight;
293 data.defgrp_index = defgrp_index;
294 data.direction = direction;
295 data.use_global_direction = use_global_direction;
296 data.tex_target = tex_target;
297 data.tex_co = tex_co;
298 data.positions = positions;
299 if (direction == MOD_DISP_DIR_NOR) {
300 data.vert_normals = mesh->vert_normals_true();
301 }
302 else if (direction == MOD_DISP_DIR_CLNOR) {
303 data.vert_normals = mesh->vert_normals();
304 }
306 use_global_direction)
307 {
308 copy_m4_m4(data.local_mat, ob->object_to_world().ptr());
309 }
310 if (tex_target != nullptr) {
311 data.pool = BKE_image_pool_new();
312 BKE_texture_fetch_images_for_pool(tex_target, data.pool);
313 }
314 TaskParallelSettings settings;
316 settings.use_threading = (positions.size() > 512);
317 BLI_task_parallel_range(0, positions.size(), &data, displaceModifier_do_task, &settings);
318
319 if (data.pool != nullptr) {
321 }
322
323 if (tex_co) {
324 MEM_freeN(tex_co);
325 }
326}
327
329 const ModifierEvalContext *ctx,
330 Mesh *mesh,
332{
333 displaceModifier_do((DisplaceModifierData *)md, ctx, mesh, positions);
334}
335
336static void panel_draw(const bContext *C, Panel *panel)
337{
338 uiLayout *col;
339 uiLayout *layout = panel->layout;
340
341 PointerRNA ob_ptr;
343
344 PointerRNA obj_data_ptr = RNA_pointer_get(&ob_ptr, "data");
345
346 PointerRNA texture_ptr = RNA_pointer_get(ptr, "texture");
347 bool has_texture = !RNA_pointer_is_null(&texture_ptr);
348 int texture_coords = RNA_enum_get(ptr, "texture_coords");
349
350 layout->use_property_split_set(true);
351
352 uiTemplateID(layout, C, ptr, "texture", "texture.new", nullptr, nullptr);
353
354 col = &layout->column(false);
355 col->active_set(has_texture);
356 col->prop(ptr, "texture_coords", UI_ITEM_NONE, IFACE_("Coordinates"), ICON_NONE);
357 if (texture_coords == MOD_DISP_MAP_OBJECT) {
358 col->prop(ptr, "texture_coords_object", UI_ITEM_NONE, IFACE_("Object"), ICON_NONE);
359 PointerRNA texture_coords_obj_ptr = RNA_pointer_get(ptr, "texture_coords_object");
360 if (!RNA_pointer_is_null(&texture_coords_obj_ptr) &&
361 (RNA_enum_get(&texture_coords_obj_ptr, "type") == OB_ARMATURE))
362 {
363 PointerRNA texture_coords_obj_data_ptr = RNA_pointer_get(&texture_coords_obj_ptr, "data");
364 col->prop_search(ptr,
365 "texture_coords_bone",
366 &texture_coords_obj_data_ptr,
367 "bones",
368 IFACE_("Bone"),
369 ICON_NONE);
370 }
371 }
372 else if (texture_coords == MOD_DISP_MAP_UV && RNA_enum_get(&ob_ptr, "type") == OB_MESH) {
373 col->prop_search(ptr, "uv_layer", &obj_data_ptr, "uv_layers", std::nullopt, ICON_GROUP_UVS);
374 }
375
376 layout->separator();
377
378 col = &layout->column(false);
379 col->prop(ptr, "direction", UI_ITEM_NONE, std::nullopt, ICON_NONE);
380 if (ELEM(RNA_enum_get(ptr, "direction"),
385 {
386 col->prop(ptr, "space", UI_ITEM_NONE, std::nullopt, ICON_NONE);
387 }
388
389 layout->separator();
390
391 col = &layout->column(false);
392 col->prop(ptr, "strength", UI_ITEM_NONE, std::nullopt, ICON_NONE);
393 col->prop(ptr, "mid_level", UI_ITEM_NONE, std::nullopt, ICON_NONE);
394
395 modifier_vgroup_ui(col, ptr, &ob_ptr, "vertex_group", "invert_vertex_group", std::nullopt);
396
398}
399
400static void panel_register(ARegionType *region_type)
401{
403}
404
406 /*idname*/ "Displace",
407 /*name*/ N_("Displace"),
408 /*struct_name*/ "DisplaceModifierData",
409 /*struct_size*/ sizeof(DisplaceModifierData),
410 /*srna*/ &RNA_DisplaceModifier,
413 /*icon*/ ICON_MOD_DISPLACE,
414
415 /*copy_data*/ BKE_modifier_copydata_generic,
416
417 /*deform_verts*/ deform_verts,
418 /*deform_matrices*/ nullptr,
419 /*deform_verts_EM*/ nullptr,
420 /*deform_matrices_EM*/ nullptr,
421 /*modify_mesh*/ nullptr,
422 /*modify_geometry_set*/ nullptr,
423
424 /*init_data*/ init_data,
425 /*required_data_mask*/ required_data_mask,
426 /*free_data*/ nullptr,
427 /*is_disabled*/ is_disabled,
428 /*update_depsgraph*/ update_depsgraph,
429 /*depends_on_time*/ depends_on_time,
430 /*depends_on_normals*/ nullptr,
431 /*foreach_ID_link*/ foreach_ID_link,
432 /*foreach_tex_link*/ foreach_tex_link,
433 /*free_runtime_data*/ nullptr,
434 /*panel_register*/ panel_register,
435 /*blend_write*/ nullptr,
436 /*blend_read*/ nullptr,
437 /*foreach_cache*/ nullptr,
438 /*foreach_working_space_color*/ nullptr,
439};
support for deformation groups and hooks.
float BKE_defvert_find_weight(const MDeformVert *dvert, int defgroup)
Definition deform.cc:774
void BKE_image_pool_free(ImagePool *pool)
ImagePool * BKE_image_pool_new()
@ IDWALK_CB_USER
@ IDWALK_CB_NOP
void(*)(void *user_data, Object *ob, ID **idpoin, LibraryForeachIDCallbackFlag cb_flag) IDWalkFunc
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
void(*)(void *user_data, Object *ob, ModifierData *md, const PointerRNA *ptr, PropertyRNA *texture_prop) TexWalkFunc
bool BKE_texture_dependsOnTime(const struct Tex *texture)
void BKE_texture_fetch_images_for_pool(struct Tex *texture, struct ImagePool *pool)
Definition texture.cc:661
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:616
#define BLI_assert(a)
Definition BLI_assert.h:46
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:221
#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
#define DNA_struct_default_get(struct_name)
@ MOD_DISP_SPACE_GLOBAL
@ eModifierType_Displace
@ MOD_DISP_MAP_OBJECT
@ MOD_DISP_MAP_GLOBAL
@ MOD_DISP_MAP_UV
@ MOD_DISP_INVERT_VGROUP
@ 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
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 deform_verts(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh, blender::MutableSpan< blender::float3 > positions)
static void panel_register(ARegionType *region_type)
static void required_data_mask(ModifierData *, CustomData_MeshMasks *r_cddata_masks)
static void panel_draw(const bContext *, Panel *panel)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static bool depends_on_time(Scene *, ModifierData *)
Definition MOD_build.cc:47
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_vgroup_ui(uiLayout *layout, PointerRNA *ptr, PointerRNA *ob_ptr, const StringRefNull vgroup_prop, const std::optional< StringRefNull > invert_vgroup_prop, const std::optional< StringRefNull > text)
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_error_message_draw(uiLayout *layout, PointerRNA *ptr)
void MOD_init_texture(MappingInfoModifierData *dmd, const ModifierEvalContext *ctx)
Definition MOD_util.cc:36
void MOD_depsgraph_update_object_bone_relation(DepsNodeHandle *node, Object *object, const char *bonename, const char *description)
Definition MOD_util.cc:186
void MOD_get_vgroup(const Object *ob, const Mesh *mesh, const char *name, const MDeformVert **dvert, int *defgrp_index)
Definition MOD_util.cc:156
void MOD_get_texture_coords(MappingInfoModifierData *dmd, const ModifierEvalContext *, Object *ob, Mesh *mesh, float(*cos)[3], float(*r_texco)[3])
Definition MOD_util.cc:49
#define C
Definition RandGen.cpp:29
void uiTemplateID(uiLayout *layout, const bContext *C, PointerRNA *ptr, blender::StringRefNull propname, const char *newop, const char *openop, const char *unlinkop, int filter=UI_TEMPLATE_ID_FILTER_ALL, bool live_icon=false, std::optional< blender::StringRef > text=std::nullopt)
#define UI_ITEM_NONE
BMesh const char void * data
constexpr int64_t size() const
Definition BLI_span.hh:493
constexpr T * data() const
Definition BLI_span.hh:539
nullptr float
uint col
static void update_depsgraph(tGraphSliderOp *gso)
void * MEM_calloc_arrayN(size_t len, size_t size, const char *str)
Definition mallocn.cc:123
void MEM_freeN(void *vmemh)
Definition mallocn.cc:113
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_discrete(ID *id, StructRNA *type, void *data)
int RNA_enum_get(PointerRNA *ptr, const char *name)
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:414
struct uiLayout * layout
float tin
Definition RE_texture.h:59
float trgba[4]
Definition RE_texture.h:60
uiLayout & column(bool align)
void separator(float factor=1.0f, LayoutSeparatorType type=LayoutSeparatorType::Auto)
void use_property_split_set(bool value)
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238