Blender  V2.93
tracking_ops_plane.c
Go to the documentation of this file.
1 /*
2  * This program is free software; you can redistribute it and/or
3  * modify it under the terms of the GNU General Public License
4  * as published by the Free Software Foundation; either version 2
5  * of the License, or (at your option) any later version.
6  *
7  * This program is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10  * GNU General Public License for more details.
11  *
12  * You should have received a copy of the GNU General Public License
13  * along with this program; if not, write to the Free Software Foundation,
14  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
15  *
16  * The Original Code is Copyright (C) 2011 Blender Foundation.
17  * All rights reserved.
18  */
19 
24 #include "MEM_guardedalloc.h"
25 
26 #include "DNA_screen_types.h"
27 #include "DNA_space_types.h"
28 
29 #include "BLI_math.h"
30 #include "BLI_utildefines.h"
31 
32 #include "BKE_context.h"
33 #include "BKE_report.h"
34 #include "BKE_tracking.h"
35 
36 #include "DEG_depsgraph.h"
37 
38 #include "WM_api.h"
39 #include "WM_types.h"
40 
41 #include "ED_clip.h"
42 
43 #include "clip_intern.h"
44 #include "tracking_ops_intern.h"
45 
46 /********************** Create plane track operator *********************/
47 
49 {
52  MovieTracking *tracking = &clip->tracking;
53  MovieTrackingPlaneTrack *plane_track;
54  ListBase *tracks_base = BKE_tracking_get_active_tracks(tracking);
55  ListBase *plane_tracks_base = BKE_tracking_get_active_plane_tracks(tracking);
56  int framenr = ED_space_clip_get_clip_frame_number(sc);
57 
58  plane_track = BKE_tracking_plane_track_add(tracking, plane_tracks_base, tracks_base, framenr);
59 
60  if (plane_track == NULL) {
61  BKE_report(op->reports, RPT_ERROR, "Need at least 4 selected point tracks to create a plane");
62  return OPERATOR_CANCELLED;
63  }
64 
66 
67  plane_track->flag |= SELECT;
68  clip->tracking.act_track = NULL;
69  clip->tracking.act_plane_track = plane_track;
70 
71  /* Compute homoraphies and apply them on marker's corner, so we've got
72  * quite nice motion from the very beginning.
73  */
75 
78 
79  return OPERATOR_FINISHED;
80 }
81 
83 {
84  /* identifiers */
85  ot->name = "Create Plane Track";
86  ot->description = "Create new plane track out of selected point tracks";
87  ot->idname = "CLIP_OT_create_plane_track";
88 
89  /* api callbacks */
92 
93  /* flags */
95 }
96 
97 /********************** Slide plane marker corner operator *********************/
98 
99 typedef struct SlidePlaneMarkerData {
103  int width, height;
105  float *corner;
107  float previous_corner[2];
108  float old_corner[2];
109  bool accurate;
111 
112 static float mouse_to_plane_slide_zone_distance_squared(const float co[2],
113  const float slide_zone[2],
114  int width,
115  int height)
116 {
117  const float pixel_co[2] = {co[0] * width, co[1] * height},
118  pixel_slide_zone[2] = {slide_zone[0] * width, slide_zone[1] * height};
119  return square_f(pixel_co[0] - pixel_slide_zone[0]) + square_f(pixel_co[1] - pixel_slide_zone[1]);
120 }
121 
123  const wmEvent *event,
124  int *r_corner)
125 {
126  const float distance_clip_squared = 12.0f * 12.0f;
128  ARegion *region = CTX_wm_region(C);
129  MovieClip *clip = ED_space_clip_get_clip(sc);
130  MovieTracking *tracking = &clip->tracking;
131  int width, height;
132  float co[2];
133  ListBase *plane_tracks_base = BKE_tracking_get_active_plane_tracks(tracking);
134  int framenr = ED_space_clip_get_clip_frame_number(sc);
135 
137  if (width == 0 || height == 0) {
138  return NULL;
139  }
140 
141  ED_clip_mouse_pos(sc, region, event->mval, co);
142 
143  float min_distance_squared = FLT_MAX;
144  int min_corner = -1;
145  MovieTrackingPlaneTrack *min_plane_track = NULL;
146  for (MovieTrackingPlaneTrack *plane_track = plane_tracks_base->first; plane_track != NULL;
147  plane_track = plane_track->next) {
148  if (PLANE_TRACK_VIEW_SELECTED(plane_track)) {
149  MovieTrackingPlaneMarker *plane_marker = BKE_tracking_plane_marker_get(plane_track, framenr);
150  for (int i = 0; i < 4; i++) {
151  float distance_squared = mouse_to_plane_slide_zone_distance_squared(
152  co, plane_marker->corners[i], width, height);
153 
154  if (distance_squared < min_distance_squared) {
155  min_distance_squared = distance_squared;
156  min_corner = i;
157  min_plane_track = plane_track;
158  }
159  }
160  }
161  }
162 
163  if (min_distance_squared < distance_clip_squared / sc->zoom) {
164  if (r_corner != NULL) {
165  *r_corner = min_corner;
166  }
167  return min_plane_track;
168  }
169 
170  return NULL;
171 }
172 
173 static void *slide_plane_marker_customdata(bContext *C, const wmEvent *event)
174 {
176  ARegion *region = CTX_wm_region(C);
177  MovieTrackingPlaneTrack *plane_track;
178  int width, height;
179  float co[2];
180  SlidePlaneMarkerData *customdata = NULL;
181  int framenr = ED_space_clip_get_clip_frame_number(sc);
182  int corner;
183 
185  if (width == 0 || height == 0) {
186  return NULL;
187  }
188 
189  ED_clip_mouse_pos(sc, region, event->mval, co);
190 
191  plane_track = tracking_plane_marker_check_slide(C, event, &corner);
192  if (plane_track) {
193  MovieTrackingPlaneMarker *plane_marker;
194 
195  customdata = MEM_callocN(sizeof(SlidePlaneMarkerData), "slide plane marker data");
196 
198 
199  plane_marker = BKE_tracking_plane_marker_ensure(plane_track, framenr);
200 
201  customdata->plane_track = plane_track;
202  customdata->plane_marker = plane_marker;
203  customdata->width = width;
204  customdata->height = height;
205 
206  customdata->previous_mval[0] = event->mval[0];
207  customdata->previous_mval[1] = event->mval[1];
208 
209  customdata->corner_index = corner;
210  customdata->corner = plane_marker->corners[corner];
211 
212  copy_v2_v2(customdata->previous_corner, customdata->corner);
213  copy_v2_v2(customdata->old_corner, customdata->corner);
214  }
215 
216  return customdata;
217 }
218 
219 static int slide_plane_marker_invoke(bContext *C, wmOperator *op, const wmEvent *event)
220 {
222 
223  if (slidedata) {
225  MovieClip *clip = ED_space_clip_get_clip(sc);
226  MovieTracking *tracking = &clip->tracking;
227 
228  tracking->act_plane_track = slidedata->plane_track;
229  tracking->act_track = NULL;
230 
231  op->customdata = slidedata;
232 
235 
237 
238  return OPERATOR_RUNNING_MODAL;
239  }
240 
241  return OPERATOR_PASS_THROUGH;
242 }
243 
245 {
246  copy_v2_v2(data->corner, data->old_corner);
247 }
248 
250 {
251  MEM_freeN(data);
252 }
253 
255 {
256  int framenr = ED_space_clip_get_clip_frame_number(sc);
257 
259 }
260 
261 static int slide_plane_marker_modal(bContext *C, wmOperator *op, const wmEvent *event)
262 {
264  MovieClip *clip = ED_space_clip_get_clip(sc);
266  float dx, dy, mdelta[2];
267  int next_corner_index, prev_corner_index, diag_corner_index;
268  const float *next_corner, *prev_corner, *diag_corner;
269  float next_edge[2], prev_edge[2], next_diag_edge[2], prev_diag_edge[2];
270 
271  switch (event->type) {
272  case EVT_LEFTCTRLKEY:
273  case EVT_RIGHTCTRLKEY:
274  case EVT_LEFTSHIFTKEY:
275  case EVT_RIGHTSHIFTKEY:
277  data->accurate = event->val == KM_PRESS;
278  }
280  case MOUSEMOVE:
281  mdelta[0] = event->mval[0] - data->previous_mval[0];
282  mdelta[1] = event->mval[1] - data->previous_mval[1];
283 
284  dx = mdelta[0] / data->width / sc->zoom;
285  dy = mdelta[1] / data->height / sc->zoom;
286 
287  if (data->accurate) {
288  dx /= 5.0f;
289  dy /= 5.0f;
290  }
291 
292  data->corner[0] = data->previous_corner[0] + dx;
293  data->corner[1] = data->previous_corner[1] + dy;
294 
295  /*
296  * prev_edge
297  * (Corner 3, current) <----------------------- (Corner 2, previous)
298  * | ^
299  * | |
300  * | |
301  * | |
302  * next_edge | | next_diag_edge
303  * | |
304  * | |
305  * | |
306  * v |
307  * (Corner 0, next) -----------------------> (Corner 1, diagonal)
308  * prev_diag_edge
309  */
310 
311  next_corner_index = (data->corner_index + 1) % 4;
312  prev_corner_index = (data->corner_index + 3) % 4;
313  diag_corner_index = (data->corner_index + 2) % 4;
314 
315  next_corner = data->plane_marker->corners[next_corner_index];
316  prev_corner = data->plane_marker->corners[prev_corner_index];
317  diag_corner = data->plane_marker->corners[diag_corner_index];
318 
319  sub_v2_v2v2(next_edge, next_corner, data->corner);
320  sub_v2_v2v2(prev_edge, data->corner, prev_corner);
321  sub_v2_v2v2(next_diag_edge, prev_corner, diag_corner);
322  sub_v2_v2v2(prev_diag_edge, diag_corner, next_corner);
323 
324  if (cross_v2v2(prev_edge, next_edge) < 0.0f) {
325  closest_to_line_v2(data->corner, data->corner, prev_corner, next_corner);
326  }
327 
328  if (cross_v2v2(next_diag_edge, prev_edge) < 0.0f) {
329  closest_to_line_v2(data->corner, data->corner, prev_corner, diag_corner);
330  }
331 
332  if (cross_v2v2(next_edge, prev_diag_edge) < 0.0f) {
333  closest_to_line_v2(data->corner, data->corner, next_corner, diag_corner);
334  }
335 
336  data->previous_mval[0] = event->mval[0];
337  data->previous_mval[1] = event->mval[1];
338  copy_v2_v2(data->previous_corner, data->corner);
339 
342 
343  break;
344 
345  case LEFTMOUSE:
346  case RIGHTMOUSE:
347  if (event->type == data->launch_event && event->val == KM_RELEASE) {
348  /* Marker is now keyframed. */
349  data->plane_marker->flag &= ~PLANE_MARKER_TRACKED;
350 
352 
354 
356 
359 
360  return OPERATOR_FINISHED;
361  }
362 
363  break;
364 
365  case EVT_ESCKEY:
367 
369 
371 
373 
374  return OPERATOR_CANCELLED;
375  }
376 
377  return OPERATOR_RUNNING_MODAL;
378 }
379 
381 {
382  /* identifiers */
383  ot->name = "Slide Plane Marker";
384  ot->description = "Slide plane marker areas";
385  ot->idname = "CLIP_OT_slide_plane_marker";
386 
387  /* api callbacks */
391 
392  /* flags */
394 }
struct SpaceClip * CTX_wm_space_clip(const bContext *C)
Definition: context.c:899
struct ARegion * CTX_wm_region(const bContext *C)
Definition: context.c:725
void BKE_report(ReportList *reports, ReportType type, const char *message)
Definition: report.c:104
struct MovieTrackingPlaneMarker * BKE_tracking_plane_marker_ensure(struct MovieTrackingPlaneTrack *plane_track, int framenr)
Definition: tracking.c:2058
struct ListBase * BKE_tracking_get_active_tracks(struct MovieTracking *tracking)
Definition: tracking.c:365
#define PLANE_TRACK_VIEW_SELECTED(plane_track)
Definition: BKE_tracking.h:503
struct MovieTrackingPlaneTrack * BKE_tracking_plane_track_add(struct MovieTracking *tracking, struct ListBase *plane_tracks_base, struct ListBase *tracks, int framenr)
Definition: tracking.c:1709
struct MovieTrackingPlaneMarker * BKE_tracking_plane_marker_get(struct MovieTrackingPlaneTrack *plane_track, int framenr)
Definition: tracking.c:1994
void BKE_tracking_track_plane_from_existing_motion(struct MovieTrackingPlaneTrack *plane_track, int start_frame)
void BKE_tracking_tracks_deselect_all(struct ListBase *tracksbase)
Definition: tracking.c:1372
struct ListBase * BKE_tracking_get_active_plane_tracks(struct MovieTracking *tracking)
Definition: tracking.c:377
#define ATTR_FALLTHROUGH
MINLINE float square_f(float a)
float closest_to_line_v2(float r_close[2], const float p[2], const float l1[2], const float l2[2])
Definition: math_geom.c:3371
MINLINE void copy_v2_v2(float r[2], const float a[2])
MINLINE float cross_v2v2(const float a[2], const float b[2]) ATTR_WARN_UNUSED_RESULT
MINLINE void sub_v2_v2v2(float r[2], const float a[2], const float b[2])
#define ELEM(...)
void DEG_id_tag_update(struct ID *id, int flag)
@ ID_RECALC_COPY_ON_WRITE
Definition: DNA_ID.h:654
@ PLANE_MARKER_TRACKED
@ OPERATOR_CANCELLED
@ OPERATOR_FINISHED
@ OPERATOR_RUNNING_MODAL
@ OPERATOR_PASS_THROUGH
int ED_space_clip_get_clip_frame_number(struct SpaceClip *sc)
Definition: clip_editor.c:227
void ED_space_clip_get_size(struct SpaceClip *sc, int *width, int *height)
Definition: clip_editor.c:141
void ED_clip_mouse_pos(struct SpaceClip *sc, struct ARegion *region, const int mval[2], float co[2])
Definition: clip_editor.c:543
bool ED_space_clip_tracking_poll(struct bContext *C)
Definition: clip_editor.c:98
struct MovieClip * ED_space_clip_get_clip(struct SpaceClip *sc)
Definition: clip_editor.c:572
_GL_VOID GLfloat value _GL_VOID_RET _GL_VOID const GLuint GLboolean *residences _GL_BOOL_RET _GL_VOID GLsizei GLfloat GLfloat GLfloat GLfloat const GLubyte *bitmap _GL_VOID_RET _GL_VOID GLenum const void *lists _GL_VOID_RET _GL_VOID const GLdouble *equation _GL_VOID_RET _GL_VOID GLdouble GLdouble blue _GL_VOID_RET _GL_VOID GLfloat GLfloat blue _GL_VOID_RET _GL_VOID GLint GLint blue _GL_VOID_RET _GL_VOID GLshort GLshort blue _GL_VOID_RET _GL_VOID GLubyte GLubyte blue _GL_VOID_RET _GL_VOID GLuint GLuint blue _GL_VOID_RET _GL_VOID GLushort GLushort blue _GL_VOID_RET _GL_VOID GLbyte GLbyte GLbyte alpha _GL_VOID_RET _GL_VOID GLdouble GLdouble GLdouble alpha _GL_VOID_RET _GL_VOID GLfloat GLfloat GLfloat alpha _GL_VOID_RET _GL_VOID GLint GLint GLint alpha _GL_VOID_RET _GL_VOID GLshort GLshort GLshort alpha _GL_VOID_RET _GL_VOID GLubyte GLubyte GLubyte alpha _GL_VOID_RET _GL_VOID GLuint GLuint GLuint alpha _GL_VOID_RET _GL_VOID GLushort GLushort GLushort alpha _GL_VOID_RET _GL_VOID GLenum mode _GL_VOID_RET _GL_VOID GLint GLsizei width
_GL_VOID GLfloat value _GL_VOID_RET _GL_VOID const GLuint GLboolean *residences _GL_BOOL_RET _GL_VOID GLsizei height
Read Guarded memory(de)allocation.
#define C
Definition: RandGen.cpp:39
#define NC_GEOM
Definition: WM_types.h:294
@ OPTYPE_BLOCKING
Definition: WM_types.h:157
@ OPTYPE_UNDO
Definition: WM_types.h:155
@ OPTYPE_GRAB_CURSOR_XY
Definition: WM_types.h:161
@ OPTYPE_REGISTER
Definition: WM_types.h:153
#define NC_MOVIECLIP
Definition: WM_types.h:298
#define NA_EDITED
Definition: WM_types.h:462
#define KM_PRESS
Definition: WM_types.h:242
#define ND_SELECT
Definition: WM_types.h:407
#define KM_RELEASE
Definition: WM_types.h:243
#define SELECT
void(* MEM_freeN)(void *vmemh)
Definition: mallocn.c:41
void *(* MEM_callocN)(size_t len, const char *str)
Definition: mallocn.c:45
void * first
Definition: DNA_listBase.h:47
struct MovieTracking tracking
MovieTrackingPlaneTrack * act_plane_track
MovieTrackingTrack * act_track
MovieTrackingPlaneTrack * plane_track
MovieTrackingPlaneMarker * plane_marker
short val
Definition: WM_types.h:579
int mval[2]
Definition: WM_types.h:583
short type
Definition: WM_types.h:577
int(* invoke)(struct bContext *, struct wmOperator *, const struct wmEvent *) ATTR_WARN_UNUSED_RESULT
Definition: WM_types.h:752
const char * name
Definition: WM_types.h:721
int(* modal)(struct bContext *, struct wmOperator *, const struct wmEvent *) ATTR_WARN_UNUSED_RESULT
Definition: WM_types.h:768
const char * idname
Definition: WM_types.h:723
bool(* poll)(struct bContext *) ATTR_WARN_UNUSED_RESULT
Definition: WM_types.h:776
const char * description
Definition: WM_types.h:726
int(* exec)(struct bContext *, struct wmOperator *) ATTR_WARN_UNUSED_RESULT
Definition: WM_types.h:736
struct ReportList * reports
void clip_tracking_hide_cursor(struct bContext *C)
void clip_tracking_show_cursor(struct bContext *C)
void CLIP_OT_create_plane_track(wmOperatorType *ot)
static int slide_plane_marker_modal(bContext *C, wmOperator *op, const wmEvent *event)
static void * slide_plane_marker_customdata(bContext *C, const wmEvent *event)
static MovieTrackingPlaneTrack * tracking_plane_marker_check_slide(bContext *C, const wmEvent *event, int *r_corner)
static float mouse_to_plane_slide_zone_distance_squared(const float co[2], const float slide_zone[2], int width, int height)
struct SlidePlaneMarkerData SlidePlaneMarkerData
static int slide_plane_marker_invoke(bContext *C, wmOperator *op, const wmEvent *event)
static void slide_plane_marker_update_homographies(SpaceClip *sc, SlidePlaneMarkerData *data)
void CLIP_OT_slide_plane_marker(wmOperatorType *ot)
static int create_plane_track_tracks_exec(bContext *C, wmOperator *op)
static void cancel_mouse_slide_plane_marker(SlidePlaneMarkerData *data)
static void free_slide_plane_marker_data(SlidePlaneMarkerData *data)
int WM_userdef_event_type_from_keymap_type(int kmitype)
wmEventHandler_Op * WM_event_add_modal_handler(bContext *C, wmOperator *op)
void WM_event_add_notifier(const bContext *C, uint type, void *reference)
@ RIGHTMOUSE
@ EVT_RIGHTCTRLKEY
@ EVT_LEFTCTRLKEY
@ MOUSEMOVE
@ LEFTMOUSE
@ EVT_ESCKEY
@ EVT_RIGHTSHIFTKEY
@ EVT_LEFTSHIFTKEY
wmOperatorType * ot
Definition: wm_files.c:3156