/* * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * The Original Code is Copyright (C) 2008 Blender Foundation. * All rights reserved. */ /** \file * \ingroup edinterface */ #include #include "MEM_guardedalloc.h" #include "DNA_userdef_types.h" #include "DNA_windowmanager_types.h" #include "BLI_blenlib.h" #include "BLI_math_base.h" #include "BLI_math_vector.h" #include "BLI_utildefines.h" #include "BKE_context.h" #include "RNA_access.h" #include "RNA_define.h" #include "WM_api.h" #include "WM_types.h" #include "ED_screen.h" #include "UI_interface.h" #include "UI_view2d.h" #include "PIL_time.h" /* USER_ZOOM_CONT */ /* -------------------------------------------------------------------- */ /** \name Internal Utilities * \{ */ static bool view2d_poll(bContext *C) { ARegion *region = CTX_wm_region(C); return (region != NULL) && (region->v2d.flag & V2D_IS_INITIALISED); } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Pan Shared Utilities * \{ */ /** * This group of operators come in several forms: * -# Modal 'dragging' with MMB - where movement of mouse dictates amount to pan view by * -# Scroll-wheel 'steps' - rolling mouse-wheel by one step moves view by predefined amount * * In order to make sure this works, each operator must define the following RNA-Operator Props: * - `deltax, deltay` - define how much to move view by (relative to zoom-correction factor) */ /** * Temporary custom-data for operator. */ typedef struct v2dViewPanData { /** screen where view pan was initiated */ bScreen *screen; /** area where view pan was initiated */ ScrArea *area; /** region where view pan was initiated */ ARegion *region; /** view2d we're operating in */ View2D *v2d; /** amount to move view relative to zoom */ float facx, facy; /* options for version 1 */ /** mouse x/y values in window when operator was initiated */ int startx, starty; /** previous x/y values of mouse in window */ int lastx, lasty; /** event starting pan, for modal exit */ int invoke_event; /** for MMB in scrollers (old feature in past, but now not that useful) */ short in_scroller; /* View2D Edge Panning */ double edge_pan_last_time; double edge_pan_start_time_x, edge_pan_start_time_y; } v2dViewPanData; /* initialize panning customdata */ static int view_pan_init(bContext *C, wmOperator *op) { ARegion *region = CTX_wm_region(C); v2dViewPanData *vpd; View2D *v2d; float winx, winy; /* regions now have v2d-data by default, so check for region */ if (region == NULL) { return 0; } /* check if panning is allowed at all */ v2d = ®ion->v2d; if ((v2d->keepofs & V2D_LOCKOFS_X) && (v2d->keepofs & V2D_LOCKOFS_Y)) { return 0; } /* set custom-data for operator */ vpd = MEM_callocN(sizeof(v2dViewPanData), "v2dViewPanData"); op->customdata = vpd; /* set pointers to owners */ vpd->screen = CTX_wm_screen(C); vpd->area = CTX_wm_area(C); vpd->v2d = v2d; vpd->region = region; /* calculate translation factor - based on size of view */ winx = (float)(BLI_rcti_size_x(®ion->winrct) + 1); winy = (float)(BLI_rcti_size_y(®ion->winrct) + 1); vpd->facx = (BLI_rctf_size_x(&v2d->cur)) / winx; vpd->facy = (BLI_rctf_size_y(&v2d->cur)) / winy; return 1; } #ifdef WITH_INPUT_NDOF static bool view_pan_poll(bContext *C) { ARegion *region = CTX_wm_region(C); View2D *v2d; /* check if there's a region in context to work with */ if (region == NULL) { return 0; } v2d = ®ion->v2d; /* check that 2d-view can pan */ if ((v2d->keepofs & V2D_LOCKOFS_X) && (v2d->keepofs & V2D_LOCKOFS_Y)) { return 0; } /* view can pan */ return 1; } #endif /* apply transform to view (i.e. adjust 'cur' rect) */ static void view_pan_apply_ex(bContext *C, v2dViewPanData *vpd, float dx, float dy) { View2D *v2d = vpd->v2d; /* calculate amount to move view by */ dx *= vpd->facx; dy *= vpd->facy; /* only move view on an axis if change is allowed */ if ((v2d->keepofs & V2D_LOCKOFS_X) == 0) { v2d->cur.xmin += dx; v2d->cur.xmax += dx; } if ((v2d->keepofs & V2D_LOCKOFS_Y) == 0) { v2d->cur.ymin += dy; v2d->cur.ymax += dy; } /* validate that view is in valid configuration after this operation */ UI_view2d_curRect_validate(v2d); /* don't rebuild full tree in outliner, since we're just changing our view */ ED_region_tag_redraw_no_rebuild(vpd->region); /* request updates to be done... */ WM_event_add_mousemove(CTX_wm_window(C)); UI_view2d_sync(vpd->screen, vpd->area, v2d, V2D_LOCK_COPY); } static void view_pan_apply(bContext *C, wmOperator *op) { v2dViewPanData *vpd = op->customdata; view_pan_apply_ex(C, vpd, RNA_int_get(op->ptr, "deltax"), RNA_int_get(op->ptr, "deltay")); } /* cleanup temp customdata */ static void view_pan_exit(wmOperator *op) { if (op->customdata) { MEM_freeN(op->customdata); op->customdata = NULL; } } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Pan Operator (modal drag-pan) * \{ */ /* for 'redo' only, with no user input */ static int view_pan_exec(bContext *C, wmOperator *op) { if (!view_pan_init(C, op)) { return OPERATOR_CANCELLED; } view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } /* set up modal operator and relevant settings */ static int view_pan_invoke(bContext *C, wmOperator *op, const wmEvent *event) { wmWindow *window = CTX_wm_window(C); v2dViewPanData *vpd; View2D *v2d; /* set up customdata */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } vpd = op->customdata; v2d = vpd->v2d; /* set initial settings */ vpd->startx = vpd->lastx = event->x; vpd->starty = vpd->lasty = event->y; vpd->invoke_event = event->type; if (event->type == MOUSEPAN) { RNA_int_set(op->ptr, "deltax", event->prevx - event->x); RNA_int_set(op->ptr, "deltay", event->prevy - event->y); view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } RNA_int_set(op->ptr, "deltax", 0); RNA_int_set(op->ptr, "deltay", 0); if (v2d->keepofs & V2D_LOCKOFS_X) { WM_cursor_modal_set(window, WM_CURSOR_NS_SCROLL); } else if (v2d->keepofs & V2D_LOCKOFS_Y) { WM_cursor_modal_set(window, WM_CURSOR_EW_SCROLL); } else { WM_cursor_modal_set(window, WM_CURSOR_NSEW_SCROLL); } /* add temp handler */ WM_event_add_modal_handler(C, op); return OPERATOR_RUNNING_MODAL; } /* handle user input - calculations of mouse-movement * need to be done here, not in the apply callback! */ static int view_pan_modal(bContext *C, wmOperator *op, const wmEvent *event) { v2dViewPanData *vpd = op->customdata; /* execute the events */ switch (event->type) { case MOUSEMOVE: { /* calculate new delta transform, then store mouse-coordinates for next-time */ RNA_int_set(op->ptr, "deltax", (vpd->lastx - event->x)); RNA_int_set(op->ptr, "deltay", (vpd->lasty - event->y)); vpd->lastx = event->x; vpd->lasty = event->y; view_pan_apply(C, op); break; } /* XXX - Mode switching isn't implemented. See comments in 36818. * switch to zoom */ #if 0 case LEFTMOUSE: if (event->val == KM_PRESS) { /* calculate overall delta mouse-movement for redo */ RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx)); RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty)); view_pan_exit(op); WM_cursor_modal_restore(CTX_wm_window(C)); WM_operator_name_call(C, "VIEW2D_OT_zoom", WM_OP_INVOKE_DEFAULT, NULL); return OPERATOR_FINISHED; } #endif default: if (event->type == vpd->invoke_event || event->type == EVT_ESCKEY) { if (event->val == KM_RELEASE) { /* calculate overall delta mouse-movement for redo */ RNA_int_set(op->ptr, "deltax", (vpd->startx - vpd->lastx)); RNA_int_set(op->ptr, "deltay", (vpd->starty - vpd->lasty)); view_pan_exit(op); WM_cursor_modal_restore(CTX_wm_window(C)); return OPERATOR_FINISHED; } } break; } return OPERATOR_RUNNING_MODAL; } static void view_pan_cancel(bContext *UNUSED(C), wmOperator *op) { view_pan_exit(op); } static void VIEW2D_OT_pan(wmOperatorType *ot) { /* identifiers */ ot->name = "Pan View"; ot->description = "Pan the view"; ot->idname = "VIEW2D_OT_pan"; /* api callbacks */ ot->exec = view_pan_exec; ot->invoke = view_pan_invoke; ot->modal = view_pan_modal; ot->cancel = view_pan_cancel; /* operator is modal */ ot->flag = OPTYPE_BLOCKING | OPTYPE_GRAB_CURSOR_XY; /* rna - must keep these in sync with the other operators */ RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX); RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX); } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Edge Pan Operator (modal) * * Scroll the region if the mouse is dragged to an edge. "Invisible" operator that always * passes through. * \{ */ /** Distance from the edge of the region within which to start panning. */ #define EDGE_PAN_REGION_PAD (U.widget_unit) /** Speed factor in pixels per second per pixel of distance from edge pan zone beginning. */ #define EDGE_PAN_SPEED_PER_PIXEL (25.0f * (float)U.dpi_fac) /** Delay before drag panning in seconds. */ #define EDGE_PAN_DELAY 1.0f /* set up modal operator and relevant settings */ static int view_edge_pan_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event)) { /* Set up customdata. */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } v2dViewPanData *vpd = op->customdata; vpd->edge_pan_start_time_x = 0.0; vpd->edge_pan_start_time_y = 0.0; vpd->edge_pan_last_time = PIL_check_seconds_timer(); WM_event_add_modal_handler(C, op); return (OPERATOR_RUNNING_MODAL | OPERATOR_PASS_THROUGH); } /** * Reset the edge pan timers if the mouse isn't in the scroll zone and * start the timers when the mouse enters a scroll zone. */ static void edge_pan_manage_delay_timers(v2dViewPanData *vpd, int pan_dir_x, int pan_dir_y, const double current_time) { if (pan_dir_x == 0) { vpd->edge_pan_start_time_x = 0.0; } else if (vpd->edge_pan_start_time_x == 0.0) { vpd->edge_pan_start_time_x = current_time; } if (pan_dir_y == 0) { vpd->edge_pan_start_time_y = 0.0; } else if (vpd->edge_pan_start_time_y == 0.0) { vpd->edge_pan_start_time_y = current_time; } } /** * Used to calculate a "fade in" factor for edge panning to make the interaction feel smooth * and more purposeful. * * \note Assumes a domain_min of 0.0f. */ static float smootherstep(const float domain_max, float x) { x = clamp_f(x / domain_max, 0.0, 1.0); return x * x * x * (x * (x * 6.0 - 15.0) + 10.0); } static float edge_pan_speed(v2dViewPanData *vpd, int event_loc, bool x_dir, const double current_time) { ARegion *region = vpd->region; /* Find the distance from the start of the drag zone. */ int min = (x_dir ? region->winrct.xmin : region->winrct.ymin) + EDGE_PAN_REGION_PAD; int max = (x_dir ? region->winrct.xmax : region->winrct.ymax) - EDGE_PAN_REGION_PAD; int distance = 0.0; if (event_loc > max) { distance = event_loc - max; } else if (event_loc < min) { distance = min - event_loc; } else { BLI_assert(!"Calculating speed outside of pan zones"); return 0.0f; } /* Apply a fade in to the speed based on a start time delay. */ double start_time = x_dir ? vpd->edge_pan_start_time_x : vpd->edge_pan_start_time_y; float delay_factor = smootherstep(EDGE_PAN_DELAY, (float)(current_time - start_time)); return distance * EDGE_PAN_SPEED_PER_PIXEL * delay_factor; } static int view_edge_pan_modal(bContext *C, wmOperator *op, const wmEvent *event) { v2dViewPanData *vpd = op->customdata; ARegion *region = vpd->region; if (event->val == KM_RELEASE || event->type == EVT_ESCKEY) { view_pan_exit(op); return (OPERATOR_FINISHED | OPERATOR_PASS_THROUGH); } /* Only mousemove events matter here, ignore others. */ if (event->type != MOUSEMOVE) { return OPERATOR_PASS_THROUGH; } /* This operator is supposed to run together with some drag action. * On successful handling, always pass events on to other handlers. */ const int success_retval = OPERATOR_PASS_THROUGH; int outside_padding = RNA_int_get(op->ptr, "outside_padding") * UI_UNIT_X; rcti padding_rect; if (outside_padding != 0) { padding_rect = region->winrct; BLI_rcti_pad(&padding_rect, outside_padding, outside_padding); } int pan_dir_x = 0; int pan_dir_y = 0; if ((outside_padding == 0) || BLI_rcti_isect_pt(&padding_rect, event->x, event->y)) { /* Find whether the mouse is beyond X and Y edges. */ if (event->x > region->winrct.xmax - EDGE_PAN_REGION_PAD) { pan_dir_x = 1; } else if (event->x < region->winrct.xmin + EDGE_PAN_REGION_PAD) { pan_dir_x = -1; } if (event->y > region->winrct.ymax - EDGE_PAN_REGION_PAD) { pan_dir_y = 1; } else if (event->y < region->winrct.ymin + EDGE_PAN_REGION_PAD) { pan_dir_y = -1; } } const double current_time = PIL_check_seconds_timer(); edge_pan_manage_delay_timers(vpd, pan_dir_x, pan_dir_y, current_time); /* Calculate the delta since the last time the operator was called. */ float dtime = (float)(current_time - vpd->edge_pan_last_time); float dx = 0.0f, dy = 0.0f; if (pan_dir_x != 0) { float speed = edge_pan_speed(vpd, event->x, true, current_time); dx = dtime * speed * (float)pan_dir_x; } if (pan_dir_y != 0) { float speed = edge_pan_speed(vpd, event->y, false, current_time); dy = dtime * speed * (float)pan_dir_y; } vpd->edge_pan_last_time = current_time; /* Pan, clamping inside the regions's total bounds. */ view_pan_apply_ex(C, vpd, dx, dy); return success_retval; } static void view_edge_pan_cancel(bContext *UNUSED(C), wmOperator *op) { view_pan_exit(op); } static void VIEW2D_OT_edge_pan(wmOperatorType *ot) { /* identifiers */ ot->name = "View Edge Pan"; ot->description = "Pan the view when the mouse is held at an edge"; ot->idname = "VIEW2D_OT_edge_pan"; /* api callbacks */ ot->invoke = view_edge_pan_invoke; ot->modal = view_edge_pan_modal; ot->cancel = view_edge_pan_cancel; /* operator is modal */ ot->flag = OPTYPE_INTERNAL; RNA_def_int(ot->srna, "outside_padding", 0, 0, 100, "Outside Padding", "Padding around the region in UI units within which panning is activated (0 to " "disable boundary)", 0, 100); } #undef EDGE_PAN_REGION_PAD #undef EDGE_PAN_SPEED_PER_PIXEL #undef EDGE_PAN_DELAY /** \} */ /* -------------------------------------------------------------------- */ /** \name View Pan Operator (single step) * \{ */ /* this operator only needs this single callback, where it calls the view_pan_*() methods */ static int view_scrollright_exec(bContext *C, wmOperator *op) { v2dViewPanData *vpd; /* initialize default settings (and validate if ok to run) */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } /* also, check if can pan in horizontal axis */ vpd = op->customdata; if (vpd->v2d->keepofs & V2D_LOCKOFS_X) { view_pan_exit(op); return OPERATOR_PASS_THROUGH; } /* set RNA-Props - only movement in positive x-direction */ RNA_int_set(op->ptr, "deltax", 40); RNA_int_set(op->ptr, "deltay", 0); /* apply movement, then we're done */ view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } static void VIEW2D_OT_scroll_right(wmOperatorType *ot) { /* identifiers */ ot->name = "Scroll Right"; ot->description = "Scroll the view right"; ot->idname = "VIEW2D_OT_scroll_right"; /* api callbacks */ ot->exec = view_scrollright_exec; /* rna - must keep these in sync with the other operators */ RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX); RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX); } /* this operator only needs this single callback, where it calls the view_pan_*() methods */ static int view_scrollleft_exec(bContext *C, wmOperator *op) { v2dViewPanData *vpd; /* initialize default settings (and validate if ok to run) */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } /* also, check if can pan in horizontal axis */ vpd = op->customdata; if (vpd->v2d->keepofs & V2D_LOCKOFS_X) { view_pan_exit(op); return OPERATOR_PASS_THROUGH; } /* set RNA-Props - only movement in negative x-direction */ RNA_int_set(op->ptr, "deltax", -40); RNA_int_set(op->ptr, "deltay", 0); /* apply movement, then we're done */ view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } static void VIEW2D_OT_scroll_left(wmOperatorType *ot) { /* identifiers */ ot->name = "Scroll Left"; ot->description = "Scroll the view left"; ot->idname = "VIEW2D_OT_scroll_left"; /* api callbacks */ ot->exec = view_scrollleft_exec; /* rna - must keep these in sync with the other operators */ RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX); RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX); } /* this operator only needs this single callback, where it calls the view_pan_*() methods */ static int view_scrolldown_exec(bContext *C, wmOperator *op) { v2dViewPanData *vpd; /* initialize default settings (and validate if ok to run) */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } /* also, check if can pan in vertical axis */ vpd = op->customdata; if (vpd->v2d->keepofs & V2D_LOCKOFS_Y) { view_pan_exit(op); return OPERATOR_PASS_THROUGH; } /* set RNA-Props */ RNA_int_set(op->ptr, "deltax", 0); RNA_int_set(op->ptr, "deltay", -40); PropertyRNA *prop = RNA_struct_find_property(op->ptr, "page"); if (RNA_property_is_set(op->ptr, prop) && RNA_property_boolean_get(op->ptr, prop)) { ARegion *region = CTX_wm_region(C); RNA_int_set(op->ptr, "deltay", region->v2d.mask.ymin - region->v2d.mask.ymax); } /* apply movement, then we're done */ view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } static void VIEW2D_OT_scroll_down(wmOperatorType *ot) { /* identifiers */ ot->name = "Scroll Down"; ot->description = "Scroll the view down"; ot->idname = "VIEW2D_OT_scroll_down"; /* api callbacks */ ot->exec = view_scrolldown_exec; /* rna - must keep these in sync with the other operators */ RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX); RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX); RNA_def_boolean(ot->srna, "page", 0, "Page", "Scroll down one page"); } /* this operator only needs this single callback, where it calls the view_pan_*() methods */ static int view_scrollup_exec(bContext *C, wmOperator *op) { v2dViewPanData *vpd; /* initialize default settings (and validate if ok to run) */ if (!view_pan_init(C, op)) { return OPERATOR_PASS_THROUGH; } /* also, check if can pan in vertical axis */ vpd = op->customdata; if (vpd->v2d->keepofs & V2D_LOCKOFS_Y) { view_pan_exit(op); return OPERATOR_PASS_THROUGH; } /* set RNA-Props */ RNA_int_set(op->ptr, "deltax", 0); RNA_int_set(op->ptr, "deltay", 40); PropertyRNA *prop = RNA_struct_find_property(op->ptr, "page"); if (RNA_property_is_set(op->ptr, prop) && RNA_property_boolean_get(op->ptr, prop)) { ARegion *region = CTX_wm_region(C); RNA_int_set(op->ptr, "deltay", BLI_rcti_size_y(®ion->v2d.mask)); } /* apply movement, then we're done */ view_pan_apply(C, op); view_pan_exit(op); return OPERATOR_FINISHED; } static void VIEW2D_OT_scroll_up(wmOperatorType *ot) { /* identifiers */ ot->name = "Scroll Up"; ot->description = "Scroll the view up"; ot->idname = "VIEW2D_OT_scroll_up"; /* api callbacks */ ot->exec = view_scrollup_exec; /* rna - must keep these in sync with the other operators */ RNA_def_int(ot->srna, "deltax", 0, INT_MIN, INT_MAX, "Delta X", "", INT_MIN, INT_MAX); RNA_def_int(ot->srna, "deltay", 0, INT_MIN, INT_MAX, "Delta Y", "", INT_MIN, INT_MAX); RNA_def_boolean(ot->srna, "page", 0, "Page", "Scroll up one page"); } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Zoom Shared Utilities * \{ */ /** * This group of operators come in several forms: * -# Scroll-wheel 'steps' - rolling mouse-wheel by one step zooms view by predefined amount. * -# Scroll-wheel 'steps' + alt + ctrl/shift - zooms view on one axis only (ctrl=x, shift=y). * XXX this could be implemented... * -# Pad +/- Keys - pressing each key moves the zooms the view by a predefined amount. * * In order to make sure this works, each operator must define the following RNA-Operator Props: * * - zoomfacx, zoomfacy - These two zoom factors allow for non-uniform scaling. * It is safe to scale by 0, as these factors are used to determine. * amount to enlarge 'cur' by. */ /** * Temporary custom-data for operator. */ typedef struct v2dViewZoomData { View2D *v2d; /* view2d we're operating in */ ARegion *region; /* needed for continuous zoom */ wmTimer *timer; double timer_lastdraw; int lastx, lasty; /* previous x/y values of mouse in window */ int invoke_event; /* event type that invoked, for modal exits */ float dx, dy; /* running tally of previous delta values (for obtaining final zoom) */ float mx_2d, my_2d; /* initial mouse location in v2d coords */ } v2dViewZoomData; /** * Clamp by convention rather then locking flags, * for ndof and +/- keys */ static void view_zoom_axis_lock_defaults(bContext *C, bool r_do_zoom_xy[2]) { ScrArea *area = CTX_wm_area(C); r_do_zoom_xy[0] = true; r_do_zoom_xy[1] = true; /* default not to zoom the sequencer vertically */ if (area && area->spacetype == SPACE_SEQ) { ARegion *region = CTX_wm_region(C); if (region && region->regiontype == RGN_TYPE_WINDOW) { r_do_zoom_xy[1] = false; } } } /* initialize panning customdata */ static int view_zoomdrag_init(bContext *C, wmOperator *op) { ARegion *region = CTX_wm_region(C); v2dViewZoomData *vzd; View2D *v2d; /* regions now have v2d-data by default, so check for region */ if (region == NULL) { return 0; } v2d = ®ion->v2d; /* check that 2d-view is zoomable */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) && (v2d->keepzoom & V2D_LOCKZOOM_Y)) { return 0; } /* set custom-data for operator */ vzd = MEM_callocN(sizeof(v2dViewZoomData), "v2dViewZoomData"); op->customdata = vzd; /* set pointers to owners */ vzd->v2d = v2d; vzd->region = region; return 1; } /* check if step-zoom can be applied */ static bool view_zoom_poll(bContext *C) { ARegion *region = CTX_wm_region(C); View2D *v2d; /* check if there's a region in context to work with */ if (region == NULL) { return false; } /* Do not show that in 3DView context. */ if (CTX_wm_region_view3d(C)) { return false; } v2d = ®ion->v2d; /* check that 2d-view is zoomable */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) && (v2d->keepzoom & V2D_LOCKZOOM_Y)) { return false; } /* view is zoomable */ return true; } /* apply transform to view (i.e. adjust 'cur' rect) */ static void view_zoomstep_apply_ex( bContext *C, v2dViewZoomData *vzd, const bool zoom_to_pos, const float facx, const float facy) { ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; const rctf cur_old = v2d->cur; float dx, dy; const int snap_test = ED_region_snap_size_test(region); /* calculate amount to move view by, ensuring symmetry so the * old zoom level is restored after zooming back the same amount */ if (facx >= 0.0f) { dx = BLI_rctf_size_x(&v2d->cur) * facx; dy = BLI_rctf_size_y(&v2d->cur) * facy; } else { dx = (BLI_rctf_size_x(&v2d->cur) / (1.0f + 2.0f * facx)) * facx; dy = (BLI_rctf_size_y(&v2d->cur) / (1.0f + 2.0f * facy)) * facy; } /* only resize view on an axis if change is allowed */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) { if (v2d->keepofs & V2D_LOCKOFS_X) { v2d->cur.xmax -= 2 * dx; } else if (v2d->keepofs & V2D_KEEPOFS_X) { if (v2d->align & V2D_ALIGN_NO_POS_X) { v2d->cur.xmin += 2 * dx; } else { v2d->cur.xmax -= 2 * dx; } } else { v2d->cur.xmin += dx; v2d->cur.xmax -= dx; if (zoom_to_pos) { /* get zoom fac the same way as in * ui_view2d_curRect_validate_resize - better keep in sync! */ const float zoomx = (float)(BLI_rcti_size_x(&v2d->mask) + 1) / BLI_rctf_size_x(&v2d->cur); /* only move view to mouse if zoom fac is inside minzoom/maxzoom */ if (((v2d->keepzoom & V2D_LIMITZOOM) == 0) || IN_RANGE_INCL(zoomx, v2d->minzoom, v2d->maxzoom)) { float mval_fac = (vzd->mx_2d - cur_old.xmin) / BLI_rctf_size_x(&cur_old); float mval_faci = 1.0f - mval_fac; float ofs = (mval_fac * dx) - (mval_faci * dx); v2d->cur.xmin += ofs; v2d->cur.xmax += ofs; } } } } if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) { if (v2d->keepofs & V2D_LOCKOFS_Y) { v2d->cur.ymax -= 2 * dy; } else if (v2d->keepofs & V2D_KEEPOFS_Y) { if (v2d->align & V2D_ALIGN_NO_POS_Y) { v2d->cur.ymin += 2 * dy; } else { v2d->cur.ymax -= 2 * dy; } } else { v2d->cur.ymin += dy; v2d->cur.ymax -= dy; if (zoom_to_pos) { /* get zoom fac the same way as in * ui_view2d_curRect_validate_resize - better keep in sync! */ const float zoomy = (float)(BLI_rcti_size_y(&v2d->mask) + 1) / BLI_rctf_size_y(&v2d->cur); /* only move view to mouse if zoom fac is inside minzoom/maxzoom */ if (((v2d->keepzoom & V2D_LIMITZOOM) == 0) || IN_RANGE_INCL(zoomy, v2d->minzoom, v2d->maxzoom)) { float mval_fac = (vzd->my_2d - cur_old.ymin) / BLI_rctf_size_y(&cur_old); float mval_faci = 1.0f - mval_fac; float ofs = (mval_fac * dy) - (mval_faci * dy); v2d->cur.ymin += ofs; v2d->cur.ymax += ofs; } } } } /* validate that view is in valid configuration after this operation */ UI_view2d_curRect_validate(v2d); if (ED_region_snap_size_apply(region, snap_test)) { ScrArea *area = CTX_wm_area(C); ED_area_tag_redraw(area); WM_event_add_notifier(C, NC_SCREEN | NA_EDITED, NULL); } /* request updates to be done... */ ED_region_tag_redraw_no_rebuild(vzd->region); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); } static void view_zoomstep_apply(bContext *C, wmOperator *op) { v2dViewZoomData *vzd = op->customdata; const bool zoom_to_pos = U.uiflag & USER_ZOOM_TO_MOUSEPOS; view_zoomstep_apply_ex( C, vzd, zoom_to_pos, RNA_float_get(op->ptr, "zoomfacx"), RNA_float_get(op->ptr, "zoomfacy")); } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Zoom Operator (single step) * \{ */ /* cleanup temp customdata */ static void view_zoomstep_exit(wmOperator *op) { UI_view2d_zoom_cache_reset(); if (op->customdata) { MEM_freeN(op->customdata); op->customdata = NULL; } } /* this operator only needs this single callback, where it calls the view_zoom_*() methods */ static int view_zoomin_exec(bContext *C, wmOperator *op) { bool do_zoom_xy[2]; /* check that there's an active region, as View2D data resides there */ if (!view_zoom_poll(C)) { return OPERATOR_PASS_THROUGH; } view_zoom_axis_lock_defaults(C, do_zoom_xy); /* set RNA-Props - zooming in by uniform factor */ RNA_float_set(op->ptr, "zoomfacx", do_zoom_xy[0] ? 0.0375f : 0.0f); RNA_float_set(op->ptr, "zoomfacy", do_zoom_xy[1] ? 0.0375f : 0.0f); /* apply movement, then we're done */ view_zoomstep_apply(C, op); view_zoomstep_exit(op); return OPERATOR_FINISHED; } static int view_zoomin_invoke(bContext *C, wmOperator *op, const wmEvent *event) { v2dViewZoomData *vzd; if (!view_zoomdrag_init(C, op)) { return OPERATOR_PASS_THROUGH; } vzd = op->customdata; if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) { ARegion *region = CTX_wm_region(C); /* store initial mouse position (in view space) */ UI_view2d_region_to_view( ®ion->v2d, event->mval[0], event->mval[1], &vzd->mx_2d, &vzd->my_2d); } return view_zoomin_exec(C, op); } static void VIEW2D_OT_zoom_in(wmOperatorType *ot) { PropertyRNA *prop; /* identifiers */ ot->name = "Zoom In"; ot->description = "Zoom in the view"; ot->idname = "VIEW2D_OT_zoom_in"; /* api callbacks */ ot->invoke = view_zoomin_invoke; ot->exec = view_zoomin_exec; // XXX, needs view_zoomdrag_init called first. ot->poll = view_zoom_poll; /* rna - must keep these in sync with the other operators */ prop = RNA_def_float( ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); prop = RNA_def_float( ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); } /* this operator only needs this single callback, where it calls the view_zoom_*() methods */ static int view_zoomout_exec(bContext *C, wmOperator *op) { bool do_zoom_xy[2]; /* check that there's an active region, as View2D data resides there */ if (!view_zoom_poll(C)) { return OPERATOR_PASS_THROUGH; } view_zoom_axis_lock_defaults(C, do_zoom_xy); /* set RNA-Props - zooming in by uniform factor */ RNA_float_set(op->ptr, "zoomfacx", do_zoom_xy[0] ? -0.0375f : 0.0f); RNA_float_set(op->ptr, "zoomfacy", do_zoom_xy[1] ? -0.0375f : 0.0f); /* apply movement, then we're done */ view_zoomstep_apply(C, op); view_zoomstep_exit(op); return OPERATOR_FINISHED; } static int view_zoomout_invoke(bContext *C, wmOperator *op, const wmEvent *event) { v2dViewZoomData *vzd; if (!view_zoomdrag_init(C, op)) { return OPERATOR_PASS_THROUGH; } vzd = op->customdata; if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) { ARegion *region = CTX_wm_region(C); /* store initial mouse position (in view space) */ UI_view2d_region_to_view( ®ion->v2d, event->mval[0], event->mval[1], &vzd->mx_2d, &vzd->my_2d); } return view_zoomout_exec(C, op); } static void VIEW2D_OT_zoom_out(wmOperatorType *ot) { PropertyRNA *prop; /* identifiers */ ot->name = "Zoom Out"; ot->description = "Zoom out the view"; ot->idname = "VIEW2D_OT_zoom_out"; /* api callbacks */ ot->invoke = view_zoomout_invoke; // ot->exec = view_zoomout_exec; // XXX, needs view_zoomdrag_init called first. ot->poll = view_zoom_poll; /* rna - must keep these in sync with the other operators */ prop = RNA_def_float( ot->srna, "zoomfacx", 0, -FLT_MAX, FLT_MAX, "Zoom Factor X", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); prop = RNA_def_float( ot->srna, "zoomfacy", 0, -FLT_MAX, FLT_MAX, "Zoom Factor Y", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Zoom Operator (modal drag-zoom) * \{ */ /** * MMB Drag - allows non-uniform scaling by dragging mouse * * In order to make sure this works, each operator must define the following RNA-Operator Props: * - `deltax, deltay` - amounts to add to each side of the 'cur' rect */ /* apply transform to view (i.e. adjust 'cur' rect) */ static void view_zoomdrag_apply(bContext *C, wmOperator *op) { v2dViewZoomData *vzd = op->customdata; View2D *v2d = vzd->v2d; float dx, dy; const int snap_test = ED_region_snap_size_test(vzd->region); const bool use_cursor_init = RNA_boolean_get(op->ptr, "use_cursor_init"); const bool zoom_to_pos = use_cursor_init && (U.uiflag & USER_ZOOM_TO_MOUSEPOS); /* get amount to move view by */ dx = RNA_float_get(op->ptr, "deltax") / U.dpi_fac; dy = RNA_float_get(op->ptr, "deltay") / U.dpi_fac; if (U.uiflag & USER_ZOOM_INVERT) { dx *= -1; dy *= -1; } /* continuous zoom shouldn't move that fast... */ if (U.viewzoom == USER_ZOOM_CONT) { // XXX store this setting as RNA prop? double time = PIL_check_seconds_timer(); float time_step = (float)(time - vzd->timer_lastdraw); dx *= time_step * 0.5f; dy *= time_step * 0.5f; vzd->timer_lastdraw = time; } /* only move view on an axis if change is allowed */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) { if (v2d->keepofs & V2D_LOCKOFS_X) { v2d->cur.xmax -= 2 * dx; } else { if (zoom_to_pos) { float mval_fac = (vzd->mx_2d - v2d->cur.xmin) / BLI_rctf_size_x(&v2d->cur); float mval_faci = 1.0f - mval_fac; float ofs = (mval_fac * dx) - (mval_faci * dx); v2d->cur.xmin += ofs + dx; v2d->cur.xmax += ofs - dx; } else { v2d->cur.xmin += dx; v2d->cur.xmax -= dx; } } } if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) { if (v2d->keepofs & V2D_LOCKOFS_Y) { v2d->cur.ymax -= 2 * dy; } else { if (zoom_to_pos) { float mval_fac = (vzd->my_2d - v2d->cur.ymin) / BLI_rctf_size_y(&v2d->cur); float mval_faci = 1.0f - mval_fac; float ofs = (mval_fac * dy) - (mval_faci * dy); v2d->cur.ymin += ofs + dy; v2d->cur.ymax += ofs - dy; } else { v2d->cur.ymin += dy; v2d->cur.ymax -= dy; } } } /* validate that view is in valid configuration after this operation */ UI_view2d_curRect_validate(v2d); if (ED_region_snap_size_apply(vzd->region, snap_test)) { ScrArea *area = CTX_wm_area(C); ED_area_tag_redraw(area); WM_event_add_notifier(C, NC_SCREEN | NA_EDITED, NULL); } /* request updates to be done... */ ED_region_tag_redraw_no_rebuild(vzd->region); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); } /* cleanup temp customdata */ static void view_zoomdrag_exit(bContext *C, wmOperator *op) { UI_view2d_zoom_cache_reset(); if (op->customdata) { v2dViewZoomData *vzd = op->customdata; if (vzd->timer) { WM_event_remove_timer(CTX_wm_manager(C), CTX_wm_window(C), vzd->timer); } MEM_freeN(op->customdata); op->customdata = NULL; } } static void view_zoomdrag_cancel(bContext *C, wmOperator *op) { view_zoomdrag_exit(C, op); } /* for 'redo' only, with no user input */ static int view_zoomdrag_exec(bContext *C, wmOperator *op) { if (!view_zoomdrag_init(C, op)) { return OPERATOR_PASS_THROUGH; } view_zoomdrag_apply(C, op); view_zoomdrag_exit(C, op); return OPERATOR_FINISHED; } /* set up modal operator and relevant settings */ static int view_zoomdrag_invoke(bContext *C, wmOperator *op, const wmEvent *event) { wmWindow *window = CTX_wm_window(C); v2dViewZoomData *vzd; View2D *v2d; /* set up customdata */ if (!view_zoomdrag_init(C, op)) { return OPERATOR_PASS_THROUGH; } vzd = op->customdata; v2d = vzd->v2d; if (event->type == MOUSEZOOM || event->type == MOUSEPAN) { float dx, dy, fac; vzd->lastx = event->prevx; vzd->lasty = event->prevy; /* As we have only 1D information (magnify value), feed both axes * with magnify information that is stored in x axis */ fac = 0.01f * (event->prevx - event->x); dx = fac * BLI_rctf_size_x(&v2d->cur) / 10.0f; if (event->type == MOUSEPAN) { fac = 0.01f * (event->prevy - event->y); } dy = fac * BLI_rctf_size_y(&v2d->cur) / 10.0f; /* support trackpad zoom to always zoom entirely - the v2d code uses portrait or * landscape exceptions */ if (v2d->keepzoom & V2D_KEEPASPECT) { if (fabsf(dx) > fabsf(dy)) { dy = dx; } else { dx = dy; } } RNA_float_set(op->ptr, "deltax", dx); RNA_float_set(op->ptr, "deltay", dy); view_zoomdrag_apply(C, op); view_zoomdrag_exit(C, op); return OPERATOR_FINISHED; } /* set initial settings */ vzd->lastx = event->x; vzd->lasty = event->y; RNA_float_set(op->ptr, "deltax", 0); RNA_float_set(op->ptr, "deltay", 0); /* for modal exit test */ vzd->invoke_event = event->type; if (U.uiflag & USER_ZOOM_TO_MOUSEPOS) { ARegion *region = CTX_wm_region(C); /* store initial mouse position (in view space) */ UI_view2d_region_to_view( ®ion->v2d, event->mval[0], event->mval[1], &vzd->mx_2d, &vzd->my_2d); } if (v2d->keepofs & V2D_LOCKOFS_X) { WM_cursor_modal_set(window, WM_CURSOR_NS_SCROLL); } else if (v2d->keepofs & V2D_LOCKOFS_Y) { WM_cursor_modal_set(window, WM_CURSOR_EW_SCROLL); } else { WM_cursor_modal_set(window, WM_CURSOR_NSEW_SCROLL); } /* add temp handler */ WM_event_add_modal_handler(C, op); if (U.viewzoom == USER_ZOOM_CONT) { /* needs a timer to continue redrawing */ vzd->timer = WM_event_add_timer(CTX_wm_manager(C), window, TIMER, 0.01f); vzd->timer_lastdraw = PIL_check_seconds_timer(); } return OPERATOR_RUNNING_MODAL; } /* handle user input - calculations of mouse-movement need to be done here, * not in the apply callback! */ static int view_zoomdrag_modal(bContext *C, wmOperator *op, const wmEvent *event) { v2dViewZoomData *vzd = op->customdata; View2D *v2d = vzd->v2d; /* execute the events */ if (event->type == TIMER && event->customdata == vzd->timer) { view_zoomdrag_apply(C, op); } else if (event->type == MOUSEMOVE) { float dx, dy; float zoomfac = 0.01f; /* some view2d's (graph) don't have min/max zoom, or extreme ones */ if (v2d->maxzoom > 0.0f) { zoomfac = clamp_f(0.001f * v2d->maxzoom, 0.001f, 0.01f); } /* calculate new delta transform, based on zooming mode */ if (U.viewzoom == USER_ZOOM_SCALE) { /* 'scale' zooming */ float dist; float len_old[2]; float len_new[2]; /* x-axis transform */ dist = BLI_rcti_size_x(&v2d->mask) / 2.0f; len_old[0] = fabsf(vzd->lastx - vzd->region->winrct.xmin - dist); len_new[0] = fabsf(event->x - vzd->region->winrct.xmin - dist); len_old[0] *= zoomfac * BLI_rctf_size_x(&v2d->cur); len_new[0] *= zoomfac * BLI_rctf_size_x(&v2d->cur); /* y-axis transform */ dist = BLI_rcti_size_y(&v2d->mask) / 2.0f; len_old[1] = fabsf(vzd->lasty - vzd->region->winrct.ymin - dist); len_new[1] = fabsf(event->y - vzd->region->winrct.ymin - dist); len_old[1] *= zoomfac * BLI_rctf_size_y(&v2d->cur); len_new[1] *= zoomfac * BLI_rctf_size_y(&v2d->cur); /* Calculate distance */ if (v2d->keepzoom & V2D_KEEPASPECT) { dist = len_v2(len_new) - len_v2(len_old); dx = dy = dist; } else { dx = len_new[0] - len_old[0]; dy = len_new[1] - len_old[1]; } } else { /* 'continuous' or 'dolly' */ float fac; /* x-axis transform */ fac = zoomfac * (event->x - vzd->lastx); dx = fac * BLI_rctf_size_x(&v2d->cur); /* y-axis transform */ fac = zoomfac * (event->y - vzd->lasty); dy = fac * BLI_rctf_size_y(&v2d->cur); /* Only respect user setting zoom axis if the view does not have any zoom restrictions * any will be scaled uniformly */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0 && (v2d->keepzoom & V2D_LOCKZOOM_Y) == 0 && (v2d->keepzoom & V2D_KEEPASPECT)) { if (U.uiflag & USER_ZOOM_HORIZ) { dy = 0; } else { dx = 0; } } } /* support zoom to always zoom entirely - the v2d code uses portrait or * landscape exceptions */ if (v2d->keepzoom & V2D_KEEPASPECT) { if (fabsf(dx) > fabsf(dy)) { dy = dx; } else { dx = dy; } } /* set transform amount, and add current deltas to stored total delta (for redo) */ RNA_float_set(op->ptr, "deltax", dx); RNA_float_set(op->ptr, "deltay", dy); vzd->dx += dx; vzd->dy += dy; /* Store mouse coordinates for next time, if not doing continuous zoom: * - Continuous zoom only depends on distance of mouse * to starting point to determine rate of change. */ if (U.viewzoom != USER_ZOOM_CONT) { // XXX store this setting as RNA prop? vzd->lastx = event->x; vzd->lasty = event->y; } /* apply zooming */ view_zoomdrag_apply(C, op); } else if (event->type == vzd->invoke_event || event->type == EVT_ESCKEY) { if (event->val == KM_RELEASE) { /* for redo, store the overall deltas - need to respect zoom-locks here... */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) { RNA_float_set(op->ptr, "deltax", vzd->dx); } else { RNA_float_set(op->ptr, "deltax", 0); } if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) { RNA_float_set(op->ptr, "deltay", vzd->dy); } else { RNA_float_set(op->ptr, "deltay", 0); } /* free customdata */ view_zoomdrag_exit(C, op); WM_cursor_modal_restore(CTX_wm_window(C)); return OPERATOR_FINISHED; } } return OPERATOR_RUNNING_MODAL; } static void VIEW2D_OT_zoom(wmOperatorType *ot) { PropertyRNA *prop; /* identifiers */ ot->name = "Zoom 2D View"; ot->description = "Zoom in/out the view"; ot->idname = "VIEW2D_OT_zoom"; /* api callbacks */ ot->exec = view_zoomdrag_exec; ot->invoke = view_zoomdrag_invoke; ot->modal = view_zoomdrag_modal; ot->cancel = view_zoomdrag_cancel; ot->poll = view_zoom_poll; /* operator is repeatable */ ot->flag = OPTYPE_BLOCKING | OPTYPE_GRAB_CURSOR_XY; /* rna - must keep these in sync with the other operators */ prop = RNA_def_float(ot->srna, "deltax", 0, -FLT_MAX, FLT_MAX, "Delta X", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); prop = RNA_def_float(ot->srna, "deltay", 0, -FLT_MAX, FLT_MAX, "Delta Y", "", -FLT_MAX, FLT_MAX); RNA_def_property_flag(prop, PROP_HIDDEN); WM_operator_properties_use_cursor_init(ot); } /** \} */ /* -------------------------------------------------------------------- */ /** \name Border Zoom Operator * \{ */ /** * The user defines a rect using standard box select tools, and we use this rect to * define the new zoom-level of the view in the following ways: * * -# LEFTMOUSE - zoom in to view * -# RIGHTMOUSE - zoom out of view * * Currently, these key mappings are hardcoded, but it shouldn't be too important to * have custom keymappings for this... */ static int view_borderzoom_exec(bContext *C, wmOperator *op) { ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; rctf rect; rctf cur_new = v2d->cur; const int smooth_viewtx = WM_operator_smooth_viewtx_get(op); /* convert coordinates of rect to 'tot' rect coordinates */ WM_operator_properties_border_to_rctf(op, &rect); UI_view2d_region_to_view_rctf(v2d, &rect, &rect); /* check if zooming in/out view */ const bool zoom_in = !RNA_boolean_get(op->ptr, "zoom_out"); if (zoom_in) { /* zoom in: * - 'cur' rect will be defined by the coordinates of the border region * - just set the 'cur' rect to have the same coordinates as the border region * if zoom is allowed to be changed */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) { cur_new.xmin = rect.xmin; cur_new.xmax = rect.xmax; } if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) { cur_new.ymin = rect.ymin; cur_new.ymax = rect.ymax; } } else { /* zoom out: * - the current 'cur' rect coordinates are going to end up where the 'rect' ones are, * but the 'cur' rect coordinates will need to be adjusted to take in more of the view * - calculate zoom factor, and adjust using center-point */ float zoom, center, size; /* TODO: is this zoom factor calculation valid? * It seems to produce same results every time... */ if ((v2d->keepzoom & V2D_LOCKZOOM_X) == 0) { size = BLI_rctf_size_x(&cur_new); zoom = size / BLI_rctf_size_x(&rect); center = BLI_rctf_cent_x(&cur_new); cur_new.xmin = center - (size * zoom); cur_new.xmax = center + (size * zoom); } if ((v2d->keepzoom & V2D_LOCKZOOM_Y) == 0) { size = BLI_rctf_size_y(&cur_new); zoom = size / BLI_rctf_size_y(&rect); center = BLI_rctf_cent_y(&cur_new); cur_new.ymin = center - (size * zoom); cur_new.ymax = center + (size * zoom); } } UI_view2d_smooth_view(C, region, &cur_new, smooth_viewtx); return OPERATOR_FINISHED; } static void VIEW2D_OT_zoom_border(wmOperatorType *ot) { /* identifiers */ ot->name = "Zoom to Border"; ot->description = "Zoom in the view to the nearest item contained in the border"; ot->idname = "VIEW2D_OT_zoom_border"; /* api callbacks */ ot->invoke = WM_gesture_box_invoke; ot->exec = view_borderzoom_exec; ot->modal = WM_gesture_box_modal; ot->cancel = WM_gesture_box_cancel; ot->poll = view_zoom_poll; /* rna */ WM_operator_properties_gesture_box_zoom(ot); } /** \} */ /* -------------------------------------------------------------------- */ /** \name NDOF Pan/Zoom Operator * \{ */ #ifdef WITH_INPUT_NDOF static int view2d_ndof_invoke(bContext *C, wmOperator *op, const wmEvent *event) { if (event->type != NDOF_MOTION) { return OPERATOR_CANCELLED; } const wmNDOFMotionData *ndof = event->customdata; /* tune these until it feels right */ const float zoom_sensitivity = 0.5f; const float speed = 10.0f; /* match view3d ortho */ const bool has_translate = (ndof->tvec[0] && ndof->tvec[1]) && view_pan_poll(C); const bool has_zoom = (ndof->tvec[2] != 0.0f) && view_zoom_poll(C); if (has_translate) { if (view_pan_init(C, op)) { v2dViewPanData *vpd; float pan_vec[3]; WM_event_ndof_pan_get(ndof, pan_vec, false); pan_vec[0] *= speed; pan_vec[1] *= speed; vpd = op->customdata; view_pan_apply_ex(C, vpd, pan_vec[0], pan_vec[1]); view_pan_exit(op); } } if (has_zoom) { if (view_zoomdrag_init(C, op)) { v2dViewZoomData *vzd; float zoom_factor = zoom_sensitivity * ndof->dt * -ndof->tvec[2]; bool do_zoom_xy[2]; if (U.ndof_flag & NDOF_ZOOM_INVERT) { zoom_factor = -zoom_factor; } view_zoom_axis_lock_defaults(C, do_zoom_xy); vzd = op->customdata; view_zoomstep_apply_ex( C, vzd, false, do_zoom_xy[0] ? zoom_factor : 0.0f, do_zoom_xy[1] ? zoom_factor : 0.0f); view_zoomstep_exit(op); } } return OPERATOR_FINISHED; } static void VIEW2D_OT_ndof(wmOperatorType *ot) { /* identifiers */ ot->name = "NDOF Pan/Zoom"; ot->idname = "VIEW2D_OT_ndof"; ot->description = "Use a 3D mouse device to pan/zoom the view"; /* api callbacks */ ot->invoke = view2d_ndof_invoke; ot->poll = view2d_poll; /* flags */ ot->flag = OPTYPE_LOCK_BYPASS; } #endif /* WITH_INPUT_NDOF */ /** \} */ /* -------------------------------------------------------------------- */ /** \name Smooth View Operator * \{ */ struct SmoothView2DStore { rctf orig_cur, new_cur; double time_allowed; }; /** * function to get a factor out of a rectangle * * note: this doesn't always work as well as it might because the target size * may not be reached because of clamping the desired rect, we _could_ * attempt to clamp the rect before working out the zoom factor but its * not really worthwhile for the few cases this happens. */ static float smooth_view_rect_to_fac(const rctf *rect_a, const rctf *rect_b) { const float size_a[2] = {BLI_rctf_size_x(rect_a), BLI_rctf_size_y(rect_a)}; const float size_b[2] = {BLI_rctf_size_x(rect_b), BLI_rctf_size_y(rect_b)}; const float cent_a[2] = {BLI_rctf_cent_x(rect_a), BLI_rctf_cent_y(rect_a)}; const float cent_b[2] = {BLI_rctf_cent_x(rect_b), BLI_rctf_cent_y(rect_b)}; float fac_max = 0.0f; float tfac; int i; for (i = 0; i < 2; i++) { /* axis translation normalized to scale */ tfac = fabsf(cent_a[i] - cent_b[i]) / min_ff(size_a[i], size_b[i]); fac_max = max_ff(fac_max, tfac); if (fac_max >= 1.0f) { break; } /* axis scale difference, x2 so doubling or half gives 1.0f */ tfac = (1.0f - (min_ff(size_a[i], size_b[i]) / max_ff(size_a[i], size_b[i]))) * 2.0f; fac_max = max_ff(fac_max, tfac); if (fac_max >= 1.0f) { break; } } return min_ff(fac_max, 1.0f); } /* will start timer if appropriate */ /* the arguments are the desired situation */ void UI_view2d_smooth_view(bContext *C, ARegion *region, const rctf *cur, const int smooth_viewtx) { wmWindowManager *wm = CTX_wm_manager(C); wmWindow *win = CTX_wm_window(C); View2D *v2d = ®ion->v2d; struct SmoothView2DStore sms = {{0}}; bool ok = false; float fac = 1.0f; /* initialize sms */ sms.new_cur = v2d->cur; /* store the options we want to end with */ if (cur) { sms.new_cur = *cur; } if (cur) { fac = smooth_view_rect_to_fac(&v2d->cur, cur); } if (smooth_viewtx && fac > FLT_EPSILON) { bool changed = false; if (BLI_rctf_compare(&sms.new_cur, &v2d->cur, FLT_EPSILON) == false) { changed = true; } /* The new view is different from the old one * so animate the view */ if (changed) { sms.orig_cur = v2d->cur; sms.time_allowed = (double)smooth_viewtx / 1000.0; /* scale the time allowed the change in view */ sms.time_allowed *= (double)fac; /* keep track of running timer! */ if (v2d->sms == NULL) { v2d->sms = MEM_mallocN(sizeof(struct SmoothView2DStore), "smoothview v2d"); } *v2d->sms = sms; if (v2d->smooth_timer) { WM_event_remove_timer(wm, win, v2d->smooth_timer); } /* TIMER1 is hardcoded in keymap */ /* max 30 frs/sec */ v2d->smooth_timer = WM_event_add_timer(wm, win, TIMER1, 1.0 / 100.0); ok = true; } } /* if we get here nothing happens */ if (ok == false) { v2d->cur = sms.new_cur; UI_view2d_curRect_validate(v2d); ED_region_tag_redraw_no_rebuild(region); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); } } /* only meant for timer usage */ static int view2d_smoothview_invoke(bContext *C, wmOperator *UNUSED(op), const wmEvent *event) { wmWindow *win = CTX_wm_window(C); ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; struct SmoothView2DStore *sms = v2d->sms; float step; /* escape if not our timer */ if (v2d->smooth_timer == NULL || v2d->smooth_timer != event->customdata) { return OPERATOR_PASS_THROUGH; } if (sms->time_allowed != 0.0) { step = (float)((v2d->smooth_timer->duration) / sms->time_allowed); } else { step = 1.0f; } /* end timer */ if (step >= 1.0f) { v2d->cur = sms->new_cur; MEM_freeN(v2d->sms); v2d->sms = NULL; WM_event_remove_timer(CTX_wm_manager(C), win, v2d->smooth_timer); v2d->smooth_timer = NULL; /* Event handling won't know if a UI item has been moved under the pointer. */ WM_event_add_mousemove(win); } else { /* ease in/out */ step = (3.0f * step * step - 2.0f * step * step * step); BLI_rctf_interp(&v2d->cur, &sms->orig_cur, &sms->new_cur, step); } UI_view2d_curRect_validate(v2d); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); ED_region_tag_redraw_no_rebuild(region); if (v2d->sms == NULL) { UI_view2d_zoom_cache_reset(); } return OPERATOR_FINISHED; } static void VIEW2D_OT_smoothview(wmOperatorType *ot) { /* identifiers */ ot->name = "Smooth View 2D"; ot->idname = "VIEW2D_OT_smoothview"; /* api callbacks */ ot->invoke = view2d_smoothview_invoke; ot->poll = view2d_poll; /* flags */ ot->flag = OPTYPE_INTERNAL; /* rna */ WM_operator_properties_gesture_box(ot); } /** \} */ /* -------------------------------------------------------------------- */ /** \name Scroll Bar Move Operator * \{ */ /** * Scrollers should behave in the following ways, when clicked on with LMB (and dragged): * -# 'Handles' on end of 'bubble' - when the axis that the scroller represents is zoomable, * enlarge 'cur' rect on the relevant side. * -# 'Bubble'/'bar' - just drag, and bar should move with mouse (view pans opposite). * * In order to make sure this works, each operator must define the following RNA-Operator Props: * - `deltax, deltay` - define how much to move view by (relative to zoom-correction factor) */ /* customdata for scroller-invoke data */ typedef struct v2dScrollerMove { /** View2D data that this operation affects */ View2D *v2d; /** region that the scroller is in */ ARegion *region; /** scroller that mouse is in ('h' or 'v') */ char scroller; /* XXX find some way to provide visual feedback of this (active color?) */ /** -1 is min zoomer, 0 is bar, 1 is max zoomer */ short zone; /** view adjustment factor, based on size of region */ float fac; /** for pixel rounding (avoid visible UI jitter) */ float fac_round; /** amount moved by mouse on axis of interest */ float delta; /** width of the scrollbar itself, used for page up/down clicks */ float scrollbarwidth; /** initial location of scrollbar x/y, mouse relative */ int scrollbar_orig; /** previous mouse coordinates (in screen coordinates) for determining movement */ int lastx, lasty; } v2dScrollerMove; /** * #View2DScrollers is typedef'd in UI_view2d.h * This is a CUT DOWN VERSION of the 'real' version, which is defined in view2d.c, * as we only need focus bubble info. * * \warning: The start of this struct must not change, * so that it stays in sync with the 'real' version. * For now, we don't need to have a separate (internal) header for structs like this... */ struct View2DScrollers { /* focus bubbles */ int vert_min, vert_max; /* vertical scrollbar */ int hor_min, hor_max; /* horizontal scrollbar */ /* These values are written into, even if we don't use them. */ rcti _hor, _vert; }; /* quick enum for vsm->zone (scroller handles) */ enum { SCROLLHANDLE_MIN = -1, SCROLLHANDLE_BAR, SCROLLHANDLE_MAX, SCROLLHANDLE_MIN_OUTSIDE, SCROLLHANDLE_MAX_OUTSIDE, } /*eV2DScrollerHandle_Zone*/; /** * Check if mouse is within scroller handle. * * \param mouse: relevant mouse coordinate in region space. * \param sc_min, sc_max: extents of scroller 'groove' (potential available space for scroller). * \param sh_min, sh_max: positions of scrollbar handles. */ static short mouse_in_scroller_handle(int mouse, int sc_min, int sc_max, int sh_min, int sh_max) { /* firstly, check if * - 'bubble' fills entire scroller * - 'bubble' completely out of view on either side */ bool in_view = true; if (sh_min <= sc_min && sc_max <= sh_max) { in_view = false; } else if (sh_max <= sc_min || sc_max <= sh_min) { in_view = false; } if (!in_view) { return SCROLLHANDLE_BAR; } /* check if mouse is in or past either handle */ /* TODO: check if these extents are still valid or not */ bool in_max = ((mouse >= (sh_max - V2D_SCROLL_HANDLE_SIZE_HOTSPOT)) && (mouse <= (sh_max + V2D_SCROLL_HANDLE_SIZE_HOTSPOT))); bool in_min = ((mouse <= (sh_min + V2D_SCROLL_HANDLE_SIZE_HOTSPOT)) && (mouse >= (sh_min - V2D_SCROLL_HANDLE_SIZE_HOTSPOT))); bool in_bar = ((mouse < (sh_max - V2D_SCROLL_HANDLE_SIZE_HOTSPOT)) && (mouse > (sh_min + V2D_SCROLL_HANDLE_SIZE_HOTSPOT))); bool out_min = mouse < (sh_min - V2D_SCROLL_HANDLE_SIZE_HOTSPOT); bool out_max = mouse > (sh_max + V2D_SCROLL_HANDLE_SIZE_HOTSPOT); if (in_bar) { return SCROLLHANDLE_BAR; } if (in_max) { return SCROLLHANDLE_MAX; } if (in_min) { return SCROLLHANDLE_MIN; } if (out_min) { return SCROLLHANDLE_MIN_OUTSIDE; } if (out_max) { return SCROLLHANDLE_MAX_OUTSIDE; } /* unlikely to happen, though we just cover it in case */ return SCROLLHANDLE_BAR; } static bool scroller_activate_poll(bContext *C) { if (!view2d_poll(C)) { return false; } wmWindow *win = CTX_wm_window(C); ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; wmEvent *event = win->eventstate; /* check if mouse in scrollbars, if they're enabled */ return (UI_view2d_mouse_in_scrollers(region, v2d, event->x, event->y) != 0); } /* initialize customdata for scroller manipulation operator */ static void scroller_activate_init(bContext *C, wmOperator *op, const wmEvent *event, const char in_scroller) { v2dScrollerMove *vsm; View2DScrollers scrollers; ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; rctf tot_cur_union; float mask_size; /* set custom-data for operator */ vsm = MEM_callocN(sizeof(v2dScrollerMove), "v2dScrollerMove"); op->customdata = vsm; /* set general data */ vsm->v2d = v2d; vsm->region = region; vsm->scroller = in_scroller; /* store mouse-coordinates, and convert mouse/screen coordinates to region coordinates */ vsm->lastx = event->x; vsm->lasty = event->y; /* 'zone' depends on where mouse is relative to bubble * - zooming must be allowed on this axis, otherwise, default to pan */ UI_view2d_scrollers_calc(v2d, NULL, &scrollers); /* Use a union of 'cur' & 'tot' in case the current view is far outside 'tot'. In this cases * moving the scroll bars has far too little effect and the view can get stuck T31476. */ tot_cur_union = v2d->tot; BLI_rctf_union(&tot_cur_union, &v2d->cur); if (in_scroller == 'h') { /* horizontal scroller - calculate adjustment factor first */ mask_size = (float)BLI_rcti_size_x(&v2d->hor); vsm->fac = BLI_rctf_size_x(&tot_cur_union) / mask_size; /* pixel rounding */ vsm->fac_round = (BLI_rctf_size_x(&v2d->cur)) / (float)(BLI_rcti_size_x(®ion->winrct) + 1); /* get 'zone' (i.e. which part of scroller is activated) */ vsm->zone = mouse_in_scroller_handle( event->mval[0], v2d->hor.xmin, v2d->hor.xmax, scrollers.hor_min, scrollers.hor_max); if ((v2d->keepzoom & V2D_LOCKZOOM_X) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) { /* default to scroll, as handles not usable */ vsm->zone = SCROLLHANDLE_BAR; } vsm->scrollbarwidth = scrollers.hor_max - scrollers.hor_min; vsm->scrollbar_orig = ((scrollers.hor_max + scrollers.hor_min) / 2) + region->winrct.xmin; } else { /* vertical scroller - calculate adjustment factor first */ mask_size = (float)BLI_rcti_size_y(&v2d->vert); vsm->fac = BLI_rctf_size_y(&tot_cur_union) / mask_size; /* pixel rounding */ vsm->fac_round = (BLI_rctf_size_y(&v2d->cur)) / (float)(BLI_rcti_size_y(®ion->winrct) + 1); /* get 'zone' (i.e. which part of scroller is activated) */ vsm->zone = mouse_in_scroller_handle( event->mval[1], v2d->vert.ymin, v2d->vert.ymax, scrollers.vert_min, scrollers.vert_max); if ((v2d->keepzoom & V2D_LOCKZOOM_Y) && ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) { /* default to scroll, as handles not usable */ vsm->zone = SCROLLHANDLE_BAR; } vsm->scrollbarwidth = scrollers.vert_max - scrollers.vert_min; vsm->scrollbar_orig = ((scrollers.vert_max + scrollers.vert_min) / 2) + region->winrct.ymin; } ED_region_tag_redraw_no_rebuild(region); } /* cleanup temp customdata */ static void scroller_activate_exit(bContext *C, wmOperator *op) { if (op->customdata) { v2dScrollerMove *vsm = op->customdata; vsm->v2d->scroll_ui &= ~(V2D_SCROLL_H_ACTIVE | V2D_SCROLL_V_ACTIVE); MEM_freeN(op->customdata); op->customdata = NULL; ED_region_tag_redraw_no_rebuild(CTX_wm_region(C)); } } static void scroller_activate_cancel(bContext *C, wmOperator *op) { scroller_activate_exit(C, op); } /* apply transform to view (i.e. adjust 'cur' rect) */ static void scroller_activate_apply(bContext *C, wmOperator *op) { v2dScrollerMove *vsm = op->customdata; View2D *v2d = vsm->v2d; float temp; /* calculate amount to move view by */ temp = vsm->fac * vsm->delta; /* round to pixel */ temp = roundf(temp / vsm->fac_round) * vsm->fac_round; /* type of movement */ switch (vsm->zone) { case SCROLLHANDLE_MIN: /* only expand view on axis if zoom is allowed */ if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X)) { v2d->cur.xmin -= temp; } if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y)) { v2d->cur.ymin -= temp; } break; case SCROLLHANDLE_MAX: /* only expand view on axis if zoom is allowed */ if ((vsm->scroller == 'h') && !(v2d->keepzoom & V2D_LOCKZOOM_X)) { v2d->cur.xmax += temp; } if ((vsm->scroller == 'v') && !(v2d->keepzoom & V2D_LOCKZOOM_Y)) { v2d->cur.ymax += temp; } break; case SCROLLHANDLE_MIN_OUTSIDE: case SCROLLHANDLE_MAX_OUTSIDE: case SCROLLHANDLE_BAR: default: /* only move view on an axis if panning is allowed */ if ((vsm->scroller == 'h') && !(v2d->keepofs & V2D_LOCKOFS_X)) { v2d->cur.xmin += temp; v2d->cur.xmax += temp; } if ((vsm->scroller == 'v') && !(v2d->keepofs & V2D_LOCKOFS_Y)) { v2d->cur.ymin += temp; v2d->cur.ymax += temp; } break; } /* validate that view is in valid configuration after this operation */ UI_view2d_curRect_validate(v2d); /* request updates to be done... */ ED_region_tag_redraw_no_rebuild(vsm->region); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); } /** * Handle user input for scrollers - calculations of mouse-movement need to be done here, * not in the apply callback! */ static int scroller_activate_modal(bContext *C, wmOperator *op, const wmEvent *event) { v2dScrollerMove *vsm = op->customdata; /* execute the events */ switch (event->type) { case MOUSEMOVE: { /* calculate new delta transform, then store mouse-coordinates for next-time */ if (ELEM(vsm->zone, SCROLLHANDLE_BAR, SCROLLHANDLE_MAX)) { /* if using bar (i.e. 'panning') or 'max' zoom widget */ switch (vsm->scroller) { case 'h': /* horizontal scroller - so only horizontal movement * ('cur' moves opposite to mouse) */ vsm->delta = (float)(event->x - vsm->lastx); break; case 'v': /* vertical scroller - so only vertical movement * ('cur' moves opposite to mouse) */ vsm->delta = (float)(event->y - vsm->lasty); break; } } else if (vsm->zone == SCROLLHANDLE_MIN) { /* using 'min' zoom widget */ switch (vsm->scroller) { case 'h': /* horizontal scroller - so only horizontal movement * ('cur' moves with mouse) */ vsm->delta = (float)(vsm->lastx - event->x); break; case 'v': /* vertical scroller - so only vertical movement * ('cur' moves with to mouse) */ vsm->delta = (float)(vsm->lasty - event->y); break; } } /* store previous coordinates */ vsm->lastx = event->x; vsm->lasty = event->y; scroller_activate_apply(C, op); break; } case LEFTMOUSE: case MIDDLEMOUSE: if (event->val == KM_RELEASE) { /* single-click was in empty space outside bubble, so scroll by 1 'page' */ if (ELEM(vsm->zone, SCROLLHANDLE_MIN_OUTSIDE, SCROLLHANDLE_MAX_OUTSIDE)) { if (vsm->zone == SCROLLHANDLE_MIN_OUTSIDE) { vsm->delta = -vsm->scrollbarwidth * 0.8f; } else if (vsm->zone == SCROLLHANDLE_MAX_OUTSIDE) { vsm->delta = vsm->scrollbarwidth * 0.8f; } scroller_activate_apply(C, op); scroller_activate_exit(C, op); return OPERATOR_FINISHED; } /* otherwise, end the drag action */ if (vsm->lastx || vsm->lasty) { scroller_activate_exit(C, op); return OPERATOR_FINISHED; } } break; } return OPERATOR_RUNNING_MODAL; } /* a click (or click drag in progress) * should have occurred, so check if it happened in scrollbar */ static int scroller_activate_invoke(bContext *C, wmOperator *op, const wmEvent *event) { ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; /* check if mouse in scrollbars, if they're enabled */ const char in_scroller = UI_view2d_mouse_in_scrollers(region, v2d, event->x, event->y); /* if in a scroller, init customdata then set modal handler which will * catch mouse-down to start doing useful stuff */ if (in_scroller) { v2dScrollerMove *vsm; /* initialize customdata */ scroller_activate_init(C, op, event, in_scroller); vsm = (v2dScrollerMove *)op->customdata; /* support for quick jump to location - gtk and qt do this on linux */ if (event->type == MIDDLEMOUSE) { switch (vsm->scroller) { case 'h': /* horizontal scroller - so only horizontal movement * ('cur' moves opposite to mouse) */ vsm->delta = (float)(event->x - vsm->scrollbar_orig); break; case 'v': /* vertical scroller - so only vertical movement * ('cur' moves opposite to mouse) */ vsm->delta = (float)(event->y - vsm->scrollbar_orig); break; } scroller_activate_apply(C, op); vsm->zone = SCROLLHANDLE_BAR; } /* check if zoom zones are inappropriate (i.e. zoom widgets not shown), so cannot continue * NOTE: see view2d.c for latest conditions, and keep this in sync with that */ if (ELEM(vsm->zone, SCROLLHANDLE_MIN, SCROLLHANDLE_MAX)) { if (((vsm->scroller == 'h') && (v2d->scroll & V2D_SCROLL_HORIZONTAL_HANDLES) == 0) || ((vsm->scroller == 'v') && (v2d->scroll & V2D_SCROLL_VERTICAL_HANDLES) == 0)) { /* switch to bar (i.e. no scaling gets handled) */ vsm->zone = SCROLLHANDLE_BAR; } } /* check if zone is inappropriate (i.e. 'bar' but panning is banned), so cannot continue */ if (vsm->zone == SCROLLHANDLE_BAR) { if (((vsm->scroller == 'h') && (v2d->keepofs & V2D_LOCKOFS_X)) || ((vsm->scroller == 'v') && (v2d->keepofs & V2D_LOCKOFS_Y))) { /* free customdata initialized */ scroller_activate_exit(C, op); /* can't catch this event for ourselves, so let it go to someone else? */ return OPERATOR_PASS_THROUGH; } } /* zone is also inappropriate if scroller is not visible... */ if (((vsm->scroller == 'h') && (v2d->scroll & V2D_SCROLL_HORIZONTAL_FULLR)) || ((vsm->scroller == 'v') && (v2d->scroll & V2D_SCROLL_VERTICAL_FULLR))) { /* free customdata initialized */ scroller_activate_exit(C, op); /* can't catch this event for ourselves, so let it go to someone else? */ /* XXX note: if handlers use mask rect to clip input, input will fail for this case */ return OPERATOR_PASS_THROUGH; } /* activate the scroller */ if (vsm->scroller == 'h') { v2d->scroll_ui |= V2D_SCROLL_H_ACTIVE; } else { v2d->scroll_ui |= V2D_SCROLL_V_ACTIVE; } /* still ok, so can add */ WM_event_add_modal_handler(C, op); return OPERATOR_RUNNING_MODAL; } /* not in scroller, so nothing happened... * (pass through let's something else catch event) */ return OPERATOR_PASS_THROUGH; } /* LMB-Drag in Scrollers - not repeatable operator! */ static void VIEW2D_OT_scroller_activate(wmOperatorType *ot) { /* identifiers */ ot->name = "Scroller Activate"; ot->description = "Scroll view by mouse click and drag"; ot->idname = "VIEW2D_OT_scroller_activate"; /* flags */ ot->flag = OPTYPE_BLOCKING; /* api callbacks */ ot->invoke = scroller_activate_invoke; ot->modal = scroller_activate_modal; ot->cancel = scroller_activate_cancel; ot->poll = scroller_activate_poll; } /** \} */ /* -------------------------------------------------------------------- */ /** \name View Reset Operator * \{ */ static int reset_exec(bContext *C, wmOperator *UNUSED(op)) { const uiStyle *style = UI_style_get(); ARegion *region = CTX_wm_region(C); View2D *v2d = ®ion->v2d; int winx, winy; const int snap_test = ED_region_snap_size_test(region); /* zoom 1.0 */ winx = (float)(BLI_rcti_size_x(&v2d->mask) + 1); winy = (float)(BLI_rcti_size_y(&v2d->mask) + 1); v2d->cur.xmax = v2d->cur.xmin + winx; v2d->cur.ymax = v2d->cur.ymin + winy; /* align */ if (v2d->align) { /* posx and negx flags are mutually exclusive, so watch out */ if ((v2d->align & V2D_ALIGN_NO_POS_X) && !(v2d->align & V2D_ALIGN_NO_NEG_X)) { v2d->cur.xmax = 0.0f; v2d->cur.xmin = -winx * style->panelzoom; } else if ((v2d->align & V2D_ALIGN_NO_NEG_X) && !(v2d->align & V2D_ALIGN_NO_POS_X)) { v2d->cur.xmax = winx * style->panelzoom; v2d->cur.xmin = 0.0f; } /* - posx and negx flags are mutually exclusive, so watch out */ if ((v2d->align & V2D_ALIGN_NO_POS_Y) && !(v2d->align & V2D_ALIGN_NO_NEG_Y)) { v2d->cur.ymax = 0.0f; v2d->cur.ymin = -winy * style->panelzoom; } else if ((v2d->align & V2D_ALIGN_NO_NEG_Y) && !(v2d->align & V2D_ALIGN_NO_POS_Y)) { v2d->cur.ymax = winy * style->panelzoom; v2d->cur.ymin = 0.0f; } } /* validate that view is in valid configuration after this operation */ UI_view2d_curRect_validate(v2d); if (ED_region_snap_size_apply(region, snap_test)) { ScrArea *area = CTX_wm_area(C); ED_area_tag_redraw(area); WM_event_add_notifier(C, NC_SCREEN | NA_EDITED, NULL); } /* request updates to be done... */ ED_region_tag_redraw(region); UI_view2d_sync(CTX_wm_screen(C), CTX_wm_area(C), v2d, V2D_LOCK_COPY); UI_view2d_zoom_cache_reset(); return OPERATOR_FINISHED; } static void VIEW2D_OT_reset(wmOperatorType *ot) { /* identifiers */ ot->name = "Reset View"; ot->description = "Reset the view"; ot->idname = "VIEW2D_OT_reset"; /* api callbacks */ ot->exec = reset_exec; ot->poll = view2d_poll; } /** \} */ /* -------------------------------------------------------------------- */ /** \name Registration * \{ */ void ED_operatortypes_view2d(void) { WM_operatortype_append(VIEW2D_OT_pan); WM_operatortype_append(VIEW2D_OT_edge_pan); WM_operatortype_append(VIEW2D_OT_scroll_left); WM_operatortype_append(VIEW2D_OT_scroll_right); WM_operatortype_append(VIEW2D_OT_scroll_up); WM_operatortype_append(VIEW2D_OT_scroll_down); WM_operatortype_append(VIEW2D_OT_zoom_in); WM_operatortype_append(VIEW2D_OT_zoom_out); WM_operatortype_append(VIEW2D_OT_zoom); WM_operatortype_append(VIEW2D_OT_zoom_border); #ifdef WITH_INPUT_NDOF WM_operatortype_append(VIEW2D_OT_ndof); #endif WM_operatortype_append(VIEW2D_OT_smoothview); WM_operatortype_append(VIEW2D_OT_scroller_activate); WM_operatortype_append(VIEW2D_OT_reset); } void ED_keymap_view2d(wmKeyConfig *keyconf) { WM_keymap_ensure(keyconf, "View2D", 0, 0); } /** \} */