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