Blender V5.0
node_composite_stabilize2d.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2011 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
10#include "BLI_math_matrix.hh"
13
14#include "RNA_enum_types.hh"
15
16#include "UI_interface.hh"
18
19#include "DNA_movieclip_types.h"
20#include "DNA_node_types.h"
21
22#include "BKE_context.hh"
23#include "BKE_lib_id.hh"
24#include "BKE_movieclip.h"
25#include "BKE_tracking.h"
26
27#include "COM_node_operation.hh"
28
30
32
34{
35 b.use_custom_socket_order();
36 b.allow_any_socket_order();
37
38 b.add_input<decl::Color>("Image")
39 .default_value({0.8f, 0.8f, 0.8f, 1.0f})
40 .hide_value()
42 .structure_type(StructureType::Dynamic);
43 b.add_output<decl::Color>("Image").structure_type(StructureType::Dynamic).align_with_previous();
44
45 b.add_layout([](uiLayout *layout, bContext *context, PointerRNA *node_pointer) {
46 uiTemplateID(layout, context, node_pointer, "clip", nullptr, "CLIP_OT_open", nullptr);
47 });
48
49 b.add_input<decl::Bool>("Invert").default_value(false).description(
50 "Invert stabilization to reintroduce motion to the image");
51
52 PanelDeclarationBuilder &sampling_panel = b.add_panel("Sampling").default_closed(true);
53 sampling_panel.add_input<decl::Menu>("Interpolation")
57 .description("Interpolation method");
58 sampling_panel.add_input<decl::Menu>("Extension X")
59 .default_value(CMP_NODE_EXTENSION_MODE_CLIP)
62 .description("The extension mode applied to the X axis");
63 sampling_panel.add_input<decl::Menu>("Extension Y")
64 .default_value(CMP_NODE_EXTENSION_MODE_CLIP)
67 .description("The extension mode applied to the Y axis");
68}
69
70static void init(const bContext *C, PointerRNA *ptr)
71{
72 bNode *node = (bNode *)ptr->data;
73 Scene *scene = CTX_data_scene(C);
74
75 node->id = (ID *)scene->clip;
76 id_us_plus(node->id);
77}
78
79using namespace blender::compositor;
80
82 public:
84
85 void execute() override
86 {
87 const Result &input = this->get_input("Image");
88 Result &output = this->get_result("Image");
89
90 MovieClip *movie_clip = get_movie_clip();
91 if (input.is_single_value() || !movie_clip) {
92 output.share_data(input);
93 return;
94 }
95
96 const int width = input.domain().size.x;
97 const int height = input.domain().size.y;
98 const int frame_number = BKE_movieclip_remap_scene_to_clip_frame(movie_clip,
99 context().get_frame_number());
100
101 float2 translation;
102 float scale, rotation;
104 movie_clip, frame_number, width, height, translation, &scale, &rotation);
105
107 translation, math::AngleRadian(rotation), float2(scale));
109 transformation = math::invert(transformation);
110 }
111
112 output.share_data(input);
113 output.transform(transformation);
114 output.get_realization_options().interpolation = this->get_interpolation();
117 }
118
120 {
121 const Result &input = this->get_input("Interpolation");
122 const MenuValue default_menu_value = MenuValue(CMP_NODE_INTERPOLATION_BILINEAR);
123 const MenuValue menu_value = input.get_single_value_default(default_menu_value);
124 const CMPNodeInterpolation interpolation = static_cast<CMPNodeInterpolation>(menu_value.value);
125 switch (interpolation) {
133 }
134
136 }
137
139 {
140 const Result &input = this->get_input("Extension X");
141 const MenuValue default_menu_value = MenuValue(CMP_NODE_EXTENSION_MODE_CLIP);
142 const MenuValue menu_value = input.get_single_value_default(default_menu_value);
143 const CMPExtensionMode extension_x = static_cast<CMPExtensionMode>(menu_value.value);
144 switch (extension_x) {
146 return ExtensionMode::Clip;
151 }
152
153 return ExtensionMode::Clip;
154 }
155
157 {
158 const Result &input = this->get_input("Extension Y");
159 const MenuValue default_menu_value = MenuValue(CMP_NODE_EXTENSION_MODE_CLIP);
160 const MenuValue menu_value = input.get_single_value_default(default_menu_value);
161 const CMPExtensionMode extension_y = static_cast<CMPExtensionMode>(menu_value.value);
162 switch (extension_y) {
164 return ExtensionMode::Clip;
169 }
170
171 return ExtensionMode::Clip;
172 }
173
175 {
176 return this->get_input("Invert").get_single_value_default(false);
177 }
178
180 {
181 return reinterpret_cast<MovieClip *>(bnode().id);
182 }
183};
184
186{
187 return new Stabilize2DOperation(context, node);
188}
189
190} // namespace blender::nodes::node_composite_stabilize2d_cc
191
193{
195
196 static blender::bke::bNodeType ntype;
197
198 cmp_node_type_base(&ntype, "CompositorNodeStabilize", CMP_NODE_STABILIZE2D);
199 ntype.ui_name = "Stabilize 2D";
200 ntype.ui_description = "Stabilize footage using 2D stabilization motion tracking settings";
201 ntype.enum_name_legacy = "STABILIZE2D";
203 ntype.declare = file_ns::cmp_node_stabilize2d_declare;
204 ntype.initfunc_api = file_ns::init;
205 ntype.get_compositor_operation = file_ns::get_compositor_operation;
206
208}
Scene * CTX_data_scene(const bContext *C)
void id_us_plus(ID *id)
Definition lib_id.cc:358
float BKE_movieclip_remap_scene_to_clip_frame(const struct MovieClip *clip, float framenr)
#define NODE_CLASS_DISTORT
Definition BKE_node.hh:455
#define CMP_NODE_STABILIZE2D
void BKE_tracking_stabilization_data_get(struct MovieClip *clip, int framenr, int width, int height, float translation[2], float *scale, float *angle)
CMPNodeInterpolation
@ CMP_NODE_INTERPOLATION_NEAREST
@ CMP_NODE_INTERPOLATION_BILINEAR
@ CMP_NODE_INTERPOLATION_BICUBIC
@ CMP_NODE_INTERPOLATION_ANISOTROPIC
CMPExtensionMode
@ CMP_NODE_EXTENSION_MODE_EXTEND
@ CMP_NODE_EXTENSION_MODE_CLIP
@ CMP_NODE_EXTENSION_MODE_REPEAT
#define NOD_REGISTER_NODE(REGISTER_FUNC)
#define C
Definition RandGen.cpp:29
void uiTemplateID(uiLayout *layout, const bContext *C, PointerRNA *ptr, blender::StringRefNull propname, const char *newop, const char *openop, const char *unlinkop, int filter=UI_TEMPLATE_ID_FILTER_ALL, bool live_icon=false, std::optional< blender::StringRef > text=std::nullopt)
void init()
NodeOperation(Context &context, DNode node)
Result & get_result(StringRef identifier)
Definition operation.cc:39
Result & get_input(StringRef identifier) const
Definition operation.cc:138
RealizationOptions & get_realization_options()
Definition result.cc:630
T get_single_value_default(const T &default_value) const
DeclType::Builder & add_input(StringRef name, StringRef identifier="")
const CompositorInputRealizationMode & compositor_realization_mode() const
#define input
#define output
void node_register_type(bNodeType &ntype)
Definition node.cc:2416
AngleRadianBase< float > AngleRadian
CartesianBasis invert(const CartesianBasis &basis)
MatT from_loc_rot_scale(const typename MatT::loc_type &location, const RotationT &rotation, const VecBase< typename MatT::base_type, ScaleDim > &scale)
static NodeOperation * get_compositor_operation(Context &context, DNode node)
static void cmp_node_stabilize2d_declare(NodeDeclarationBuilder &b)
VecBase< float, 2 > float2
MatBase< float, 3, 3 > float3x3
static void register_node_type_cmp_stabilize2d()
void cmp_node_type_base(blender::bke::bNodeType *ntype, std::string idname, const std::optional< int16_t > legacy_type)
const EnumPropertyItem rna_enum_node_compositor_extension_items[]
const EnumPropertyItem rna_enum_node_compositor_interpolation_items[]
Definition DNA_ID.h:414
struct MovieClip * clip
struct ID * id
Defines a node type.
Definition BKE_node.hh:238
std::string ui_description
Definition BKE_node.hh:244
NodeGetCompositorOperationFunction get_compositor_operation
Definition BKE_node.hh:348
const char * enum_name_legacy
Definition BKE_node.hh:247
NodeDeclareFunction declare
Definition BKE_node.hh:362
void(* initfunc_api)(const bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:302
PointerRNA * ptr
Definition wm_files.cc:4238