Blender V4.3
node_geo_curve_fillet.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 "NOD_rna_define.hh"
6
8
10
11#include "UI_interface.hh"
12#include "UI_resources.hh"
13
14#include "GEO_fillet_curves.hh"
15
16#include "node_geometry_util.hh"
17
19
21
23{
24 b.add_input<decl::Geometry>("Curve").supported_type(
26 auto &count_input = b.add_input<decl::Int>("Count")
27 .default_value(1)
28 .min(1)
29 .max(1000)
30 .field_on_all()
31 .make_available([](bNode &node) {
32 node_storage(node).mode = GEO_NODE_CURVE_FILLET_POLY;
33 });
34 b.add_input<decl::Float>("Radius")
35 .min(0.0f)
36 .max(FLT_MAX)
38 .default_value(0.25f)
39 .field_on_all();
40 b.add_input<decl::Bool>("Limit Radius")
41 .description("Limit the maximum value of the radius in order to avoid overlapping fillets");
42 b.add_output<decl::Geometry>("Curve").propagate_all();
43
44 const bNode *node = b.node_or_null();
45 if (node != nullptr) {
46 const NodeGeometryCurveFillet &storage = node_storage(*node);
47 count_input.available(GeometryNodeCurveFilletMode(storage.mode) == GEO_NODE_CURVE_FILLET_POLY);
48 }
49}
50
51static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
52{
53 uiItemR(layout, ptr, "mode", UI_ITEM_R_EXPAND, nullptr, ICON_NONE);
54}
55
56static void node_init(bNodeTree * /*tree*/, bNode *node)
57{
58 NodeGeometryCurveFillet *data = MEM_cnew<NodeGeometryCurveFillet>(__func__);
60 node->storage = data;
61}
62
65 const fn::FieldContext &field_context,
66 const std::optional<Field<int>> &count_field,
67 const Field<float> &radius_field,
68 const bool limit_radius,
69 const AttributeFilter &attribute_filter)
70{
71 fn::FieldEvaluator evaluator{field_context, src_curves.points_num()};
72 evaluator.add(radius_field);
73
74 switch (mode) {
76 evaluator.evaluate();
77 return geometry::fillet_curves_bezier(src_curves,
78 src_curves.curves_range(),
79 evaluator.get_evaluated<float>(0),
80 limit_radius,
81 attribute_filter);
82 }
84 evaluator.add(*count_field);
85 evaluator.evaluate();
86 return geometry::fillet_curves_poly(src_curves,
87 src_curves.curves_range(),
88 evaluator.get_evaluated<float>(0),
89 evaluator.get_evaluated<int>(1),
90 limit_radius,
91 attribute_filter);
92 }
93 }
94 return bke::CurvesGeometry();
95}
96
97static void fillet_grease_pencil(GreasePencil &grease_pencil,
99 const std::optional<Field<int>> &count_field,
100 const Field<float> &radius_field,
101 const bool limit_radius,
102 const AttributeFilter &attribute_filter)
103{
104 using namespace blender::bke::greasepencil;
105 for (const int layer_index : grease_pencil.layers().index_range()) {
106 Drawing *drawing = grease_pencil.get_eval_drawing(grease_pencil.layer(layer_index));
107 if (drawing == nullptr) {
108 continue;
109 }
110 const bke::CurvesGeometry &src_curves = drawing->strokes();
111 if (src_curves.points_num() == 0) {
112 continue;
113 }
114 const bke::GreasePencilLayerFieldContext field_context(
115 grease_pencil, AttrDomain::Curve, layer_index);
116 bke::CurvesGeometry dst_curves = fillet_curve(src_curves,
117 mode,
118 field_context,
119 count_field,
120 radius_field,
121 limit_radius,
122 attribute_filter);
123 drawing->strokes_for_write() = std::move(dst_curves);
124 drawing->tag_topology_changed();
125 }
126}
127
129{
130 GeometrySet geometry_set = params.extract_input<GeometrySet>("Curve");
131
132 const NodeGeometryCurveFillet &storage = node_storage(params.node());
134
135 Field<float> radius_field = params.extract_input<Field<float>>("Radius");
136 const bool limit_radius = params.extract_input<bool>("Limit Radius");
137
138 std::optional<Field<int>> count_field;
139 if (mode == GEO_NODE_CURVE_FILLET_POLY) {
140 count_field.emplace(params.extract_input<Field<int>>("Count"));
141 }
142
143 const NodeAttributeFilter &attribute_filter = params.get_attribute_filter("Curve");
144
145 geometry_set.modify_geometry_sets([&](GeometrySet &geometry_set) {
146 if (geometry_set.has_curves()) {
147 const Curves &curves_id = *geometry_set.get_curves();
148 const bke::CurvesGeometry &src_curves = curves_id.geometry.wrap();
149 const bke::CurvesFieldContext field_context{curves_id, AttrDomain::Point};
150 bke::CurvesGeometry dst_curves = fillet_curve(src_curves,
151 mode,
152 field_context,
153 count_field,
154 radius_field,
155 limit_radius,
156 attribute_filter);
157 Curves *dst_curves_id = bke::curves_new_nomain(std::move(dst_curves));
158 bke::curves_copy_parameters(curves_id, *dst_curves_id);
159 geometry_set.replace_curves(dst_curves_id);
160 }
161 if (geometry_set.has_grease_pencil()) {
162 GreasePencil &grease_pencil = *geometry_set.get_grease_pencil_for_write();
164 grease_pencil, mode, count_field, radius_field, limit_radius, attribute_filter);
165 }
166 });
167
168 params.set_output("Curve", std::move(geometry_set));
169}
170
171static void node_rna(StructRNA *srna)
172{
173 static EnumPropertyItem mode_items[] = {
175 "BEZIER",
176 0,
177 "Bézier",
178 "Align Bézier handles to create circular arcs at each control point"},
180 "POLY",
181 0,
182 "Poly",
183 "Add control points along a circular arc (handle type is vector if Bézier Spline)"},
184 {0, nullptr, 0, nullptr, nullptr},
185 };
186
188 "mode",
189 "Mode",
190 "How to choose number of vertices on fillet",
191 mode_items,
194}
195
196static void node_register()
197{
198 static blender::bke::bNodeType ntype;
199
200 geo_node_type_base(&ntype, GEO_NODE_FILLET_CURVE, "Fillet Curve", NODE_CLASS_GEOMETRY);
203 &ntype, "NodeGeometryCurveFillet", node_free_standard_storage, node_copy_standard_storage);
204 ntype.declare = node_declare;
205 ntype.initfunc = node_init;
208
209 node_rna(ntype.rna_ext.srna);
210}
212
213} // namespace blender::nodes::node_geo_curve_fillet_cc
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
GeometryNodeCurveFilletMode
@ GEO_NODE_CURVE_FILLET_BEZIER
@ GEO_NODE_CURVE_FILLET_POLY
#define NOD_REGISTER_NODE(REGISTER_FUNC)
#define NOD_storage_enum_accessors(member)
@ PROP_DISTANCE
Definition RNA_types.hh:159
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
@ UI_ITEM_R_EXPAND
IndexRange curves_range() const
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
void make_available(bNode &node) const
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 fillet_curves_bezier(const bke::CurvesGeometry &src_curves, const IndexMask &curve_selection, const VArray< float > &radius, bool limit_radius, const bke::AttributeFilter &attribute_filter)
bke::CurvesGeometry fillet_curves_poly(const bke::CurvesGeometry &src_curves, const IndexMask &curve_selection, const VArray< float > &radius, const VArray< int > &counts, bool limit_radius, const bke::AttributeFilter &attribute_filter)
static void node_init(bNodeTree *, bNode *node)
static bke::CurvesGeometry fillet_curve(const bke::CurvesGeometry &src_curves, const GeometryNodeCurveFilletMode mode, const fn::FieldContext &field_context, const std::optional< Field< int > > &count_field, const Field< float > &radius_field, const bool limit_radius, const AttributeFilter &attribute_filter)
static void node_geo_exec(GeoNodeExecParams params)
static void node_declare(NodeDeclarationBuilder &b)
static void node_layout(uiLayout *layout, bContext *, PointerRNA *ptr)
static void fillet_grease_pencil(GreasePencil &grease_pencil, const GeometryNodeCurveFilletMode mode, const std::optional< Field< int > > &count_field, const Field< float > &radius_field, const bool limit_radius, const AttributeFilter &attribute_filter)
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
#define FLT_MAX
Definition stdcycles.h:14
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
NodeDeclareFunction declare
Definition BKE_node.hh:347
PointerRNA * ptr
Definition wm_files.cc:4126