svn merge ^/trunk/blender -r49082:49104
[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
40 #include "BKE_context.h"
41
42 #include "RNA_access.h"
43 #include "RNA_define.h"
44
45 #include "WM_api.h"
46 #include "WM_types.h"
47
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)(ar->winrct.xmax - ar->winrct.xmin + 1);
122         winy = (float)(ar->winrct.ymax - ar->winrct.ymin + 1);
123         vpd->facx = (v2d->cur.xmax - v2d->cur.xmin) / winx;
124         vpd->facy = (v2d->cur.ymax - v2d->cur.ymin) / winy;
125         
126         return 1;
127 }
128
129 /* apply transform to view (i.e. adjust 'cur' rect) */
130 static void view_pan_apply(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         
156         UI_view2d_sync(vpd->sc, vpd->sa, v2d, V2D_LOCK_COPY);
157         
158         /* exceptions */
159         if (vpd->sa->spacetype == SPACE_OUTLINER) {
160                 /* don't rebuild full tree, since we're just changing our view */
161                 SpaceOops *soops = vpd->sa->spacedata.first;
162                 soops->storeflag |= SO_TREESTORE_REDRAW;
163         }
164 }
165
166 /* cleanup temp customdata  */
167 static void view_pan_exit(wmOperator *op)
168 {
169         if (op->customdata) {
170                 MEM_freeN(op->customdata);
171                 op->customdata = NULL;
172         }
173
174  
175 /* ------------------ Modal Drag Version (1) ---------------------- */
176
177 /* for 'redo' only, with no user input */
178 static int view_pan_exec(bContext *C, wmOperator *op)
179 {
180         if (!view_pan_init(C, op))
181                 return OPERATOR_CANCELLED;
182         
183         view_pan_apply(op);
184         view_pan_exit(op);
185         return OPERATOR_FINISHED;
186 }
187
188 /* set up modal operator and relevant settings */
189 static int view_pan_invoke(bContext *C, wmOperator *op, wmEvent *event)
190 {
191         wmWindow *window = CTX_wm_window(C);
192         v2dViewPanData *vpd;
193         View2D *v2d;
194         
195         /* set up customdata */
196         if (!view_pan_init(C, op))
197                 return OPERATOR_PASS_THROUGH;
198         
199         vpd = op->customdata;
200         v2d = vpd->v2d;
201         
202         /* set initial settings */
203         vpd->startx = vpd->lastx = event->x;
204         vpd->starty = vpd->lasty = event->y;
205         vpd->invoke_event = event->type;
206         
207         if (event->type == MOUSEPAN) {
208                 RNA_int_set(op->ptr, "deltax", event->prevx - event->x);
209                 RNA_int_set(op->ptr, "deltay", event->prevy - event->y);
210                 
211                 view_pan_apply(op);
212                 view_pan_exit(op);
213                 return OPERATOR_FINISHED;
214         }
215         
216         RNA_int_set(op->ptr, "deltax", 0);
217         RNA_int_set(op->ptr, "deltay", 0);
218         
219         if (v2d->keepofs & V2D_LOCKOFS_X)
220                 WM_cursor_modal(window, BC_NS_SCROLLCURSOR);
221         else if (v2d->keepofs & V2D_LOCKOFS_Y)
222                 WM_cursor_modal(window, BC_EW_SCROLLCURSOR);
223         else
224                 WM_cursor_modal(window, BC_NSEW_SCROLLCURSOR);
225         
226         /* add temp handler */
227         WM_event_add_modal_handler(C, op);
228
229         return OPERATOR_RUNNING_MODAL;
230 }
231
232 /* handle user input - calculations of mouse-movement need to be done here, not in the apply callback! */
233 static int view_pan_modal(bContext *C, wmOperator *op, wmEvent *event)
234 {
235         v2dViewPanData *vpd = op->customdata;
236         
237         /* execute the events */
238         switch (event->type) {
239                 case MOUSEMOVE:
240                 {
241                         /* calculate new delta transform, then store mouse-coordinates for next-time */
242                         RNA_int_set(op->ptr, "deltax", (vpd->lastx - event->x));
243                         RNA_int_set(op->ptr, "deltay", (vpd->lasty - event->y));
244                         
245                         vpd->lastx = event->x;
246                         vpd->lasty = event->y;
247                         
248                         view_pan_apply(op);
249                 }
250                 break;
251                         /* XXX - Mode switching isn't implemented. See comments in 36818.
252                          * switch to zoom */
253 #if 0
254                 case LEFTMOUSE:
255                         if (event->val == KM_PRESS) {
256                                 /* calculate overall delta mouse-movement for redo */
257                                 RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx));
258                                 RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty));
259                                 
260                                 view_pan_exit(op);
261                                 WM_cursor_restore(CTX_wm_window(C));
262                                 WM_operator_name_call(C, "VIEW2D_OT_zoom", WM_OP_INVOKE_DEFAULT, NULL);
263                                 return OPERATOR_FINISHED;
264                         }
265 #endif
266                 default:
267                         if (event->type == vpd->invoke_event || event->type == ESCKEY) {
268                                 if (event->val == KM_RELEASE) {
269                                         /* calculate overall delta mouse-movement for redo */
270                                         RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx));
271                                         RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty));
272                                         
273                                         view_pan_exit(op);
274                                         WM_cursor_restore(CTX_wm_window(C));
275                                         
276                                         return OPERATOR_FINISHED;
277                                 }
278                         }
279                         break;
280         }
281
282         return OPERATOR_RUNNING_MODAL;
283 }
284
285 static int view_pan_cancel(bContext *UNUSED(C), wmOperator *op)
286 {
287         view_pan_exit(op);
288         return OPERATOR_CANCELLED;
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 callsthe 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(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 callsthe 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(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 callsthe 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(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 callsthe 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", ar->v2d.mask.ymax - ar->v2d.mask.ymin);
474         }
475         
476         /* apply movement, then we're done */
477         view_pan_apply(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 = (v2d->cur.xmax - v2d->cur.xmin) * facx;
593                 dy = (v2d->cur.ymax - v2d->cur.ymin) * facy;
594         }
595         else {
596                 dx = ((v2d->cur.xmax - v2d->cur.xmin) / (1.0f + 2.0f * facx)) * facx;
597                 dy = ((v2d->cur.ymax - v2d->cur.ymin) / (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) / (v2d->cur.xmax - v2d->cur.xmin);
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) / (v2d->cur.ymax - v2d->cur.ymin);
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         if (op->customdata) {
666                 MEM_freeN(op->customdata);
667                 op->customdata = NULL;
668         }
669 }
670
671 /* this operator only needs this single callback, where it calls the view_zoom_*() methods */
672 static int view_zoomin_exec(bContext *C, wmOperator *op)
673 {
674         /* check that there's an active region, as View2D data resides there */
675         if (!view_zoom_poll(C))
676                 return OPERATOR_PASS_THROUGH;
677         
678         /* set RNA-Props - zooming in by uniform factor */
679         RNA_float_set(op->ptr, "zoomfacx", 0.0375f);
680         RNA_float_set(op->ptr, "zoomfacy", 0.0375f);
681         
682         /* apply movement, then we're done */
683         view_zoomstep_apply(C, op);
684         
685         view_zoomstep_exit(op);
686         
687         return OPERATOR_FINISHED;
688 }
689
690 static int view_zoomin_invoke(bContext *C, wmOperator *op, wmEvent *event)
691 {
692         v2dViewZoomData *vzd;
693         
694         if (!view_zoomdrag_init(C, op))
695                 return OPERATOR_PASS_THROUGH;
696         
697         vzd = op->customdata;
698         
699         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
700                 ARegion *ar = CTX_wm_region(C);
701                 
702                 /* store initial mouse position (in view space) */
703                 UI_view2d_region_to_view(&ar->v2d, 
704                                          event->mval[0], event->mval[1],
705                                          &vzd->mx_2d, &vzd->my_2d);
706         }
707         
708         return view_zoomin_exec(C, op);
709 }
710
711 static void VIEW2D_OT_zoom_in(wmOperatorType *ot)
712 {
713         /* identifiers */
714         ot->name = "Zoom In";
715         ot->description = "Zoom in the view";
716         ot->idname = "VIEW2D_OT_zoom_in";
717         
718         /* api callbacks */
719         ot->invoke = view_zoomin_invoke;
720 //      ot->exec = view_zoomin_exec;  // XXX, needs view_zoomdrag_init called first.
721         ot->poll = view_zoom_poll;
722         
723         /* rna - must keep these in sync with the other operators */
724         RNA_def_float(ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX);
725         RNA_def_float(ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX);
726 }
727         
728 /* this operator only needs this single callback, where it callsthe view_zoom_*() methods */
729 static int view_zoomout_exec(bContext *C, wmOperator *op)
730 {
731         /* check that there's an active region, as View2D data resides there */
732         if (!view_zoom_poll(C))
733                 return OPERATOR_PASS_THROUGH;
734         
735         /* set RNA-Props - zooming in by uniform factor */
736         RNA_float_set(op->ptr, "zoomfacx", -0.0375f);
737         RNA_float_set(op->ptr, "zoomfacy", -0.0375f);
738         
739         /* apply movement, then we're done */
740         view_zoomstep_apply(C, op);
741
742         view_zoomstep_exit(op);
743         
744         return OPERATOR_FINISHED;
745 }
746
747 static int view_zoomout_invoke(bContext *C, wmOperator *op, wmEvent *event)
748 {
749         v2dViewZoomData *vzd;
750         
751         if (!view_zoomdrag_init(C, op))
752                 return OPERATOR_PASS_THROUGH;
753
754         vzd = op->customdata;
755         
756         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
757                 ARegion *ar = CTX_wm_region(C);
758                 
759                 /* store initial mouse position (in view space) */
760                 UI_view2d_region_to_view(&ar->v2d, 
761                                          event->mval[0], event->mval[1],
762                                          &vzd->mx_2d, &vzd->my_2d);
763         }
764         
765         return view_zoomout_exec(C, op);
766 }
767
768 static void VIEW2D_OT_zoom_out(wmOperatorType *ot)
769 {
770         /* identifiers */
771         ot->name = "Zoom Out";
772         ot->description = "Zoom out the view";
773         ot->idname = "VIEW2D_OT_zoom_out";
774         
775         /* api callbacks */
776         ot->invoke = view_zoomout_invoke;
777 //      ot->exec = view_zoomout_exec; // XXX, needs view_zoomdrag_init called first.
778         ot->poll = view_zoom_poll;
779         
780         /* rna - must keep these in sync with the other operators */
781         RNA_def_float(ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX);
782         RNA_def_float(ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX);
783 }
784
785 /* ********************************************************* */
786 /* DRAG-ZOOM OPERATOR                                                                    */
787
788 /*  MMB Drag - allows non-uniform scaling by dragging mouse
789  *
790  *      In order to make sure this works, each operator must define the following RNA-Operator Props:
791  *              deltax, deltay  - amounts to add to each side of the 'cur' rect
792  */
793
794 /* apply transform to view (i.e. adjust 'cur' rect) */
795 static void view_zoomdrag_apply(bContext *C, wmOperator *op)
796 {
797         v2dViewZoomData *vzd = op->customdata;
798         View2D *v2d = vzd->v2d;
799         float dx, dy;
800         
801         /* get amount to move view by */
802         dx = RNA_float_get(op->ptr, "deltax");
803         dy = RNA_float_get(op->ptr, "deltay");
804
805         /* continuous zoom shouldn't move that fast... */
806         if (U.viewzoom == USER_ZOOM_CONT) { // XXX store this setting as RNA prop?
807                 double time = PIL_check_seconds_timer();
808                 float time_step = (float)(time - vzd->timer_lastdraw);
809
810                 dx *= time_step * 0.5f;
811                 dy *= time_step * 0.5f;
812                 
813                 vzd->timer_lastdraw = time;
814         }
815
816         /* only move view on an axis if change is allowed */
817         if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
818                 if (v2d->keepofs & V2D_LOCKOFS_X) {
819                         v2d->cur.xmax -= 2 * dx;
820                 }
821                 else {
822                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
823                                 float mval_fac = (vzd->mx_2d - v2d->cur.xmin) / (v2d->cur.xmax - v2d->cur.xmin);
824                                 float mval_faci = 1.0f - mval_fac;
825                                 float ofs = (mval_fac * dx) - (mval_faci * dx);
826                                 
827                                 v2d->cur.xmin += ofs + dx;
828                                 v2d->cur.xmax += ofs - dx;
829                         }
830                         else {
831                                 v2d->cur.xmin += dx;
832                                 v2d->cur.xmax -= dx;
833                         }
834                 }
835         }
836         if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
837                 if (v2d->keepofs & V2D_LOCKOFS_Y) {
838                         v2d->cur.ymax -= 2 * dy;
839                 }
840                 else {
841                         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
842                                 float mval_fac = (vzd->my_2d - v2d->cur.ymin) / (v2d->cur.ymax - v2d->cur.ymin);
843                                 float mval_faci = 1.0f - mval_fac;
844                                 float ofs = (mval_fac * dy) - (mval_faci * dy);
845                                 
846                                 v2d->cur.ymin += ofs + dy;
847                                 v2d->cur.ymax += ofs - dy;
848                         }
849                         else {
850                                 v2d->cur.ymin += dy;
851                                 v2d->cur.ymax -= dy;
852                         }
853                 }
854         }
855         
856         /* validate that view is in valid configuration after this operation */
857         UI_view2d_curRect_validate(v2d);
858         
859         /* request updates to be done... */
860         ED_region_tag_redraw(vzd->ar);
861         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
862 }
863
864 /* cleanup temp customdata  */
865 static void view_zoomdrag_exit(bContext *C, wmOperator *op)
866 {
867         if (op->customdata) {
868                 v2dViewZoomData *vzd = op->customdata;
869                 
870                 if (vzd->timer)
871                         WM_event_remove_timer(CTX_wm_manager(C), CTX_wm_window(C), vzd->timer);
872                 
873                 MEM_freeN(op->customdata);
874                 op->customdata = NULL;
875         }
876
877
878 static int view_zoomdrag_cancel(bContext *C, wmOperator *op)
879 {
880         view_zoomdrag_exit(C, op);
881
882         return OPERATOR_CANCELLED;
883 }
884
885 /* for 'redo' only, with no user input */
886 static int view_zoomdrag_exec(bContext *C, wmOperator *op)
887 {
888         if (!view_zoomdrag_init(C, op))
889                 return OPERATOR_PASS_THROUGH;
890         
891         view_zoomdrag_apply(C, op);
892         view_zoomdrag_exit(C, op);
893         return OPERATOR_FINISHED;
894 }
895
896 /* set up modal operator and relevant settings */
897 static int view_zoomdrag_invoke(bContext *C, wmOperator *op, wmEvent *event)
898 {
899         wmWindow *window = CTX_wm_window(C);
900         v2dViewZoomData *vzd;
901         View2D *v2d;
902         
903         /* set up customdata */
904         if (!view_zoomdrag_init(C, op))
905                 return OPERATOR_PASS_THROUGH;
906         
907         vzd = op->customdata;
908         v2d = vzd->v2d;
909         
910         if (event->type == MOUSEZOOM) {
911                 float dx, dy, fac;
912                 
913                 vzd->lastx = event->prevx;
914                 vzd->lasty = event->prevy;
915                 
916                 /* As we have only 1D information (magnify value), feed both axes
917                  * with magnify information that is stored in x axis 
918                  */
919                 fac = 0.01f * (event->x - event->prevx);
920                 dx = fac * (v2d->cur.xmax - v2d->cur.xmin) / 10.0f;
921                 dy = fac * (v2d->cur.ymax - v2d->cur.ymin) / 10.0f;
922
923                 RNA_float_set(op->ptr, "deltax", dx);
924                 RNA_float_set(op->ptr, "deltay", dy);
925                 
926                 view_zoomdrag_apply(C, op);
927                 view_zoomdrag_exit(C, op);
928                 return OPERATOR_FINISHED;
929         }       
930         
931         /* set initial settings */
932         vzd->lastx = event->x;
933         vzd->lasty = event->y;
934         RNA_float_set(op->ptr, "deltax", 0);
935         RNA_float_set(op->ptr, "deltay", 0);
936         
937         /* for modal exit test */
938         vzd->invoke_event = event->type;
939         
940         if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) {
941                 ARegion *ar = CTX_wm_region(C);
942                 
943                 /* store initial mouse position (in view space) */
944                 UI_view2d_region_to_view(&ar->v2d, 
945                                          event->mval[0], event->mval[1],
946                                          &vzd->mx_2d, &vzd->my_2d);
947         }
948
949         if (v2d->keepofs & V2D_LOCKOFS_X)
950                 WM_cursor_modal(window, BC_NS_SCROLLCURSOR);
951         else if (v2d->keepofs & V2D_LOCKOFS_Y)
952                 WM_cursor_modal(window, BC_EW_SCROLLCURSOR);
953         else
954                 WM_cursor_modal(window, BC_NSEW_SCROLLCURSOR);
955         
956         /* add temp handler */
957         WM_event_add_modal_handler(C, op);
958
959         if (U.viewzoom == USER_ZOOM_CONT) {
960                 /* needs a timer to continue redrawing */
961                 vzd->timer = WM_event_add_timer(CTX_wm_manager(C), CTX_wm_window(C), TIMER, 0.01f);
962                 vzd->timer_lastdraw = PIL_check_seconds_timer();
963         }
964
965         return OPERATOR_RUNNING_MODAL;
966 }
967
968 /* handle user input - calculations of mouse-movement need to be done here, not in the apply callback! */
969 static int view_zoomdrag_modal(bContext *C, wmOperator *op, wmEvent *event)
970 {
971         v2dViewZoomData *vzd = op->customdata;
972         View2D *v2d = vzd->v2d;
973         
974         /* execute the events */
975         if (event->type == TIMER && event->customdata == vzd->timer) {
976                 view_zoomdrag_apply(C, op);
977         }
978         else if (event->type == MOUSEMOVE) {
979                 float dx, dy;
980                 
981                 /* calculate new delta transform, based on zooming mode */
982                 if (U.viewzoom == USER_ZOOM_SCALE) {
983                         /* 'scale' zooming */
984                         float dist;
985                         
986                         /* x-axis transform */
987                         dist = (v2d->mask.xmax - v2d->mask.xmin) / 2.0f;
988                         dx = 1.0f - (fabsf(vzd->lastx - dist) + 2.0f) / (fabsf(event->x - dist) + 2.0f);
989                         dx *= 0.5f * (v2d->cur.xmax - v2d->cur.xmin);
990                         
991                         /* y-axis transform */
992                         dist = (v2d->mask.ymax - v2d->mask.ymin) / 2.0f;
993                         dy = 1.0f - (fabsf(vzd->lasty - dist) + 2.0f) / (fabsf(event->y - dist) + 2.0f);
994                         dy *= 0.5f * (v2d->cur.ymax - v2d->cur.ymin);
995                 }
996                 else {
997                         /* 'continuous' or 'dolly' */
998                         float fac;
999                         
1000                         /* x-axis transform */
1001                         fac = 0.01f * (event->x - vzd->lastx);
1002                         dx = fac * (v2d->cur.xmax - v2d->cur.xmin);
1003                         
1004                         /* y-axis transform */
1005                         fac = 0.01f * (event->y - vzd->lasty);
1006                         dy = fac * (v2d->cur.ymax - v2d->cur.ymin);
1007 #if 0
1008                         /* continuous zoom shouldn't move that fast... */
1009                         if (U.viewzoom == USER_ZOOM_CONT) { // XXX store this setting as RNA prop?
1010                                 double time = PIL_check_seconds_timer();
1011                                 float time_step = (float)(time - vzd->timer_lastdraw);
1012
1013                                 dx /= (0.1f / time_step);
1014                                 dy /= (0.1f / time_step);
1015                                 
1016                                 vzd->timer_lastdraw = time;
1017                         }
1018 #endif
1019                 }
1020                 
1021                 /* set transform amount, and add current deltas to stored total delta (for redo) */
1022                 RNA_float_set(op->ptr, "deltax", dx);
1023                 RNA_float_set(op->ptr, "deltay", dy);
1024                 vzd->dx += dx;
1025                 vzd->dy += dy;
1026                 
1027                 /* store mouse coordinates for next time, if not doing continuous zoom
1028                  *      - continuous zoom only depends on distance of mouse to starting point to determine rate of change
1029                  */
1030                 if (U.viewzoom != USER_ZOOM_CONT) { // XXX store this setting as RNA prop?
1031                         vzd->lastx = event->x;
1032                         vzd->lasty = event->y;
1033                 }
1034                 
1035                 /* apply zooming */
1036                 view_zoomdrag_apply(C, op);
1037         } 
1038         else if (event->type == vzd->invoke_event || event->type == ESCKEY) {
1039                 if (event->val == KM_RELEASE) {
1040                         
1041                         /* for redo, store the overall deltas - need to respect zoom-locks here... */
1042                         if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0)
1043                                 RNA_float_set(op->ptr, "deltax", vzd->dx);
1044                         else
1045                                 RNA_float_set(op->ptr, "deltax", 0);
1046                                 
1047                         if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0)
1048                                 RNA_float_set(op->ptr, "deltay", vzd->dy);
1049                         else
1050                                 RNA_float_set(op->ptr, "deltay", 0);
1051                         
1052                         /* free customdata */
1053                         view_zoomdrag_exit(C, op);
1054                         WM_cursor_restore(CTX_wm_window(C));
1055                         
1056                         return OPERATOR_FINISHED;
1057                 }
1058         }
1059
1060         return OPERATOR_RUNNING_MODAL;
1061 }
1062
1063 static void VIEW2D_OT_zoom(wmOperatorType *ot)
1064 {
1065         /* identifiers */
1066         ot->name = "Zoom 2D View";
1067         ot->description = "Zoom in/out the view";
1068         ot->idname = "VIEW2D_OT_zoom";
1069         
1070         /* api callbacks */
1071         ot->exec = view_zoomdrag_exec;
1072         ot->invoke = view_zoomdrag_invoke;
1073         ot->modal = view_zoomdrag_modal;
1074         ot->cancel = view_zoomdrag_cancel;
1075         
1076         ot->poll = view_zoom_poll;
1077         
1078         /* operator is repeatable */
1079         // ot->flag = OPTYPE_BLOCKING;
1080         
1081         /* rna - must keep these in sync with the other operators */
1082         RNA_def_float(ot->srna, "deltax", 0, -FLT_MAX, FLT_MAX, "Delta X", "", -FLT_MAX, FLT_MAX);
1083         RNA_def_float(ot->srna, "deltay", 0, -FLT_MAX, FLT_MAX, "Delta Y", "", -FLT_MAX, FLT_MAX);
1084 }
1085
1086 /* ********************************************************* */
1087 /* BORDER-ZOOM */
1088
1089 /* The user defines a rect using standard borderselect tools, and we use this rect to 
1090  * define the new zoom-level of the view in the following ways:
1091  *      1) LEFTMOUSE - zoom in to view
1092  *      2) RIGHTMOUSE - zoom out of view
1093  *
1094  * Currently, these key mappings are hardcoded, but it shouldn't be too important to
1095  * have custom keymappings for this...
1096  */
1097  
1098 static int view_borderzoom_exec(bContext *C, wmOperator *op)
1099 {
1100         ARegion *ar = CTX_wm_region(C);
1101         View2D *v2d = &ar->v2d;
1102         rctf rect;
1103         int gesture_mode;
1104         
1105         /* convert coordinates of rect to 'tot' rect coordinates */
1106         UI_view2d_region_to_view(v2d, RNA_int_get(op->ptr, "xmin"), RNA_int_get(op->ptr, "ymin"), &rect.xmin, &rect.ymin);
1107         UI_view2d_region_to_view(v2d, RNA_int_get(op->ptr, "xmax"), RNA_int_get(op->ptr, "ymax"), &rect.xmax, &rect.ymax);
1108         
1109         /* check if zooming in/out view */
1110         gesture_mode = RNA_int_get(op->ptr, "gesture_mode");
1111         
1112         if (gesture_mode == GESTURE_MODAL_IN) {
1113                 /* zoom in: 
1114                  *      - 'cur' rect will be defined by the coordinates of the border region 
1115                  *      - just set the 'cur' rect to have the same coordinates as the border region
1116                  *        if zoom is allowed to be changed
1117                  */
1118                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
1119                         v2d->cur.xmin = rect.xmin;
1120                         v2d->cur.xmax = rect.xmax;
1121                 }
1122                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
1123                         v2d->cur.ymin = rect.ymin;
1124                         v2d->cur.ymax = rect.ymax;
1125                 }
1126         }
1127         else { /* if (gesture_mode == GESTURE_MODAL_OUT) */
1128
1129                 /* zoom out:
1130                  *      - the current 'cur' rect coordinates are going to end up where the 'rect' ones are,
1131                  *        but the 'cur' rect coordinates will need to be adjusted to take in more of the view
1132                  *      - calculate zoom factor, and adjust using center-point
1133                  */
1134                 float zoom, center, size;
1135                 
1136                 /* TODO: is this zoom factor calculation valid? It seems to produce same results everytime... */
1137                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) {
1138                         size = (v2d->cur.xmax - v2d->cur.xmin);
1139                         zoom = size / (rect.xmax - rect.xmin);
1140                         center = (v2d->cur.xmax + v2d->cur.xmin) * 0.5f;
1141                         
1142                         v2d->cur.xmin = center - (size * zoom);
1143                         v2d->cur.xmax = center + (size * zoom);
1144                 }
1145                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) {
1146                         size = (v2d->cur.ymax - v2d->cur.ymin);
1147                         zoom = size / (rect.ymax - rect.ymin);
1148                         center = (v2d->cur.ymax + v2d->cur.ymin) * 0.5f;
1149                         
1150                         v2d->cur.ymin = center - (size * zoom);
1151                         v2d->cur.ymax = center + (size * zoom);
1152                 }
1153         }
1154         
1155         /* validate that view is in valid configuration after this operation */
1156         UI_view2d_curRect_validate(v2d);
1157         
1158         /* request updates to be done... */
1159         ED_region_tag_redraw(ar);
1160         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1161         
1162         return OPERATOR_FINISHED;
1163
1164
1165 static void VIEW2D_OT_zoom_border(wmOperatorType *ot)
1166 {
1167         /* identifiers */
1168         ot->name = "Zoom to Border";
1169         ot->description = "Zoom in the view to the nearest item contained in the border";
1170         ot->idname = "VIEW2D_OT_zoom_border";
1171         
1172         /* api callbacks */
1173         ot->invoke = WM_border_select_invoke;
1174         ot->exec = view_borderzoom_exec;
1175         ot->modal = WM_border_select_modal;
1176         ot->cancel = WM_border_select_cancel;
1177         
1178         ot->poll = view_zoom_poll;
1179         
1180         /* rna */
1181         RNA_def_int(ot->srna, "gesture_mode", 0, INT_MIN, INT_MAX, "Gesture Mode", "", INT_MIN, INT_MAX);
1182         RNA_def_int(ot->srna, "xmin", 0, INT_MIN, INT_MAX, "X Min", "", INT_MIN, INT_MAX);
1183         RNA_def_int(ot->srna, "xmax", 0, INT_MIN, INT_MAX, "X Max", "", INT_MIN, INT_MAX);
1184         RNA_def_int(ot->srna, "ymin", 0, INT_MIN, INT_MAX, "Y Min", "", INT_MIN, INT_MAX);
1185         RNA_def_int(ot->srna, "ymax", 0, INT_MIN, INT_MAX, "Y Max", "", INT_MIN, INT_MAX);
1186 }
1187
1188 /* ********************************************************* */
1189 /* SCROLLERS */
1190
1191 /*  Scrollers should behave in the following ways, when clicked on with LMB (and dragged):
1192  *              1) 'Handles' on end of 'bubble' - when the axis that the scroller represents is zoomable, 
1193  *                      enlarge 'cur' rect on the relevant side 
1194  *              2) 'Bubble'/'bar' - just drag, and bar should move with mouse (view pans opposite)
1195  *
1196  *      In order to make sure this works, each operator must define the following RNA-Operator Props:
1197  *              deltax, deltay  - define how much to move view by (relative to zoom-correction factor)
1198  */
1199
1200 /* customdata for scroller-invoke data */
1201 typedef struct v2dScrollerMove {
1202         View2D *v2d;            /* View2D data that this operation affects */
1203         ARegion *ar;            /* region that the scroller is in */
1204         
1205         short scroller;         /* scroller that mouse is in ('h' or 'v') */
1206         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?)
1207         
1208         float fac;              /* view adjustment factor, based on size of region */
1209         float delta;            /* amount moved by mouse on axis of interest */
1210         
1211         float scrollbarwidth;   /* width of the scrollbar itself, used for page up/down clicks */
1212         int scrollbar_orig;      /* initial location of scrollbar x/y, mouse relative */
1213         
1214         int lastx, lasty;       /* previous mouse coordinates (in screen coordinates) for determining movement */
1215 } v2dScrollerMove;
1216
1217
1218 /* View2DScrollers is typedef'd in UI_view2d.h 
1219  * This is a CUT DOWN VERSION of the 'real' version, which is defined in view2d.c, as we only need focus bubble info
1220  * WARNING: the start of this struct must not change, so that it stays in sync with the 'real' version
1221  *         For now, we don't need to have a separate (internal) header for structs like this...
1222  */
1223 struct View2DScrollers {        
1224         /* focus bubbles */
1225         int vert_min, vert_max; /* vertical scrollbar */
1226         int hor_min, hor_max;   /* horizontal scrollbar */
1227 };
1228
1229 /* quick enum for vsm->zone (scroller handles) */
1230 enum {
1231         SCROLLHANDLE_MIN = -1,
1232         SCROLLHANDLE_BAR,
1233         SCROLLHANDLE_MAX,
1234         SCROLLHANDLE_MIN_OUTSIDE,
1235         SCROLLHANDLE_MAX_OUTSIDE
1236 } /*eV2DScrollerHandle_Zone*/;
1237
1238 /* ------------------------ */
1239
1240 /* check if mouse is within scroller handle 
1241  *      - mouse                 =   relevant mouse coordinate in region space
1242  *      - sc_min, sc_max        =   extents of scroller 'groove' (potential available space for scroller)
1243  *      - sh_min, sh_max        =   positions of scrollbar handles
1244  */
1245 static short mouse_in_scroller_handle(int mouse, int sc_min, int sc_max, int sh_min, int sh_max)
1246 {
1247         short in_min, in_max, in_bar, out_min, out_max, in_view = 1;
1248         
1249         /* firstly, check if 
1250          *      - 'bubble' fills entire scroller 
1251          *      - 'bubble' completely out of view on either side 
1252          */
1253         if ((sh_min <= sc_min) && (sh_max >= sc_max)) in_view = 0;
1254         if (sh_min == sh_max) {
1255                 if (sh_min <= sc_min) in_view = 0;
1256                 if (sh_max >= sc_max) in_view = 0;
1257         }
1258         else {
1259                 if (sh_max <= sc_min) in_view = 0;
1260                 if (sh_min >= sc_max) in_view = 0;
1261         }
1262         
1263         
1264         if (in_view == 0) {
1265                 return SCROLLHANDLE_BAR;
1266         }
1267         
1268         /* check if mouse is in or past either handle */
1269         /* TODO: check if these extents are still valid or not */
1270         in_max = ( (mouse >= (sh_max - V2D_SCROLLER_HANDLE_SIZE)) && (mouse <= (sh_max + V2D_SCROLLER_HANDLE_SIZE)) );
1271         in_min = ( (mouse <= (sh_min + V2D_SCROLLER_HANDLE_SIZE)) && (mouse >= (sh_min - V2D_SCROLLER_HANDLE_SIZE)) );
1272         in_bar = ( (mouse < (sh_max - V2D_SCROLLER_HANDLE_SIZE)) && (mouse > (sh_min + V2D_SCROLLER_HANDLE_SIZE)) );
1273         out_min = mouse < (sh_min - V2D_SCROLLER_HANDLE_SIZE);
1274         out_max = mouse > (sh_max + V2D_SCROLLER_HANDLE_SIZE);
1275         
1276         if (in_bar)
1277                 return SCROLLHANDLE_BAR;
1278         else if (in_max)
1279                 return SCROLLHANDLE_MAX;
1280         else if (in_min)
1281                 return SCROLLHANDLE_MIN;
1282         else if (out_min)
1283                 return SCROLLHANDLE_MIN_OUTSIDE;                          
1284         else if (out_max)
1285                 return SCROLLHANDLE_MAX_OUTSIDE;
1286         
1287         /* unlikely to happen, though we just cover it in case */
1288         return SCROLLHANDLE_BAR;
1289
1290
1291 /* initialize customdata for scroller manipulation operator */
1292 static void scroller_activate_init(bContext *C, wmOperator *op, wmEvent *event, short in_scroller)
1293 {
1294         v2dScrollerMove *vsm;
1295         View2DScrollers *scrollers;
1296         ARegion *ar = CTX_wm_region(C);
1297         View2D *v2d = &ar->v2d;
1298         float mask_size;
1299         
1300         /* set custom-data for operator */
1301         vsm = MEM_callocN(sizeof(v2dScrollerMove), "v2dScrollerMove");
1302         op->customdata = vsm;
1303         
1304         /* set general data */
1305         vsm->v2d = v2d;
1306         vsm->ar = ar;
1307         vsm->scroller = in_scroller;
1308
1309         /* store mouse-coordinates, and convert mouse/screen coordinates to region coordinates */
1310         vsm->lastx = event->x;
1311         vsm->lasty = event->y;
1312         /* 'zone' depends on where mouse is relative to bubble 
1313          *      - zooming must be allowed on this axis, otherwise, default to pan
1314          */
1315         scrollers = UI_view2d_scrollers_calc(C, v2d, V2D_ARG_DUMMY, V2D_ARG_DUMMY, V2D_ARG_DUMMY, V2D_ARG_DUMMY);
1316
1317
1318         if (in_scroller == 'h') {
1319                 /* horizontal scroller - calculate adjustment factor first */
1320                 mask_size = (float)(v2d->hor.xmax - v2d->hor.xmin);
1321                 vsm->fac = (v2d->tot.xmax - v2d->tot.xmin) / mask_size;
1322                 
1323                 /* get 'zone' (i.e. which part of scroller is activated) */
1324                 vsm->zone = mouse_in_scroller_handle(event->mval[0], v2d->hor.xmin, v2d->hor.xmax, scrollers->hor_min, scrollers->hor_max);
1325                 
1326                 if ((v2d->keepzoom & V2D_LOCKZOOM_X) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1327                         /* default to scroll, as handles not usable */
1328                         vsm->zone = SCROLLHANDLE_BAR;
1329                 }
1330
1331                 vsm->scrollbarwidth = scrollers->hor_max - scrollers->hor_min;
1332                 vsm->scrollbar_orig = ((scrollers->hor_max + scrollers->hor_min) / 2) + ar->winrct.xmin;
1333         }
1334         else {
1335                 /* vertical scroller - calculate adjustment factor first */
1336                 mask_size = (float)(v2d->vert.ymax - v2d->vert.ymin);
1337                 vsm->fac = (v2d->tot.ymax - v2d->tot.ymin) / mask_size;
1338                 
1339                 /* get 'zone' (i.e. which part of scroller is activated) */
1340                 vsm->zone = mouse_in_scroller_handle(event->mval[1], v2d->vert.ymin, v2d->vert.ymax, scrollers->vert_min, scrollers->vert_max);
1341                         
1342                 if ((v2d->keepzoom & V2D_LOCKZOOM_Y) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1343                         /* default to scroll, as handles not usable */
1344                         vsm->zone = SCROLLHANDLE_BAR;
1345                 }
1346                 
1347                 vsm->scrollbarwidth = scrollers->vert_max - scrollers->vert_min;
1348                 vsm->scrollbar_orig = ((scrollers->vert_max + scrollers->vert_min) / 2) +  +ar->winrct.ymin;
1349         }
1350         
1351         UI_view2d_scrollers_free(scrollers);
1352         ED_region_tag_redraw(ar);
1353 }
1354
1355 /* cleanup temp customdata  */
1356 static void scroller_activate_exit(bContext *C, wmOperator *op)
1357 {
1358         if (op->customdata) {
1359                 v2dScrollerMove *vsm = op->customdata;
1360
1361                 vsm->v2d->scroll_ui &= ~(V2D_SCROLL_H_ACTIVE | V2D_SCROLL_V_ACTIVE);
1362                 
1363                 MEM_freeN(op->customdata);
1364                 op->customdata = NULL;
1365                 
1366                 ED_region_tag_redraw(CTX_wm_region(C));
1367         }
1368 }
1369
1370 static int scroller_activate_cancel(bContext *C, wmOperator *op)
1371 {
1372         scroller_activate_exit(C, op);
1373
1374         return OPERATOR_CANCELLED;
1375 }
1376
1377 /* apply transform to view (i.e. adjust 'cur' rect) */
1378 static void scroller_activate_apply(bContext *C, wmOperator *op)
1379 {
1380         v2dScrollerMove *vsm = op->customdata;
1381         View2D *v2d = vsm->v2d;
1382         float temp;
1383         
1384         /* calculate amount to move view by */
1385         temp = vsm->fac * vsm->delta;
1386         
1387         /* type of movement */
1388         switch (vsm->zone) {
1389                 case SCROLLHANDLE_MIN:
1390                         /* only expand view on axis if zoom is allowed */
1391                         if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X))
1392                                 v2d->cur.xmin -= temp;
1393                         if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y))
1394                                 v2d->cur.ymin -= temp;
1395                         break;
1396                         
1397                 case SCROLLHANDLE_MAX:
1398                         
1399                         /* only expand view on axis if zoom is allowed */
1400                         if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X))
1401                                 v2d->cur.xmax += temp;
1402                         if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y))
1403                                 v2d->cur.ymax += temp;
1404                         break;
1405                         
1406                 case SCROLLHANDLE_MIN_OUTSIDE:
1407                 case SCROLLHANDLE_MAX_OUTSIDE:
1408                 case SCROLLHANDLE_BAR:
1409                 default:
1410                         /* only move view on an axis if panning is allowed */
1411                         if ((vsm->scroller == 'h') && !(v2d->keepofs & V2D_LOCKOFS_X)) {
1412                                 v2d->cur.xmin += temp;
1413                                 v2d->cur.xmax += temp;
1414                         }
1415                         if ((vsm->scroller == 'v') && !(v2d->keepofs & V2D_LOCKOFS_Y)) {
1416                                 v2d->cur.ymin += temp;
1417                                 v2d->cur.ymax += temp;
1418                         }
1419                         break;
1420                         
1421         }
1422         
1423         /* validate that view is in valid configuration after this operation */
1424         UI_view2d_curRect_validate(v2d);
1425         
1426         /* request updates to be done... */
1427         ED_region_tag_redraw(vsm->ar);
1428         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1429 }
1430
1431 /* handle user input for scrollers - calculations of mouse-movement need to be done here, not in the apply callback! */
1432 static int scroller_activate_modal(bContext *C, wmOperator *op, wmEvent *event)
1433 {
1434         v2dScrollerMove *vsm = op->customdata;
1435         
1436         /* execute the events */
1437         switch (event->type) {
1438                 case MOUSEMOVE:
1439                 {
1440                         /* calculate new delta transform, then store mouse-coordinates for next-time */
1441                         if (ELEM(vsm->zone, SCROLLHANDLE_BAR, SCROLLHANDLE_MAX)) {
1442                                 /* if using bar (i.e. 'panning') or 'max' zoom widget */
1443                                 switch (vsm->scroller) {
1444                                         case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves opposite to mouse) */
1445                                                 vsm->delta = (float)(event->x - vsm->lastx);
1446                                                 break;
1447                                         case 'v': /* vertical scroller - so only vertical movement ('cur' moves opposite to mouse) */
1448                                                 vsm->delta = (float)(event->y - vsm->lasty);
1449                                                 break;
1450                                 }
1451                         }
1452                         else if (vsm->zone == SCROLLHANDLE_MIN) {
1453                                 /* using 'min' zoom widget */
1454                                 switch (vsm->scroller) {
1455                                         case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves with mouse) */
1456                                                 vsm->delta = (float)(vsm->lastx - event->x);
1457                                                 break;
1458                                         case 'v': /* vertical scroller - so only vertical movement ('cur' moves with to mouse) */
1459                                                 vsm->delta = (float)(vsm->lasty - event->y);
1460                                                 break;
1461                                 }
1462                         }
1463                         
1464                         /* store previous coordinates */
1465                         vsm->lastx = event->x;
1466                         vsm->lasty = event->y;
1467                         
1468                         scroller_activate_apply(C, op);
1469                 }
1470                 break;
1471                         
1472                 case LEFTMOUSE:
1473                 case MIDDLEMOUSE:
1474                         if (event->val == KM_RELEASE) {
1475                                 /* single-click was in empty space outside bubble, so scroll by 1 'page' */
1476                                 if (ELEM(vsm->zone, SCROLLHANDLE_MIN_OUTSIDE, SCROLLHANDLE_MAX_OUTSIDE)) {
1477                                         if (vsm->zone == SCROLLHANDLE_MIN_OUTSIDE)
1478                                                 vsm->delta = -vsm->scrollbarwidth * 0.8f;
1479                                         else if (vsm->zone == SCROLLHANDLE_MAX_OUTSIDE)
1480                                                 vsm->delta = vsm->scrollbarwidth * 0.8f;
1481                                         
1482                                         scroller_activate_apply(C, op);
1483                                         scroller_activate_exit(C, op);
1484                                         return OPERATOR_FINISHED;
1485                                 }
1486                                 
1487                                 /* otherwise, end the drag action  */
1488                                 if (vsm->lastx || vsm->lasty) {
1489                                         scroller_activate_exit(C, op);
1490                                         return OPERATOR_FINISHED;
1491                                 }
1492                         }
1493                         break;
1494
1495         }
1496
1497         return OPERATOR_RUNNING_MODAL;
1498 }
1499
1500
1501 /* a click (or click drag in progress) should have occurred, so check if it happened in scrollbar */
1502 static int scroller_activate_invoke(bContext *C, wmOperator *op, wmEvent *event)
1503 {
1504         ARegion *ar = CTX_wm_region(C);
1505         View2D *v2d = &ar->v2d;
1506         short in_scroller = 0;
1507                 
1508         /* check if mouse in scrollbars, if they're enabled */
1509         in_scroller = UI_view2d_mouse_in_scrollers(C, v2d, event->x, event->y);
1510         
1511         /* if in a scroller, init customdata then set modal handler which will catch mousedown to start doing useful stuff */
1512         if (in_scroller) {
1513                 v2dScrollerMove *vsm;
1514                 
1515                 /* initialize customdata */
1516                 scroller_activate_init(C, op, event, in_scroller);
1517                 vsm = (v2dScrollerMove *)op->customdata;
1518                 
1519                 /* support for quick jump to location - gtk and qt do this on linux */
1520                 if (event->type == MIDDLEMOUSE) {
1521                         switch (vsm->scroller) {
1522                                 case 'h': /* horizontal scroller - so only horizontal movement ('cur' moves opposite to mouse) */
1523                                         vsm->delta = (float)(event->x - vsm->scrollbar_orig);
1524                                         break;
1525                                 case 'v': /* vertical scroller - so only vertical movement ('cur' moves opposite to mouse) */
1526                                         vsm->delta = (float)(event->y - vsm->scrollbar_orig);
1527                                         break;
1528                         }
1529                         scroller_activate_apply(C, op);
1530
1531                         vsm->zone = SCROLLHANDLE_BAR;
1532                 }
1533
1534                 /* check if zoom zones are inappropriate (i.e. zoom widgets not shown), so cannot continue
1535                  * NOTE: see view2d.c for latest conditions, and keep this in sync with that
1536                  */
1537                 if (ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) {
1538                         if ( ((vsm->scroller == 'h') && (v2d->scroll & V2D_SCROLL_SCALE_HORIZONTAL) == 0) ||
1539                              ((vsm->scroller == 'v') && (v2d->scroll & V2D_SCROLL_SCALE_VERTICAL) == 0) )
1540                         {
1541                                 /* switch to bar (i.e. no scaling gets handled) */
1542                                 vsm->zone = SCROLLHANDLE_BAR;
1543                         }
1544                 }
1545                 
1546                 /* check if zone is inappropriate (i.e. 'bar' but panning is banned), so cannot continue */
1547                 if (vsm->zone == SCROLLHANDLE_BAR) {
1548                         if ( ((vsm->scroller == 'h') && (v2d->keepofs & V2D_LOCKOFS_X)) ||
1549                              ((vsm->scroller == 'v') && (v2d->keepofs & V2D_LOCKOFS_Y)) )
1550                         {
1551                                 /* free customdata initialized */
1552                                 scroller_activate_exit(C, op);
1553                                 
1554                                 /* can't catch this event for ourselves, so let it go to someone else? */
1555                                 return OPERATOR_PASS_THROUGH;
1556                         }                       
1557                 }
1558                 
1559                 /* zone is also inappropriate if scroller is not visible... */
1560                 if ( ((vsm->scroller == 'h') && (v2d->scroll & (V2D_SCROLL_HORIZONTAL_HIDE | V2D_SCROLL_HORIZONTAL_FULLR))) ||
1561                      ((vsm->scroller == 'v') && (v2d->scroll & (V2D_SCROLL_VERTICAL_HIDE | V2D_SCROLL_VERTICAL_FULLR))) )
1562                 {
1563                         /* free customdata initialized */
1564                         scroller_activate_exit(C, op);
1565                                 
1566                         /* can't catch this event for ourselves, so let it go to someone else? */
1567                         /* XXX note: if handlers use mask rect to clip input, input will fail for this case */
1568                         return OPERATOR_PASS_THROUGH;
1569                 }
1570                 
1571                 /* activate the scroller */
1572                 if (vsm->scroller == 'h')
1573                         v2d->scroll_ui |= V2D_SCROLL_H_ACTIVE;
1574                 else
1575                         v2d->scroll_ui |= V2D_SCROLL_V_ACTIVE;
1576                 
1577                 /* still ok, so can add */
1578                 WM_event_add_modal_handler(C, op);
1579                 return OPERATOR_RUNNING_MODAL;
1580         }
1581         else {
1582                 /* not in scroller, so nothing happened... (pass through let's something else catch event) */
1583                 return OPERATOR_PASS_THROUGH;
1584         }
1585 }
1586
1587 /* LMB-Drag in Scrollers - not repeatable operator! */
1588 static void VIEW2D_OT_scroller_activate(wmOperatorType *ot)
1589 {
1590         /* identifiers */
1591         ot->name = "Scroller Activate";
1592         ot->description = "Scroll view by mouse click and drag";
1593         ot->idname = "VIEW2D_OT_scroller_activate";
1594
1595         /* flags */
1596         ot->flag = OPTYPE_BLOCKING;
1597         
1598         /* api callbacks */
1599         ot->invoke = scroller_activate_invoke;
1600         ot->modal = scroller_activate_modal;
1601         ot->cancel = scroller_activate_cancel;
1602
1603         ot->poll = view2d_poll;
1604 }
1605
1606 /* ********************************************************* */
1607 /* RESET */
1608
1609 static int reset_exec(bContext *C, wmOperator *UNUSED(op))
1610 {
1611         uiStyle *style = UI_GetStyle();
1612         ARegion *ar = CTX_wm_region(C);
1613         View2D *v2d = &ar->v2d;
1614         int winx, winy;
1615
1616         /* zoom 1.0 */
1617         winx = (float)(v2d->mask.xmax - v2d->mask.xmin + 1);
1618         winy = (float)(v2d->mask.ymax - v2d->mask.ymin + 1);
1619
1620         v2d->cur.xmax = v2d->cur.xmin + winx;
1621         v2d->cur.ymax = v2d->cur.ymin + winy;
1622         
1623         /* align */
1624         if (v2d->align) {
1625                 /* posx and negx flags are mutually exclusive, so watch out */
1626                 if ((v2d->align & V2D_ALIGN_NO_POS_X) && !(v2d->align & V2D_ALIGN_NO_NEG_X)) {
1627                         v2d->cur.xmax = 0.0f;
1628                         v2d->cur.xmin = -winx * style->panelzoom;
1629                 }
1630                 else if ((v2d->align & V2D_ALIGN_NO_NEG_X) && !(v2d->align & V2D_ALIGN_NO_POS_X)) {
1631                         v2d->cur.xmax = winx * style->panelzoom;
1632                         v2d->cur.xmin = 0.0f;
1633                 }
1634
1635                 /* - posx and negx flags are mutually exclusive, so watch out */
1636                 if ((v2d->align & V2D_ALIGN_NO_POS_Y) && !(v2d->align & V2D_ALIGN_NO_NEG_Y)) {
1637                         v2d->cur.ymax = 0.0f;
1638                         v2d->cur.ymin = -winy * style->panelzoom;
1639                 }
1640                 else if ((v2d->align & V2D_ALIGN_NO_NEG_Y) && !(v2d->align & V2D_ALIGN_NO_POS_Y)) {
1641                         v2d->cur.ymax = winy * style->panelzoom;
1642                         v2d->cur.ymin = 0.0f;
1643                 }
1644         }
1645
1646         /* validate that view is in valid configuration after this operation */
1647         UI_view2d_curRect_validate(v2d);
1648         
1649         /* request updates to be done... */
1650         ED_region_tag_redraw(ar);
1651         UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY);
1652         
1653         return OPERATOR_FINISHED;
1654 }
1655  
1656 static void VIEW2D_OT_reset(wmOperatorType *ot)
1657 {
1658         /* identifiers */
1659         ot->name = "Reset View";
1660         ot->description = "Reset the view";
1661         ot->idname = "VIEW2D_OT_reset";
1662         
1663         /* api callbacks */
1664         ot->exec = reset_exec;
1665         ot->poll = view2d_poll;
1666 }
1667  
1668 /* ********************************************************* */
1669 /* Registration */
1670
1671 void UI_view2d_operatortypes(void)
1672 {
1673         WM_operatortype_append(VIEW2D_OT_pan);
1674         
1675         WM_operatortype_append(VIEW2D_OT_scroll_left);
1676         WM_operatortype_append(VIEW2D_OT_scroll_right);
1677         WM_operatortype_append(VIEW2D_OT_scroll_up);
1678         WM_operatortype_append(VIEW2D_OT_scroll_down);
1679         
1680         WM_operatortype_append(VIEW2D_OT_zoom_in);
1681         WM_operatortype_append(VIEW2D_OT_zoom_out);
1682         
1683         WM_operatortype_append(VIEW2D_OT_zoom);
1684         WM_operatortype_append(VIEW2D_OT_zoom_border);
1685         
1686         WM_operatortype_append(VIEW2D_OT_scroller_activate);
1687
1688         WM_operatortype_append(VIEW2D_OT_reset);
1689 }
1690
1691 void UI_view2d_keymap(wmKeyConfig *keyconf)
1692 {
1693         wmKeyMap *keymap = WM_keymap_find(keyconf, "View2D", 0, 0);
1694         wmKeyMapItem *kmi;
1695
1696         /* scrollers */
1697         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", LEFTMOUSE, KM_PRESS, 0, 0);
1698         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", MIDDLEMOUSE, KM_PRESS, 0, 0);
1699
1700         /* pan/scroll */
1701         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, 0, 0);
1702         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, KM_SHIFT, 0);
1703         
1704         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MOUSEPAN, 0, 0, 0);
1705         
1706         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", WHEELDOWNMOUSE, KM_PRESS, KM_CTRL, 0);
1707         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", WHEELUPMOUSE, KM_PRESS, KM_CTRL, 0);
1708         
1709         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, KM_SHIFT, 0);
1710         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, KM_SHIFT, 0);
1711         
1712         /* zoom - single step */
1713         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", WHEELOUTMOUSE, KM_PRESS, 0, 0);
1714         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", WHEELINMOUSE, KM_PRESS, 0, 0);
1715         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", PADMINUS, KM_PRESS, 0, 0);
1716         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", PADPLUSKEY, KM_PRESS, 0, 0);
1717         
1718         /* scroll up/down - no modifiers, only when zoom fails */
1719         /* these may fail if zoom is disallowed, in which case they should pass on event */
1720         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1721         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, 0, 0);
1722         /* these may be necessary if vertical scroll is disallowed */
1723         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1724         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", WHEELUPMOUSE, KM_PRESS, 0, 0);
1725         
1726         /* alternatives for page up/down to scroll */
1727 #if 0 // XXX disabled, since this causes conflicts with hotkeys in animation editors
1728         /* scroll up/down may fall through to left/right */
1729         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", PAGEDOWNKEY, KM_PRESS, 0, 0);
1730         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", PAGEUPKEY, KM_PRESS, 0, 0);
1731         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", PAGEDOWNKEY, KM_PRESS, 0, 0);
1732         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", PAGEUPKEY, KM_PRESS, 0, 0);
1733         /* shift for moving view left/right with page up/down */
1734         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_right", PAGEDOWNKEY, KM_PRESS, KM_SHIFT, 0);
1735         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_left", PAGEUPKEY, KM_PRESS, KM_SHIFT, 0);
1736 #endif
1737         
1738         /* zoom - drag */
1739         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MIDDLEMOUSE, KM_PRESS, KM_CTRL, 0);
1740         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEZOOM, 0, 0, 0);
1741         
1742         /* borderzoom - drag */
1743         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_border", BKEY, KM_PRESS, KM_SHIFT, 0);
1744
1745         /* Alternative keymap for buttons listview */
1746         keymap = WM_keymap_find(keyconf, "View2D Buttons List", 0, 0);
1747
1748         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", LEFTMOUSE, KM_PRESS, 0, 0);
1749         WM_keymap_add_item(keymap, "VIEW2D_OT_scroller_activate", MIDDLEMOUSE, KM_PRESS, 0, 0);
1750
1751         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MIDDLEMOUSE, KM_PRESS, 0, 0);
1752         WM_keymap_add_item(keymap, "VIEW2D_OT_pan", MOUSEPAN, 0, 0, 0);
1753         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", WHEELDOWNMOUSE, KM_PRESS, 0, 0);
1754         WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", WHEELUPMOUSE, KM_PRESS, 0, 0);
1755         
1756         kmi = WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_down", PAGEDOWNKEY, KM_PRESS, 0, 0);
1757         RNA_boolean_set(kmi->ptr, "page", TRUE);
1758         kmi = WM_keymap_add_item(keymap, "VIEW2D_OT_scroll_up", PAGEUPKEY, KM_PRESS, 0, 0);
1759         RNA_boolean_set(kmi->ptr, "page", TRUE);
1760         
1761         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MIDDLEMOUSE, KM_PRESS, KM_CTRL, 0);
1762         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom", MOUSEZOOM, 0, 0, 0);
1763         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_out", PADMINUS, KM_PRESS, 0, 0);
1764         WM_keymap_add_item(keymap, "VIEW2D_OT_zoom_in", PADPLUSKEY, KM_PRESS, 0, 0);
1765         WM_keymap_add_item(keymap, "VIEW2D_OT_reset", HOMEKEY, KM_PRESS, 0, 0);
1766 }
1767