Blender V5.0
node_geo_curve_trim.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
9#include "UI_resources.hh"
10
12
14#include "GEO_trim_curves.hh"
15
16#include "NOD_rna_define.hh"
17
18#include "node_geometry_util.hh"
19
21
23
25{
26 b.use_custom_socket_order();
27 b.allow_any_socket_order();
28 b.add_default_layout();
29 b.add_input<decl::Geometry>("Curve")
30 .supported_type({GeometryComponent::Type::Curve, GeometryComponent::Type::GreasePencil})
31 .description("Curves to shorten");
32 b.add_output<decl::Geometry>("Curve").propagate_all().align_with_previous();
33 b.add_input<decl::Bool>("Selection").default_value(true).hide_value().field_on_all();
34 auto &start_fac = b.add_input<decl::Float>("Start")
35 .min(0.0f)
36 .max(1.0f)
37 .subtype(PROP_FACTOR)
38 .make_available([](bNode &node) {
39 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_FACTOR;
40 })
41 .field_on_all();
42 auto &end_fac = b.add_input<decl::Float>("End")
43 .min(0.0f)
44 .max(1.0f)
45 .default_value(1.0f)
46 .subtype(PROP_FACTOR)
47 .make_available([](bNode &node) {
48 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_FACTOR;
49 })
50 .field_on_all();
51 auto &start_len = b.add_input<decl::Float>("Start", "Start_001")
52 .min(0.0f)
53 .subtype(PROP_DISTANCE)
54 .make_available([](bNode &node) {
55 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_LENGTH;
56 })
57 .field_on_all();
58 auto &end_len = b.add_input<decl::Float>("End", "End_001")
59 .min(0.0f)
60 .default_value(1.0f)
61 .subtype(PROP_DISTANCE)
62 .make_available([](bNode &node) {
63 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_LENGTH;
64 })
65 .field_on_all();
66
67 const bNode *node = b.node_or_null();
68 if (node != nullptr) {
69 const NodeGeometryCurveTrim &storage = node_storage(*node);
71
72 start_fac.available(mode == GEO_NODE_CURVE_SAMPLE_FACTOR);
73 end_fac.available(mode == GEO_NODE_CURVE_SAMPLE_FACTOR);
74 start_len.available(mode == GEO_NODE_CURVE_SAMPLE_LENGTH);
75 end_len.available(mode == GEO_NODE_CURVE_SAMPLE_LENGTH);
76 }
77}
78
79static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
80{
81 layout->prop(ptr, "mode", UI_ITEM_R_EXPAND, std::nullopt, ICON_NONE);
82}
83
84static void node_init(bNodeTree * /*tree*/, bNode *node)
85{
87
89 node->storage = data;
90}
91
93 public:
97 {
98 bNode &node = params.add_node("GeometryNodeTrimCurve");
99 node_storage(node).mode = mode;
100 params.update_and_connect_available_socket(node, socket_name);
101 }
102};
103
105{
106 const NodeDeclaration &declaration = *params.node_type().static_declaration;
107
109 search_link_ops_for_declarations(params, declaration.inputs.as_span().take_front(1));
110
111 if (params.in_out() == SOCK_IN) {
112 if (params.node_tree().typeinfo->validate_link(eNodeSocketDatatype(params.other_socket().type),
113 SOCK_FLOAT))
114 {
115 params.add_item(IFACE_("Start (Factor)"),
117 params.add_item(IFACE_("End (Factor)"), SocketSearchOp{"End", GEO_NODE_CURVE_SAMPLE_FACTOR});
118 params.add_item(IFACE_("Start (Length)"),
120 params.add_item(IFACE_("End (Length)"), SocketSearchOp{"End", GEO_NODE_CURVE_SAMPLE_LENGTH});
121 }
122 }
123}
124
125static bool trim_curves(const bke::CurvesGeometry &src_curves,
127 const fn::FieldContext &field_context,
128 Field<bool> &selection_field,
129 Field<float> &start_field,
130 Field<float> &end_field,
131 const AttributeFilter &attribute_filter,
132 bke::CurvesGeometry &dst_curves)
133{
134 if (src_curves.is_empty()) {
135 return false;
136 }
137 fn::FieldEvaluator evaluator{field_context, src_curves.curves_num()};
138 evaluator.add(selection_field);
139 evaluator.add(start_field);
140 evaluator.add(end_field);
141 evaluator.evaluate();
142
143 const IndexMask selection = evaluator.get_evaluated_as_mask(0);
144 const VArray<float> starts = evaluator.get_evaluated<float>(1);
145 const VArray<float> ends = evaluator.get_evaluated<float>(2);
146
147 if (selection.is_empty()) {
148 return false;
149 }
150
151 dst_curves = geometry::trim_curves(src_curves, selection, starts, ends, mode, attribute_filter);
152 return true;
153}
154
155static void geometry_set_curve_trim(GeometrySet &geometry_set,
157 Field<bool> &selection_field,
158 Field<float> &start_field,
159 Field<float> &end_field,
160 const AttributeFilter &attribute_filter)
161{
162 if (geometry_set.has_curves()) {
163 const Curves &src_curves_id = *geometry_set.get_curves();
164 const bke::CurvesGeometry &src_curves = src_curves_id.geometry.wrap();
165 const bke::CurvesFieldContext field_context{src_curves_id, AttrDomain::Curve};
166 bke::CurvesGeometry dst_curves;
167 if (trim_curves(src_curves,
168 mode,
169 field_context,
170 selection_field,
171 start_field,
172 end_field,
173 attribute_filter,
174 dst_curves))
175 {
176 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
177 bke::curves_copy_parameters(src_curves_id, *dst_curves_id);
178 geometry_set.replace_curves(dst_curves_id);
179 }
180 }
181 if (geometry_set.has_grease_pencil()) {
182 using namespace bke::greasepencil;
183 GreasePencil &grease_pencil = *geometry_set.get_grease_pencil_for_write();
184 for (const int layer_index : grease_pencil.layers().index_range()) {
185 Drawing *drawing = grease_pencil.get_eval_drawing(grease_pencil.layer(layer_index));
186 if (drawing == nullptr) {
187 continue;
188 }
189 const bke::CurvesGeometry &src_curves = drawing->strokes();
190 const bke::GreasePencilLayerFieldContext field_context{
191 grease_pencil, AttrDomain::Curve, layer_index};
192 bke::CurvesGeometry dst_curves;
193 if (trim_curves(src_curves,
194 mode,
195 field_context,
196 selection_field,
197 start_field,
198 end_field,
199 attribute_filter,
200 dst_curves))
201 {
202 drawing->strokes_for_write() = std::move(dst_curves);
203 drawing->tag_topology_changed();
204 }
205 }
206 }
207}
208
210{
211 const NodeGeometryCurveTrim &storage = node_storage(params.node());
213
214 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
216
217 const NodeAttributeFilter &attribute_filter = params.get_attribute_filter("Curve");
218
219 Field<bool> selection_field = params.extract_input<Field<bool>>("Selection");
220 if (mode == GEO_NODE_CURVE_SAMPLE_FACTOR) {
221 Field<float> start_field = params.extract_input<Field<float>>("Start");
222 Field<float> end_field = params.extract_input<Field<float>>("End");
223 geometry::foreach_real_geometry(geometry_set, [&](GeometrySet &geometry_set) {
225 geometry_set, mode, selection_field, start_field, end_field, attribute_filter);
226 });
227 }
228 else if (mode == GEO_NODE_CURVE_SAMPLE_LENGTH) {
229 Field<float> start_field = params.extract_input<Field<float>>("Start_001");
230 Field<float> end_field = params.extract_input<Field<float>>("End_001");
231 geometry::foreach_real_geometry(geometry_set, [&](GeometrySet &geometry_set) {
233 geometry_set, mode, selection_field, start_field, end_field, attribute_filter);
234 });
235 }
236
237 params.set_output("Curve", std::move(geometry_set));
238}
239
240static void node_rna(StructRNA *srna)
241{
242 static EnumPropertyItem mode_items[] = {
244 "FACTOR",
245 0,
246 "Factor",
247 "Find the endpoint positions using a factor of each spline's length"},
249 "LENGTH",
250 0,
251 "Length",
252 "Find the endpoint positions using a length from the start of each spline"},
253 {0, nullptr, 0, nullptr, nullptr},
254 };
255
257 "mode",
258 "Mode",
259 "How to find endpoint positions for the trimmed spline",
260 mode_items,
262}
263
264static void node_register()
265{
266 static blender::bke::bNodeType ntype;
267 geo_node_type_base(&ntype, "GeometryNodeTrimCurve", GEO_NODE_TRIM_CURVE);
268 ntype.ui_name = "Trim Curve";
269 ntype.ui_description = "Shorten curves by removing portions at the start or end";
270 ntype.enum_name_legacy = "TRIM_CURVE";
274 ntype.declare = node_declare;
276 ntype, "NodeGeometryCurveTrim", node_free_standard_storage, node_copy_standard_storage);
277 ntype.initfunc = node_init;
280
281 node_rna(ntype.rna_ext.srna);
282}
284
285} // namespace blender::nodes::node_geo_curve_trim_cc
Low-level operations for curves.
Low-level operations for grease pencil.
#define NODE_STORAGE_FUNCS(StorageT)
Definition BKE_node.hh:1240
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:461
#define GEO_NODE_TRIM_CURVE
#define IFACE_(msgid)
@ GEO_NODE_CURVE_RESAMPLE_LENGTH
GeometryNodeCurveSampleMode
@ GEO_NODE_CURVE_SAMPLE_FACTOR
@ GEO_NODE_CURVE_SAMPLE_LENGTH
@ SOCK_IN
eNodeSocketDatatype
@ SOCK_FLOAT
#define NOD_REGISTER_NODE(REGISTER_FUNC)
#define NOD_storage_enum_accessors(member)
@ PROP_DISTANCE
Definition RNA_types.hh:256
@ PROP_FACTOR
Definition RNA_types.hh:251
@ UI_ITEM_R_EXPAND
BMesh const char void * data
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
int add(GField field, GVArray *varray_ptr)
Definition field.cc:751
IndexMask get_evaluated_as_mask(int field_index)
Definition field.cc:804
const GVArray & get_evaluated(const int field_index) const
Definition FN_field.hh:448
static void remember_deformed_positions_if_necessary(GeometrySet &geometry)
Vector< SocketDeclaration * > inputs
Vector< SocketDeclaration * > outputs
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
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:2416
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:5414
Curves * curves_new_nomain(int points_num, int curves_num)
bke::CurvesGeometry trim_curves(const bke::CurvesGeometry &src_curves, const IndexMask &selection, const VArray< float > &starts, const VArray< float > &ends, GeometryNodeCurveSampleMode mode, const bke::AttributeFilter &attribute_filter)
void foreach_real_geometry(bke::GeometrySet &geometry, FunctionRef< void(bke::GeometrySet &geometry_set)> fn)
static void node_init(bNodeTree *, bNode *node)
static bool trim_curves(const bke::CurvesGeometry &src_curves, const GeometryNodeCurveSampleMode mode, const fn::FieldContext &field_context, Field< bool > &selection_field, Field< float > &start_field, Field< float > &end_field, const AttributeFilter &attribute_filter, bke::CurvesGeometry &dst_curves)
static void node_declare(NodeDeclarationBuilder &b)
static void node_gather_link_searches(GatherLinkSearchOpParams &params)
static void geometry_set_curve_trim(GeometrySet &geometry_set, const GeometryNodeCurveSampleMode mode, Field< bool > &selection_field, Field< float > &start_field, Field< float > &end_field, const AttributeFilter &attribute_filter)
static void node_layout(uiLayout *layout, bContext *, PointerRNA *ptr)
static void node_geo_exec(GeoNodeExecParams params)
void search_link_ops_for_declarations(GatherLinkSearchOpParams &params, Span< SocketDeclaration * > declarations)
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, 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
CurvesGeometry geometry
StructRNA * srna
void * storage
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
void(* initfunc)(bNodeTree *ntree, bNode *node)
Definition BKE_node.hh:289
NodeGeometryExecFunction geometry_node_execute
Definition BKE_node.hh:354
const char * enum_name_legacy
Definition BKE_node.hh:247
void(* draw_buttons)(uiLayout *, bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:259
NodeGatherSocketLinkOperationsFunction gather_link_search_ops
Definition BKE_node.hh:378
NodeDeclareFunction declare
Definition BKE_node.hh:362
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:4238