Exemple #1
0
 def UpdateFontSize(self):
     old_font = self.__fnt_layout
     self.__fnt_layout = GetMonospaceFont(self.GetSettings().layout_font_size)
     if old_font.GetPointSize() != self.__fnt_layout.GetPointSize():
         self.__set_renderer_font = False
         self.__UpdateFontScaling()
         self.__CalcCanvasSize()
         self.__context.FullRedraw()
         self.FullUpdate()
Exemple #2
0
    def __init__(self,
                 parent,
                 id,
                 canvas,
                 name='',
                 style=wx.LC_REPORT | wx.SUNKEN_BORDER,
                 properties=['element']):
        wx.ListCtrl.__init__(self,
                             parent=parent,
                             id=wx.NewId(),
                             name=name,
                             style=style)
        self.SetFont(GetMonospaceFont(12))

        # used for coloring
        self.__canvas = canvas
        # list of dictionary of properties
        self.__elements = []
        # list of element pointers
        self.__element_ptr = []
        # properties to show.
        self.__properties = properties[:]  #must have at least 1
        #insertion point for elements at end
        self.__current_new_idx = 0
        self.RefreshAll()
Exemple #3
0
    def __init__(self, parent, id, canvas, name = '', style = wx.LC_REPORT | wx.SUNKEN_BORDER):
        wx.ListCtrl.__init__(self, parent = parent, id = wx.NewId(), name = name, style = style)
        self.SetFont(GetMonospaceFont(12))

        # used for coloring
        self.__canvas = canvas
        # list of dictionary of properties
        self.__transactions = []
        # properties to show.
        self.__properties = ['start', 'location', 'annotation'] # must have at least 1
        # insertion point for elements at end
        self.__current_new_idx = 0
        self.__colorize = True
        self.RefreshAll()
Exemple #4
0
 def __init__(self, canvas, handler):
     super(self.__class__, self).__init__(
         canvas.GetParent(),
         style=wx.BORDER_NONE | wx.WS_EX_TRANSIENT | wx.STAY_ON_TOP)
     self.Show(False)
     self.__canvas = canvas
     self.__handler = handler
     self.__element = None
     self.__annotation = None
     self.SetFont(GetMonospaceFont(11))
     self.SetBackgroundStyle(wx.BG_STYLE_PAINT)
     self.SetForegroundColour(wx.BLACK)
     self.Bind(wx.EVT_PAINT, self.OnPaint)
     self.Bind(wx.EVT_ENTER_WINDOW, self.OnMouse)
Exemple #5
0
 def __init__(self, canvas, handler):
     super(self.__class__, self).__init__(canvas.GetParent())
     self.Show(False)
     self.__canvas = canvas
     self.__handler = handler
     self.Bind(wx.EVT_ENTER_WINDOW, self.OnMouse)
     sizer = wx.BoxSizer(wx.HORIZONTAL)
     panel_sizer = wx.BoxSizer(wx.HORIZONTAL)
     self.__panel = wx.Panel(self)
     self.__panel.SetFont(GetMonospaceFont(11))
     self.__panel.SetForegroundColour(wx.BLACK)
     self.__text_ctrl = wx.StaticText(self.__panel)
     panel_sizer.Add(self.__text_ctrl)
     self.__panel.SetSizer(panel_sizer)
     sizer.Add(self.__panel,
               flag=wx.TOP | wx.BOTTOM | wx.LEFT | wx.RIGHT
               | wx.ALIGN_CENTER,
               border=1)
     self.SetSizer(sizer)
Exemple #6
0
    def __init__(self,
                 parent,
                 context,
                 dialog,
                 ID=-1,
                 pos=wx.DefaultPosition,
                 size=wx.DefaultSize,
                 style=wx.NO_FULL_REPAINT_ON_RESIZE):

        wx.ScrolledWindow.__init__(self, parent)
        self.SetBackgroundStyle(wx.BG_STYLE_PAINT)
        self.SetBackgroundColour(wx.WHITE)
        self.__renderer = core.Renderer()
        assert len(autocoloring.REASON_BRUSHES) > 0 and len(
            autocoloring.BACKGROUND_BRUSHES) > 0
        self.SetRendererBrushes()
        self.__renderer.setExtensions(context.GetExtensionManager())
        self.__dragscrl = drgscrlr.DragScroller(self,
                                                rate=300,
                                                sensitivity=.01)
        self.__WIDTH = 2000
        self.__HEIGHT = 2000

        # full canvas scale
        self.__canvas_scale = 1.0

        self.__SCROLL_RATE = 20
        self.__scroll_ratios = (0, 0)
        self.__UpdateScrollbars()
        self.__parent = parent
        self.__context = context
        self.__layout = context.GetLayout()
        self.__dlg = dialog
        self.__draw_grid = False
        #TODO, make neat ####################################
        self.__gridsize = 14
        self.__gridlines = []
        self.__UpdateGrid(self.__gridsize)
        #End TODO ###########################################
        self.__selection = Selection_Mgr(self, dialog)
        self.__user_handler = Input_Decoder(self, self.__selection)

        self.__set_renderer_font = False
        self.__schedule_line_draw_style = 4  # classic

        # used for color highlighting of transactions
        self.__colored_transactions = {}

        self.__hover_preview = HoverPreview(self, context)
        # Load images

        self.__fnt_layout = GetMonospaceFont(11)

        # Disable background erasing
        def disable_event(*pargs, **kwargs):
            pass

        self.Bind(wx.EVT_ERASE_BACKGROUND, disable_event)
        self.Bind(wx.EVT_PAINT, self.OnPaint)
        self.Bind(wx.EVT_SIZE, self.OnSize)

        # Bindings for mouse events
        self.Bind(
            wx.EVT_LEFT_DOWN,
            partial(self.__user_handler.LeftDown,
                    canvas=self,
                    selection=self.__selection,
                    dialog=self.__dlg))
        self.Bind(
            wx.EVT_MOTION,
            partial(self.__user_handler.MouseMove,
                    canvas=self,
                    selection=self.__selection,
                    context=self.__context,
                    mouse_over_preview=self.__hover_preview))
        self.Bind(
            wx.EVT_LEFT_DCLICK,
            partial(self.__user_handler.LeftDouble,
                    canvas=self,
                    selection=self.__selection,
                    dialog=self.__dlg))
        self.Bind(
            wx.EVT_LEFT_UP,
            partial(self.__user_handler.LeftUp,
                    canvas=self,
                    selection=self.__selection,
                    dialog=self.__dlg))
        self.Bind(
            wx.EVT_RIGHT_DOWN,
            partial(self.__user_handler.RightDown,
                    canvas=self,
                    drgscrl=self.__dragscrl))
        self.Bind(
            wx.EVT_RIGHT_UP,
            partial(self.__user_handler.RightUp,
                    canvas=self,
                    drgscrl=self.__dragscrl))
        # Bindings for key events
        self.Bind(
            wx.EVT_KEY_DOWN,
            partial(self.__user_handler.KeyDown,
                    canvas=self,
                    selection=self.__selection,
                    dialog=self.__dlg,
                    context=self.__context))
        self.Bind(
            wx.EVT_KEY_UP,
            partial(self.__user_handler.KeyUp,
                    canvas=self,
                    selection=self.__selection,
                    dialog=self.__dlg,
                    context=self.__context))

        self.Bind(wx.EVT_MOUSEWHEEL,
                  partial(self.__user_handler.MouseWheel, canvas=self))

        # required to actually receive key events
        self.SetFocus()

        self.Bind(wx.EVT_SCROLLWIN, self.ScrollWin)
Exemple #7
0
class Layout_Canvas(wx.ScrolledWindow):

    __AUTO_CANVAS_SIZE_MARGIN = 40 # Pixels

    MIN_WIDTH = 500
    MIN_HEIGHT = 500
    MIN_ZOOM = 0.1
    MAX_ZOOM = 2

    # # Construct a Layout_Canvas
    #  @param parent Parent Layout_Frame
    #  @param context Associated Layout_Context
    #  @param dialog Element Properties Dialog for this canvas/frame
    #  @param ID Window ID
    def __init__(self,
                 parent,
                 context,
                 dialog,
                 ID = -1,
                 pos = wx.DefaultPosition,
                 size = wx.DefaultSize,
                 style = wx.NO_FULL_REPAINT_ON_RESIZE):

        wx.ScrolledWindow.__init__(self, parent)
        self.SetBackgroundStyle(wx.BG_STYLE_PAINT)
        self.SetBackgroundColour(wx.WHITE)
        self.__renderer = core.Renderer()
        assert len(autocoloring.REASON_BRUSHES) > 0 and len(autocoloring.BACKGROUND_BRUSHES) > 0
        self.SetRendererBrushes()
        self.__renderer.setExtensions(context.GetExtensionManager())
        self.__dragscrl = drgscrlr.DragScroller(self, rate = 300, sensitivity = .01)
        self.__WIDTH = 2000
        self.__HEIGHT = 2000

        # full canvas scale
        self.__canvas_scale = 1.0
        self.__font_scale = (1.0, 1.0)

        self.__SCROLL_RATE = 20
        self.__scroll_ratios = (0, 0)
        self.__UpdateScrollbars()
        self.__parent = parent
        self.__context = context
        self.__layout = context.GetLayout()
        self.__dlg = dialog
        self.__draw_grid = False
        #TODO, make neat ####################################
        self.__gridsize = 14
        self.__gridlines = []
        self.__UpdateGrid(self.__gridsize)
        #End TODO ###########################################
        self.__selection = Selection_Mgr(self, dialog)
        self.__user_handler = Input_Decoder(self, self.__selection)

        self.__set_renderer_font = False
        self.__schedule_line_draw_style = 4 # classic

        # used for color highlighting of transactions
        self.__colored_transactions = {}

        self.__hover_preview = HoverPreview(self, context)
        # Load images

        self.__fnt_layout = GetMonospaceFont(self.GetSettings().layout_font_size)
        self.__UpdateFontScaling()

        # Disable background erasing
        def disable_event(*pargs, **kwargs):
            pass

        self.Bind(wx.EVT_ERASE_BACKGROUND, disable_event)
        self.Bind(wx.EVT_PAINT, self.OnPaint)
        self.Bind(wx.EVT_SIZE, self.OnSize)

        # Bindings for mouse events
        self.Bind(wx.EVT_LEFT_DOWN,
                  partial(self.__user_handler.LeftDown, canvas = self,
                          selection = self.__selection, dialog = self.__dlg))
        self.Bind(wx.EVT_MOTION,
                  partial(self.__user_handler.MouseMove, canvas = self,
                  selection = self.__selection, context = self.__context,
                  mouse_over_preview = self.__hover_preview))
        self.Bind(wx.EVT_LEFT_DCLICK,
                  partial(self.__user_handler.LeftDouble, canvas = self,
                          selection = self.__selection, dialog = self.__dlg))
        self.Bind(wx.EVT_LEFT_UP,
                  partial(self.__user_handler.LeftUp, canvas = self,
                          selection = self.__selection, dialog = self.__dlg))
        self.Bind(wx.EVT_RIGHT_DOWN,
                  partial(self.__user_handler.RightDown, canvas = self, drgscrl = self.__dragscrl))
        self.Bind(wx.EVT_RIGHT_UP,
                  partial(self.__user_handler.RightUp, canvas = self, drgscrl = self.__dragscrl))
        # Bindings for key events
        self.Bind(wx.EVT_KEY_DOWN,
                  partial(self.__user_handler.KeyDown, canvas = self,
                          selection = self.__selection, dialog = self.__dlg,
                          context = self.__context))
        self.Bind(wx.EVT_KEY_UP,
                  partial(self.__user_handler.KeyUp, canvas = self,
                          selection = self.__selection, dialog = self.__dlg,
                          context = self.__context))

        self.Bind(wx.EVT_MOUSEWHEEL, partial(self.__user_handler.MouseWheel, canvas = self))

        # required to actually receive key events
        self.SetFocus()

        self.Bind(wx.EVT_SCROLLWIN, self.ScrollWin)

    # # Size of the canvas grid
    @property
    def gridsize(self):
        return self.__gridsize

    # # Snap sensitivity range (+/- each gridline)
    @property
    def range(self):
        return self.__snap_capture_delta

    @property
    def scrollrate(self):
        return self.__SCROLL_RATE

    @property
    def context(self):
        return self.__context

    # # Returns the parent frame which owns this Canvas
    def GetFrame(self):
        return self.__parent

    # # Returns a tuple showing the area of the canvas visible to the screen
    #  @return (x,y,w,h)
    def GetVisibleArea(self):
        x, y = self.GetViewStart()
        w, h = self.GetClientSize()
        return (x, y, w, h)

    # # Updates color for a transaction
    def UpdateTransactionColor(self, el, annotation):
        if el.HasProperty('auto_color_basis') \
          and el.HasProperty('color_basis_type'):
            auto_color_basis = el.GetProperty('auto_color_basis')
            color_basis_type = el.GetProperty('color_basis_type')
            content_type = el.GetProperty('Content')
            record = self.GetTransactionColor(annotation, content_type, color_basis_type, auto_color_basis)
            if record:
                string_to_display, brush, _, _, _, _ = record
            else:
                try:
                    tick = el.GetVisibilityTick()
                except AttributeError:
                    tick = 0
                string_to_display, brush, _, _ = self.AddColoredTransaction(annotation,
                                                                            content_type,
                                                                            color_basis_type,
                                                                            auto_color_basis,
                                                                            tick,
                                                                            el)
        else:
            string_to_display = self.__hover_preview.annotation
            brush = wx.Brush((200, 200, 200))

        return string_to_display, brush

    # # Here the canvas does all it's drawing
    def DoDraw(self, dc):
        logging.debug('xxx: Started C drawing')

        # Draw grid
        # This has effectively 0 cost
        if self.__draw_grid:
            brush = dc.GetBackground()
            brush.SetColour('black')
            dc.SetBackground(brush)
            dc.Clear()
            dc.SetPen(wx.Pen((220, 220, 220), 1))
            # dc.SetLogicalFunction(wx.INVERT)
            xoff, yoff = self.GetRenderOffsets()
            for x, y, x1, y1 in self.__gridlines:
                dc.DrawLine(x - xoff, y - yoff, x1 - xoff, y1 - yoff)

        t_start = time.monotonic()
        self.__renderer.drawElements(dc, self, self.__context.GetHC())

        logging.debug('{0}s: C drawing'.format(time.monotonic() - t_start))

    def ScrollWin(self, evt):
        super(self.__class__, self).Refresh()

        bounds = self.__GetScrollBounds()
        x_bound = bounds[0] - self.scrollrate
        y_bound = bounds[1] - self.scrollrate

        w_pix, h_pix = self.GetClientSize()
        percent_bar_x = w_pix / self.__WIDTH
        percent_bar_y = h_pix / self.__HEIGHT

        self.__scroll_ratios = (self.GetScrollPos(wx.HORIZONTAL) / x_bound + percent_bar_x / 4.0,
                                self.GetScrollPos(wx.VERTICAL) / y_bound + percent_bar_y / 4.0)

        # If we're in edit mode, update the cursor location in the toolbar
        if self.__user_handler.GetEditMode():
            (x, y) = self.GetMousePosition()
            offset = self.CalcScrollInc(evt)
            scroll_units = self.GetScrollPixelsPerUnit()

            if evt.GetOrientation() == wx.HORIZONTAL:
                x += offset * scroll_units[0]
            elif evt.GetOrientation() == wx.VERTICAL:
                y += offset * scroll_units[1]

            self.__parent.UpdateMouseLocation(x, y)

        mousePos = wx.GetMousePosition()
        self.__hover_preview.HandleMouseMove(mousePos, self)
        evt.Skip()

    # # Returns the position of the mouse within the canvas
    def GetMousePosition(self):
        return self.CalcUnscrolledPosition(self.ScreenToClient(wx.GetMousePosition()))

    # # Execute all drawing logic
    def FullUpdate(self):

        # update quad tree (if needed)
        self.__context.MicroUpdate()
        self.Refresh()
        # Tell the Element Prop's Dlg window to update itself
        self.__dlg.Refresh()

    # # Execute all drawing logic
    def OnPaint(self, event):
        paint_dc = wx.AutoBufferedPaintDC(self)
        paint_dc.Clear();
        context = wx.GraphicsContext.Create(paint_dc)
        dc = wx.GCDC(context)
        dc.SetLogicalScale(self.__canvas_scale, self.__canvas_scale)
        dc.SetFont(self.__fnt_layout)
        if not self.__set_renderer_font:
            self.__renderer.setFontFromDC(dc)
            self.__set_renderer_font = True
        self.DoDraw(dc)
        self.__selection.Draw(dc)

    # # Returns Hover Preview
    def GetHoverPreview(self):
        return self.__hover_preview

    # # Turns on or off the drawing of the grid
    def ToggleGrid(self):
        self.__draw_grid = not self.__draw_grid
        self.Refresh()

    # # Refresh. Recalc size and forward to superclass
    def Refresh(self):
        # Recalculate canvas size
        if self.__CalcCanvasSize():
            self.__UpdateScrollbars()
            self.__UpdateGrid(self.__gridsize)

        super(self.__class__, self).Refresh()

    # # Gets called when the window is resized, and when the canvas is initialized
    def OnSize(self, event):
        # update the screen
        self.FullUpdate()

    # # Gets the selection manager for this canvas
    def GetSelectionManager(self):
        return self.__selection

    # # Gets the input decoder for this canvas
    def GetInputDecoder(self):
        return self.__user_handler

    # # Forward on the draw-orderd list of all Elements in this
    #  frame/canvas/layout...
    def GetElements(self):
        return self.__context.GetElements()

    # # Return Element/Value pairs
    def GetElementPairs(self):
        return self.__context.GetElementPairs()

    # # Compute and returns display bounds
    def GetBounds(self):
        posx, posy, width, height = self.GetVisibleArea()
        mins = self.CalcUnscrolledPosition((0, 0))
        bounds = (mins[0],
                  mins[1],
                  mins[0] + width / self.__canvas_scale,
                  mins[1] + height / self.__canvas_scale)
        return bounds

    # Gets the update region relative to the current visible area
    # Coordinates are scaled to account for current zoom factor
    # Used by schedule elements to determine where to blit
    def GetScaledUpdateRegion(self):
        box = self.GetUpdateRegion().GetBox()
        box = wx.Rect(math.floor(box[0] / self.__canvas_scale),
                      math.floor(box[1] / self.__canvas_scale),
                      math.ceil(box[2] / self.__canvas_scale),
                      math.ceil(box[3] / self.__canvas_scale))

        box.Inflate(10, 10) # Fudge factor to ensure no dirt is left behind
        return box

    # Gets the update region relative to the origin of the scrolled area
    # Region is scaled to account for current zoom factor
    # Used to determine which elements need to be updated
    def GetScrolledUpdateRegion(self):
        box = self.GetUpdateRegion().GetBox()
        box.SetPosition(self.CalcUnscrolledPosition(box.GetTopLeft()))
        box.SetHeight(math.ceil(box.GetHeight() / self.__canvas_scale))
        box.SetWidth(math.ceil(box.GetWidth() / self.__canvas_scale))

        box.Inflate(10, 10) # Fudge factor to ensure no dirt is left behind
        return box

    # # Returns element pairs suitable for drawing
    def GetDrawPairs(self):
        box = self.GetScrolledUpdateRegion()
        top_left = box.GetTopLeft()
        bottom_right = box.GetBottomRight()
        bounds = (top_left[0], top_left[1], bottom_right[0], bottom_right[1])
        return self.__context.GetDrawPairs(bounds)

    def GetVisibilityTick(self):
        return self.__context.GetVisibilityTick()

    # # Set the brushes in the renderer to the current brush set
    def SetRendererBrushes(self):
        self.__renderer.setBrushes(autocoloring.REASON_BRUSHES.as_dict(), autocoloring.BACKGROUND_BRUSHES.as_dict())

    # # Resets the renderer brushes and purges the brush cache - needed whenever the palette or color shuffle mode changes
    def RefreshBrushes(self):
        self.SetRendererBrushes()
        self.PurgeBrushCache()

    # # Returns brushes needed to draw elements
    def GetBrushes(self):
        return autocoloring.BACKGROUND_BRUSHES, autocoloring.REASON_BRUSHES

    # #Returns offsets needed to render
    def GetRenderOffsets(self):
        return self.CalcUnscrolledPosition((0, 0))

    # # Returns the Cython renderer
    def GetRenderer(self):
        return self.__renderer

    # # Returns element settings dialog
    def GetDialog(self):
        return self.__dlg

    # # Returns global setting for schedule line drawing
    def GetScheduleLineStyle(self):
        return self.__schedule_line_draw_style

    def GetScale(self):
        return self.__canvas_scale

    # # Updates the highlighting state of all cached transactions
    def UpdateTransactionHighlighting(self):
        for key, val in self.__colored_transactions.items():
            self.__colored_transactions[key] = val[:2] + \
                                               (self.__context.IsUopUidHighlighted(val[4]),) + \
                                               (self.__context.IsSearchResult(val[5]),) + \
                                               val[4:]

    def GetTransactionColor(self, annotation, content_type, color_basis_type, auto_color_basis):
        return self.__colored_transactions.get('%s:%s:%s:%s' % (annotation, color_basis_type, auto_color_basis, hash(content_type)))

    # # Track the tagging of one transaction
    #  @note OThis can be called mutiple times to override existing colors
    def AddColoredTransaction(self, annotation, content_type, color_basis_type, auto_color_basis, start_tick, element):
        if len(self.__colored_transactions) > 10000:
            self.__colored_transactions.clear()

        string_to_display, brush = self.__renderer.parseAnnotationAndGetColor(annotation,
                                                                              content_type,
                                                                              color_basis_type,
                                                                              auto_color_basis)
        key = '%s:%s:%s:%s' % (annotation, color_basis_type, auto_color_basis, hash(content_type))
        uop_uid = highlighting_utils.GetUopUid(annotation)
        highlighted = self.__context.IsUopUidHighlighted(uop_uid)

        if element.HasProperty('LocationString'):
            search_result_hash = self.__context.SearchResultHash(start_tick, element.GetProperty('LocationString'))
        else:
            search_result_hash = None

        search_result = self.__context.IsSearchResult(search_result_hash)
        self.__colored_transactions[key] = (string_to_display, brush, highlighted, search_result, uop_uid, search_result_hash)
        return string_to_display, brush, highlighted, search_result

    # # Gets old-style coloring (no basis configured)
    def GetAutocolorColor(self, annotation):
        string_to_display, brush = self.__renderer.parseAnnotationAndGetColor(annotation,
                                                        'auto_color_annotation')
        return brush.GetColour()

    # # remove all entries from the dictionary of colored transactions
    # will auto regenerate next render frame
    def PurgeBrushCache(self):
        self.__colored_transactions.clear()
        self.__context.FullRedraw()
        self.Refresh()

    # # Set the global style for schedule lines
    def SetScheduleLineStyle(self, style):
        if style != self.__schedule_line_draw_style:
            self.__schedule_line_draw_style = style
            self.__context.FullUpdate()
            self.Refresh()

    def SetScale(self, scale):
        # cap at 2x zoom in
        self.__canvas_scale = min(max(self.MIN_ZOOM, scale), self.MAX_ZOOM)

        self.__CalcCanvasSize()
        self.__UpdateScrollbars()
        self.__UpdateGrid(self.__gridsize)

        super(self.__class__, self).Refresh()

    # # Returns a list of all Elements beneath the given point
    def DetectCollision(self, pt):
        return self.__context.DetectCollision(pt)

    # # override to handle full-canvas scale
    def CalcUnscrolledPosition(self, position):
        view_start = self.GetViewStart()
        x0 = view_start[0] * self.scrollrate / self.__canvas_scale
        y0 = view_start[1] * self.scrollrate / self.__canvas_scale
        return x0 + position[0] / self.__canvas_scale, y0 + position[1] / self.__canvas_scale

    # # Calculate the canvas size based on all elements
    #  @return True if size changed, False if not
    def __CalcCanvasSize(self):

        l, t, r, b = self.__context.GetElementExtents()
        r *= self.__canvas_scale * self.__font_scale[0]
        b *= self.__canvas_scale * self.__font_scale[1]

        width = max(self.MIN_WIDTH, r + self.__AUTO_CANVAS_SIZE_MARGIN)
        height = max(self.MIN_HEIGHT, b + self.__AUTO_CANVAS_SIZE_MARGIN)

        if self.__WIDTH == width and self.__HEIGHT == height:
            return False # No change

        self.__WIDTH = width
        self.__HEIGHT = height
        return True

    def __GetScrollBounds(self):
        sr = float(self.scrollrate)
        return self.__WIDTH / sr, self.__HEIGHT / sr

    # # Update the scrollbars based on a new canvas size
    #  Restores prior scroll offsets
    #  Uses instance attributes __SCROLL_RATE, __WIDTH, __HEIGHT
    def __UpdateScrollbars(self):
        sr = self.scrollrate

        w_pix, h_pix = self.GetClientSize()
        x_bound, y_bound = self.__GetScrollBounds()
        x, y = self.__scroll_ratios
        percent_bar_x = w_pix / self.__WIDTH
        percent_bar_y = h_pix / self.__HEIGHT
        if percent_bar_x > 1:
            percent_bar_x = 1
        if percent_bar_y > 1:
            percent_bar_y = 1

        self.SetScrollbars(sr,
                           sr,
                           x_bound,
                           y_bound,
                           x * x_bound * (1 - percent_bar_x),
                           y * y_bound * (1 - percent_bar_y),
                           True)

    # # Regenerates the gridlines based on __WIDTH and __HEIGHT
    #  @param gridsize Space between each gridline
    def __UpdateGrid(self, gridsize):
        self.__gridlines = []
        for x in range(round(self.__WIDTH / gridsize)):
            self.__gridlines.append((x * gridsize, 0, x * gridsize, self.__HEIGHT))
        for y in range(round(self.__HEIGHT / gridsize)):
            self.__gridlines.append((0, y * gridsize, self.__WIDTH, y * gridsize))
        assert gridsize % 2 == 0
        self.__snap_capture_delta = 7
        assert self.__snap_capture_delta <= gridsize / 2

    def GetSettings(self):
        return self.__parent.GetSettings()

    def __UpdateFontScaling(self):
        default_font_w, default_font_h = GetDefaultFont().GetPixelSize()
        cur_font_w, cur_font_h = self.__fnt_layout.GetPixelSize()
        self.__font_scale = (cur_font_w / default_font_w, cur_font_h / default_font_h)
        for e in self.__context.GetElementPairs():
            e.GetElement().SetProperty('scale_factor', self.__font_scale)

    def UpdateFontSize(self):
        old_font = self.__fnt_layout
        self.__fnt_layout = GetMonospaceFont(self.GetSettings().layout_font_size)
        if old_font.GetPointSize() != self.__fnt_layout.GetPointSize():
            self.__set_renderer_font = False
            self.__UpdateFontScaling()
            self.__CalcCanvasSize()
            self.__context.FullRedraw()
            self.FullUpdate()
    def __init__(self, parent):
        self.__layout_frame = parent
        self.__context = parent.GetContext()
        self.__sel_mgr = parent.GetCanvas().GetSelectionManager()

        # initialize graphical part
        wx.Dialog.__init__(self,
                           parent,
                           -1,
                           'Translate Elements - {}'.format(
                               self.__layout_frame.ComputeTitle()),
                           size=(-1, -1),
                           style=wx.MAXIMIZE_BOX | wx.RESIZE_BORDER
                           | wx.CAPTION | wx.CLOSE_BOX | wx.SYSTEM_MENU)

        self.__fnt_numbers = GetMonospaceFont(12)

        panel = wx.Panel(self)

        lbl_info = wx.StaticText(panel, -1,
                                 '(Press ENTER to appy changes, ESC to exit)')
        lbl_info.SetFont(self.__fnt_numbers)
        lbl_info.SetForegroundColour((120, 120, 120))

        lbl_heading = wx.StaticText(
            panel, -1,
            '{} selected elements'.format(len(self.__sel_mgr.GetSelection())))

        box_dims = (60, 40)
        panel_box = wx.Panel(panel, size=box_dims, style=wx.BORDER_SIMPLE)
        panel_box.SetMinSize(box_dims)
        self.__lbl_sel_lt = wx.StaticText(panel, -1, '')
        self.__lbl_sel_lt.SetFont(self.__fnt_numbers)
        self.__lbl_sel_wh = wx.StaticText(panel_box, -1, '')  # Inside box
        self.__lbl_sel_wh.SetFont(self.__fnt_numbers)
        self.__lbl_sel_rb = wx.StaticText(panel, -1, '')
        self.__lbl_sel_rb.SetFont(self.__fnt_numbers)

        self.__UpdateDesc()

        lbl_x = wx.StaticText(panel, -1, "X: ", size=(20, 20))
        self.__txt_x = wx.TextCtrl(panel, -1, "0", size=(70, 20))
        self.__chk_rel_x = wx.CheckBox(panel, -1, label='relative')
        self.__chk_rel_x.SetValue(True)

        lbl_y = wx.StaticText(panel, -1, "Y: ", size=(20, 20))
        self.__txt_y = wx.TextCtrl(panel, -1, "0", size=(70, 20))
        self.__chk_rel_y = wx.CheckBox(panel, -1, label='relative')
        self.__chk_rel_y.SetValue(True)

        self.__btn_exit = wx.Button(panel, -1, 'Close')
        self.__btn_apply = wx.Button(panel, -1, 'Apply')

        panel_box_sizer = wx.BoxSizer(wx.VERTICAL)
        panel_box_sizer.Add((1, 1), 1, wx.EXPAND)
        panel_box_sizer.Add(
            self.__lbl_sel_wh, 0,
            wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER_HORIZONTAL)
        panel_box_sizer.Add((1, 1), 1, wx.EXPAND)
        panel_box.SetSizer(panel_box_sizer)

        position_sizer = wx.BoxSizer(wx.HORIZONTAL)
        position_sizer.Add(self.__lbl_sel_lt, 0,
                           wx.ALIGN_TOP | wx.ALIGN_RIGHT | wx.RIGHT, 4)
        position_sizer.Add(panel_box, 1, wx.EXPAND | wx.TOP | wx.BOTTOM, 10)
        position_sizer.Add(self.__lbl_sel_rb, 0,
                           wx.ALIGN_BOTTOM | wx.ALIGN_RIGHT | wx.LEFT, 4)

        sizer_x = wx.BoxSizer(wx.HORIZONTAL)
        sizer_x.Add(lbl_x, 0, wx.ALL, 4)
        sizer_x.Add(self.__txt_x, 0, wx.ALL, 4)
        sizer_x.Add(self.__chk_rel_x, 0, wx.ALL, 4)

        sizer_y = wx.BoxSizer(wx.HORIZONTAL)
        sizer_y.Add(lbl_y, 0, wx.ALL, 4)
        sizer_y.Add(self.__txt_y, 0, wx.ALL, 4)
        sizer_y.Add(self.__chk_rel_y, 0, wx.ALL, 4)

        self.__sizer_buttons = wx.BoxSizer(wx.HORIZONTAL)
        self.__sizer_buttons.Add(self.__btn_exit, 0, 0, 4)
        self.__sizer_buttons.Add((1, 1), 1, wx.EXPAND)
        self.__sizer_buttons.Add(self.__btn_apply, 0, 0, 4)

        main_sizer = wx.BoxSizer(wx.VERTICAL)
        main_sizer.Add(lbl_heading, 0, wx.ALIGN_CENTER_HORIZONTAL)
        # main_sizer.Add((1,6), 0)
        main_sizer.Add(position_sizer, 0, wx.EXPAND | wx.ALL, 10)  # Indent
        main_sizer.Add(wx.StaticLine(panel), 0, wx.EXPAND)
        main_sizer.Add((1, 4), 0)
        main_sizer.Add(sizer_x, 0, wx.ALIGN_LEFT)
        main_sizer.Add(sizer_y, 0, wx.ALIGN_LEFT)
        main_sizer.Add((1, 4), 0, wx.EXPAND)
        main_sizer.Add(wx.StaticLine(panel), 0, wx.EXPAND)
        main_sizer.Add((1, 4), 0)
        main_sizer.Add(lbl_info, 0, wx.ALIGN_CENTER)
        main_sizer.Add((1, 18), 0)
        main_sizer.Add(self.__sizer_buttons, 0, wx.EXPAND)

        panel.SetSizer(main_sizer)

        padding_sizer = wx.BoxSizer(wx.VERTICAL)
        padding_sizer.Add(panel, 1, wx.EXPAND | wx.ALL, 10)
        self.SetSizer(padding_sizer)

        self.Fit()

        self.__txt_x.SetFocus()

        # TODO: bind escape to exit!

        # Could show currently selected elements for excluding some from the xlate operation
        # #self.__results_box = ElementList(panel,
        # #            wx.NewId(),
        # #            parent.GetCanvas(),
        # #            name='listbox',
        # #            properties=[''])
        # #main_sizer.Add(self.__results_box, 1, wx.EXPAND, 5)

        # bind to events
        self.__btn_apply.Bind(wx.EVT_BUTTON, self.OnApply)
        self.__btn_exit.Bind(wx.EVT_BUTTON, self.OnExit)
        panel.Bind(wx.EVT_KEY_UP, self.OnKeyPress)
        # #panel.Bind(wx.EVT_KEY_DOWN, self.OnKeyPress)
        # #panel.Bind(wx.EVT_CHAR, self.OnKeyPress)
        self.__txt_x.Bind(wx.EVT_TEXT, self.OnCoordChange)
        self.__txt_y.Bind(wx.EVT_TEXT, self.OnCoordChange)
        self.__chk_rel_x.Bind(wx.EVT_CHECKBOX, self.OnCoordChange)
        self.__chk_rel_y.Bind(wx.EVT_CHECKBOX, self.OnCoordChange)