Blender V5.0
tracking_plane_tracker.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
10
11#include "MEM_guardedalloc.h"
12
13#include "DNA_tracking_types.h"
14
15#include "BLI_math_matrix.h"
16#include "BLI_math_vector.h"
17
18#include "BKE_tracking.h"
19
20#include "libmv-capi.h"
21
22typedef double Vec2[2];
23
25 MovieTrackingPlaneTrack *plane_track, int frame1, int frame2, Vec2 **r_x1, Vec2 **r_x2)
26{
27 Vec2 *x1, *x2;
28
29 *r_x1 = x1 = MEM_calloc_arrayN<Vec2>(plane_track->point_tracksnr, "point correspondences x1");
30 *r_x2 = x2 = MEM_calloc_arrayN<Vec2>(plane_track->point_tracksnr, "point correspondences x2");
31
32 int correspondence_index = 0;
33 for (int i = 0; i < plane_track->point_tracksnr; i++) {
34 MovieTrackingTrack *point_track = plane_track->point_tracks[i];
35 MovieTrackingMarker *point_marker1, *point_marker2;
36
37 point_marker1 = BKE_tracking_marker_get_exact(point_track, frame1);
38 point_marker2 = BKE_tracking_marker_get_exact(point_track, frame2);
39
40 if (point_marker1 != nullptr && point_marker2 != nullptr) {
41 /* Here conversion from float to double happens. */
42 x1[correspondence_index][0] = point_marker1->pos[0];
43 x1[correspondence_index][1] = point_marker1->pos[1];
44
45 x2[correspondence_index][0] = point_marker2->pos[0];
46 x2[correspondence_index][1] = point_marker2->pos[1];
47
48 correspondence_index++;
49 }
50 }
51
52 return correspondence_index;
53}
54
55/* NOTE: frame number should be in clip space, not scene space */
57 int start_frame,
58 int direction,
59 bool retrack)
60{
61 MovieTrackingPlaneMarker *start_plane_marker = BKE_tracking_plane_marker_get(plane_track,
62 start_frame);
63 MovieTrackingPlaneMarker *keyframe_plane_marker = nullptr;
64 MovieTrackingPlaneMarker new_plane_marker;
65 int frame_delta = direction > 0 ? 1 : -1;
66
67 if (plane_track->flag & PLANE_TRACK_AUTOKEY) {
68 /* Find a keyframe in given direction. */
69 for (int current_frame = start_frame;; current_frame += frame_delta) {
71 plane_track, current_frame + frame_delta);
72
73 if (next_plane_marker == nullptr) {
74 break;
75 }
76
77 if ((next_plane_marker->flag & PLANE_MARKER_TRACKED) == 0) {
78 keyframe_plane_marker = next_plane_marker;
79 break;
80 }
81 }
82 }
83 else {
84 start_plane_marker->flag |= PLANE_MARKER_TRACKED;
85 }
86
87 new_plane_marker = *start_plane_marker;
88 new_plane_marker.flag |= PLANE_MARKER_TRACKED;
89
90 for (int current_frame = start_frame;; current_frame += frame_delta) {
92 plane_track, current_frame + frame_delta);
93 Vec2 *x1, *x2;
94 double H_double[3][3];
95 float H[3][3];
96
97 /* As soon as we meet keyframed plane, we stop updating the sequence. */
98 if (next_plane_marker && (next_plane_marker->flag & PLANE_MARKER_TRACKED) == 0) {
99 /* Don't override keyframes if track is in auto-keyframe mode */
100 if (plane_track->flag & PLANE_TRACK_AUTOKEY) {
101 break;
102 }
103 }
104
105 const int num_correspondences = point_markers_correspondences_on_both_image(
106 plane_track, current_frame, current_frame + frame_delta, &x1, &x2);
107 if (num_correspondences < 4) {
108 MEM_freeN(x1);
109 MEM_freeN(x2);
110 break;
111 }
112
113 libmv_homography2DFromCorrespondencesEuc(x1, x2, num_correspondences, H_double);
114
115 copy_m3_m3d(H, H_double);
116
117 for (int i = 0; i < 4; i++) {
118 float vec[3] = {0.0f, 0.0f, 1.0f}, vec2[3];
119 copy_v2_v2(vec, new_plane_marker.corners[i]);
120
121 /* Apply homography */
122 mul_v3_m3v3(vec2, H, vec);
123
124 /* Normalize. */
125 vec2[0] /= vec2[2];
126 vec2[1] /= vec2[2];
127
128 copy_v2_v2(new_plane_marker.corners[i], vec2);
129 }
130
131 new_plane_marker.framenr = current_frame + frame_delta;
132
133 if (!retrack && keyframe_plane_marker && next_plane_marker &&
134 (plane_track->flag & PLANE_TRACK_AUTOKEY))
135 {
136 float fac = (float(next_plane_marker->framenr) - start_plane_marker->framenr) /
137 (float(keyframe_plane_marker->framenr) - start_plane_marker->framenr);
138
139 fac = 3 * fac * fac - 2 * fac * fac * fac;
140
141 for (int i = 0; i < 4; i++) {
142 interp_v2_v2v2(new_plane_marker.corners[i],
143 new_plane_marker.corners[i],
144 next_plane_marker->corners[i],
145 fac);
146 }
147 }
148
149 BKE_tracking_plane_marker_insert(plane_track, &new_plane_marker);
150
151 MEM_freeN(x1);
152 MEM_freeN(x2);
153 }
154}
155
157 int start_frame)
158{
159 track_plane_from_existing_motion(plane_track, start_frame, 1, false);
160 track_plane_from_existing_motion(plane_track, start_frame, -1, false);
161}
162
164 int start_frame,
165 int direction)
166{
167 MovieTrackingPlaneMarker *plane_marker = BKE_tracking_plane_marker_get(plane_track, start_frame);
168 int index = plane_marker - plane_track->markers;
169 int frame_delta = direction > 0 ? 1 : -1;
170
171 while (index >= 0 && index < plane_track->markersnr) {
172 if ((plane_marker->flag & PLANE_MARKER_TRACKED) == 0) {
173 return plane_marker;
174 }
175 plane_marker += frame_delta;
176 }
177
178 return nullptr;
179}
180
182 MovieTrackingPlaneTrack *plane_track, int start_frame)
183{
184 MovieTrackingPlaneMarker *prev_plane_keyframe, *next_plane_keyframe;
185
186 prev_plane_keyframe = find_plane_keyframe(plane_track, start_frame, -1);
187 next_plane_keyframe = find_plane_keyframe(plane_track, start_frame, 1);
188
189 if (prev_plane_keyframe != nullptr && next_plane_keyframe != nullptr) {
190 /* First we track from left keyframe to the right one without any blending. */
191 track_plane_from_existing_motion(plane_track, prev_plane_keyframe->framenr, 1, true);
192
193 /* And then we track from the right keyframe to the left one, so shape blends in nicely */
194 track_plane_from_existing_motion(plane_track, next_plane_keyframe->framenr, -1, false);
195 }
196 else if (prev_plane_keyframe != nullptr) {
197 track_plane_from_existing_motion(plane_track, prev_plane_keyframe->framenr, 1, true);
198 }
199 else if (next_plane_keyframe != nullptr) {
200 track_plane_from_existing_motion(plane_track, next_plane_keyframe->framenr, -1, true);
201 }
202}
203
204BLI_INLINE void float_corners_to_double(/*const*/ float corners[4][2], double double_corners[4][2])
205{
206 copy_v2db_v2fl(double_corners[0], corners[0]);
207 copy_v2db_v2fl(double_corners[1], corners[1]);
208 copy_v2db_v2fl(double_corners[2], corners[2]);
209 copy_v2db_v2fl(double_corners[3], corners[3]);
210}
211
212void BKE_tracking_homography_between_two_quads(/*const*/ float reference_corners[4][2],
213 /*const*/ float corners[4][2],
214 float H[3][3])
215{
216 Vec2 x1[4], x2[4];
217 double H_double[3][3];
218
219 float_corners_to_double(reference_corners, x1);
220 float_corners_to_double(corners, x2);
221
222 libmv_homography2DFromCorrespondencesEuc(x1, x2, 4, H_double);
223
224 copy_m3_m3d(H, H_double);
225}
struct MovieTrackingPlaneMarker * BKE_tracking_plane_marker_insert(struct MovieTrackingPlaneTrack *plane_track, struct MovieTrackingPlaneMarker *plane_marker)
Definition tracking.cc:1719
struct MovieTrackingPlaneMarker * BKE_tracking_plane_marker_get_exact(struct MovieTrackingPlaneTrack *plane_track, int framenr)
Definition tracking.cc:1837
struct MovieTrackingMarker * BKE_tracking_marker_get_exact(struct MovieTrackingTrack *track, int framenr)
Definition tracking.cc:1391
struct MovieTrackingPlaneMarker * BKE_tracking_plane_marker_get(struct MovieTrackingPlaneTrack *plane_track, int framenr)
Definition tracking.cc:1791
#define BLI_INLINE
void copy_m3_m3d(float m1[3][3], const double m2[3][3])
void mul_v3_m3v3(float r[3], const float M[3][3], const float a[3])
MINLINE void copy_v2db_v2fl(double r[2], const float a[2])
void interp_v2_v2v2(float r[2], const float a[2], const float b[2], float t)
MINLINE void copy_v2_v2(float r[2], const float a[2])
@ PLANE_MARKER_TRACKED
@ PLANE_TRACK_AUTOKEY
Read Guarded memory(de)allocation.
nullptr float
void libmv_homography2DFromCorrespondencesEuc(double(*x1)[2], double(*x2)[2], int num_points, double H[3][3])
void * MEM_calloc_arrayN(size_t len, size_t size, const char *str)
Definition mallocn.cc:123
void MEM_freeN(void *vmemh)
Definition mallocn.cc:113
#define H(x, y, z)
MovieTrackingTrack ** point_tracks
MovieTrackingPlaneMarker * markers
i
Definition text_draw.cc:230
double Vec2[2]
static void track_plane_from_existing_motion(MovieTrackingPlaneTrack *plane_track, int start_frame, int direction, bool retrack)
static int point_markers_correspondences_on_both_image(MovieTrackingPlaneTrack *plane_track, int frame1, int frame2, Vec2 **r_x1, Vec2 **r_x2)
BLI_INLINE void float_corners_to_double(float corners[4][2], double double_corners[4][2])
void BKE_tracking_retrack_plane_from_existing_motion_at_segment(MovieTrackingPlaneTrack *plane_track, int start_frame)
static MovieTrackingPlaneMarker * find_plane_keyframe(MovieTrackingPlaneTrack *plane_track, int start_frame, int direction)
void BKE_tracking_track_plane_from_existing_motion(MovieTrackingPlaneTrack *plane_track, int start_frame)
void BKE_tracking_homography_between_two_quads(float reference_corners[4][2], float corners[4][2], float H[3][3])