]> git.saurik.com Git - wxWidgets.git/blobdiff - wxPython/demo/wxDragImage.py
Added XML simplification scripts for generating the wxPython metadata xml.
[wxWidgets.git] / wxPython / demo / wxDragImage.py
index 19985b7dd36a3b478b6a662b0e11b4ab45b52be7..ad8f880a5d4fb3ceeefeb1b0973a3953f9a882b7 100644 (file)
+# 11/15/2003 - Jeff Grimmett (grimmtooth@softhome.net)
+#
+# o Updated for wx namespace
+# 
 
 
-from wxPython.wx import *
+import  wx
+import  images
 
 #----------------------------------------------------------------------
 
 class DragShape:
     def __init__(self, bmp):
         self.bmp = bmp
 
 #----------------------------------------------------------------------
 
 class DragShape:
     def __init__(self, bmp):
         self.bmp = bmp
-        self.pos = wxPoint(0,0)
-        self.shown = true
+        self.pos = (0,0)
+        self.shown = True
         self.text = None
         self.text = None
-        self.fullscreen = false
-
+        self.fullscreen = False
 
     def HitTest(self, pt):
         rect = self.GetRect()
 
     def HitTest(self, pt):
         rect = self.GetRect()
-        return rect.Inside(pt.x, pt.y)
-
+        return rect.InsideXY(pt.x, pt.y)
 
     def GetRect(self):
 
     def GetRect(self):
-        return wxRect(self.pos.x, self.pos.y,
+        return wx.Rect(self.pos[0], self.pos[1],
                       self.bmp.GetWidth(), self.bmp.GetHeight())
 
                       self.bmp.GetWidth(), self.bmp.GetHeight())
 
-
-    def Draw(self, dc, op = wxCOPY):
+    def Draw(self, dc, op = wx.COPY):
         if self.bmp.Ok():
         if self.bmp.Ok():
-            memDC = wxMemoryDC()
+            memDC = wx.MemoryDC()
             memDC.SelectObject(self.bmp)
 
             memDC.SelectObject(self.bmp)
 
-            dc.Blit(self.pos.x, self.pos.y,
-                    self.bmp.GetWidth(), self.bmp.GetHeight(),
-                    memDC, 0, 0, op, true)
+            dc.Blit((self.pos[0], self.pos[1]),
+                    (self.bmp.GetWidth(), self.bmp.GetHeight()),
+                    memDC, (0, 0), op, True)
 
 
-            return true
+            return True
         else:
         else:
-            return false
+            return False
 
 
 
 #----------------------------------------------------------------------
 
 
 
 
 #----------------------------------------------------------------------
 
-class DragCanvas(wxScrolledWindow):
+class DragCanvas(wx.ScrolledWindow):
     def __init__(self, parent, ID):
     def __init__(self, parent, ID):
-        wxScrolledWindow.__init__(self, parent, ID)
+        wx.ScrolledWindow.__init__(self, parent, ID)
         self.shapes = []
         self.dragImage = None
         self.dragShape = None
         self.shapes = []
         self.dragImage = None
         self.dragShape = None
+        self.hiliteShape = None
 
 
-        self.SetCursor(wxStockCursor(wxCURSOR_ARROW))
-        self.bg_bmp = wxBitmap('bitmaps/backgrnd.png', wxBITMAP_TYPE_PNG)
-
+        self.SetCursor(wx.StockCursor(wx.CURSOR_ARROW))
+        self.bg_bmp = images.getBackgroundBitmap()
 
         # Make a shape from an image and mask.  This one will demo
         # dragging outside the window
 
         # Make a shape from an image and mask.  This one will demo
         # dragging outside the window
-        bmp = wxBitmap('bitmaps/test_image.png', wxBITMAP_TYPE_PNG)
-        mask = wxMaskColour(bmp, wxWHITE)
-        bmp.SetMask(mask)
+        bmp = images.getTestStarBitmap()
         shape = DragShape(bmp)
         shape = DragShape(bmp)
-        shape.pos = wxPoint(5, 5)
-        shape.fullscreen = true
+        shape.pos = (5, 5)
+        shape.fullscreen = True
         self.shapes.append(shape)
 
         self.shapes.append(shape)
 
-
         # Make a shape from some text
         text = "Some Text"
         # Make a shape from some text
         text = "Some Text"
-        font = wxFont(15, wxROMAN, wxNORMAL, wxBOLD)
+        bg_colour = wx.Colour(57, 115, 57)  # matches the bg image
+        font = wx.Font(15, wx.ROMAN, wx.NORMAL, wx.BOLD)
         textExtent = self.GetFullTextExtent(text, font)
         textExtent = self.GetFullTextExtent(text, font)
-        bmp = wxEmptyBitmap(textExtent[0], textExtent[1])
-        dc = wxMemoryDC()
+
+        # create a bitmap the same size as our text
+        bmp = wx.EmptyBitmap(textExtent[0], textExtent[1])
+
+        # 'draw' the text onto the bitmap
+        dc = wx.MemoryDC()
         dc.SelectObject(bmp)
         dc.SelectObject(bmp)
+        dc.SetBackground(wx.Brush(bg_colour, wx.SOLID))
         dc.Clear()
         dc.Clear()
-        dc.SetTextForeground(wxRED)
+        dc.SetTextForeground(wx.RED)
         dc.SetFont(font)
         dc.SetFont(font)
-        dc.DrawText(text, 0, 0)
-        dc.SelectObject(wxNullBitmap)
-        del dc
-        mask = wxMaskColour(bmp, wxWHITE)
+        dc.DrawText(text, (0, 0))
+        dc.SelectObject(wx.NullBitmap)
+        mask = wx.MaskColour(bmp, bg_colour)
         bmp.SetMask(mask)
         shape = DragShape(bmp)
         bmp.SetMask(mask)
         shape = DragShape(bmp)
-        shape.pos = wxPoint(5, 100)
+        shape.pos = (5, 100)
         shape.text = "Some dragging text"
         self.shapes.append(shape)
 
 
         # Make some shapes from some playing card images.
         x = 200
         shape.text = "Some dragging text"
         self.shapes.append(shape)
 
 
         # Make some shapes from some playing card images.
         x = 200
-        for card in ['01c.gif', '10s.gif', '12h.gif', '13d.gif']:
-            bmp = wxBitmap('bitmaps/'+card, wxBITMAP_TYPE_GIF)
+
+        for card in ['_01c_', '_12h_', '_13d_', '_10s_']:
+            bmpFunc = getattr(images, "get%sBitmap" % card)
+            bmp = bmpFunc()
             shape = DragShape(bmp)
             shape = DragShape(bmp)
-            shape.pos = wxPoint(x, 5)
+            shape.pos = (x, 5)
             self.shapes.append(shape)
             x = x + 80
 
 
             self.shapes.append(shape)
             x = x + 80
 
 
-        EVT_ERASE_BACKGROUND(self, self.OnEraseBackground)
-        EVT_PAINT(self, self.OnPaint)
-        EVT_LEFT_DOWN(self, self.OnLeftDown)
-        EVT_LEFT_UP(self, self.OnLeftUp)
-        EVT_MOTION(self, self.OnMotion)
+        self.Bind(wx.EVT_ERASE_BACKGROUND, self.OnEraseBackground)
+        self.Bind(wx.EVT_PAINT, self.OnPaint)
+        self.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
+        self.Bind(wx.EVT_LEFT_UP, self.OnLeftUp)
+        self.Bind(wx.EVT_MOTION, self.OnMotion)
+        self.Bind(wx.EVT_LEAVE_WINDOW, self.OnLeaveWindow)
 
 
+    
+    # We're not doing anything here, but you might have reason to.
+    # for example, if you were dragging something, you might elect to
+    # 'drop it' when the cursor left the window.
+    def OnLeaveWindow(self, evt):
+        pass
 
 
 
 
+    # tile the background bitmap
     def TileBackground(self, dc):
     def TileBackground(self, dc):
-        # tile the background bitmap
         sz = self.GetClientSize()
         w = self.bg_bmp.GetWidth()
         h = self.bg_bmp.GetHeight()
 
         x = 0
         sz = self.GetClientSize()
         w = self.bg_bmp.GetWidth()
         h = self.bg_bmp.GetHeight()
 
         x = 0
+
         while x < sz.width:
             y = 0
         while x < sz.width:
             y = 0
+
             while y < sz.height:
             while y < sz.height:
-                dc.DrawBitmap(self.bg_bmp, x, y)
+                dc.DrawBitmap(self.bg_bmp, (x, y))
                 y = y + h
                 y = y + h
+
             x = x + w
 
 
             x = x + w
 
 
+    # Go through our list of shapes and draw them in whatever place they are.
     def DrawShapes(self, dc):
         for shape in self.shapes:
             if shape.shown:
                 shape.Draw(dc)
 
     def DrawShapes(self, dc):
         for shape in self.shapes:
             if shape.shown:
                 shape.Draw(dc)
 
-
+    # This is actually a sophisticated 'hit test', but in this
+    # case we're also determining which shape, if any, was 'hit'.
     def FindShape(self, pt):
         for shape in self.shapes:
             if shape.HitTest(pt):
                 return shape
         return None
 
     def FindShape(self, pt):
         for shape in self.shapes:
             if shape.HitTest(pt):
                 return shape
         return None
 
-
+    # Remove a shape from the display
     def EraseShape(self, shape, dc):
         r = shape.GetRect()
     def EraseShape(self, shape, dc):
         r = shape.GetRect()
-        dc.SetClippingRegion(r.x, r.y, r.width, r.height)
+        dc.SetClippingRect(r)
         self.TileBackground(dc)
         self.DrawShapes(dc)
         dc.DestroyClippingRegion()
 
         self.TileBackground(dc)
         self.DrawShapes(dc)
         dc.DestroyClippingRegion()
 
-
-
-
+    # Clears the background, then redraws it. If the DC is passed, then
+    # we only do so in the area so designated. Otherwise, it's the whole thing.
     def OnEraseBackground(self, evt):
         dc = evt.GetDC()
     def OnEraseBackground(self, evt):
         dc = evt.GetDC()
+
         if not dc:
             dc = wxClientDC(self)
         if not dc:
             dc = wxClientDC(self)
+            rect = self.GetUpdateRegion().GetBox()
+            dc.SetClippingRect(rect)
         self.TileBackground(dc)
 
         self.TileBackground(dc)
 
-
+    # Fired whenever a paint event occurs
     def OnPaint(self, evt):
     def OnPaint(self, evt):
-        dc = wxPaintDC(self)
+        dc = wx.PaintDC(self)
         self.PrepareDC(dc)
         self.DrawShapes(dc)
 
         self.PrepareDC(dc)
         self.DrawShapes(dc)
 
-
+    # Left mouse button is down.
     def OnLeftDown(self, evt):
     def OnLeftDown(self, evt):
+        # Did the mouse go down on one of our shapes?
         shape = self.FindShape(evt.GetPosition())
         shape = self.FindShape(evt.GetPosition())
+
+        # If a shape was 'hit', then set that as the shape we're going to
+        # drag around. Get our start position. Dragging has not yet started.
+        # That will happen once the mouse moves, OR the mouse is released.
         if shape:
         if shape:
-            # get ready to start dragging, but wait for the user to
-            # move it a bit first
             self.dragShape = shape
             self.dragStartPos = evt.GetPosition()
 
             self.dragShape = shape
             self.dragStartPos = evt.GetPosition()
 
-
+    # Left mouse button up.
     def OnLeftUp(self, evt):
         if not self.dragImage or not self.dragShape:
             self.dragImage = None
             self.dragShape = None
             return
 
     def OnLeftUp(self, evt):
         if not self.dragImage or not self.dragShape:
             self.dragImage = None
             self.dragShape = None
             return
 
-        # end the dragging
+        # Hide the image, end dragging, and nuke out the drag image.
         self.dragImage.Hide()
         self.dragImage.EndDrag()
         self.dragImage = None
 
         self.dragImage.Hide()
         self.dragImage.EndDrag()
         self.dragImage = None
 
+        dc = wx.ClientDC(self)
+
+        if self.hiliteShape:
+            self.hiliteShape.Draw(dc)
+            self.hiliteShape = None
+
         # reposition and draw the shape
         # reposition and draw the shape
-        pt = evt.GetPosition()
-        newPos = wxPoint(self.dragShape.pos.x + (pt.x - self.dragStartPos.x),
-                         self.dragShape.pos.y + (pt.y - self.dragStartPos.y))
 
 
-        dc = wxClientDC(self)
-        self.dragShape.pos = newPos
-        self.dragShape.shown = true
+        # Note by jmg 11/28/03 
+        # Here's the original:
+        #
+        # self.dragShape.pos = self.dragShape.pos + evt.GetPosition() - self.dragStartPos
+        #
+        # So if there are any problems associated with this, use that as
+        # a starting place in your investigation. I've tried to simulate the
+        # wx.Point __add__ method here -- it won't work for tuples as we
+        # have now from the various methods
+        #
+        # There must be a better way to do this :-)
+        #
+        
+        self.dragShape.pos = (
+            self.dragShape.pos[0] + evt.GetPosition()[0] - self.dragStartPos[0],
+            self.dragShape.pos[1] + evt.GetPosition()[1] - self.dragStartPos[1]
+            )
+            
+        self.dragShape.shown = True
         self.dragShape.Draw(dc)
         self.dragShape = None
 
         self.dragShape.Draw(dc)
         self.dragShape = None
 
-
+    # The mouse is moving
     def OnMotion(self, evt):
     def OnMotion(self, evt):
+        # Ignore mouse movement if we're not dragging.
         if not self.dragShape or not evt.Dragging() or not evt.LeftIsDown():
             return
 
         if not self.dragShape or not evt.Dragging() or not evt.LeftIsDown():
             return
 
-        # if we have a shape, but havn't started dragging yet
+        # if we have a shape, but haven't started dragging yet
         if self.dragShape and not self.dragImage:
 
             # only start the drag after having moved a couple pixels
         if self.dragShape and not self.dragImage:
 
             # only start the drag after having moved a couple pixels
-            tolerance = 4
+            tolerance = 2
             pt = evt.GetPosition()
             dx = abs(pt.x - self.dragStartPos.x)
             dy = abs(pt.y - self.dragStartPos.y)
             if dx <= tolerance and dy <= tolerance:
                 return
 
             pt = evt.GetPosition()
             dx = abs(pt.x - self.dragStartPos.x)
             dy = abs(pt.y - self.dragStartPos.y)
             if dx <= tolerance and dy <= tolerance:
                 return
 
-            if self.dragShape.text:
-                self.dragImage = wxDragString(self.dragShape.text,
-                                              wxStockCursor(wxCURSOR_HAND))
-            else:
-                self.dragImage = wxDragImage(self.dragShape.bmp,
-                                             wxStockCursor(wxCURSOR_HAND))
+            # erase the shape since it will be drawn independently now
+            dc = wx.ClientDC(self)
+            self.dragShape.shown = False
+            self.EraseShape(self.dragShape, dc)
 
 
-            newPos = wxPoint(self.dragShape.pos.x + (pt.x - self.dragStartPos.x),
-                             self.dragShape.pos.y + (pt.y - self.dragStartPos.y))
 
 
-            if self.dragShape.fullscreen:
-                newPos = self.ClientToScreen(newPos)
-                self.dragImage.BeginDrag((0,0), self, true)
+            if self.dragShape.text:
+                self.dragImage = wx.DragString(self.dragShape.text,
+                                              wx.StockCursor(wx.CURSOR_HAND))
             else:
             else:
-                self.dragImage.BeginDrag((0,0), self)
-
+                self.dragImage = wx.DragImage(self.dragShape.bmp,
+                                             wx.StockCursor(wx.CURSOR_HAND))
 
 
-            # erase the shape since it will be drawn independently now
-            dc = wxClientDC(self)
-            self.dragShape.shown = false
-            self.EraseShape(self.dragShape, dc)
+            hotspot = self.dragStartPos - self.dragShape.pos
+            self.dragImage.BeginDrag(hotspot, self, self.dragShape.fullscreen)
 
 
-            self.dragImage.Move(newPos)
+            self.dragImage.Move(pt)
             self.dragImage.Show()
 
 
             self.dragImage.Show()
 
 
-        # if we have shape and image then move it.
+        # if we have shape and image then move it, posibly highlighting another shape.
         elif self.dragShape and self.dragImage:
         elif self.dragShape and self.dragImage:
-            pt = evt.GetPosition()
-            newPos = wxPoint(self.dragShape.pos.x + (pt.x - self.dragStartPos.x),
-                             self.dragShape.pos.y + (pt.y - self.dragStartPos.y))
-            if self.dragShape.fullscreen:
-                newPos = self.ClientToScreen(newPos)
+            onShape = self.FindShape(evt.GetPosition())
+            unhiliteOld = False
+            hiliteNew = False
 
 
-            self.dragImage.Move(newPos)
+            # figure out what to hilite and what to unhilite
+            if self.hiliteShape:
+                if onShape is None or self.hiliteShape is not onShape:
+                    unhiliteOld = True
+
+            if onShape and onShape is not self.hiliteShape and onShape.shown:
+                hiliteNew = True
+
+            # if needed, hide the drag image so we can update the window
+            if unhiliteOld or hiliteNew:
+                self.dragImage.Hide()
+
+            if unhiliteOld:
+                dc = wx.ClientDC(self)
+                self.hiliteShape.Draw(dc)
+                self.hiliteShape = None
+
+            if hiliteNew:
+                dc = wx.ClientDC(self)
+                self.hiliteShape = onShape
+                self.hiliteShape.Draw(dc, wx.INVERT)
+
+            # now move it and show it again if needed
+            self.dragImage.Move(evt.GetPosition())
+            if unhiliteOld or hiliteNew:
+                self.dragImage.Show()
 
 
 #----------------------------------------------------------------------
 
 def runTest(frame, nb, log):
 
 
 #----------------------------------------------------------------------
 
 def runTest(frame, nb, log):
-    win = DragCanvas(nb, -1)
+
+    win = wx.Panel(nb, -1)
+    canvas = DragCanvas(win, -1)
+
+    def onSize(evt, panel=win, canvas=canvas): 
+        canvas.SetSize(panel.GetSize())
+
+    win.Bind(wx.EVT_SIZE, onSize)
     return win
 
 #----------------------------------------------------------------------
     return win
 
 #----------------------------------------------------------------------
@@ -246,4 +313,33 @@ def runTest(frame, nb, log):
 
 
 overview = """\
 
 
 overview = """\
+DragImage is used when you wish to drag an object on the screen, and a simple 
+cursor is not enough.
+
+On Windows, the WIN32 API is used to do achieve smooth dragging. On other 
+platforms, <code>GenericDragImage</code> is used. Applications may also prefer to use 
+<code>GenericDragImage</code> on Windows, too.
+
+<b>wxPython note</b>: wxPython uses <code>GenericDragImage</code> on all 
+platforms, but uses the <code>DragImage</code> name.
+
+To use this class, when you wish to start dragging an image, create a 
+<code>DragImage</code> object and store it somewhere you can access it as the 
+drag progresses. Call BeginDrag to start, and EndDrag to stop the drag. To move 
+the image, initially call Show and then Move. If you wish to update the screen 
+contents during the drag (for example, highlight an item as in the example), first 
+call Hide, update the screen, call Move, and then call Show.
+
+You can drag within one window, or you can use full-screen dragging either across 
+the whole screen, or just restricted to one area of the screen to save resources. 
+If you want the user to drag between two windows, then you will need to use 
+full-screen dragging.
 """
 """
+
+
+if __name__ == '__main__':
+    import sys,os
+    import run
+    run.main(['', os.path.basename(sys.argv[0])])
+