Blender V4.5
node_geo_curve_resample.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
6
7#include "BKE_curves.hh"
9
10#include "UI_interface.hh"
11#include "UI_resources.hh"
12
13#include "NOD_rna_define.hh"
14
15#include "node_geometry_util.hh"
16
18
20
22{
23 b.use_custom_socket_order();
24 b.allow_any_socket_order();
25 b.add_default_layout();
26 b.add_input<decl::Geometry>("Curve").supported_type(
27 {GeometryComponent::Type::Curve, GeometryComponent::Type::GreasePencil});
28 b.add_output<decl::Geometry>("Curve").propagate_all().align_with_previous();
29 b.add_input<decl::Bool>("Selection").default_value(true).field_on_all().hide_value();
30 auto &count =
31 b.add_input<decl::Int>("Count").default_value(10).min(1).max(100000).field_on_all();
32 auto &length = b.add_input<decl::Float>("Length")
33 .default_value(0.1f)
34 .min(0.01f)
35 .subtype(PROP_DISTANCE)
36 .field_on_all();
37
38 const bNode *node = b.node_or_null();
39 if (node != nullptr) {
40 const NodeGeometryCurveResample &storage = node_storage(*node);
42
43 count.available(mode == GEO_NODE_CURVE_RESAMPLE_COUNT);
44 length.available(mode == GEO_NODE_CURVE_RESAMPLE_LENGTH);
45 }
46}
47
48static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
49{
50 layout->prop(ptr, "mode", UI_ITEM_NONE, "", ICON_NONE);
51}
52
53static void node_layout_ex(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
54{
55 layout->prop(ptr, "keep_last_segment", UI_ITEM_NONE, std::nullopt, ICON_NONE);
56}
57
58static void node_init(bNodeTree * /*tree*/, bNode *node)
59{
61
63 data->keep_last_segment = true;
64 node->storage = data;
65}
66
68{
69 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
70
71 const NodeGeometryCurveResample &storage = node_storage(params.node());
73
74 const Field<bool> selection = params.extract_input<Field<bool>>("Selection");
75
77
78 switch (mode) {
80 Field<int> count = params.extract_input<Field<int>>("Count");
81 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
82 if (const Curves *src_curves_id = geometry.get_curves()) {
83 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
84 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
86 src_curves, field_context, selection, count);
87 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
88 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
89 geometry.replace_curves(dst_curves_id);
90 }
91 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
92 using namespace blender::bke::greasepencil;
93 for (const int layer_index : grease_pencil->layers().index_range()) {
94 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
95
96 if (drawing == nullptr) {
97 continue;
98 }
99 const bke::CurvesGeometry &src_curves = drawing->strokes();
100 const bke::GreasePencilLayerFieldContext field_context(
101 *grease_pencil, AttrDomain::Curve, layer_index);
103 src_curves, field_context, selection, count);
104 drawing->strokes_for_write() = std::move(dst_curves);
105 drawing->tag_topology_changed();
106 }
107 }
108 });
109 break;
110 }
112 Field<float> length = params.extract_input<Field<float>>("Length");
113 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
114 if (const Curves *src_curves_id = geometry.get_curves()) {
115 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
116 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
118 src_curves, field_context, selection, length, {}, storage.keep_last_segment);
119 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
120 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
121 geometry.replace_curves(dst_curves_id);
122 }
123 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
124 using namespace blender::bke::greasepencil;
125 for (const int layer_index : grease_pencil->layers().index_range()) {
126 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
127 if (drawing == nullptr) {
128 continue;
129 }
130 const bke::CurvesGeometry &src_curves = drawing->strokes();
131 const bke::GreasePencilLayerFieldContext field_context(
132 *grease_pencil, AttrDomain::Curve, layer_index);
134 src_curves, field_context, selection, length, {}, storage.keep_last_segment);
135 drawing->strokes_for_write() = std::move(dst_curves);
136 drawing->tag_topology_changed();
137 }
138 }
139 });
140 break;
141 }
143 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
144 if (const Curves *src_curves_id = geometry.get_curves()) {
145 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
146 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
148 src_curves, field_context, selection);
149 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
150 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
151 geometry.replace_curves(dst_curves_id);
152 }
153 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
154 using namespace blender::bke::greasepencil;
155 for (const int layer_index : grease_pencil->layers().index_range()) {
156 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
157 if (drawing == nullptr) {
158 continue;
159 }
160 const bke::CurvesGeometry &src_curves = drawing->strokes();
161 const bke::GreasePencilLayerFieldContext field_context(
162 *grease_pencil, AttrDomain::Curve, layer_index);
164 src_curves, field_context, selection);
165 drawing->strokes_for_write() = std::move(dst_curves);
166 drawing->tag_topology_changed();
167 }
168 }
169 });
170 break;
171 }
172
173 params.set_output("Curve", std::move(geometry_set));
174}
175
176static void node_rna(StructRNA *srna)
177{
178 static EnumPropertyItem mode_items[] = {
180 "EVALUATED",
181 0,
182 "Evaluated",
183 "Output the input spline's evaluated points, based on the resolution attribute for NURBS "
184 "and Bézier splines. Poly splines are unchanged"},
186 "COUNT",
187 0,
188 "Count",
189 "Sample the specified number of points along each spline"},
191 "LENGTH",
192 0,
193 "Length",
194 "Calculate the number of samples by splitting each spline into segments with the specified "
195 "length"},
196 {0, nullptr, 0, nullptr, nullptr},
197 };
198
200 "mode",
201 "Mode",
202 "How to specify the amount of samples",
203 mode_items,
205
207 "keep_last_segment",
208 "Keep Last Segment",
209 "Do not collapse curves to single points if they are shorter than the "
210 "given length. The collapsing behavior exists for compatibility reasons.",
211 NOD_storage_boolean_accessors(keep_last_segment, 1));
212}
213
214static void node_register()
215{
216 static blender::bke::bNodeType ntype;
217
218 geo_node_type_base(&ntype, "GeometryNodeResampleCurve", GEO_NODE_RESAMPLE_CURVE);
219 ntype.ui_name = "Resample Curve";
220 ntype.ui_description = "Generate a poly spline for each input spline";
221 ntype.enum_name_legacy = "RESAMPLE_CURVE";
223 ntype.declare = node_declare;
227 ntype, "NodeGeometryCurveResample", node_free_standard_storage, node_copy_standard_storage);
228 ntype.initfunc = node_init;
231
232 node_rna(ntype.rna_ext.srna);
233}
235
236} // namespace blender::nodes::node_geo_curve_resample_cc
Low-level operations for curves.
Low-level operations for grease pencil.
#define NODE_STORAGE_FUNCS(StorageT)
Definition BKE_node.hh:1215
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:447
#define GEO_NODE_RESAMPLE_CURVE
GeometryNodeCurveResampleMode
@ GEO_NODE_CURVE_RESAMPLE_LENGTH
@ GEO_NODE_CURVE_RESAMPLE_EVALUATED
@ GEO_NODE_CURVE_RESAMPLE_COUNT
#define NOD_REGISTER_NODE(REGISTER_FUNC)
#define NOD_storage_boolean_accessors(member, flag)
#define NOD_storage_enum_accessors(member)
@ PROP_DISTANCE
Definition RNA_types.hh:244
#define UI_ITEM_NONE
BMesh const char void * data
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
static void remember_deformed_positions_if_necessary(GeometrySet &geometry)
float length(VecOp< float, D >) RET
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
int count
void * MEM_callocN(size_t len, const char *str)
Definition mallocn.cc:118
void curves_copy_parameters(const Curves &src, Curves &dst)
void node_register_type(bNodeType &ntype)
Definition node.cc:2748
void node_type_storage(bNodeType &ntype, std::optional< StringRefNull > storagename, void(*freefunc)(bNode *node), void(*copyfunc)(bNodeTree *dest_ntree, bNode *dest_node, const bNode *src_node))
Definition node.cc:5603
Curves * curves_new_nomain(int points_num, int curves_num)
CurvesGeometry resample_to_count(const CurvesGeometry &src_curves, const IndexMask &selection, const VArray< int > &counts, const ResampleCurvesOutputAttributeIDs &output_ids={})
CurvesGeometry resample_to_evaluated(const CurvesGeometry &src_curves, const IndexMask &selection, const ResampleCurvesOutputAttributeIDs &output_ids={})
CurvesGeometry resample_to_length(const CurvesGeometry &src_curves, const IndexMask &selection, const VArray< float > &sample_lengths, const ResampleCurvesOutputAttributeIDs &output_ids={}, bool keep_last_segment=false)
static void node_init(bNodeTree *, bNode *node)
static void node_declare(NodeDeclarationBuilder &b)
static void node_layout_ex(uiLayout *layout, bContext *, PointerRNA *ptr)
static void node_geo_exec(GeoNodeExecParams params)
static void node_layout(uiLayout *layout, bContext *, PointerRNA *ptr)
PropertyRNA * RNA_def_node_enum(StructRNA *srna, const char *identifier, const char *ui_name, const char *ui_description, const EnumPropertyItem *static_items, const EnumRNAAccessors accessors, std::optional< int > default_value, const EnumPropertyItemFunc item_func, const bool allow_animation)
PropertyRNA * RNA_def_node_boolean(StructRNA *srna, const char *identifier, const char *ui_name, const char *ui_description, const BooleanRNAAccessors accessors, std::optional< bool > default_value, bool allow_animation)
void geo_node_type_base(blender::bke::bNodeType *ntype, std::string idname, const std::optional< int16_t > legacy_type)
void node_free_standard_storage(bNode *node)
Definition node_util.cc:42
void node_copy_standard_storage(bNodeTree *, bNode *dest_node, const bNode *src_node)
Definition node_util.cc:54
#define min(a, b)
Definition sort.cc:36
StructRNA * srna
Definition RNA_types.hh:909
void * storage
void modify_geometry_sets(ForeachSubGeometryCallback callback)
GreasePencil * get_grease_pencil_for_write()
Defines a node type.
Definition BKE_node.hh:226
std::string ui_description
Definition BKE_node.hh:232
void(* initfunc)(bNodeTree *ntree, bNode *node)
Definition BKE_node.hh:277
void(* draw_buttons_ex)(uiLayout *, bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:249
NodeGeometryExecFunction geometry_node_execute
Definition BKE_node.hh:347
const char * enum_name_legacy
Definition BKE_node.hh:235
void(* draw_buttons)(uiLayout *, bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:247
NodeDeclareFunction declare
Definition BKE_node.hh:355
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)
max
Definition text_draw.cc:251
PointerRNA * ptr
Definition wm_files.cc:4226