Blender V5.0
MOD_mesh_to_volume.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2023 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
9#include <vector>
10
11#include "BKE_geometry_set.hh"
12#include "BKE_lib_id.hh"
13#include "BKE_lib_query.hh"
14#include "BKE_mesh.hh"
15#include "BKE_mesh_wrapper.hh"
16#include "BKE_modifier.hh"
17#include "BKE_volume.hh"
18
19#include "BLT_translation.hh"
20
21#include "DNA_mesh_types.h"
22#include "DNA_object_types.h"
23#include "DNA_screen_types.h"
24
25#include "GEO_mesh_to_volume.hh"
26
28#include "UI_resources.hh"
29
30#include "MOD_ui_common.hh"
31
33
34#include "RNA_prototypes.hh"
35#include "RNA_types.hh"
36
37static void init_data(ModifierData *md)
38{
39 MeshToVolumeModifierData *mvmd = reinterpret_cast<MeshToVolumeModifierData *>(md);
40 mvmd->object = nullptr;
42 mvmd->voxel_size = 0.1f;
43 mvmd->voxel_amount = 32;
44 mvmd->interior_band_width = 0.2f;
45 mvmd->density = 1.0f;
46}
47
49{
50 MeshToVolumeModifierData *mvmd = reinterpret_cast<MeshToVolumeModifierData *>(md);
51 DEG_add_depends_on_transform_relation(ctx->node, "Mesh to Volume Modifier");
52 if (mvmd->object) {
54 ctx->node, mvmd->object, DEG_OB_COMP_GEOMETRY, "Mesh to Volume Modifier");
56 ctx->node, mvmd->object, DEG_OB_COMP_TRANSFORM, "Mesh to Volume Modifier");
57 }
58}
59
60static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
61{
62 MeshToVolumeModifierData *mvmd = reinterpret_cast<MeshToVolumeModifierData *>(md);
63 walk(user_data, ob, (ID **)&mvmd->object, IDWALK_CB_NOP);
64}
65
66static void panel_draw(const bContext * /*C*/, Panel *panel)
67{
68 uiLayout *layout = panel->layout;
69
71 MeshToVolumeModifierData *mvmd = static_cast<MeshToVolumeModifierData *>(ptr->data);
72
73 layout->use_property_split_set(true);
74
75 layout->prop(ptr, "object", UI_ITEM_NONE, std::nullopt, ICON_NONE);
76 layout->prop(ptr, "density", UI_ITEM_NONE, std::nullopt, ICON_NONE);
77
78 {
79 uiLayout *col = &layout->column(false);
80 col->prop(ptr, "interior_band_width", UI_ITEM_NONE, std::nullopt, ICON_NONE);
81 }
82 {
83 uiLayout *col = &layout->column(false);
84 col->prop(ptr, "resolution_mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
86 col->prop(ptr, "voxel_amount", UI_ITEM_NONE, std::nullopt, ICON_NONE);
87 }
88 else {
89 col->prop(ptr, "voxel_size", UI_ITEM_NONE, std::nullopt, ICON_NONE);
90 }
91 }
92
94}
95
96static void panel_register(ARegionType *region_type)
97{
99}
100
102 const ModifierEvalContext *ctx,
103 Volume *input_volume)
104{
105#ifdef WITH_OPENVDB
106 using namespace blender;
107
108 MeshToVolumeModifierData *mvmd = reinterpret_cast<MeshToVolumeModifierData *>(md);
109 Object *object_to_convert = mvmd->object;
110
111 if (object_to_convert == nullptr) {
112 return input_volume;
113 }
115 if (mesh == nullptr) {
116 return input_volume;
117 }
119 if (mesh->verts_num == 0) {
120 return input_volume;
121 }
122
123 const float4x4 mesh_to_own_object_space_transform = ctx->object->world_to_object() *
124 object_to_convert->object_to_world();
128 resolution.settings.voxel_amount = mvmd->voxel_amount;
129 if (resolution.settings.voxel_amount < 1.0f) {
130 return input_volume;
131 }
132 }
133 else if (resolution.mode == MESH_TO_VOLUME_RESOLUTION_MODE_VOXEL_SIZE) {
134 resolution.settings.voxel_size = mvmd->voxel_size;
135 if (resolution.settings.voxel_size < 1e-5f) {
136 return input_volume;
137 }
138 }
139
140 const float voxel_size = geometry::volume_compute_voxel_size(
141 ctx->depsgraph,
142 [&]() { return *mesh->bounds_min_max(); },
143 resolution,
144 0.0f,
145 mesh_to_own_object_space_transform);
146
147 /* Create a new volume. */
148 Volume *volume;
149 if (input_volume == nullptr) {
150 volume = BKE_id_new_nomain<Volume>("Volume");
151 }
152 else {
153 volume = BKE_volume_new_for_eval(input_volume);
154 }
155
156 /* Convert mesh to grid and add to volume. */
157 geometry::fog_volume_grid_add_from_mesh(volume,
158 "density",
159 mesh->vert_positions(),
160 mesh->corner_verts(),
161 mesh->corner_tris(),
162 mesh_to_own_object_space_transform,
163 voxel_size,
165 mvmd->density);
166
167 return volume;
168
169#else
170 UNUSED_VARS(md);
171 BKE_modifier_set_error(ctx->object, md, "Compiled without OpenVDB");
172 return input_volume;
173#endif
174}
175
177 const ModifierEvalContext *ctx,
178 blender::bke::GeometrySet *geometry_set)
179{
180 Volume *input_volume = geometry_set->get_volume_for_write();
181 Volume *result_volume = mesh_to_volume(md, ctx, input_volume);
182 if (result_volume != input_volume) {
183 geometry_set->replace_volume(result_volume);
184 }
185}
186
188 /*idname*/ "Mesh to Volume",
189 /*name*/ N_("Mesh to Volume"),
190 /*struct_name*/ "MeshToVolumeModifierData",
191 /*struct_size*/ sizeof(MeshToVolumeModifierData),
192 /*srna*/ &RNA_MeshToVolumeModifier,
194 /*flags*/ static_cast<ModifierTypeFlag>(0),
195 /*icon*/ ICON_VOLUME_DATA, /* TODO: Use correct icon. */
196
197 /*copy_data*/ BKE_modifier_copydata_generic,
198
199 /*deform_verts*/ nullptr,
200 /*deform_matrices*/ nullptr,
201 /*deform_verts_EM*/ nullptr,
202 /*deform_matrices_EM*/ nullptr,
203 /*modify_mesh*/ nullptr,
204 /*modify_geometry_set*/ modify_geometry_set,
205
206 /*init_data*/ init_data,
207 /*required_data_mask*/ nullptr,
208 /*free_data*/ nullptr,
209 /*is_disabled*/ nullptr,
210 /*update_depsgraph*/ update_depsgraph,
211 /*depends_on_time*/ nullptr,
212 /*depends_on_normals*/ nullptr,
213 /*foreach_ID_link*/ foreach_ID_link,
214 /*foreach_tex_link*/ nullptr,
215 /*free_runtime_data*/ nullptr,
216 /*panel_register*/ panel_register,
217 /*blend_write*/ nullptr,
218 /*blend_read*/ nullptr,
219 /*foreach_cache*/ nullptr,
220 /*foreach_working_space_color*/ nullptr,
221};
void * BKE_id_new_nomain(short type, const char *name)
Definition lib_id.cc:1519
@ IDWALK_CB_NOP
void BKE_mesh_wrapper_ensure_mdata(Mesh *mesh)
void(*)(void *user_data, Object *ob, ID **idpoin, LibraryForeachIDCallbackFlag cb_flag) IDWalkFunc
Mesh * BKE_modifier_get_evaluated_mesh_from_evaluated_object(Object *ob_eval)
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
ModifierTypeFlag
void BKE_modifier_set_error(const Object *ob, ModifierData *md, const char *format,...) ATTR_PRINTF_FORMAT(3
Volume data-block.
Volume * BKE_volume_new_for_eval(const Volume *volume_src)
#define UNUSED_VARS(...)
void DEG_add_depends_on_transform_relation(DepsNodeHandle *node_handle, const char *description)
void DEG_add_object_relation(DepsNodeHandle *node_handle, Object *object, eDepsObjectComponentType component, const char *description)
@ DEG_OB_COMP_GEOMETRY
@ DEG_OB_COMP_TRANSFORM
@ eModifierType_MeshToVolume
MeshToVolumeModifierResolutionMode
@ MESH_TO_VOLUME_RESOLUTION_MODE_VOXEL_SIZE
@ MESH_TO_VOLUME_RESOLUTION_MODE_VOXEL_AMOUNT
Object is a sort of wrapper for general info.
static void init_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static void panel_draw(const bContext *, Panel *panel)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static void init_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
ModifierTypeInfo modifierType_MeshToVolume
static void panel_draw(const bContext *, Panel *panel)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, blender::bke::GeometrySet *geometry_set)
static Volume * mesh_to_volume(ModifierData *md, const ModifierEvalContext *ctx, Volume *input_volume)
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)
#define UI_ITEM_NONE
uint col
MatBase< 4, 4 > float4x4
static void update_depsgraph(tGraphSliderOp *gso)
Definition DNA_ID.h:414
int verts_num
struct uiLayout * layout
void replace_volume(Volume *volume, GeometryOwnershipType ownership=GeometryOwnershipType::Owned)
union blender::geometry::MeshToVolumeResolution::@117353126071235122164361234052245271154241141031 settings
MeshToVolumeModifierResolutionMode mode
uiLayout & column(bool align)
void use_property_split_set(bool value)
void prop(PointerRNA *ptr, PropertyRNA *prop, int index, int value, eUI_Item_Flag flag, std::optional< blender::StringRef > name_opt, int icon, std::optional< blender::StringRef > placeholder=std::nullopt)
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238