Blender V4.3
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.add_input<decl::Geometry>("Curve").supported_type(
25 b.add_input<decl::Bool>("Selection").default_value(true).field_on_all().hide_value();
26 auto &count =
27 b.add_input<decl::Int>("Count").default_value(10).min(1).max(100000).field_on_all();
28 auto &length = b.add_input<decl::Float>("Length")
29 .default_value(0.1f)
30 .min(0.01f)
32 .field_on_all();
33 b.add_output<decl::Geometry>("Curve").propagate_all();
34
35 const bNode *node = b.node_or_null();
36 if (node != nullptr) {
37 const NodeGeometryCurveResample &storage = node_storage(*node);
39
40 count.available(mode == GEO_NODE_CURVE_RESAMPLE_COUNT);
41 length.available(mode == GEO_NODE_CURVE_RESAMPLE_LENGTH);
42 }
43}
44
45static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
46{
47 uiItemR(layout, ptr, "mode", UI_ITEM_NONE, "", ICON_NONE);
48}
49
50static void node_init(bNodeTree * /*tree*/, bNode *node)
51{
52 NodeGeometryCurveResample *data = MEM_cnew<NodeGeometryCurveResample>(__func__);
53
55 node->storage = data;
56}
57
59{
60 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
61
62 const NodeGeometryCurveResample &storage = node_storage(params.node());
64
65 const Field<bool> selection = params.extract_input<Field<bool>>("Selection");
66
68
69 switch (mode) {
71 Field<int> count = params.extract_input<Field<int>>("Count");
72 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
73 if (const Curves *src_curves_id = geometry.get_curves()) {
74 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
75 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
77 src_curves, field_context, selection, count);
78 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
79 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
80 geometry.replace_curves(dst_curves_id);
81 }
82 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
83 using namespace blender::bke::greasepencil;
84 for (const int layer_index : grease_pencil->layers().index_range()) {
85 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
86
87 if (drawing == nullptr) {
88 continue;
89 }
90 const bke::CurvesGeometry &src_curves = drawing->strokes();
91 const bke::GreasePencilLayerFieldContext field_context(
92 *grease_pencil, AttrDomain::Curve, layer_index);
94 src_curves, field_context, selection, count);
95 drawing->strokes_for_write() = std::move(dst_curves);
96 drawing->tag_topology_changed();
97 }
98 }
99 });
100 break;
101 }
103 Field<float> length = params.extract_input<Field<float>>("Length");
104 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
105 if (const Curves *src_curves_id = geometry.get_curves()) {
106 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
107 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
109 src_curves, field_context, selection, length);
110 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
111 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
112 geometry.replace_curves(dst_curves_id);
113 }
114 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
115 using namespace blender::bke::greasepencil;
116 for (const int layer_index : grease_pencil->layers().index_range()) {
117 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
118 if (drawing == nullptr) {
119 continue;
120 }
121 const bke::CurvesGeometry &src_curves = drawing->strokes();
122 const bke::GreasePencilLayerFieldContext field_context(
123 *grease_pencil, AttrDomain::Curve, layer_index);
125 src_curves, field_context, selection, length);
126 drawing->strokes_for_write() = std::move(dst_curves);
127 drawing->tag_topology_changed();
128 }
129 }
130 });
131 break;
132 }
134 geometry_set.modify_geometry_sets([&](GeometrySet &geometry) {
135 if (const Curves *src_curves_id = geometry.get_curves()) {
136 const bke::CurvesGeometry &src_curves = src_curves_id->geometry.wrap();
137 const bke::CurvesFieldContext field_context{*src_curves_id, AttrDomain::Curve};
139 src_curves, field_context, selection);
140 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
141 bke::curves_copy_parameters(*src_curves_id, *dst_curves_id);
142 geometry.replace_curves(dst_curves_id);
143 }
144 if (GreasePencil *grease_pencil = geometry_set.get_grease_pencil_for_write()) {
145 using namespace blender::bke::greasepencil;
146 for (const int layer_index : grease_pencil->layers().index_range()) {
147 Drawing *drawing = grease_pencil->get_eval_drawing(grease_pencil->layer(layer_index));
148 if (drawing == nullptr) {
149 continue;
150 }
151 const bke::CurvesGeometry &src_curves = drawing->strokes();
152 const bke::GreasePencilLayerFieldContext field_context(
153 *grease_pencil, AttrDomain::Curve, layer_index);
155 src_curves, field_context, selection);
156 drawing->strokes_for_write() = std::move(dst_curves);
157 drawing->tag_topology_changed();
158 }
159 }
160 });
161 break;
162 }
163
164 params.set_output("Curve", std::move(geometry_set));
165}
166
167static void node_rna(StructRNA *srna)
168{
169 static EnumPropertyItem mode_items[] = {
171 "EVALUATED",
172 0,
173 "Evaluated",
174 "Output the input spline's evaluated points, based on the resolution attribute for NURBS "
175 "and Bézier splines. Poly splines are unchanged"},
177 "COUNT",
178 0,
179 "Count",
180 "Sample the specified number of points along each spline"},
182 "LENGTH",
183 0,
184 "Length",
185 "Calculate the number of samples by splitting each spline into segments with the specified "
186 "length"},
187 {0, nullptr, 0, nullptr, nullptr},
188 };
189
191 "mode",
192 "Mode",
193 "How to specify the amount of samples",
194 mode_items,
196}
197
198static void node_register()
199{
200 static blender::bke::bNodeType ntype;
201
202 geo_node_type_base(&ntype, GEO_NODE_RESAMPLE_CURVE, "Resample Curve", NODE_CLASS_GEOMETRY);
203 ntype.declare = node_declare;
206 &ntype, "NodeGeometryCurveResample", node_free_standard_storage, node_copy_standard_storage);
207 ntype.initfunc = node_init;
210
211 node_rna(ntype.rna_ext.srna);
212}
214
215} // 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:1799
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:418
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_enum_accessors(member)
@ PROP_DISTANCE
Definition RNA_types.hh:159
#define UI_ITEM_NONE
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
static void remember_deformed_positions_if_necessary(GeometrySet &geometry)
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
local_group_size(16, 16) .push_constant(Type b
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
int count
void curves_copy_parameters(const Curves &src, Curves &dst)
void node_type_storage(bNodeType *ntype, const char *storagename, void(*freefunc)(bNode *node), void(*copyfunc)(bNodeTree *dest_ntree, bNode *dest_node, const bNode *src_node))
Definition node.cc:4632
void node_register_type(bNodeType *ntype)
Definition node.cc:1708
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={})
static void node_init(bNodeTree *, bNode *node)
static void node_declare(NodeDeclarationBuilder &b)
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)
void geo_node_type_base(blender::bke::bNodeType *ntype, int type, const char *name, short nclass)
void node_free_standard_storage(bNode *node)
Definition node_util.cc:46
void node_copy_standard_storage(bNodeTree *, bNode *dest_node, const bNode *src_node)
Definition node_util.cc:58
StructRNA * srna
Definition RNA_types.hh:780
void modify_geometry_sets(ForeachSubGeometryCallback callback)
GreasePencil * get_grease_pencil_for_write()
Defines a node type.
Definition BKE_node.hh:218
void(* initfunc)(bNodeTree *ntree, bNode *node)
Definition BKE_node.hh:267
NodeGeometryExecFunction geometry_node_execute
Definition BKE_node.hh:339
void(* draw_buttons)(uiLayout *, bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:238
NodeDeclareFunction declare
Definition BKE_node.hh:347
PointerRNA * ptr
Definition wm_files.cc:4126