Blender V5.0
MOD_volume_to_mesh.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 "BKE_lib_query.hh"
10#include "BKE_mesh.hh"
11#include "BKE_modifier.hh"
12#include "BKE_volume.hh"
13#include "BKE_volume_grid.hh"
14#include "BKE_volume_to_mesh.hh"
15
16#include "BLT_translation.hh"
17
18#include "MOD_ui_common.hh"
19
20#include "DNA_modifier_types.h"
21#include "DNA_object_types.h"
22#include "DNA_screen_types.h"
23
25#include "UI_resources.hh"
26
27#include "RNA_prototypes.hh"
28#include "RNA_types.hh"
29
31#include "BLI_span.hh"
32#include "BLI_string.h"
33
35
36#ifdef WITH_OPENVDB
37# include <openvdb/tools/GridTransformer.h>
38# include <openvdb/tools/VolumeToMesh.h>
39#endif
40
41using blender::float3;
43using blender::Span;
44
45static void init_data(ModifierData *md)
46{
47 VolumeToMeshModifierData *vmmd = reinterpret_cast<VolumeToMeshModifierData *>(md);
48 vmmd->object = nullptr;
49 vmmd->threshold = 0.1f;
50 STRNCPY(vmmd->grid_name, "density");
51 vmmd->adaptivity = 0.0f;
53 vmmd->voxel_amount = 32;
54 vmmd->voxel_size = 0.1f;
55 vmmd->flag = 0;
56}
57
59{
60 VolumeToMeshModifierData *vmmd = reinterpret_cast<VolumeToMeshModifierData *>(md);
61 DEG_add_depends_on_transform_relation(ctx->node, "Volume to Mesh Modifier");
62 if (vmmd->object) {
64 ctx->node, vmmd->object, DEG_OB_COMP_GEOMETRY, "Volume to Mesh Modifier");
66 ctx->node, vmmd->object, DEG_OB_COMP_TRANSFORM, "Volume to Mesh Modifier");
67 }
68}
69
70static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
71{
72 VolumeToMeshModifierData *vmmd = reinterpret_cast<VolumeToMeshModifierData *>(md);
73 walk(user_data, ob, (ID **)&vmmd->object, IDWALK_CB_NOP);
74}
75
76static void panel_draw(const bContext * /*C*/, Panel *panel)
77{
78 uiLayout *layout = panel->layout;
79
81 VolumeToMeshModifierData *vmmd = static_cast<VolumeToMeshModifierData *>(ptr->data);
82
83 layout->use_property_split_set(true);
84
85 {
86 uiLayout *col = &layout->column(false);
87 col->prop(ptr, "object", UI_ITEM_NONE, std::nullopt, ICON_NONE);
88 col->prop(ptr, "grid_name", UI_ITEM_NONE, std::nullopt, ICON_NONE);
89 }
90
91 {
92 uiLayout *col = &layout->column(false);
93 col->prop(ptr, "resolution_mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
95 col->prop(ptr, "voxel_amount", UI_ITEM_NONE, std::nullopt, ICON_NONE);
96 }
98 col->prop(ptr, "voxel_size", UI_ITEM_NONE, std::nullopt, ICON_NONE);
99 }
100 }
101
102 {
103 uiLayout *col = &layout->column(false);
104 col->prop(ptr, "threshold", UI_ITEM_NONE, std::nullopt, ICON_NONE);
105 col->prop(ptr, "adaptivity", UI_ITEM_NONE, std::nullopt, ICON_NONE);
106 col->prop(ptr, "use_smooth_shade", UI_ITEM_NONE, std::nullopt, ICON_NONE);
107 }
108
110}
111
112static void panel_register(ARegionType *region_type)
113{
115}
116
117static Mesh *create_empty_mesh(const Mesh *input_mesh)
118{
119 Mesh *new_mesh = BKE_mesh_new_nomain(0, 0, 0, 0);
120 BKE_mesh_copy_parameters_for_eval(new_mesh, input_mesh);
121 return new_mesh;
122}
123
124static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *input_mesh)
125{
126 using namespace blender;
127#ifdef WITH_OPENVDB
128 VolumeToMeshModifierData *vmmd = reinterpret_cast<VolumeToMeshModifierData *>(md);
129 if (vmmd->object == nullptr) {
130 return create_empty_mesh(input_mesh);
131 }
132 if (vmmd->object->type != OB_VOLUME) {
133 return create_empty_mesh(input_mesh);
134 }
136 vmmd->voxel_size == 0.0f)
137 {
138 return create_empty_mesh(input_mesh);
139 }
141 vmmd->voxel_amount == 0)
142 {
143 return create_empty_mesh(input_mesh);
144 }
145
146 const Volume *volume = static_cast<Volume *>(vmmd->object->data);
147
149 const blender::bke::VolumeGridData *volume_grid = BKE_volume_grid_find(volume, vmmd->grid_name);
150 if (volume_grid == nullptr) {
151 BKE_modifier_set_error(ctx->object, md, "Cannot find '%s' grid", vmmd->grid_name);
152 return create_empty_mesh(input_mesh);
153 }
154
155 blender::bke::VolumeTreeAccessToken tree_token;
156 const openvdb::GridBase &local_grid = volume_grid->grid(tree_token);
157
158 openvdb::math::Transform::Ptr transform = local_grid.transform().copy();
159 transform->postMult(openvdb::Mat4d(vmmd->object->object_to_world().base_ptr()));
160 openvdb::Mat4d imat = openvdb::Mat4d(ctx->object->world_to_object().base_ptr());
161 /* `imat` had floating point issues and wasn't affine. */
162 imat.setCol(3, openvdb::Vec4d(0, 0, 0, 1));
163 transform->postMult(imat);
164
165 /* Create a temporary transformed grid. The underlying tree is shared. */
166 openvdb::GridBase::ConstPtr transformed_grid = local_grid.copyGridReplacingTransform(transform);
167
171 resolution.settings.voxel_amount = vmmd->voxel_amount;
172 }
174 resolution.settings.voxel_size = vmmd->voxel_size;
175 }
176
177 Mesh *mesh = bke::volume_to_mesh(
178 *transformed_grid, resolution, vmmd->threshold, vmmd->adaptivity);
179 if (mesh == nullptr) {
180 BKE_modifier_set_error(ctx->object, md, "Could not generate mesh from grid");
181 return create_empty_mesh(input_mesh);
182 }
183
184 BKE_mesh_copy_parameters_for_eval(mesh, input_mesh);
186 return mesh;
187#else
188 UNUSED_VARS(md);
189 BKE_modifier_set_error(ctx->object, md, "Compiled without OpenVDB");
190 return create_empty_mesh(input_mesh);
191#endif
192}
193
195 /*idname*/ "Volume to Mesh",
196 /*name*/ N_("Volume to Mesh"),
197 /*struct_name*/ "VolumeToMeshModifierData",
198 /*struct_size*/ sizeof(VolumeToMeshModifierData),
199 /*srna*/ &RNA_VolumeToMeshModifier,
202 /*icon*/ ICON_VOLUME_DATA, /* TODO: Use correct icon. */
203
204 /*copy_data*/ BKE_modifier_copydata_generic,
205
206 /*deform_verts*/ nullptr,
207 /*deform_matrices*/ nullptr,
208 /*deform_verts_EM*/ nullptr,
209 /*deform_matrices_EM*/ nullptr,
210 /*modify_mesh*/ modify_mesh,
211 /*modify_geometry_set*/ nullptr,
212
213 /*init_data*/ init_data,
214 /*required_data_mask*/ nullptr,
215 /*free_data*/ nullptr,
216 /*is_disabled*/ nullptr,
217 /*update_depsgraph*/ update_depsgraph,
218 /*depends_on_time*/ nullptr,
219 /*depends_on_normals*/ nullptr,
220 /*foreach_ID_link*/ foreach_ID_link,
221 /*foreach_tex_link*/ nullptr,
222 /*free_runtime_data*/ nullptr,
223 /*panel_register*/ panel_register,
224 /*blend_write*/ nullptr,
225 /*blend_read*/ nullptr,
226 /*foreach_cache*/ nullptr,
227 /*foreach_working_space_color*/ nullptr,
228};
@ IDWALK_CB_NOP
void BKE_mesh_copy_parameters_for_eval(Mesh *me_dst, const Mesh *me_src)
Mesh * BKE_mesh_new_nomain(int verts_num, int edges_num, int faces_num, int corners_num)
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_AcceptsMesh
void BKE_modifier_set_error(const Object *ob, ModifierData *md, const char *format,...) ATTR_PRINTF_FORMAT(3
Volume data-block.
bool BKE_volume_load(const Volume *volume, const Main *bmain)
const blender::bke::VolumeGridData * BKE_volume_grid_find(const Volume *volume, blender::StringRef name)
char * STRNCPY(char(&dst)[N], const char *src)
Definition BLI_string.h:693
#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
Main * DEG_get_bmain(const Depsgraph *graph)
@ eModifierType_VolumeToMesh
@ VOLUME_TO_MESH_USE_SMOOTH_SHADE
VolumeToMeshResolutionMode
@ VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_SIZE
@ VOLUME_TO_MESH_RESOLUTION_MODE_GRID
@ VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_AMOUNT
Object is a sort of wrapper for general info.
@ OB_VOLUME
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 Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
Definition MOD_array.cc:862
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)
static void init_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static Mesh * create_empty_mesh(const Mesh *input_mesh)
ModifierTypeInfo modifierType_VolumeToMesh
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 Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *input_mesh)
#define UI_ITEM_NONE
SIMD_FORCE_INLINE btVector3 transform(const btVector3 &point) const
uint col
static void update_depsgraph(tGraphSliderOp *gso)
void mesh_smooth_set(Mesh &mesh, bool use_smooth, bool keep_sharp_edges=false)
MatBase< float, 4, 4 > float4x4
VecBase< float, 3 > float3
Definition DNA_ID.h:414
struct uiLayout * layout
union blender::bke::VolumeToMeshResolution::@103176042355372060356022076152214323072307172372 settings
uiLayout & column(bool align)
void use_property_split_set(bool value)
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238