Blender V4.3
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#include "BLI_task.hh"
8
9#include "UI_interface.hh"
10#include "UI_resources.hh"
11
13
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.add_input<decl::Geometry>("Curve").supported_type(
28 b.add_input<decl::Bool>("Selection").default_value(true).hide_value().supports_field();
29 auto &start_fac = b.add_input<decl::Float>("Start")
30 .min(0.0f)
31 .max(1.0f)
33 .make_available([](bNode &node) {
34 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_FACTOR;
35 })
36 .field_on_all();
37 auto &end_fac = b.add_input<decl::Float>("End")
38 .min(0.0f)
39 .max(1.0f)
40 .default_value(1.0f)
41 .subtype(PROP_FACTOR)
42 .make_available([](bNode &node) {
43 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_FACTOR;
44 })
45 .field_on_all();
46 auto &start_len = b.add_input<decl::Float>("Start", "Start_001")
47 .min(0.0f)
49 .make_available([](bNode &node) {
50 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_LENGTH;
51 })
52 .field_on_all();
53 auto &end_len = b.add_input<decl::Float>("End", "End_001")
54 .min(0.0f)
55 .default_value(1.0f)
56 .subtype(PROP_DISTANCE)
57 .make_available([](bNode &node) {
58 node_storage(node).mode = GEO_NODE_CURVE_SAMPLE_LENGTH;
59 })
60 .field_on_all();
61 b.add_output<decl::Geometry>("Curve").propagate_all();
62
63 const bNode *node = b.node_or_null();
64 if (node != nullptr) {
65 const NodeGeometryCurveTrim &storage = node_storage(*node);
67
68 start_fac.available(mode == GEO_NODE_CURVE_SAMPLE_FACTOR);
69 end_fac.available(mode == GEO_NODE_CURVE_SAMPLE_FACTOR);
70 start_len.available(mode == GEO_NODE_CURVE_SAMPLE_LENGTH);
71 end_len.available(mode == GEO_NODE_CURVE_SAMPLE_LENGTH);
72 }
73}
74
75static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
76{
77 uiItemR(layout, ptr, "mode", UI_ITEM_R_EXPAND, nullptr, ICON_NONE);
78}
79
80static void node_init(bNodeTree * /*tree*/, bNode *node)
81{
82 NodeGeometryCurveTrim *data = MEM_cnew<NodeGeometryCurveTrim>(__func__);
83
85 node->storage = data;
86}
87
89 public:
93 {
94 bNode &node = params.add_node("GeometryNodeTrimCurve");
95 node_storage(node).mode = mode;
96 params.update_and_connect_available_socket(node, socket_name);
97 }
98};
99
101{
102 const NodeDeclaration &declaration = *params.node_type().static_declaration;
103
105 search_link_ops_for_declarations(params, declaration.inputs.as_span().take_front(1));
106
107 if (params.in_out() == SOCK_IN) {
108 if (params.node_tree().typeinfo->validate_link(eNodeSocketDatatype(params.other_socket().type),
109 SOCK_FLOAT))
110 {
111 params.add_item(IFACE_("Start (Factor)"),
113 params.add_item(IFACE_("End (Factor)"), SocketSearchOp{"End", GEO_NODE_CURVE_SAMPLE_FACTOR});
114 params.add_item(IFACE_("Start (Length)"),
116 params.add_item(IFACE_("End (Length)"), SocketSearchOp{"End", GEO_NODE_CURVE_SAMPLE_LENGTH});
117 }
118 }
119}
120
121static bool trim_curves(const bke::CurvesGeometry &src_curves,
123 const fn::FieldContext &field_context,
124 Field<bool> &selection_field,
125 Field<float> &start_field,
126 Field<float> &end_field,
127 const AttributeFilter &attribute_filter,
128 bke::CurvesGeometry &dst_curves)
129{
130 if (src_curves.curves_num() == 0) {
131 return false;
132 }
133 fn::FieldEvaluator evaluator{field_context, src_curves.curves_num()};
134 evaluator.add(selection_field);
135 evaluator.add(start_field);
136 evaluator.add(end_field);
137 evaluator.evaluate();
138
139 const IndexMask selection = evaluator.get_evaluated_as_mask(0);
140 const VArray<float> starts = evaluator.get_evaluated<float>(1);
141 const VArray<float> ends = evaluator.get_evaluated<float>(2);
142
143 if (selection.is_empty()) {
144 return false;
145 }
146
147 dst_curves = geometry::trim_curves(src_curves, selection, starts, ends, mode, attribute_filter);
148 return true;
149}
150
151static void geometry_set_curve_trim(GeometrySet &geometry_set,
153 Field<bool> &selection_field,
154 Field<float> &start_field,
155 Field<float> &end_field,
156 const AttributeFilter &attribute_filter)
157{
158 if (geometry_set.has_curves()) {
159 const Curves &src_curves_id = *geometry_set.get_curves();
160 const bke::CurvesGeometry &src_curves = src_curves_id.geometry.wrap();
161 const bke::CurvesFieldContext field_context{src_curves_id, AttrDomain::Curve};
162 bke::CurvesGeometry dst_curves;
163 if (trim_curves(src_curves,
164 mode,
165 field_context,
166 selection_field,
167 start_field,
168 end_field,
169 attribute_filter,
170 dst_curves))
171 {
172 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
173 bke::curves_copy_parameters(src_curves_id, *dst_curves_id);
174 geometry_set.replace_curves(dst_curves_id);
175 }
176 }
177 if (geometry_set.has_grease_pencil()) {
178 using namespace bke::greasepencil;
179 GreasePencil &grease_pencil = *geometry_set.get_grease_pencil_for_write();
180 for (const int layer_index : grease_pencil.layers().index_range()) {
181 Drawing *drawing = grease_pencil.get_eval_drawing(grease_pencil.layer(layer_index));
182 if (drawing == nullptr) {
183 continue;
184 }
185 const bke::CurvesGeometry &src_curves = drawing->strokes();
186 const bke::GreasePencilLayerFieldContext field_context{
187 grease_pencil, AttrDomain::Curve, layer_index};
188 bke::CurvesGeometry dst_curves;
189 if (trim_curves(src_curves,
190 mode,
191 field_context,
192 selection_field,
193 start_field,
194 end_field,
195 attribute_filter,
196 dst_curves))
197 {
198 drawing->strokes_for_write() = std::move(dst_curves);
199 drawing->tag_topology_changed();
200 }
201 }
202 }
203}
204
206{
207 const NodeGeometryCurveTrim &storage = node_storage(params.node());
209
210 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
212
213 const NodeAttributeFilter &attribute_filter = params.get_attribute_filter("Curve");
214
215 Field<bool> selection_field = params.extract_input<Field<bool>>("Selection");
216 if (mode == GEO_NODE_CURVE_SAMPLE_FACTOR) {
217 Field<float> start_field = params.extract_input<Field<float>>("Start");
218 Field<float> end_field = params.extract_input<Field<float>>("End");
219 geometry_set.modify_geometry_sets([&](GeometrySet &geometry_set) {
221 geometry_set, mode, selection_field, start_field, end_field, attribute_filter);
222 });
223 }
224 else if (mode == GEO_NODE_CURVE_SAMPLE_LENGTH) {
225 Field<float> start_field = params.extract_input<Field<float>>("Start_001");
226 Field<float> end_field = params.extract_input<Field<float>>("End_001");
227 geometry_set.modify_geometry_sets([&](GeometrySet &geometry_set) {
229 geometry_set, mode, selection_field, start_field, end_field, attribute_filter);
230 });
231 }
232
233 params.set_output("Curve", std::move(geometry_set));
234}
235
236static void node_rna(StructRNA *srna)
237{
238 static EnumPropertyItem mode_items[] = {
240 "FACTOR",
241 0,
242 "Factor",
243 "Find the endpoint positions using a factor of each spline's length"},
245 "LENGTH",
246 0,
247 "Length",
248 "Find the endpoint positions using a length from the start of each spline"},
249 {0, nullptr, 0, nullptr, nullptr},
250 };
251
253 "mode",
254 "Mode",
255 "How to find endpoint positions for the trimmed spline",
256 mode_items,
258}
259
260static void node_register()
261{
262 static blender::bke::bNodeType ntype;
263 geo_node_type_base(&ntype, GEO_NODE_TRIM_CURVE, "Trim Curve", NODE_CLASS_GEOMETRY);
266 ntype.declare = node_declare;
268 &ntype, "NodeGeometryCurveTrim", node_free_standard_storage, node_copy_standard_storage);
269 ntype.initfunc = node_init;
272
273 node_rna(ntype.rna_ext.srna);
274}
276
277} // 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:1799
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:418
#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:159
@ PROP_FACTOR
Definition RNA_types.hh:154
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
@ UI_ITEM_R_EXPAND
static void remember_deformed_positions_if_necessary(GeometrySet &geometry)
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
Vector< SocketDeclaration * > inputs
Vector< SocketDeclaration * > outputs
local_group_size(16, 16) .push_constant(Type b
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
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)
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)
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, 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
#define min(a, b)
Definition sort.c:32
CurvesGeometry geometry
StructRNA * srna
Definition RNA_types.hh:780
const Curves * get_curves() const
void replace_curves(Curves *curves, GeometryOwnershipType ownership=GeometryOwnershipType::Owned)
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
NodeGatherSocketLinkOperationsFunction gather_link_search_ops
Definition BKE_node.hh:363
NodeDeclareFunction declare
Definition BKE_node.hh:347
PointerRNA * ptr
Definition wm_files.cc:4126