Correction for recent commit clearing font cache
[blender.git] / source / blender / editors / interface / view2d_ops.c
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version. 
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * The Original Code is Copyright (C) 2008 Blender Foundation.
19  * All rights reserved.
20  * 
21  * Contributor(s): Blender Foundation, Joshua Leung
22  *
23  * ***** END GPL LICENSE BLOCK *****
24  */
25
26 /** \file blender/editors/interface/view2d_ops.c
27  *  \ingroup edinterface
28  */
29
30
31 #include <math.h>
32
33 #include "MEM_guardedalloc.h"
34
35 #include "DNA_userdef_types.h"
36
37 #include "BLI_blenlib.h"
38 #include "BLI_utildefines.h"
39 #include "BLI_math_base.h"
40
41 #include "BKE_context.h"
42
43 #include "RNA_access.h"
44 #include "RNA_define.h"
45
46 #include "WM_api.h"
47 #include "WM_types.h"
48
49 #include "ED_screen.h"
50
51 #include "UI_view2d.h"
52 #include "UI_interface.h"
53
54 #include "PIL_time.h" /* USER_ZOOM_CONT */
55
56 static int view2d_poll(bContext *C)
57 {
58         ARegion *ar = CTX_wm_region(C);
59
60         return (ar != NULL) && (ar->v2d.flag & V2D_IS_INITIALISED);
61 }
62
63 /* ********************************************************* */
64 /* VIEW PANNING OPERATOR                                                                 */
65
66 /*  This group of operators come in several forms:
67  *              1) Modal 'dragging' with MMB - where movement of mouse dictates amount to pan view by
68  *              2) Scrollwheel 'steps' - rolling mousewheel by one step moves view by predefined amount
69  *
70  *      In order to make sure this works, each operator must define the following RNA-Operator Props:
71  *              deltax, deltay  - define how much to move view by (relative to zoom-correction factor)
72  */
73
74 /* ------------------ Shared 'core' stuff ---------------------- */
75  
76 /* temp customdata for operator */
77 typedef struct v2dViewPanData {
78         bScreen *sc;            /* screen where view pan was initiated */
79         ScrArea *sa;            /* area where view pan was initiated */
80         ARegion *ar;            /* region where view pan was initiated */
81         View2D *v2d;            /* view2d we're operating in */
82
83         float facx, facy;       /* amount to move view relative to zoom */
84
85         /* options for version 1 */
86         int startx, starty;     /* mouse x/y values in window when operator was initiated */
87         int lastx, lasty;       /* previous x/y values of mouse in window */
88         int invoke_event;       /* event starting pan, for modal exit */
89         
90         short in_scroller;      /* for MMB in scrollers (old feature in past, but now not that useful) */
91 } v2dViewPanData;
92  
93 /* initialize panning customdata */
94 static int view_pan_init(bContext *C, wmOperator *op)
95 {
96         ARegion *ar = CTX_wm_region(C);
97         v2dViewPanData *vpd;
98         View2D *v2d;
99         float winx, winy;
100         
101         /* regions now have v2d-data by default, so check for region */
102         if (ar == NULL)
103                 return 0;
104                 
105         /* check if panning is allowed at all */
106         v2d = &ar->v2d;
107         if ((v2d->keepofs & V2D_LOCKOFS_X) && (v2d->keepofs & V2D_LOCKOFS_Y))
108                 return 0;
109         
110         /* set custom-data for operator */
111         vpd = MEM_callocN(sizeof(v2dViewPanData), "v2dViewPanData");
112         op->customdata = vpd;
113         
114         /* set pointers to owners */
115         vpd->sc = CTX_wm_screen(C);
116         vpd->sa = CTX_wm_area(C);
117         vpd->v2d = v2d;
118         vpd->ar = ar;
119         
120         /* calculate translation factor - based on size of view */
121         winx = (float)(BLI_rcti_size_x(&ar->winrct) + 1);
122         winy = (float)(BLI_rcti_size_y(&ar->winrct) + 1);
123         vpd->facx = (BLI_rctf_size_x(&v2d->cur)) / winx;
124         vpd->facy = (BLI_rctf_size_y(&v2d->cur)) / winy;
125         
126         return 1;
127 }
128
129 /* apply transform to view (i.e. adjust 'cur' rect) */
130 static void view_pan_apply(bContext *C, wmOperator *op)
131 {
132         v2dViewPanData *vpd = op->customdata;
133         View2D *v2d = vpd->v2d;
134         float dx, dy;
135         
136         /* calculate amount to move view by */
137         dx = vpd->facx * (float)RNA_int_get(op->ptr, "deltax");
138         dy = vpd->facy * (float)RNA_int_get(op->ptr, "deltay");
139         
140         /* only move view on an axis if change is allowed */
141         if ((v2d->keepofs & V2D_LOCKOFS_X) == 0) {
142                 v2d->cur.xmin += dx;
143                 v2d->cur.xmax += dx;
144         }
145         if ((v2d->keepofs & V2D_LOCKOFS_Y) == 0) {
146                 v2d->cur.ymin += dy;
147                 v2d->cur.ymax += dy;
148         }
149         
150         /* validate that view is in valid configuration after this operation */
151         UI_view2d_curRect_validate(v2d);
152         
153         /* request updates to be done... */
154         ED_region_tag_redraw(vpd->ar);
155         WM_event_add_mousemove(C);
156         
157         UI_view2d_sync(vpd->sc, vpd->sa, v2d, V2D_LOCK_COPY);
158         
159         /* exceptions */
160         if (vpd->sa->spacetype == SPACE_OUTLINER) {
161                 /* don't rebuild full tree, since we're just changing our view */
162                 SpaceOops *soops = vpd->sa->spacedata.first;
163                 soops->storeflag |= SO_TREESTORE_REDRAW;
164         }
165 }
166
167 /* cleanup temp customdata  */
168 static void view_pan_exit(wmOperator *op)
169 {
170         if (op->customdata) {
171                 MEM_freeN(op->customdata);
172                 op->customdata = NULL;
173         }
174
175  
176 /* ------------------ Modal Drag Version (1) ---------------------- */
177
178 /* for 'redo' only, with no user input */
179 static int view_pan_exec(bContext *C, wmOperator *op)
180 {
181         if (!view_pan_init(C, op))
182                 return OPERATOR_CANCELLED;
183         
184         view_pan_apply(C, op);
185         view_pan_exit(op);
186         return OPERATOR_FINISHED;
187 }
188
189 /* set up modal operator and relevant settings */
190 static int view_pan_invoke(bContext *C, wmOperator *op, const wmEvent *event)
191 {
192         wmWindow *window = CTX_wm_window(C);
193         v2dViewPanData *vpd;
194         View2D *v2d;
195         
196         /* set up customdata */
197         if (!view_pan_init(C, op))
198                 return OPERATOR_PASS_THROUGH;
199         
200         vpd = op->customdata;
201         v2d = vpd->v2d;
202         
203         /* set initial settings */
204         vpd->startx = vpd->lastx = event->x;
205         vpd->starty = vpd->lasty = event->y;
206         vpd->invoke_event = event->type;
207         
208         if (event->type == MOUSEPAN) {
209                 RNA_int_set(op->ptr, "deltax", event->prevx - event->x);
210                 RNA_int_set(op->ptr, "deltay", event->prevy - event->y);
211                 
212                 view_pan_apply(C, op);
213                 view_pan_exit(op);
214                 return OPERATOR_FINISHED;
215         }
216         
217         RNA_int_set(op->ptr, "deltax", 0);
218         RNA_int_set(op->ptr, "deltay", 0);
219         
220         if (v2d->keepofs & V2D_LOCKOFS_X)
221                 WM_cursor_modal_set(window, BC_NS_SCROLLCURSOR);
222         else if (v2d->keepofs & V2D_LOCKOFS_Y)
223                 WM_cursor_modal_set(window, BC_EW_SCROLLCURSOR);
224         else
225                 WM_cursor_modal_set(window, BC_NSEW_SCROLLCURSOR);
226         
227         /* add temp handler */
228         WM_event_add_modal_handler(C, op);
229
230         return OPERATOR_RUNNING_MODAL;
231 }
232
233 /* handle user input - calculations of mouse-movement need to be done here, not in the apply callback! */
234 static int view_pan_modal(bContext *C, wmOperator *op, const wmEvent *event)
235 {
236         v2dViewPanData *vpd = op->customdata;
237         
238         /* execute the events */
239         switch (event->type) {
240                 case MOUSEMOVE:
241                 {
242                         /* calculate new delta transform, then store mouse-coordinates for next-time */
243                         RNA_int_set(op->ptr, "deltax", (vpd->lastx - event->x));
244                         RNA_int_set(op->ptr, "deltay", (vpd->lasty - event->y));
245                         
246                         vpd->lastx = event->x;
247                         vpd->lasty = event->y;
248                         
249                         view_pan_apply(C, op);
250                         break;
251                 }
252                         /* XXX - Mode switching isn't implemented. See comments in 36818.
253                          * switch to zoom */
254 #if 0
255                 case LEFTMOUSE:
256                         if (event->val == KM_PRESS) {
257                                 /* calculate overall delta mouse-movement for redo */
258                                 RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx));
259                                 RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty));
260                                 
261                                 view_pan_exit(op);
262                                 WM_cursor_modal_restore(CTX_wm_window(C));
263                                 WM_operator_name_call(C, "VIEW2D_OT_zoom", WM_OP_INVOKE_DEFAULT, NULL);
264                                 return OPERATOR_FINISHED;
265                         }
266 #endif
267                 default:
268                         if (event->type == vpd->invoke_event || event->type == ESCKEY) {
269                                 if (event->val == KM_RELEASE) {
270                                         /* calculate overall delta mouse-movement for redo */
271                                         RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx));
272                                         RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty));
273                                         
274                                         view_pan_exit(op);
275                                         WM_cursor_modal_restore(CTX_wm_window(C));
276                                         
277                                         return OPERATOR_FINISHED;
278                                 }
279                         }
280                         break;
281         }
282
283         return OPERATOR_RUNNING_MODAL;
284 }
285
286 static void view_pan_cancel(bContext *UNUSED(C), wmOperator *op)
287 {
288         view_pan_exit(op);
289 }
290
291 static void VIEW2D_OT_pan(wmOperatorType *ot)
292 {
293         /* identifiers */
294         ot->name = "Pan View";
295         ot->description = "Pan the view";
296         ot->idname = "VIEW2D_OT_pan";
297         
298         /* api callbacks */
299         ot->exec = view_pan_exec;
300         ot->invoke = view_pan_invoke;
301         ot->modal = view_pan_modal;
302         ot->cancel = view_pan_cancel;
303         
304         /* operator is modal */
305         ot->flag = OPTYPE_BLOCKING | OPTYPE_GRAB_POINTER;
306         
307         /* rna - must keep these in sync with the other operators */
308         RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX);
309         RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX);
310 }
311
312 /* ------------------ Scrollwheel Versions (2) ---------------------- */
313
314 /* this operator only needs this single callback, where it calls the view_pan_*() methods */
315 static int view_scrollright_exec(bContext *C, wmOperator *op)
316 {
317         v2dViewPanData *vpd;
318         
319         /* initialize default settings (and validate if ok to run) */
320         if (!view_pan_init(C, op))
321                 return OPERATOR_PASS_THROUGH;
322                 
323         /* also, check if can pan in horizontal axis */
324         vpd = op->customdata;
325         if (vpd->v2d->keepofs & V2D_LOCKOFS_X) {
326                 view_pan_exit(op);
327                 return OPERATOR_PASS_THROUGH;
328         }
329         
330         /* set RNA-Props - only movement in positive x-direction */
331         RNA_int_set(op->ptr, "deltax", 20);
332         RNA_int_set(op->ptr, "deltay", 0);
333         
334         /* apply movement, then we're done */
335         view_pan_apply(C, op);
336         view_pan_exit(op);
337         
338         return OPERATOR_FINISHED;
339 }
340
341 static void VIEW2D_OT_scroll_right(wmOperatorType *ot)
342 {
343         /* identifiers */
344         ot->name = "Scroll Right";
345         ot->description = "Scroll the view right";
346         ot->idname = "VIEW2D_OT_scroll_right";
347         
348         /* api callbacks */
349         ot->exec = view_scrollright_exec;
350         
351         /* rna - must keep these in sync with the other operators */
352         RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX);
353         RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX);
354 }
355
356
357
358 /* this operator only needs this single callback, where it calls the view_pan_*() methods */
359 static int view_scrollleft_exec(bContext *C, wmOperator *op)
360 {
361         v2dViewPanData *vpd;
362         
363         /* initialize default settings (and validate if ok to run) */
364         if (!view_pan_init(C, op))
365                 return OPERATOR_PASS_THROUGH;
366                 
367         /* also, check if can pan in horizontal axis */
368         vpd = op->customdata;
369         if (vpd->v2d->keepofs & V2D_LOCKOFS_X) {
370                 view_pan_exit(op);
371                 return OPERATOR_PASS_THROUGH;
372         }
373         
374         /* set RNA-Props - only movement in negative x-direction */
375         RNA_int_set(op->ptr, "deltax", -20);
376         RNA_int_set(op->ptr, "deltay", 0);
377         
378         /* apply movement, then we're done */
379         view_pan_apply(C, op);
380         view_pan_exit(op);
381         
382         return OPERATOR_FINISHED;
383 }
384
385 static void VIEW2D_OT_scroll_left(wmOperatorType *ot)
386 {
387         /* identifiers */
388         ot->name = "Scroll Left";
389         ot->description = "Scroll the view left";
390         ot->idname = "VIEW2D_OT_scroll_left";
391         
392         /* api callbacks */
393         ot->exec = view_scrollleft_exec;
394         
395         /* rna - must keep these in sync with the other operators */
396         RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX);
397         RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX);
398 }
399
400
401 /* this operator only needs this single callback, where it calls the view_pan_*() methods */
402 static int view_scrolldown_exec(bContext *C, wmOperator *op)
403 {
404         v2dViewPanData *vpd;
405         
406         /* initialize default settings (and validate if ok to run) */
407         if (!view_pan_init(C, op))
408                 return OPERATOR_PASS_THROUGH;
409                 
410         /* also, check if can pan in vertical axis */
411         vpd = op->customdata;
412         if (vpd->v2d->keepofs & V2D_LOCKOFS_Y) {
413                 view_pan_exit(op);
414                 return OPERATOR_PASS_THROUGH;
415         }
416         
417         /* set RNA-Props */
418         RNA_int_set(op->ptr, "deltax", 0);
419         RNA_int_set(op->ptr, "deltay", -40);
420         
421         if (RNA_boolean_get(op->ptr, "page")) {
422                 ARegion *ar = CTX_wm_region(C);
423                 RNA_int_set(op->ptr, "deltay", ar->v2d.mask.ymin - ar->v2d.mask.ymax);
424         }
425         
426         /* apply movement, then we're done */
427         view_pan_apply(C, op);
428         view_pan_exit(op);
429         
430         return OPERATOR_FINISHED;
431 }
432
433 static void VIEW2D_OT_scroll_down(wmOperatorType *ot)
434 {
435         /* identifiers */
436         ot->name = "Scroll Down";
437         ot->description = "Scroll the view down";
438         ot->idname = "VIEW2D_OT_scroll_down";
439         
440         /* api callbacks */
441         ot->exec = view_scrolldown_exec;
442         
443         /* rna - must keep these in sync with the other operators */
444         RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX);
445         RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX);
446         RNA_def_boolean(ot->srna, "page", 0, "Page", "Scroll down one page");
447 }
448
449
450
451 /* this operator only needs this single callback, where it calls the view_pan_*() methods */
452 static int view_scrollup_exec(bContext *C, wmOperator *op)
453 {
454         v2dViewPanData *vpd;
455         
456         /* initialize default settings (and validate if ok to run) */
457         if (!view_pan_init(C, op))
458                 return OPERATOR_PASS_THROUGH;
459                 
460         /* also, check if can pan in vertical axis */
461         vpd = op->customdata;
462         if (vpd->v2d->keepofs & V2D_LOCKOFS_Y) {
463                 view_pan_exit(op);
464                 return OPERATOR_PASS_THROUGH;
465         }
466         
467         /* set RNA-Props */
468         RNA_int_set(op->ptr, "deltax", 0);
469         RNA_int_set(op->ptr, "deltay", 40);
470         
471         if (RNA_boolean_get(op->ptr, "page")) {
472                 ARegion *ar = CTX_wm_region(C);
473                 RNA_int_set(op->ptr, "deltay", BLI_rcti_size_y(&ar->v2d.mask));
474         }
475         
476         /* apply movement, then we're done */
477         view_pan_apply(C, op);
478         view_pan_exit(op);
479         
480         return OPERATOR_FINISHED;
481 }
482
483 static void VIEW2D_OT_scroll_up(wmOperatorType *ot)
484 {
485         /* identifiers */
486         ot->name = "Scroll Up";
487         ot->description = "Scroll the view up";
488         ot->idname = "VIEW2D_OT_scroll_up";
489         
490         /* api callbacks */
491         ot->exec = view_scrollup_exec;
492         
493         /* rna - must keep these in sync with the other operators */
494         RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX);
495         RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX);
496         RNA_def_boolean(ot->srna, "page", 0, "Page", "Scroll up one page");
497 }
498
499 /* ********************************************************* */
500 /* SINGLE-STEP VIEW ZOOMING OPERATOR                                             */
501
502 /* This group of operators come in several forms:
503  * 1) Scrollwheel 'steps' - rolling mousewheel by one step zooms view by predefined amount
504  * 2) Scrollwheel 'steps' + alt + ctrl/shift - zooms view on one axis only (ctrl=x, shift=y)  // XXX this could be implemented...
505  * 3) Pad +/- Keys - pressing each key moves the zooms the view by a predefined amount
506  *
507  * In order to make sure this works, each operator must define the following RNA-Operator Props:
508  * zoomfacx, zoomfacy - These two zoom factors allow for non-uniform scaling.
509  *                      It is safe to scale by 0, as these factors are used to determine
510  *                      amount to enlarge 'cur' by
511  */
512
513 /* ------------------ 'Shared' stuff ------------------------ */
514
515 /* temp customdata for operator */
516 typedef struct v2dViewZoomData {
517         View2D *v2d;            /* view2d we're operating in */
518         ARegion *ar;
519
520         /* needed for continuous zoom */
521         wmTimer *timer;
522         double timer_lastdraw;
523
524         int lastx, lasty;       /* previous x/y values of mouse in window */
525         int invoke_event;       /* event type that invoked, for modal exits */
526         float dx, dy;           /* running tally of previous delta values (for obtaining final zoom) */
527         float mx_2d, my_2d;     /* initial mouse location in v2d coords */
528 } v2dViewZoomData;
529
530
531 /* initialize panning customdata */
532 static int view_zoomdrag_init(bContext *C, wmOperator *op)
533 {
534         ARegion *ar = CTX_wm_region(C);
535         v2dViewZoomData *vzd;
536         View2D *v2d;
537         
538         /* regions now have v2d-data by default, so check for region */
539         if (ar == NULL)
540                 return 0;
541         v2d = &ar->v2d;
542         
543         /* check that 2d-view is zoomable */
544         if ((v2d->keepzoom & V2D_LOCKZOOM_X) && (v2d->keepzoom & V2D_LOCKZOOM_Y))
545                 return 0;
546         
547         /* set custom-data for operator */
548         vzd = MEM_callocN(sizeof(v2dViewZoomData), "v2dViewZoomData");
549         op->customdata = vzd;
550         
551         /* set pointers to owners */
552         vzd->v2d = v2d;
553         vzd->ar = ar;
554         
555         return 1;
556 }
557
558 /* check if step-zoom can be applied */
559 static int view_zoom_poll(bContext *C)
560 {
561         ARegion *ar = CTX_wm_region(C);
562         View2D *v2d;
563         
564         /* check if there's a region in context to work with */
565         if (ar == NULL)
566                 return 0;
567         v2d = &ar->v2d;
568         
569         /* check that 2d-view is zoomable */
570         if ((v2d->keepzoom & V2D_LOCKZOOM_X) && (v2d->keepzoom & V2D_LOCKZOOM_Y))
571                 return 0;
572                 
573         /* view is zoomable */
574         return 1;
575 }
576  
577 /* apply transform to view (i.e. adjust 'cur' rect) */
578 static void view_zoomstep_apply(bContext *C, wmOperator *op)
579 {
580         v2dViewZoomData *vzd = op->customdata;
581         ARegion *ar = CTX_wm_region(C);
582         View2D *v2d = &ar->v2d;
583         float dx, dy, facx, facy;
584         
585         /* calculate amount to move view by, ensuring symmetry so the
586          * old zoom level is restored after zooming back the same amount 
587          */
588         facx = RNA_float_get(op->ptr, "zoomfacx");
589         facy = RNA_float_get(op->ptr, "zoomfacy");
590
591         if (facx >= 0.0f) {
592                 dx = BLI_rctf_size_x(&v2d->cur) * facx;
593                 dy = BLI_rctf_size_y(&v2d->cur) * facy;
594         }
595         else {
596                 dx = (BLI_rctf_size_x(&v2d->cur) / (1.0f + 2.0f * facx)) * facx;
597                 dy = (BLI_rctf_size_y(&v2d->cur) / (1.0f + 2.0f * facy)) * facy;
598         }
599
600         /* only resize view on an axis if change is allowed */
601         if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
602                 if (v2d->keepofs & V2D_LOCKOFS_X) {
603                         v2d->cur.xmax -= 2 * dx;
604                 }
605                 else if (v2d->keepofs & V2D_KEEPOFS_X) {
606                         if (v2d->align & V2D_ALIGN_NO_POS_X)
607                                 v2d->cur.xmin += 2 * dx;
608                         else
609                                 v2d->cur.xmax -= 2 * dx;
610                 }
611                 else {
612                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
613                                 float mval_fac = (vzd->mx_2d - v2d->cur.xmin) / BLI_rctf_size_x(&v2d->cur);
614                                 float mval_faci = 1.0f - mval_fac;
615                                 float ofs = (mval_fac * dx) - (mval_faci * dx);
616                                 
617                                 v2d->cur.xmin += ofs + dx;
618                                 v2d->cur.xmax += ofs - dx;
619                         }
620                         else {
621                                 v2d->cur.xmin += dx;
622                                 v2d->cur.xmax -= dx;
623                         }
624                 }
625         }
626         if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
627                 if (v2d->keepofs & V2D_LOCKOFS_Y) {
628                         v2d->cur.ymax -= 2 * dy;
629                 }
630                 else if (v2d->keepofs & V2D_KEEPOFS_Y) {
631                         if (v2d->align & V2D_ALIGN_NO_POS_Y)
632                                 v2d->cur.ymin += 2 * dy;
633                         else
634                                 v2d->cur.ymax -= 2 * dy;
635                 }
636                 else {
637                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
638                                 float mval_fac = (vzd->my_2d - v2d->cur.ymin) / BLI_rctf_size_y(&v2d->cur);
639                                 float mval_faci = 1.0f - mval_fac;
640                                 float ofs = (mval_fac * dy) - (mval_faci * dy);
641                                 
642                                 v2d->cur.ymin += ofs + dy;
643                                 v2d->cur.ymax += ofs - dy;
644                         }
645                         else {
646                                 v2d->cur.ymin += dy;
647                                 v2d->cur.ymax -= dy;
648                         }
649                 }
650         }
651
652         /* validate that view is in valid configuration after this operation */
653         UI_view2d_curRect_validate(v2d);
654
655         /* request updates to be done... */
656         ED_region_tag_redraw(vzd->ar);
657         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
658 }
659
660 /* --------------- Individual Operators ------------------- */
661
662 /* cleanup temp customdata  */
663 static void view_zoomstep_exit(wmOperator *op)
664 {
665         UI_view2d_zoom_cache_reset();
666
667         if (op->customdata) {
668                 MEM_freeN(op->customdata);
669                 op->customdata = NULL;
670         }
671 }
672
673 /* this operator only needs this single callback, where it calls the view_zoom_*() methods */
674 static int view_zoomin_exec(bContext *C, wmOperator *op)
675 {
676         ScrArea *sa = CTX_wm_area(C);
677         bool do_zoom_x = true;
678         bool do_zoom_y = true;
679
680         /* check that there's an active region, as View2D data resides there */
681         if (!view_zoom_poll(C))
682                 return OPERATOR_PASS_THROUGH;
683         
684         /* default not to zoom the sequencer vertically */
685         if (sa && sa->spacetype == SPACE_SEQ) {
686                 ARegion *ar = CTX_wm_region(C);
687
688                 if (ar && ar->regiontype != RGN_TYPE_PREVIEW)
689                         do_zoom_y = false;
690         }
691
692         /* set RNA-Props - zooming in by uniform factor */
693         RNA_float_set(op->ptr, "zoomfacx", do_zoom_x ? 0.0375f : 0.0f);
694         RNA_float_set(op->ptr, "zoomfacy", do_zoom_y ? 0.0375f : 0.0f);
695         
696         /* apply movement, then we're done */
697         view_zoomstep_apply(C, op);
698         
699         view_zoomstep_exit(op);
700         
701         return OPERATOR_FINISHED;
702 }
703
704 static int view_zoomin_invoke(bContext *C, wmOperator *op, const wmEvent *event)
705 {
706         v2dViewZoomData *vzd;
707         
708         if (!view_zoomdrag_init(C, op))
709                 return OPERATOR_PASS_THROUGH;
710         
711         vzd = op->customdata;
712         
713         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
714                 ARegion *ar = CTX_wm_region(C);
715                 
716                 /* store initial mouse position (in view space) */
717                 UI_view2d_region_to_view(&ar->v2d, 
718                                          event->mval[0], event->mval[1],
719                                          &vzd->mx_2d, &vzd->my_2d);
720         }
721         
722         return view_zoomin_exec(C, op);
723 }
724
725 static void VIEW2D_OT_zoom_in(wmOperatorType *ot)
726 {
727         /* identifiers */
728         ot->name = "Zoom In";
729         ot->description = "Zoom in the view";
730         ot->idname = "VIEW2D_OT_zoom_in";
731         
732         /* api callbacks */
733         ot->invoke = view_zoomin_invoke;
734         ot->exec = view_zoomin_exec;  // XXX, needs view_zoomdrag_init called first.
735         ot->poll = view_zoom_poll;
736         
737         /* rna - must keep these in sync with the other operators */
738         RNA_def_float(ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX);
739         RNA_def_float(ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX);
740 }
741         
742 /* this operator only needs this single callback, where it calls the view_zoom_*() methods */
743 static int view_zoomout_exec(bContext *C, wmOperator *op)
744 {
745         ScrArea *sa = CTX_wm_area(C);
746         short do_zoom_x = true;
747         short do_zoom_y = true;
748
749         /* check that there's an active region, as View2D data resides there */
750         if (!view_zoom_poll(C))
751                 return OPERATOR_PASS_THROUGH;
752         
753         /* default not to zoom the sequencer vertically */
754         if (sa && sa->spacetype == SPACE_SEQ) {
755                 ARegion *ar = CTX_wm_region(C);
756
757                 if (ar && ar->regiontype != RGN_TYPE_PREVIEW)
758                         do_zoom_y = false;
759         }
760
761         /* set RNA-Props - zooming in by uniform factor */
762         RNA_float_set(op->ptr, "zoomfacx", do_zoom_x ? -0.0375f : 0.0f);
763         RNA_float_set(op->ptr, "zoomfacy", do_zoom_y ? -0.0375f : 0.0f);
764         
765         /* apply movement, then we're done */
766         view_zoomstep_apply(C, op);
767
768         view_zoomstep_exit(op);
769         
770         return OPERATOR_FINISHED;
771 }
772
773 static int view_zoomout_invoke(bContext *C, wmOperator *op, const wmEvent *event)
774 {
775         v2dViewZoomData *vzd;
776         
777         if (!view_zoomdrag_init(C, op))
778                 return OPERATOR_PASS_THROUGH;
779
780         vzd = op->customdata;
781         
782         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
783                 ARegion *ar = CTX_wm_region(C);
784                 
785                 /* store initial mouse position (in view space) */
786                 UI_view2d_region_to_view(&ar->v2d, 
787                                          event->mval[0], event->mval[1],
788                                          &vzd->mx_2d, &vzd->my_2d);
789         }
790         
791         return view_zoomout_exec(C, op);
792 }
793
794 static void VIEW2D_OT_zoom_out(wmOperatorType *ot)
795 {
796         /* identifiers */
797         ot->name = "Zoom Out";
798         ot->description = "Zoom out the view";
799         ot->idname = "VIEW2D_OT_zoom_out";
800         
801         /* api callbacks */
802         ot->invoke = view_zoomout_invoke;
803 //      ot->exec = view_zoomout_exec; // XXX, needs view_zoomdrag_init called first.
804         ot->poll = view_zoom_poll;
805         
806         /* rna - must keep these in sync with the other operators */
807         RNA_def_float(ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX);
808         RNA_def_float(ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX);
809 }
810
811 /* ********************************************************* */
812 /* DRAG-ZOOM OPERATOR                                                                    */
813
814 /*  MMB Drag - allows non-uniform scaling by dragging mouse
815  *
816  *      In order to make sure this works, each operator must define the following RNA-Operator Props:
817  *              deltax, deltay  - amounts to add to each side of the 'cur' rect
818  */
819
820 /* apply transform to view (i.e. adjust 'cur' rect) */
821 static void view_zoomdrag_apply(bContext *C, wmOperator *op)
822 {
823         v2dViewZoomData *vzd = op->customdata;
824         View2D *v2d = vzd->v2d;
825         float dx, dy;
826         
827         /* get amount to move view by */
828         dx = RNA_float_get(op->ptr, "deltax");
829         dy = RNA_float_get(op->ptr, "deltay");
830
831         if (U.uiflag & USER_ZOOM_INVERT) {
832                 dx *= -1;
833                 dy *= -1;
834         }
835         
836         /* continuous zoom shouldn't move that fast... */
837         if (U.viewzoom == USER_ZOOM_CONT) { // XXX store this setting as RNA prop?
838                 double time = PIL_check_seconds_timer();
839                 float time_step = (float)(time - vzd->timer_lastdraw);
840
841                 dx *= time_step * 0.5f;
842                 dy *= time_step * 0.5f;
843                 
844                 vzd->timer_lastdraw = time;
845         }
846
847         /* only move view on an axis if change is allowed */
848         if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
849                 if (v2d->keepofs & V2D_LOCKOFS_X) {
850                         v2d->cur.xmax -= 2 * dx;
851                 }
852                 else {
853                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
854                                 float mval_fac = (vzd->mx_2d - v2d->cur.xmin) / BLI_rctf_size_x(&v2d->cur);
855                                 float mval_faci = 1.0f - mval_fac;
856                                 float ofs = (mval_fac * dx) - (mval_faci * dx);
857                                 
858                                 v2d->cur.xmin += ofs + dx;
859                                 v2d->cur.xmax += ofs - dx;
860                         }
861                         else {
862                                 v2d->cur.xmin += dx;
863                                 v2d->cur.xmax -= dx;
864                         }
865                 }
866         }
867         if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
868                 if (v2d->keepofs & V2D_LOCKOFS_Y) {
869                         v2d->cur.ymax -= 2 * dy;
870                 }
871                 else {
872                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
873                                 float mval_fac = (vzd->my_2d - v2d->cur.ymin) / BLI_rctf_size_y(&v2d->cur);
874                                 float mval_faci = 1.0f - mval_fac;
875                                 float ofs = (mval_fac * dy) - (mval_faci * dy);
876                                 
877                                 v2d->cur.ymin += ofs + dy;
878                                 v2d->cur.ymax += ofs - dy;
879                         }
880                         else {
881                                 v2d->cur.ymin += dy;
882                                 v2d->cur.ymax -= dy;
883                         }
884                 }
885         }
886         
887         /* validate that view is in valid configuration after this operation */
888         UI_view2d_curRect_validate(v2d);
889         
890         /* request updates to be done... */
891         ED_region_tag_redraw(vzd->ar);
892         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
893 }
894
895 /* cleanup temp customdata  */
896 static void view_zoomdrag_exit(bContext *C, wmOperator *op)
897 {
898         UI_view2d_zoom_cache_reset();
899
900         if (op->customdata) {
901                 v2dViewZoomData *vzd = op->customdata;
902                 
903                 if (vzd->timer)
904                         WM_event_remove_timer(CTX_wm_manager(C), CTX_wm_window(C), vzd->timer);
905                 
906                 MEM_freeN(op->customdata);
907                 op->customdata = NULL;
908         }
909
910
911 static void view_zoomdrag_cancel(bContext *C, wmOperator *op)
912 {
913         view_zoomdrag_exit(C, op);
914 }
915
916 /* for 'redo' only, with no user input */
917 static int view_zoomdrag_exec(bContext *C, wmOperator *op)
918 {
919         if (!view_zoomdrag_init(C, op))
920                 return OPERATOR_PASS_THROUGH;
921         
922         view_zoomdrag_apply(C, op);
923         view_zoomdrag_exit(C, op);
924         return OPERATOR_FINISHED;
925 }
926
927 /* set up modal operator and relevant settings */
928 static int view_zoomdrag_invoke(bContext *C, wmOperator *op, const wmEvent *event)
929 {
930         wmWindow *window = CTX_wm_window(C);
931         v2dViewZoomData *vzd;
932         View2D *v2d;
933         
934         /* set up customdata */
935         if (!view_zoomdrag_init(C, op))
936                 return OPERATOR_PASS_THROUGH;
937         
938         vzd = op->customdata;
939         v2d = vzd->v2d;
940         
941         if (event->type == MOUSEZOOM || event->type == MOUSEPAN) {
942                 float dx, dy, fac;
943                 
944                 vzd->lastx = event->prevx;
945                 vzd->lasty = event->prevy;
946                 
947                 /* As we have only 1D information (magnify value), feed both axes
948                  * with magnify information that is stored in x axis 
949                  */
950                 fac = 0.01f * (event->prevx - event->x);
951                 dx = fac * BLI_rctf_size_x(&v2d->cur) / 10.0f;
952                 if (event->type == MOUSEPAN)
953                         fac = 0.01f * (event->prevy - event->y);
954                 dy = fac * BLI_rctf_size_y(&v2d->cur) / 10.0f;
955
956                 /* support trackpad zoom to always zoom entirely - the v2d code uses portrait or landscape exceptions */
957                 if (v2d->keepzoom & V2D_KEEPASPECT) {
958                         if (fabsf(dx) > fabsf(dy))
959                                 dy = dx;
960                         else
961                                 dx = dy;
962                 }
963                 RNA_float_set(op->ptr, "deltax", dx);
964                 RNA_float_set(op->ptr, "deltay", dy);
965                 
966                 view_zoomdrag_apply(C, op);
967                 view_zoomdrag_exit(C, op);
968                 return OPERATOR_FINISHED;
969         }
970         
971         /* set initial settings */
972         vzd->lastx = event->x;
973         vzd->lasty = event->y;
974         RNA_float_set(op->ptr, "deltax", 0);
975         RNA_float_set(op->ptr, "deltay", 0);
976         
977         /* for modal exit test */
978         vzd->invoke_event = event->type;
979         
980         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
981                 ARegion *ar = CTX_wm_region(C);
982                 
983                 /* store initial mouse position (in view space) */
984                 UI_view2d_region_to_view(&ar->v2d, 
985                                          event->mval[0], event->mval[1],
986                                          &vzd->mx_2d, &vzd->my_2d);
987         }
988
989         if (v2d->keepofs & V2D_LOCKOFS_X)
990                 WM_cursor_modal_set(window, BC_NS_SCROLLCURSOR);
991         else if (v2d->keepofs & V2D_LOCKOFS_Y)
992                 WM_cursor_modal_set(window, BC_EW_SCROLLCURSOR);
993         else
994                 WM_cursor_modal_set(window, BC_NSEW_SCROLLCURSOR);
995         
996         /* add temp handler */
997         WM_event_add_modal_handler(C, op);
998
999         if (U.viewzoom == USER_ZOOM_CONT) {
1000                 /* needs a timer to continue redrawing */
1001                 vzd->timer = WM_event_add_timer(CTX_wm_manager(C), CTX_wm_window(C), TIMER, 0.01f);
1002                 vzd->timer_lastdraw = PIL_check_seconds_timer();
1003         }
1004
1005         return OPERATOR_RUNNING_MODAL;
1006 }
1007
1008 /* handle user input - calculations of mouse-movement need to be done here, not in the apply callback! */
1009 static int view_zoomdrag_modal(bContext *C, wmOperator *op, const wmEvent *event)
1010 {
1011         v2dViewZoomData *vzd = op->customdata;
1012         View2D *v2d = vzd->v2d;
1013         
1014         /* execute the events */
1015         if (event->type == TIMER && event->customdata == vzd->timer) {
1016                 view_zoomdrag_apply(C, op);
1017         }
1018         else if (event->type == MOUSEMOVE) {
1019                 float dx, dy;
1020                 
1021                 /* calculate new delta transform, based on zooming mode */
1022                 if (U.viewzoom == USER_ZOOM_SCALE) {
1023                         /* 'scale' zooming */
1024                         float dist;
1025                         
1026                         /* x-axis transform */
1027                         dist = BLI_rcti_size_x(&v2d->mask) / 2.0f;
1028                         dx = 1.0f - (fabsf(vzd->lastx - vzd->ar->winrct.xmin - dist) + 2.0f) / (fabsf(event->mval[0] - dist) + 2.0f);
1029                         dx *= 0.5f * BLI_rctf_size_x(&v2d->cur);
1030                         
1031                         /* y-axis transform */
1032                         dist = BLI_rcti_size_y(&v2d->mask) / 2.0f;
1033                         dy = 1.0f - (fabsf(vzd->lasty - vzd->ar->winrct.ymin - dist) + 2.0f) / (fabsf(event->mval[1] - dist) + 2.0f);
1034                         dy *= 0.5f * BLI_rctf_size_y(&v2d->cur);
1035                 }
1036                 else {
1037                         /* 'continuous' or 'dolly' */
1038                         float fac, zoomfac = 0.01f;
1039                         
1040                         /* some view2d's (graph) don't have min/max zoom, or extreme ones */
1041                         if (v2d->maxzoom > 0.0f)
1042                                 zoomfac = CLAMPIS(0.001f * v2d->maxzoom, 0.001f, 0.01f);
1043                         
1044                         /* x-axis transform */
1045                         fac = zoomfac * (event->x - vzd->lastx);
1046                         dx = fac * BLI_rctf_size_x(&v2d->cur);
1047                         
1048                         /* y-axis transform */
1049                         fac = zoomfac * (event->y - vzd->lasty);
1050                         dy = fac * BLI_rctf_size_y(&v2d->cur);
1051                         
1052                 }
1053                 
1054                 /* support zoom to always zoom entirely - the v2d code uses portrait or landscape exceptions */
1055                 if (v2d->keepzoom & V2D_KEEPASPECT) {
1056                         if (fabsf(dx) > fabsf(dy))
1057                                 dy = dx;
1058                         else
1059                                 dx = dy;
1060                 }
1061                 
1062                 /* set transform amount, and add current deltas to stored total delta (for redo) */
1063                 RNA_float_set(op->ptr, "deltax", dx);
1064                 RNA_float_set(op->ptr, "deltay", dy);
1065
1066                 vzd->dx += dx;
1067                 vzd->dy += dy;
1068                 
1069                 /* store mouse coordinates for next time, if not doing continuous zoom
1070                  *      - continuous zoom only depends on distance of mouse to starting point to determine rate of change
1071                  */
1072                 if (U.viewzoom != USER_ZOOM_CONT) { // XXX store this setting as RNA prop?
1073                         vzd->lastx = event->x;
1074                         vzd->lasty = event->y;
1075                 }
1076                 
1077                 /* apply zooming */
1078                 view_zoomdrag_apply(C, op);
1079         }
1080         else if (event->type == vzd->invoke_event || event->type == ESCKEY) {
1081                 if (event->val == KM_RELEASE) {
1082                         
1083                         /* for redo, store the overall deltas - need to respect zoom-locks here... */
1084                         if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0)
1085                                 RNA_float_set(op->ptr, "deltax", vzd->dx);
1086                         else
1087                                 RNA_float_set(op->ptr, "deltax", 0);
1088                                 
1089                         if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0)
1090                                 RNA_float_set(op->ptr, "deltay", vzd->dy);
1091                         else
1092                                 RNA_float_set(op->ptr, "deltay", 0);
1093                         
1094                         /* free customdata */
1095                         view_zoomdrag_exit(C, op);
1096                         WM_cursor_modal_restore(CTX_wm_window(C));
1097                         
1098                         return OPERATOR_FINISHED;
1099                 }
1100         }
1101
1102         return OPERATOR_RUNNING_MODAL;
1103 }
1104
1105 static void VIEW2D_OT_zoom(wmOperatorType *ot)
1106 {
1107         /* identifiers */
1108         ot->name = "Zoom 2D View";
1109         ot->description = "Zoom in/out the view";
1110         ot->idname = "VIEW2D_OT_zoom";
1111         
1112         /* api callbacks */
1113         ot->exec = view_zoomdrag_exec;
1114         ot->invoke = view_zoomdrag_invoke;
1115         ot->modal = view_zoomdrag_modal;
1116         ot->cancel = view_zoomdrag_cancel;
1117         
1118         ot->poll = view_zoom_poll;
1119         
1120         /* operator is repeatable */
1121         ot->flag = OPTYPE_BLOCKING | OPTYPE_GRAB_POINTER;
1122         
1123         /* rna - must keep these in sync with the other operators */
1124         RNA_def_float(ot->srna, "deltax", 0, -FLT_MAX, FLT_MAX, "Delta X", "", -FLT_MAX, FLT_MAX);
1125         RNA_def_float(ot->srna, "deltay", 0, -FLT_MAX, FLT_MAX, "Delta Y", "", -FLT_MAX, FLT_MAX);
1126 }
1127
1128 /* ********************************************************* */
1129 /* BORDER-ZOOM */
1130
1131 /* The user defines a rect using standard borderselect tools, and we use this rect to 
1132  * define the new zoom-level of the view in the following ways:
1133  *      1) LEFTMOUSE - zoom in to view
1134  *      2) RIGHTMOUSE - zoom out of view
1135  *
1136  * Currently, these key mappings are hardcoded, but it shouldn't be too important to
1137  * have custom keymappings for this...
1138  */
1139  
1140 static int view_borderzoom_exec(bContext *C, wmOperator *op)
1141 {
1142         ARegion *ar = CTX_wm_region(C);
1143         View2D *v2d = &ar->v2d;
1144         rctf rect;
1145         rctf cur_new = v2d->cur;
1146         int gesture_mode;
1147         const int smooth_viewtx = WM_operator_smooth_viewtx_get(op);
1148         
1149         /* convert coordinates of rect to 'tot' rect coordinates */
1150         UI_view2d_region_to_view(v2d, RNA_int_get(op->ptr, "xmin"), RNA_int_get(op->ptr, "ymin"), &rect.xmin, &rect.ymin);
1151         UI_view2d_region_to_view(v2d, RNA_int_get(op->ptr, "xmax"), RNA_int_get(op->ptr, "ymax"), &rect.xmax, &rect.ymax);
1152         
1153         /* check if zooming in/out view */
1154         gesture_mode = RNA_int_get(op->ptr, "gesture_mode");
1155         
1156         if (gesture_mode == GESTURE_MODAL_IN) {
1157                 /* zoom in: 
1158                  *      - 'cur' rect will be defined by the coordinates of the border region 
1159                  *      - just set the 'cur' rect to have the same coordinates as the border region
1160                  *        if zoom is allowed to be changed
1161                  */
1162                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
1163                         cur_new.xmin = rect.xmin;
1164                         cur_new.xmax = rect.xmax;
1165                 }
1166                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
1167                         cur_new.ymin = rect.ymin;
1168                         cur_new.ymax = rect.ymax;
1169                 }
1170         }
1171         else { /* if (gesture_mode == GESTURE_MODAL_OUT) */
1172
1173                 /* zoom out:
1174                  *      - the current 'cur' rect coordinates are going to end up where the 'rect' ones are,
1175                  *        but the 'cur' rect coordinates will need to be adjusted to take in more of the view
1176                  *      - calculate zoom factor, and adjust using center-point
1177                  */
1178                 float zoom, center, size;
1179                 
1180                 /* TODO: is this zoom factor calculation valid? It seems to produce same results every time... */
1181                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
1182                         size = BLI_rctf_size_x(&cur_new);
1183                         zoom = size / BLI_rctf_size_x(&rect);
1184                         center = BLI_rctf_cent_x(&cur_new);
1185                         
1186                         cur_new.xmin = center - (size * zoom);
1187                         cur_new.xmax = center + (size * zoom);
1188                 }
1189                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
1190                         size = BLI_rctf_size_y(&cur_new);
1191                         zoom = size / BLI_rctf_size_y(&rect);
1192                         center = BLI_rctf_cent_y(&cur_new);
1193                         
1194                         cur_new.ymin = center - (size * zoom);
1195                         cur_new.ymax = center + (size * zoom);
1196                 }
1197         }
1198         
1199         UI_view2d_smooth_view(C, ar, &cur_new, smooth_viewtx);
1200         
1201         return OPERATOR_FINISHED;
1202
1203
1204 static void VIEW2D_OT_zoom_border(wmOperatorType *ot)
1205 {
1206         /* identifiers */
1207         ot->name = "Zoom to Border";
1208         ot->description = "Zoom in the view to the nearest item contained in the border";
1209         ot->idname = "VIEW2D_OT_zoom_border";
1210         
1211         /* api callbacks */
1212         ot->invoke = WM_border_select_invoke;
1213         ot->exec = view_borderzoom_exec;
1214         ot->modal = WM_border_select_modal;
1215         ot->cancel = WM_border_select_cancel;
1216         
1217         ot->poll = view_zoom_poll;
1218         
1219         /* rna */
1220         WM_operator_properties_gesture_border(ot, false);
1221 }
1222
1223 /* ********************************************************* */
1224 /* SMOOTH VIEW */
1225
1226 struct SmoothView2DStore {
1227         rctf orig_cur, new_cur;
1228
1229         double time_allowed;
1230 };
1231
1232 /**
1233  * function to get a factor out of a rectangle
1234  *
1235  * note: this doesn't always work as well as it might because the target size
1236  *       may not be reached because of clamping the desired rect, we _could_
1237  *       attempt to clamp the rect before working out the zoom factor but its
1238  *       not really worthwhile for the few cases this happens.
1239  */
1240 static float smooth_view_rect_to_fac(const rctf *rect_a, const rctf *rect_b)
1241 {
1242         float size_a[2] = {BLI_rctf_size_x(rect_a),
1243                            BLI_rctf_size_y(rect_a)};
1244         float size_b[2] = {BLI_rctf_size_x(rect_b),
1245                            BLI_rctf_size_y(rect_b)};
1246         float cent_a[2] = {BLI_rctf_cent_x(rect_a),
1247                            BLI_rctf_cent_y(rect_a)};
1248         float cent_b[2] = {BLI_rctf_cent_x(rect_b),
1249                            BLI_rctf_cent_y(rect_b)};
1250
1251         float fac_max = 0.0f;
1252         float tfac;
1253
1254         int i;
1255
1256         for (i = 0; i < 2; i++) {
1257                 /* axis translation normalized to scale */
1258                 tfac = fabsf(cent_a[i] - cent_b[i]) / min_ff(size_a[i], size_b[i]);
1259                 fac_max = max_ff(fac_max, tfac);
1260                 if (fac_max >= 1.0f) break;
1261
1262                 /* axis scale difference, x2 so doubling or half gives 1.0f */
1263                 tfac = (1.0f - (min_ff(size_a[i], size_b[i]) / max_ff(size_a[i], size_b[i]))) * 2.0f;
1264                 fac_max = max_ff(fac_max, tfac);
1265                 if (fac_max >= 1.0f) break;
1266         }
1267         return min_ff(fac_max, 1.0f);
1268 }
1269
1270 /* will start timer if appropriate */
1271 /* the arguments are the desired situation */
1272 void UI_view2d_smooth_view(bContext *C, ARegion *ar,
1273                            const rctf *cur, const int smooth_viewtx)
1274 {
1275         wmWindowManager *wm = CTX_wm_manager(C);
1276         wmWindow *win = CTX_wm_window(C);
1277
1278         View2D *v2d = &ar->v2d;
1279         struct SmoothView2DStore sms = {{0}};
1280         bool ok = false;
1281         float fac = 1.0f;
1282
1283         /* initialize sms */
1284         sms.new_cur = v2d->cur;
1285
1286         /* store the options we want to end with */
1287         if (cur) sms.new_cur = *cur;
1288
1289         if (cur) {
1290                 fac = smooth_view_rect_to_fac(&v2d->cur, cur);
1291         }
1292
1293         if (smooth_viewtx && fac > FLT_EPSILON) {
1294                 bool changed = false;
1295
1296                 if (BLI_rctf_compare(&sms.new_cur, &v2d->cur, FLT_EPSILON) == false)
1297                         changed = true;
1298
1299                 /* The new view is different from the old one
1300                  * so animate the view */
1301                 if (changed) {
1302                         sms.orig_cur = v2d->cur;
1303
1304                         sms.time_allowed = (double)smooth_viewtx / 1000.0;
1305
1306                         /* scale the time allowed the change in view */
1307                         sms.time_allowed *= (double)fac;
1308
1309                         /* keep track of running timer! */
1310                         if (v2d->sms == NULL)
1311                                 v2d->sms = MEM_mallocN(sizeof(struct SmoothView2DStore), "smoothview v2d");
1312                         *v2d->sms = sms;
1313                         if (v2d->smooth_timer)
1314                                 WM_event_remove_timer(wm, win, v2d->smooth_timer);
1315                         /* TIMER1 is hardcoded in keymap */
1316                         v2d->smooth_timer = WM_event_add_timer(wm, win, TIMER1, 1.0 / 100.0); /* max 30 frs/sec */
1317
1318                         ok = true;
1319                 }
1320         }
1321
1322         /* if we get here nothing happens */
1323         if (ok == false) {
1324                 v2d->cur = sms.new_cur;
1325
1326                 UI_view2d_curRect_validate(v2d);
1327                 ED_region_tag_redraw(ar);
1328                 UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1329         }
1330 }
1331
1332 /* only meant for timer usage */
1333 static int view2d_smoothview_invoke(bContext *C, wmOperator *UNUSED(op), const wmEvent *event)
1334 {
1335         ARegion *ar = CTX_wm_region(C);
1336         View2D *v2d = &ar->v2d;
1337         struct SmoothView2DStore *sms = v2d->sms;
1338         float step;
1339
1340         /* escape if not our timer */
1341         if (v2d->smooth_timer == NULL || v2d->smooth_timer != event->customdata)
1342                 return OPERATOR_PASS_THROUGH;
1343
1344         if (sms->time_allowed != 0.0)
1345                 step = (float)((v2d->smooth_timer->duration) / sms->time_allowed);
1346         else
1347                 step = 1.0f;
1348
1349         /* end timer */
1350         if (step >= 1.0f) {
1351                 v2d->cur = sms->new_cur;
1352
1353                 MEM_freeN(v2d->sms);
1354                 v2d->sms = NULL;
1355
1356                 WM_event_remove_timer(CTX_wm_manager(C), CTX_wm_window(C), v2d->smooth_timer);
1357                 v2d->smooth_timer = NULL;
1358         }
1359         else {
1360                 /* ease in/out */
1361                 step = (3.0f * step * step - 2.0f * step * step * step);
1362
1363                 BLI_rctf_interp(&v2d->cur, &sms->orig_cur, &sms->new_cur, step);
1364         }
1365
1366         UI_view2d_curRect_validate(v2d);
1367         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1368         ED_region_tag_redraw(ar);
1369
1370         if (v2d->sms == NULL) {
1371                 UI_view2d_zoom_cache_reset();
1372         }
1373
1374         return OPERATOR_FINISHED;
1375 }
1376
1377 static void VIEW2D_OT_smoothview(wmOperatorType *ot)
1378 {
1379         /* identifiers */
1380         ot->name = "Smooth View 2D";
1381         ot->description = "";
1382         ot->idname = "VIEW2D_OT_smoothview";
1383
1384         /* api callbacks */
1385         ot->invoke = view2d_smoothview_invoke;
1386         ot->poll = view2d_poll;
1387
1388         /* flags */
1389         ot->flag = OPTYPE_INTERNAL;
1390
1391         /* rna */
1392         WM_operator_properties_gesture_border(ot, false);
1393 }
1394
1395 /* ********************************************************* */
1396 /* SCROLLERS */
1397
1398 /*  Scrollers should behave in the following ways, when clicked on with LMB (and dragged):
1399  *              1) 'Handles' on end of 'bubble' - when the axis that the scroller represents is zoomable, 
1400  *                      enlarge 'cur' rect on the relevant side 
1401  *              2) 'Bubble'/'bar' - just drag, and bar should move with mouse (view pans opposite)
1402  *
1403  *      In order to make sure this works, each operator must define the following RNA-Operator Props:
1404  *              deltax, deltay  - define how much to move view by (relative to zoom-correction factor)
1405  */
1406
1407 /* customdata for scroller-invoke data */
1408 typedef struct v2dScrollerMove {
1409         View2D *v2d;            /* View2D data that this operation affects */
1410         ARegion *ar;            /* region that the scroller is in */
1411         
1412         short scroller;         /* scroller that mouse is in ('h' or 'v') */
1413         short zone; /* -1 is min zoomer, 0 is bar, 1 is max zoomer */             // XXX find some way to provide visual feedback of this (active color?)
1414         
1415         float fac;              /* view adjustment factor, based on size of region */
1416         float delta;            /* amount moved by mouse on axis of interest */
1417         
1418         float scrollbarwidth;   /* width of the scrollbar itself, used for page up/down clicks */
1419         int scrollbar_orig;      /* initial location of scrollbar x/y, mouse relative */
1420         
1421         int lastx, lasty;       /* previous mouse coordinates (in screen coordinates) for determining movement */
1422 } v2dScrollerMove;
1423
1424
1425 /* View2DScrollers is typedef'd in UI_view2d.h 
1426  * This is a CUT DOWN VERSION of the 'real' version, which is defined in view2d.c, as we only need focus bubble info
1427  * WARNING: the start of this struct must not change, so that it stays in sync with the 'real' version
1428  *         For now, we don't need to have a separate (internal) header for structs like this...
1429  */
1430 struct View2DScrollers {        
1431         /* focus bubbles */
1432         int vert_min, vert_max; /* vertical scrollbar */
1433         int hor_min, hor_max;   /* horizontal scrollbar */
1434 };
1435
1436 /* quick enum for vsm->zone (scroller handles) */
1437 enum {
1438         SCROLLHANDLE_MIN = -1,
1439         SCROLLHANDLE_BAR,
1440         SCROLLHANDLE_MAX,
1441         SCROLLHANDLE_MIN_OUTSIDE,
1442         SCROLLHANDLE_MAX_OUTSIDE
1443 } /*eV2DScrollerHandle_Zone*/;
1444
1445 /* ------------------------ */
1446
1447 /* check if mouse is within scroller handle 
1448  *      - mouse                 =   relevant mouse coordinate in region space
1449  *      - sc_min, sc_max        =   extents of scroller 'groove' (potential available space for scroller)
1450  *      - sh_min, sh_max        =   positions of scrollbar handles
1451  */
1452 static short mouse_in_scroller_handle(int mouse, int sc_min, int sc_max, int sh_min, int sh_max)
1453 {
1454         short in_min, in_max, in_bar, out_min, out_max, in_view = 1;
1455         
1456         /* firstly, check if 
1457          *      - 'bubble' fills entire scroller 
1458          *      - 'bubble' completely out of view on either side 
1459          */
1460         if ((sh_min <= sc_min) && (sh_max >= sc_max)) in_view = 0;
1461         if (sh_min == sh_max) {
1462                 if (sh_min <= sc_min) in_view = 0;
1463                 if (sh_max >= sc_max) in_view = 0;
1464         }
1465         else {
1466                 if (sh_max <= sc_min) in_view = 0;
1467                 if (sh_min >= sc_max) in_view = 0;
1468         }
1469         
1470         
1471         if (in_view == 0) {
1472                 return SCROLLHANDLE_BAR;
1473         }
1474         
1475         /* check if mouse is in or past either handle */
1476         /* TODO: check if these extents are still valid or not */
1477         in_max = ( (mouse >= (sh_max - V2D_SCROLLER_HANDLE_SIZE)) && (mouse <= (sh_max + V2D_SCROLLER_HANDLE_SIZE)) );
1478         in_min = ( (mouse <= (sh_min + V2D_SCROLLER_HANDLE_SIZE)) && (mouse >= (sh_min - V2D_SCROLLER_HANDLE_SIZE)) );
1479         in_bar = ( (mouse < (sh_max - V2D_SCROLLER_HANDLE_SIZE)) && (mouse > (sh_min + V2D_SCROLLER_HANDLE_SIZE)) );
1480         out_min = mouse < (sh_min - V2D_SCROLLER_HANDLE_SIZE);
1481         out_max = mouse > (sh_max + V2D_SCROLLER_HANDLE_SIZE);
1482         
1483         if (in_bar)
1484                 return SCROLLHANDLE_BAR;
1485         else if (in_max)
1486                 return SCROLLHANDLE_MAX;
1487         else if (in_min)
1488                 return SCROLLHANDLE_MIN;
1489         else if (out_min)
1490                 return SCROLLHANDLE_MIN_OUTSIDE;
1491         else if (out_max)
1492                 return SCROLLHANDLE_MAX_OUTSIDE;
1493         
1494         /* unlikely to happen, though we just cover it in case */
1495         return SCROLLHANDLE_BAR;
1496
1497
1498 /* initialize customdata for scroller manipulation operator */
1499 static void scroller_activate_init(bContext *C, wmOperator *op, const wmEvent *event, short in_scroller)
1500 {
1501         v2dScrollerMove *vsm;
1502         View2DScrollers *scrollers;
1503         ARegion *ar = CTX_wm_region(C);
1504         View2D *v2d = &ar->v2d;
1505         rctf tot_cur_union;
1506         float mask_size;
1507         
1508         /* set custom-data for operator */
1509         vsm = MEM_callocN(sizeof(v2dScrollerMove), "v2dScrollerMove");
1510         op->customdata = vsm;
1511         
1512         /* set general data */
1513         vsm->v2d = v2d;
1514         vsm->ar = ar;
1515         vsm->scroller = in_scroller;
1516
1517         /* store mouse-coordinates, and convert mouse/screen coordinates to region coordinates */
1518         vsm->lastx = event->x;
1519         vsm->lasty = event->y;
1520         /* 'zone' depends on where mouse is relative to bubble 
1521          *      - zooming must be allowed on this axis, otherwise, default to pan
1522          */
1523         scrollers = UI_view2d_scrollers_calc(C, v2d, V2D_ARG_DUMMY, V2D_ARG_DUMMY, V2D_ARG_DUMMY, V2D_ARG_DUMMY);
1524
1525         /* use a union of 'cur' & 'tot' incase the current view is far outside 'tot'.
1526          * In this cases moving the scroll bars has far too little effect and the view can get stuck [#31476] */
1527         tot_cur_union = v2d->tot;
1528         BLI_rctf_union(&tot_cur_union, &v2d->cur);
1529
1530         if (in_scroller == 'h') {
1531                 /* horizontal scroller - calculate adjustment factor first */
1532                 mask_size = (float)BLI_rcti_size_x(&v2d->hor);
1533                 vsm->fac = BLI_rctf_size_x(&tot_cur_union) / mask_size;
1534                 
1535                 /* get 'zone' (i.e. which part of scroller is activated) */
1536                 vsm->zone = mouse_in_scroller_handle(event->mval[0],
1537                                                      v2d->hor.xmin, v2d->hor.xmax,
1538                                                      scrollers->hor_min, scrollers->hor_max);
1539                 
1540                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1541                         /* default to scroll, as handles not usable */
1542                         vsm->zone = SCROLLHANDLE_BAR;
1543                 }
1544
1545                 vsm->scrollbarwidth = scrollers->hor_max - scrollers->hor_min;
1546                 vsm->scrollbar_orig = ((scrollers->hor_max + scrollers->hor_min) / 2) + ar->winrct.xmin;
1547         }
1548         else {
1549                 /* vertical scroller - calculate adjustment factor first */
1550                 mask_size = (float)BLI_rcti_size_y(&v2d->vert);
1551                 vsm->fac = BLI_rctf_size_y(&tot_cur_union) / mask_size;
1552                 
1553                 /* get 'zone' (i.e. which part of scroller is activated) */
1554                 vsm->zone = mouse_in_scroller_handle(event->mval[1],
1555                                                      v2d->vert.ymin, v2d->vert.ymax,
1556                                                      scrollers->vert_min, scrollers->vert_max);
1557                         
1558                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1559                         /* default to scroll, as handles not usable */
1560                         vsm->zone = SCROLLHANDLE_BAR;
1561                 }
1562                 
1563                 vsm->scrollbarwidth = scrollers->vert_max - scrollers->vert_min;
1564                 vsm->scrollbar_orig = ((scrollers->vert_max + scrollers->vert_min) / 2) + ar->winrct.ymin;
1565         }
1566         
1567         UI_view2d_scrollers_free(scrollers);
1568         ED_region_tag_redraw(ar);
1569 }
1570
1571 /* cleanup temp customdata  */
1572 static void scroller_activate_exit(bContext *C, wmOperator *op)
1573 {
1574         if (op->customdata) {
1575                 v2dScrollerMove *vsm = op->customdata;
1576
1577                 vsm->v2d->scroll_ui &= ~(V2D_SCROLL_H_ACTIVE | V2D_SCROLL_V_ACTIVE);
1578                 
1579                 MEM_freeN(op->customdata);
1580                 op->customdata = NULL;
1581                 
1582                 ED_region_tag_redraw(CTX_wm_region(C));
1583         }
1584 }
1585
1586 static void scroller_activate_cancel(bContext *C, wmOperator *op)
1587 {
1588         scroller_activate_exit(C, op);
1589 }
1590
1591 /* apply transform to view (i.e. adjust 'cur' rect) */
1592 static void scroller_activate_apply(bContext *C, wmOperator *op)
1593 {
1594         v2dScrollerMove *vsm = op->customdata;
1595         View2D *v2d = vsm->v2d;
1596         float temp;
1597         
1598         /* calculate amount to move view by */
1599         temp = vsm->fac * vsm->delta;
1600         
1601         /* type of movement */
1602         switch (vsm->zone) {
1603                 case SCROLLHANDLE_MIN:
1604                         /* only expand view on axis if zoom is allowed */
1605                         if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X))
1606                                 v2d->cur.xmin -= temp;
1607                         if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y))
1608                                 v2d->cur.ymin -= temp;
1609                         break;
1610                         
1611                 case SCROLLHANDLE_MAX:
1612                         
1613                         /* only expand view on axis if zoom is allowed */
1614                         if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X))
1615                                 v2d->cur.xmax += temp;
1616                         if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y))
1617                                 v2d->cur.ymax += temp;
1618                         break;
1619                         
1620                 case SCROLLHANDLE_MIN_OUTSIDE:
1621                 case SCROLLHANDLE_MAX_OUTSIDE:
1622                 case SCROLLHANDLE_BAR:
1623                 default:
1624                         /* only move view on an axis if panning is allowed */
1625                         if ((vsm->scroller == 'h') && !(v2d->keepofs & V2D_LOCKOFS_X)) {
1626                                 v2d->cur.xmin += temp;
1627                                 v2d->cur.xmax += temp;
1628                         }
1629                         if ((vsm->scroller == 'v') && !(v2d->keepofs & V2D_LOCKOFS_Y)) {
1630                                 v2d->cur.ymin += temp;
1631                                 v2d->cur.ymax += temp;
1632                         }
1633                         break;
1634                         
1635         }
1636         
1637         /* validate that view is in valid configuration after this operation */
1638         UI_view2d_curRect_validate(v2d);
1639         
1640         /* request updates to be done... */
1641         ED_region_tag_redraw(vsm->ar);
1642         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1643 }
1644
1645 /* handle user input for scrollers - calculations of mouse-movement need to be done here, not in the apply callback! */
1646 static int scroller_activate_modal(bContext *C, wmOperator *op, const wmEvent *event)
1647 {
1648         v2dScrollerMove *vsm = op->customdata;
1649         
1650         /* execute the events */
1651         switch (event->type) {
1652                 case MOUSEMOVE:
1653                 {
1654                         /* calculate new delta transform, then store mouse-coordinates for next-time */
1655                         if (ELEM(vsm->zone, SCROLLHANDLE_BAR, SCROLLHANDLE_MAX)) {
1656                                 /* if using bar (i.e. 'panning') or 'max' zoom widget */
1657                                 switch (vsm->scroller) {
1658                                         case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves opposite to mouse) */
1659                                                 vsm->delta = (float)(event->x - vsm->lastx);
1660                                                 break;
1661                                         case 'v': /* vertical scroller - so only vertical movement ('cur' moves opposite to mouse) */
1662                                                 vsm->delta = (float)(event->y - vsm->lasty);
1663                                                 break;
1664                                 }
1665                         }
1666                         else if (vsm->zone == SCROLLHANDLE_MIN) {
1667                                 /* using 'min' zoom widget */
1668                                 switch (vsm->scroller) {
1669                                         case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves with mouse) */
1670                                                 vsm->delta = (float)(vsm->lastx - event->x);
1671                                                 break;
1672                                         case 'v': /* vertical scroller - so only vertical movement ('cur' moves with to mouse) */
1673                                                 vsm->delta = (float)(vsm->lasty - event->y);
1674                                                 break;
1675                                 }
1676                         }
1677                         
1678                         /* store previous coordinates */
1679                         vsm->lastx = event->x;
1680                         vsm->lasty = event->y;
1681                         
1682                         scroller_activate_apply(C, op);
1683                         break;
1684                 }
1685                 case LEFTMOUSE:
1686                 case MIDDLEMOUSE:
1687                         if (event->val == KM_RELEASE) {
1688                                 /* single-click was in empty space outside bubble, so scroll by 1 'page' */
1689                                 if (ELEM(vsm->zone, SCROLLHANDLE_MIN_OUTSIDE, SCROLLHANDLE_MAX_OUTSIDE)) {
1690                                         if (vsm->zone == SCROLLHANDLE_MIN_OUTSIDE)
1691                                                 vsm->delta = -vsm->scrollbarwidth * 0.8f;
1692                                         else if (vsm->zone == SCROLLHANDLE_MAX_OUTSIDE)
1693                                                 vsm->delta = vsm->scrollbarwidth * 0.8f;
1694                                         
1695                                         scroller_activate_apply(C, op);
1696                                         scroller_activate_exit(C, op);
1697                                         return OPERATOR_FINISHED;
1698                                 }
1699                                 
1700                                 /* otherwise, end the drag action  */
1701                                 if (vsm->lastx || vsm->lasty) {
1702                                         scroller_activate_exit(C, op);
1703                                         return OPERATOR_FINISHED;
1704                                 }
1705                         }
1706                         break;
1707
1708         }
1709
1710         return OPERATOR_RUNNING_MODAL;
1711 }
1712
1713
1714 /* a click (or click drag in progress) should have occurred, so check if it happened in scrollbar */
1715 static int scroller_activate_invoke(bContext *C, wmOperator *op, const wmEvent *event)
1716 {
1717         ARegion *ar = CTX_wm_region(C);
1718         View2D *v2d = &ar->v2d;
1719         short in_scroller = 0;
1720                 
1721         /* check if mouse in scrollbars, if they're enabled */
1722         in_scroller = UI_view2d_mouse_in_scrollers(C, v2d, event->x, event->y);
1723         
1724         /* if in a scroller, init customdata then set modal handler which will catch mousedown to start doing useful stuff */
1725         if (in_scroller) {
1726                 v2dScrollerMove *vsm;
1727                 
1728                 /* initialize customdata */
1729                 scroller_activate_init(C, op, event, in_scroller);
1730                 vsm = (v2dScrollerMove *)op->customdata;
1731                 
1732                 /* support for quick jump to location - gtk and qt do this on linux */
1733                 if (event->type == MIDDLEMOUSE) {
1734                         switch (vsm->scroller) {
1735                                 case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves opposite to mouse) */
1736                                         vsm->delta = (float)(event->x - vsm->scrollbar_orig);
1737                                         break;
1738                                 case 'v': /* vertical scroller - so only vertical movement ('cur' moves opposite to mouse) */
1739                                         vsm->delta = (float)(event->y - vsm->scrollbar_orig);
1740                                         break;
1741                         }
1742                         scroller_activate_apply(C, op);
1743
1744                         vsm->zone = SCROLLHANDLE_BAR;
1745                 }
1746
1747                 /* check if zoom zones are inappropriate (i.e. zoom widgets not shown), so cannot continue
1748                  * NOTE: see view2d.c for latest conditions, and keep this in sync with that
1749                  */
1750                 if (ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1751                         if (((vsm->scroller == 'h') && (v2d->scroll & V2D_SCROLL_SCALE_HORIZONTAL) == 0) ||
1752                             ((vsm->scroller == 'v') && (v2d->scroll & V2D_SCROLL_SCALE_VERTICAL) == 0))
1753                         {
1754                                 /* switch to bar (i.e. no scaling gets handled) */
1755                                 vsm->zone = SCROLLHANDLE_BAR;
1756                         }
1757                 }
1758                 
1759                 /* check if zone is inappropriate (i.e. 'bar' but panning is banned), so cannot continue */
1760                 if (vsm->zone == SCROLLHANDLE_BAR) {
1761                         if (((vsm->scroller == 'h') && (v2d->keepofs & V2D_LOCKOFS_X)) ||
1762                             ((vsm->scroller == 'v') && (v2d->keepofs & V2D_LOCKOFS_Y)))
1763                         {
1764                                 /* free customdata initialized */
1765                                 scroller_activate_exit(C, op);
1766                                 
1767                                 /* can't catch this event for ourselves, so let it go to someone else? */
1768                                 return OPERATOR_PASS_THROUGH;
1769                         }
1770                 }
1771                 
1772                 /* zone is also inappropriate if scroller is not visible... */
1773                 if (((vsm->scroller == 'h') && (v2d->scroll & (V2D_SCROLL_HORIZONTAL_FULLR))) ||
1774                     ((vsm->scroller == 'v') && (v2d->scroll & (V2D_SCROLL_VERTICAL_FULLR))) )
1775                 {
1776                         /* free customdata initialized */
1777                         scroller_activate_exit(C, op);
1778                                 
1779                         /* can't catch this event for ourselves, so let it go to someone else? */
1780                         /* XXX note: if handlers use mask rect to clip input, input will fail for this case */
1781                         return OPERATOR_PASS_THROUGH;
1782                 }
1783                 
1784                 /* activate the scroller */
1785                 if (vsm->scroller == 'h')
1786                         v2d->scroll_ui |= V2D_SCROLL_H_ACTIVE;
1787                 else
1788                         v2d->scroll_ui |= V2D_SCROLL_V_ACTIVE;
1789                 
1790                 /* still ok, so can add */
1791                 WM_event_add_modal_handler(C, op);
1792                 return OPERATOR_RUNNING_MODAL;
1793         }
1794         else {
1795                 /* not in scroller, so nothing happened... (pass through let's something else catch event) */
1796                 return OPERATOR_PASS_THROUGH;
1797         }
1798 }
1799
1800 /* LMB-Drag in Scrollers - not repeatable operator! */
1801 static void VIEW2D_OT_scroller_activate(wmOperatorType *ot)
1802 {
1803         /* identifiers */
1804         ot->name = "Scroller Activate";
1805         ot->description = "Scroll view by mouse click and drag";
1806         ot->idname = "VIEW2D_OT_scroller_activate";
1807
1808         /* flags */
1809         ot->flag = OPTYPE_BLOCKING;
1810         
1811         /* api callbacks */
1812         ot->invoke = scroller_activate_invoke;
1813         ot->modal = scroller_activate_modal;
1814         ot->cancel = scroller_activate_cancel;
1815
1816         ot->poll = view2d_poll;
1817 }
1818
1819 /* ********************************************************* */
1820 /* RESET */
1821
1822 static int reset_exec(bContext *C, wmOperator *UNUSED(op))
1823 {
1824         uiStyle *style = UI_GetStyle();
1825         ARegion *ar = CTX_wm_region(C);
1826         View2D *v2d = &ar->v2d;
1827         int winx, winy;
1828
1829         /* zoom 1.0 */
1830         winx = (float)(BLI_rcti_size_x(&v2d->mask) + 1);
1831         winy = (float)(BLI_rcti_size_y(&v2d->mask) + 1);
1832
1833         v2d->cur.xmax = v2d->cur.xmin + winx;
1834         v2d->cur.ymax = v2d->cur.ymin + winy;
1835         
1836         /* align */
1837         if (v2d->align) {
1838                 /* posx and negx flags are mutually exclusive, so watch out */
1839                 if ((v2d->align & V2D_ALIGN_NO_POS_X) && !(v2d->align & V2D_ALIGN_NO_NEG_X)) {
1840                         v2d->cur.xmax = 0.0f;
1841                         v2d->cur.xmin = -winx * style->panelzoom;
1842                 }
1843                 else if ((v2d->align & V2D_ALIGN_NO_NEG_X) && !(v2d->align & V2D_ALIGN_NO_POS_X)) {
1844                         v2d->cur.xmax = winx * style->panelzoom;
1845                         v2d->cur.xmin = 0.0f;
1846                 }
1847
1848                 /* - posx and negx flags are mutually exclusive, so watch out */
1849                 if ((v2d->align & V2D_ALIGN_NO_POS_Y) && !(v2d->align & V2D_ALIGN_NO_NEG_Y)) {
1850                         v2d->cur.ymax = 0.0f;
1851                         v2d->cur.ymin = -winy * style->panelzoom;
1852                 }
1853                 else if ((v2d->align & V2D_ALIGN_NO_NEG_Y) && !(v2d->align & V2D_ALIGN_NO_POS_Y)) {
1854                         v2d->cur.ymax = winy * style->panelzoom;
1855                         v2d->cur.ymin = 0.0f;
1856                 }
1857         }
1858
1859         /* validate that view is in valid configuration after this operation */
1860         UI_view2d_curRect_validate(v2d);
1861         
1862         /* request updates to be done... */
1863         ED_region_tag_redraw(ar);
1864         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1865         
1866         UI_view2d_zoom_cache_reset();
1867
1868         return OPERATOR_FINISHED;
1869 }
1870
1871 static void VIEW2D_OT_reset(wmOperatorType *ot)
1872 {
1873         /* identifiers */
1874         ot->name = "Reset View";
1875         ot->description = "Reset the view";
1876         ot->idname = "VIEW2D_OT_reset";
1877         
1878         /* api callbacks */
1879         ot->exec = reset_exec;
1880         ot->poll = view2d_poll;
1881 }
1882  
1883 /* ********************************************************* */
1884 /* Registration */
1885
1886 void UI_view2d_operatortypes(void)
1887 {
1888         WM_operatortype_append(VIEW2D_OT_pan);
1889         
1890         WM_operatortype_append(VIEW2D_OT_scroll_left);
1891         WM_operatortype_append(VIEW2D_OT_scroll_right);
1892         WM_operatortype_append(VIEW2D_OT_scroll_up);
1893         WM_operatortype_append(VIEW2D_OT_scroll_down);
1894         
1895         WM_operatortype_append(VIEW2D_OT_zoom_in);
1896         WM_operatortype_append(VIEW2D_OT_zoom_out);
1897         
1898         WM_operatortype_append(VIEW2D_OT_zoom);
1899         WM_operatortype_append(VIEW2D_OT_zoom_border);
1900
1901         WM_operatortype_append(VIEW2D_OT_smoothview);
1902         
1903         WM_operatortype_append(VIEW2D_OT_scroller_activate);
1904
1905         WM_operatortype_append(VIEW2D_OT_reset);
1906 }
1907
1908 void UI_view2d_keymap(wmKeyConfig *keyconf)
1909 {
1910         wmKeyMap *keymap = WM_keymap_find(keyconf, "View2D", 0, 0);
1911         wmKeyMapItem *kmi;
1912
1913         /* scrollers */
1914         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", LEFTMOUSE, KM_PRESS, 0, 0);
1915         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", MIDDLEMOUSE, KM_PRESS, 0, 0);
1916
1917         /* pan/scroll */
1918         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, 0, 0);
1919         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, KM_SHIFT, 0);
1920         
1921         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MOUSEPAN, 0, 0, 0);
1922         
1923         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", WHEELDOWNMOUSE, KM_PRESS, KM_CTRL, 0);
1924         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", WHEELUPMOUSE, KM_PRESS, KM_CTRL, 0);
1925         
1926         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, KM_SHIFT, 0);
1927         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, KM_SHIFT, 0);
1928         
1929         /* zoom - single step */
1930         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", WHEELOUTMOUSE, KM_PRESS, 0, 0);
1931         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", WHEELINMOUSE, KM_PRESS, 0, 0);
1932         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", PADMINUS, KM_PRESS, 0, 0);
1933         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", PADPLUSKEY, KM_PRESS, 0, 0);
1934         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEPAN, 0, KM_CTRL, 0);
1935         
1936         WM_keymap_verify_item(keymap, "VIEW2D_OT_smoothview", TIMER1, KM_ANY, KM_ANY, 0);
1937
1938         /* scroll up/down - no modifiers, only when zoom fails */
1939         /* these may fail if zoom is disallowed, in which case they should pass on event */
1940         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1941         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, 0, 0);
1942         /* these may be necessary if vertical scroll is disallowed */
1943         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1944         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", WHEELUPMOUSE, KM_PRESS, 0, 0);
1945         
1946         /* alternatives for page up/down to scroll */
1947 #if 0 // XXX disabled, since this causes conflicts with hotkeys in animation editors
1948         /* scroll up/down may fall through to left/right */
1949         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", PAGEDOWNKEY, KM_PRESS, 0, 0);
1950         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", PAGEUPKEY, KM_PRESS, 0, 0);
1951         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", PAGEDOWNKEY, KM_PRESS, 0, 0);
1952         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", PAGEUPKEY, KM_PRESS, 0, 0);
1953         /* shift for moving view left/right with page up/down */
1954         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", PAGEDOWNKEY, KM_PRESS, KM_SHIFT, 0);
1955         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", PAGEUPKEY, KM_PRESS, KM_SHIFT, 0);
1956 #endif
1957         
1958         /* zoom - drag */
1959         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MIDDLEMOUSE, KM_PRESS, KM_CTRL, 0);
1960         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEZOOM, 0, 0, 0);
1961         
1962         /* borderzoom - drag */
1963         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_border", BKEY, KM_PRESS, KM_SHIFT, 0);
1964
1965         /* Alternative keymap for buttons listview */
1966         keymap = WM_keymap_find(keyconf, "View2D Buttons List", 0, 0);
1967
1968         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", LEFTMOUSE, KM_PRESS, 0, 0);
1969         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", MIDDLEMOUSE, KM_PRESS, 0, 0);
1970
1971         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, 0, 0);
1972         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MOUSEPAN, 0, 0, 0);
1973         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1974         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, 0, 0);
1975         
1976         kmi = WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", PAGEDOWNKEY, KM_PRESS, 0, 0);
1977         RNA_boolean_set(kmi->ptr, "page", true);
1978         kmi = WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", PAGEUPKEY, KM_PRESS, 0, 0);
1979         RNA_boolean_set(kmi->ptr, "page", true);
1980         
1981         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MIDDLEMOUSE, KM_PRESS, KM_CTRL, 0);
1982         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEZOOM, 0, 0, 0);
1983         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEPAN, 0, KM_CTRL, 0);
1984         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", PADMINUS, KM_PRESS, 0, 0);
1985         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", PADPLUSKEY, KM_PRESS, 0, 0);
1986         WM_keymap_add_item(keymap, "VIEW2D_OT_reset", HOMEKEY, KM_PRESS, 0, 0);
1987 }
1988