/* SPDX-License-Identifier: GPL-2.0-or-later */ #include "DEG_depsgraph_query.h" #ifdef WITH_OPENVDB # include # include #endif #include "node_geometry_util.hh" #include "BKE_lib_id.h" #include "BKE_material.h" #include "BKE_mesh.h" #include "BKE_mesh_runtime.h" #include "BKE_volume.h" #include "BKE_volume_to_mesh.hh" #include "DNA_mesh_types.h" #include "DNA_meshdata_types.h" #include "UI_interface.h" #include "UI_resources.h" namespace blender::nodes::node_geo_volume_to_mesh_cc { NODE_STORAGE_FUNCS(NodeGeometryVolumeToMesh) static void node_declare(NodeDeclarationBuilder &b) { b.add_input(N_("Volume")).supported_type(GEO_COMPONENT_TYPE_VOLUME); b.add_input(N_("Voxel Size")) .default_value(0.3f) .min(0.01f) .subtype(PROP_DISTANCE) .make_available([](bNode &node) { node_storage(node).resolution_mode = VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_SIZE; }); b.add_input(N_("Voxel Amount")) .default_value(64.0f) .min(0.0f) .make_available([](bNode &node) { node_storage(node).resolution_mode = VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_AMOUNT; }); b.add_input(N_("Threshold")) .default_value(0.1f) .description(N_("Values larger than the threshold are inside the generated mesh")); b.add_input(N_("Adaptivity")).min(0.0f).max(1.0f).subtype(PROP_FACTOR); b.add_output(N_("Mesh")); } static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr) { uiLayoutSetPropSep(layout, true); uiLayoutSetPropDecorate(layout, false); uiItemR(layout, ptr, "resolution_mode", 0, IFACE_("Resolution"), ICON_NONE); } static void node_init(bNodeTree * /*tree*/, bNode *node) { NodeGeometryVolumeToMesh *data = MEM_cnew(__func__); data->resolution_mode = VOLUME_TO_MESH_RESOLUTION_MODE_GRID; node->storage = data; } static void node_update(bNodeTree *ntree, bNode *node) { const NodeGeometryVolumeToMesh &storage = node_storage(*node); bNodeSocket *voxel_size_socket = nodeFindSocket(node, SOCK_IN, "Voxel Size"); bNodeSocket *voxel_amount_socket = nodeFindSocket(node, SOCK_IN, "Voxel Amount"); nodeSetSocketAvailability(ntree, voxel_amount_socket, storage.resolution_mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_AMOUNT); nodeSetSocketAvailability(ntree, voxel_size_socket, storage.resolution_mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_SIZE); } #ifdef WITH_OPENVDB static bke::VolumeToMeshResolution get_resolution_param(const GeoNodeExecParams ¶ms) { const NodeGeometryVolumeToMesh &storage = node_storage(params.node()); bke::VolumeToMeshResolution resolution; resolution.mode = (VolumeToMeshResolutionMode)storage.resolution_mode; if (resolution.mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_AMOUNT) { resolution.settings.voxel_amount = std::max(params.get_input("Voxel Amount"), 0.0f); } else if (resolution.mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_SIZE) { resolution.settings.voxel_size = std::max(params.get_input("Voxel Size"), 0.0f); } return resolution; } static Mesh *create_mesh_from_volume_grids(Span grids, const float threshold, const float adaptivity, const bke::VolumeToMeshResolution &resolution) { Array mesh_data(grids.size()); for (const int i : grids.index_range()) { mesh_data[i] = bke::volume_to_mesh_data(*grids[i], resolution, threshold, adaptivity); } int vert_offset = 0; int poly_offset = 0; int loop_offset = 0; Array vert_offsets(mesh_data.size()); Array poly_offsets(mesh_data.size()); Array loop_offsets(mesh_data.size()); for (const int i : grids.index_range()) { const bke::OpenVDBMeshData &data = mesh_data[i]; vert_offsets[i] = vert_offset; poly_offsets[i] = poly_offset; loop_offsets[i] = loop_offset; vert_offset += data.verts.size(); poly_offset += (data.tris.size() + data.quads.size()); loop_offset += (3 * data.tris.size() + 4 * data.quads.size()); } Mesh *mesh = BKE_mesh_new_nomain(vert_offset, 0, 0, loop_offset, poly_offset); BKE_id_material_eval_ensure_default_slot(&mesh->id); MutableSpan positions = mesh->positions_for_write(); MutableSpan polys = mesh->polys_for_write(); MutableSpan loops = mesh->loops_for_write(); for (const int i : grids.index_range()) { const bke::OpenVDBMeshData &data = mesh_data[i]; bke::fill_mesh_from_openvdb_data(data.verts, data.tris, data.quads, vert_offsets[i], poly_offsets[i], loop_offsets[i], positions, polys, loops); } BKE_mesh_calc_edges(mesh, false, false); return mesh; } static Mesh *create_mesh_from_volume(GeometrySet &geometry_set, GeoNodeExecParams ¶ms) { const Volume *volume = geometry_set.get_volume_for_read(); if (volume == nullptr) { return nullptr; } const bke::VolumeToMeshResolution resolution = get_resolution_param(params); if (resolution.mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_SIZE && resolution.settings.voxel_size <= 0.0f) { return nullptr; } if (resolution.mode == VOLUME_TO_MESH_RESOLUTION_MODE_VOXEL_AMOUNT && resolution.settings.voxel_amount <= 0) { return nullptr; } const Main *bmain = DEG_get_bmain(params.depsgraph()); BKE_volume_load(volume, bmain); Vector grids; for (const int i : IndexRange(BKE_volume_num_grids(volume))) { const VolumeGrid *volume_grid = BKE_volume_grid_get_for_read(volume, i); openvdb::GridBase::ConstPtr grid = BKE_volume_grid_openvdb_for_read(volume, volume_grid); grids.append(std::move(grid)); } if (grids.is_empty()) { return nullptr; } return create_mesh_from_volume_grids(grids, params.get_input("Threshold"), params.get_input("Adaptivity"), resolution); } #endif /* WITH_OPENVDB */ static void node_geo_exec(GeoNodeExecParams params) { #ifdef WITH_OPENVDB GeometrySet geometry_set = params.extract_input("Volume"); geometry_set.modify_geometry_sets([&](GeometrySet &geometry_set) { Mesh *mesh = create_mesh_from_volume(geometry_set, params); geometry_set.replace_mesh(mesh); geometry_set.keep_only_during_modify({GEO_COMPONENT_TYPE_MESH}); }); params.set_output("Mesh", std::move(geometry_set)); #else params.set_default_remaining_outputs(); params.error_message_add(NodeWarningType::Error, TIP_("Disabled, Blender was compiled without OpenVDB")); #endif } } // namespace blender::nodes::node_geo_volume_to_mesh_cc void register_node_type_geo_volume_to_mesh() { namespace file_ns = blender::nodes::node_geo_volume_to_mesh_cc; static bNodeType ntype; geo_node_type_base(&ntype, GEO_NODE_VOLUME_TO_MESH, "Volume to Mesh", NODE_CLASS_GEOMETRY); ntype.declare = file_ns::node_declare; node_type_storage( &ntype, "NodeGeometryVolumeToMesh", node_free_standard_storage, node_copy_standard_storage); node_type_size(&ntype, 170, 120, 700); ntype.initfunc = file_ns::node_init; ntype.updatefunc = file_ns::node_update; ntype.geometry_node_execute = file_ns::node_geo_exec; ntype.draw_buttons = file_ns::node_layout; nodeRegisterType(&ntype); }