Blender V5.0
node_geo_curve_subdivide.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
5#include "BKE_curves.hh"
7
10
11#include "node_geometry_util.hh"
12
14
16{
17 b.use_custom_socket_order();
18 b.allow_any_socket_order();
19 b.add_input<decl::Geometry>("Curve")
20 .supported_type({GeometryComponent::Type::Curve, GeometryComponent::Type::GreasePencil})
21 .description("Curves to subdivide");
22 b.add_output<decl::Geometry>("Curve").propagate_all().align_with_previous();
23 b.add_input<decl::Int>("Cuts").default_value(1).min(0).max(1000).field_on_all().description(
24 "The number of control points to create on the segment following each point");
25}
26
27static Curves *subdivide_curves(const Curves &src_curves_id,
28 Field<int> &cuts_field,
29 const bke::AttributeFilter &attribute_filter)
30{
31 const bke::CurvesGeometry &src_curves = src_curves_id.geometry.wrap();
32
33 const bke::CurvesFieldContext field_context{src_curves_id, AttrDomain::Point};
34 fn::FieldEvaluator evaluator{field_context, src_curves.points_num()};
35 evaluator.add(cuts_field);
36 evaluator.evaluate();
37 const VArray<int> cuts = evaluator.get_evaluated<int>(0);
38
39 if (cuts.is_single() && cuts.get_internal_single() < 1) {
40 return nullptr;
41 }
42
44 src_curves, src_curves.curves_range(), cuts, attribute_filter);
45
46 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
47 bke::curves_copy_parameters(src_curves_id, *dst_curves_id);
48 return dst_curves_id;
49}
50
52 Field<int> &cuts_field,
53 const AttributeFilter &attribute_filter)
54{
55 using namespace bke::greasepencil;
56 for (const int layer_index : grease_pencil.layers().index_range()) {
57 Drawing *drawing = grease_pencil.get_eval_drawing(grease_pencil.layer(layer_index));
58
59 if (drawing == nullptr) {
60 continue;
61 }
62
63 const bke::CurvesGeometry &src_curves = drawing->strokes();
64 const bke::GreasePencilLayerFieldContext field_context{
65 grease_pencil, AttrDomain::Point, layer_index};
66
67 fn::FieldEvaluator evaluator{field_context, src_curves.points_num()};
68 evaluator.add(cuts_field);
69 evaluator.evaluate();
70 const VArray<int> cuts = evaluator.get_evaluated<int>(0);
71
72 if (cuts.is_single() && cuts.get_internal_single() < 1) {
73 continue;
74 }
75
77 src_curves, src_curves.curves_range(), cuts, attribute_filter);
78
79 drawing->strokes_for_write() = std::move(dst_curves);
80 drawing->tag_topology_changed();
81 }
82}
83
85{
86 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
87 Field<int> cuts_field = params.extract_input<Field<int>>("Cuts");
88
90 const NodeAttributeFilter &attribute_filter = params.get_attribute_filter("Curve");
91
92 geometry::foreach_real_geometry(geometry_set, [&](GeometrySet &geometry_set) {
93 if (geometry_set.has_curves()) {
94 const Curves &src_curves_id = *geometry_set.get_curves();
95 Curves *dst_curves_id = subdivide_curves(src_curves_id, cuts_field, attribute_filter);
96 if (dst_curves_id) {
97 geometry_set.replace_curves(dst_curves_id);
98 }
99 }
100 if (geometry_set.has_grease_pencil()) {
101 GreasePencil &grease_pencil = *geometry_set.get_grease_pencil_for_write();
102 subdivide_grease_pencil_curves(grease_pencil, cuts_field, attribute_filter);
103 }
104 });
105 params.set_output("Curve", geometry_set);
106}
107
108static void node_register()
109{
110 static blender::bke::bNodeType ntype;
111 geo_node_type_base(&ntype, "GeometryNodeSubdivideCurve", GEO_NODE_SUBDIVIDE_CURVE);
112 ntype.ui_name = "Subdivide Curve";
113 ntype.ui_description = "Dividing each curve segment into a specified number of pieces";
114 ntype.enum_name_legacy = "SUBDIVIDE_CURVE";
116 ntype.declare = node_declare;
119}
121
122} // namespace blender::nodes::node_geo_curve_subdivide_cc
Low-level operations for curves.
Low-level operations for grease pencil.
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:461
#define GEO_NODE_SUBDIVIDE_CURVE
#define NOD_REGISTER_NODE(REGISTER_FUNC)
IndexRange curves_range() const
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
int add(GField field, GVArray *varray_ptr)
Definition field.cc:751
const GVArray & get_evaluated(const int field_index) const
Definition FN_field.hh:448
static void remember_deformed_positions_if_necessary(GeometrySet &geometry)
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
void curves_copy_parameters(const Curves &src, Curves &dst)
void node_register_type(bNodeType &ntype)
Definition node.cc:2416
Curves * curves_new_nomain(int points_num, int curves_num)
void foreach_real_geometry(bke::GeometrySet &geometry, FunctionRef< void(bke::GeometrySet &geometry_set)> fn)
bke::CurvesGeometry subdivide_curves(const bke::CurvesGeometry &src_curves, const IndexMask &selection, const VArray< int > &cuts, const bke::AttributeFilter &attribute_filter={})
static void node_declare(NodeDeclarationBuilder &b)
static void node_geo_exec(GeoNodeExecParams params)
static void subdivide_grease_pencil_curves(GreasePencil &grease_pencil, Field< int > &cuts_field, const AttributeFilter &attribute_filter)
static Curves * subdivide_curves(const Curves &src_curves_id, Field< int > &cuts_field, const bke::AttributeFilter &attribute_filter)
void geo_node_type_base(blender::bke::bNodeType *ntype, std::string idname, const std::optional< int16_t > legacy_type)
CurvesGeometry geometry
const Curves * get_curves() const
void replace_curves(Curves *curves, GeometryOwnershipType ownership=GeometryOwnershipType::Owned)
GreasePencil * get_grease_pencil_for_write()
Defines a node type.
Definition BKE_node.hh:238
std::string ui_description
Definition BKE_node.hh:244
NodeGeometryExecFunction geometry_node_execute
Definition BKE_node.hh:354
const char * enum_name_legacy
Definition BKE_node.hh:247
NodeDeclareFunction declare
Definition BKE_node.hh:362