]> git.saurik.com Git - wxWidgets.git/blob - wxPython/wx/lib/floatcanvas/FloatCanvas.py
override GetLabe/SetLabel as on wxMac it will draw its own label that
[wxWidgets.git] / wxPython / wx / lib / floatcanvas / FloatCanvas.py
1
2 try:
3 from Numeric import array,asarray,Float,cos,pi,sum,minimum,maximum,Int32,zeros
4 except ImportError:
5 from numarray import array, asarray, Float, cos, pi, sum, minimum, maximum, Int32, zeros
6
7 from time import clock, sleep
8
9 import wx
10
11 import types
12 import os
13
14 import Resources
15
16 ## A global variable to hold the Pixels per inch that wxWindows thinks is in use
17 ## This is used for scaling fonts.
18 ## This can't be computed on module __init__, because a wx.App might not have iniitalized yet.
19 global ScreenPPI
20
21 ## a custom Exceptions:
22
23 class FloatCanvasException(Exception):
24 pass
25
26 ## All the mouse events
27 #EVT_FC_ENTER_WINDOW = wx.NewEventType()
28 #EVT_FC_LEAVE_WINDOW = wx.NewEventType()
29 EVT_FC_LEFT_DOWN = wx.NewEventType()
30 EVT_FC_LEFT_UP = wx.NewEventType()
31 EVT_FC_LEFT_DCLICK = wx.NewEventType()
32 EVT_FC_MIDDLE_DOWN = wx.NewEventType()
33 EVT_FC_MIDDLE_UP = wx.NewEventType()
34 EVT_FC_MIDDLE_DCLICK = wx.NewEventType()
35 EVT_FC_RIGHT_DOWN = wx.NewEventType()
36 EVT_FC_RIGHT_UP = wx.NewEventType()
37 EVT_FC_RIGHT_DCLICK = wx.NewEventType()
38 EVT_FC_MOTION = wx.NewEventType()
39 EVT_FC_MOUSEWHEEL = wx.NewEventType()
40 ## these two are for the hit-test stuff, I never make them real Events
41 EVT_FC_ENTER_OBJECT = wx.NewEventType()
42 EVT_FC_LEAVE_OBJECT = wx.NewEventType()
43
44 #def EVT_ENTER_WINDOW( window, function ):
45 # window.Connect( -1, -1, EVT_FC_ENTER_WINDOW, function )
46 #def EVT_LEAVE_WINDOW( window, function ):
47 # window.Connect( -1, -1,EVT_FC_LEAVE_WINDOW , function )
48 def EVT_LEFT_DOWN( window, function ):
49 window.Connect( -1, -1,EVT_FC_LEFT_DOWN , function )
50 def EVT_LEFT_UP( window, function ):
51 window.Connect( -1, -1,EVT_FC_LEFT_UP , function )
52 def EVT_LEFT_DCLICK ( window, function ):
53 window.Connect( -1, -1,EVT_FC_LEFT_DCLICK , function )
54 def EVT_MIDDLE_DOWN ( window, function ):
55 window.Connect( -1, -1,EVT_FC_MIDDLE_DOWN , function )
56 def EVT_MIDDLE_UP ( window, function ):
57 window.Connect( -1, -1,EVT_FC_MIDDLE_UP , function )
58 def EVT_MIDDLE_DCLICK ( window, function ):
59 window.Connect( -1, -1,EVT_FC_MIDDLE_DCLICK , function )
60 def EVT_RIGHT_DOWN ( window, function ):
61 window.Connect( -1, -1,EVT_FC_RIGHT_DOWN , function )
62 def EVT_RIGHT_UP( window, function ):
63 window.Connect( -1, -1,EVT_FC_RIGHT_UP , function )
64 def EVT_RIGHT_DCLICK( window, function ):
65 window.Connect( -1, -1,EVT_FC_RIGHT_DCLICK , function )
66 def EVT_MOTION( window, function ):
67 window.Connect( -1, -1,EVT_FC_MOTION , function )
68 def EVT_MOUSEWHEEL( window, function ):
69 window.Connect( -1, -1,EVT_FC_MOUSEWHEEL , function )
70
71 class MouseEvent(wx.PyCommandEvent):
72
73 """
74
75 This event class takes a regular wxWindows mouse event as a parameter,
76 and wraps it so that there is access to all the original methods. This
77 is similar to subclassing, but you can't subclass a wxWindows event
78
79 The goal is to be able to it just like a regular mouse event.
80
81 It adds the method:
82
83 GetCoords() , which returns and (x,y) tuple in world coordinates.
84
85 Another differnce is that it is a CommandEvent, which propagates up
86 the window hierarchy until it is handled.
87
88 """
89
90 def __init__(self, EventType, NativeEvent, WinID, Coords = None):
91 wx.PyCommandEvent.__init__(self)
92
93 self.SetEventType( EventType )
94 self._NativeEvent = NativeEvent
95 self.Coords = Coords
96
97 def SetCoords(self,Coords):
98 self.Coords = Coords
99
100 def GetCoords(self):
101 return self.Coords
102
103 def __getattr__(self, name):
104 #return eval(self.NativeEvent.__getattr__(name) )
105 return getattr(self._NativeEvent, name)
106
107 #### ColorGEnerator class is now obsolete. I'm using a python generator function instead.
108 ##class ColorGenerator:
109
110 ## """
111
112 ## An instance of this class generates a unique color each time
113 ## GetNextColor() is called. Someday I will use a proper Python
114 ## generator for this class.
115
116 ## The point of this generator is for the hit-test bitmap, each object
117 ## needs to be a unique color. Also, each system can be running a
118 ## different number of colors, and it doesn't appear to be possible to
119 ## have a wxMemDC with a different colordepth as the screen so this
120 ## generates colors far enough apart that they can be distinguished on
121 ## a 16bit screen. Anything less than 16bits won't work. It could, but
122 ## I havn't written the code that way. You also wouldn't get many
123 ## distict colors
124
125 ## """
126
127 ## def __init__(self):
128 ## import sys
129 ## ## figure out the color depth of the screen
130 ## ## for some bizare reason, thisdoesn't work on OS-X
131 ## if sys.platform == 'darwin':
132 ## depth = 24
133 ## else:
134 ## b = wx.EmptyBitmap(1,1)
135 ## depth = b.GetDepth()
136 ## self.r = 0
137 ## self.g = 0
138 ## self.b = 0
139 ## if depth == 16:
140 ## self.step = 8
141 ## elif depth >= 24:
142 ## self.step = 1
143 ## else:
144 ## raise FloatCanvasException("ColorGenerator does not work with depth = %s"%depth )
145
146 ## def GetNextColor(self):
147 ## step = self.step
148 ## ##r,g,b = self.r,self.g,self.b
149 ## self.r += step
150 ## if self.r > 255:
151 ## self.r = step
152 ## self.g += step
153 ## if self.g > 255:
154 ## self.g = step
155 ## self.b += step
156 ## if self.b > 255:
157 ## ## fixme: this should be a derived exception
158 ## raise FloatCanvasException("Too many objects in colorgenerator for HitTest")
159 ## return (self.r,self.g,self.b)
160
161 ## def Reset(self):
162 ## self.r = 0
163 ## self.g = 0
164 ## self.b = 0
165
166 def cycleidxs(indexcount, maxvalue, step):
167 if indexcount == 0:
168 yield ()
169 else:
170 for idx in xrange(0, maxvalue, step):
171 for tail in cycleidxs(indexcount - 1, maxvalue, step):
172 yield (idx, ) + tail
173
174 def colorGenerator():
175 import sys
176 if sys.platform == 'darwin':
177 depth = 24
178 else:
179 b = wx.EmptyBitmap(1,1)
180 depth = b.GetDepth()
181 if depth == 16:
182 step = 8
183 elif depth >= 24:
184 step = 1
185 else:
186 raise "ColorGenerator does not work with depth = %s" % depth
187 return cycleidxs(indexcount=3, maxvalue=256, step=step)
188
189
190 #### I don't know if the Set objects are useful, beyond the pointset object
191 #### The problem is that when zoomed in, the BB is checked to see whether to draw the object.
192 #### A Set object can defeat this
193
194 ##class ObjectSetMixin:
195 ## """
196 ## A mix-in class for draw objects that are sets of objects
197
198 ## It contains methods for setting lists of pens and brushes
199
200 ## """
201 ## def SetPens(self,LineColors,LineStyles,LineWidths):
202 ## """
203 ## This method used when an object could have a list of pens, rather than just one
204 ## It is used for LineSet, and perhaps others in the future.
205
206 ## fixme: this should be in a mixin
207
208 ## fixme: this is really kludgy, there has got to be a better way!
209
210 ## """
211
212 ## length = 1
213 ## if type(LineColors) == types.ListType:
214 ## length = len(LineColors)
215 ## else:
216 ## LineColors = [LineColors]
217
218 ## if type(LineStyles) == types.ListType:
219 ## length = len(LineStyles)
220 ## else:
221 ## LineStyles = [LineStyles]
222
223 ## if type(LineWidths) == types.ListType:
224 ## length = len(LineWidths)
225 ## else:
226 ## LineWidths = [LineWidths]
227
228 ## if length > 1:
229 ## if len(LineColors) == 1:
230 ## LineColors = LineColors*length
231 ## if len(LineStyles) == 1:
232 ## LineStyles = LineStyles*length
233 ## if len(LineWidths) == 1:
234 ## LineWidths = LineWidths*length
235
236 ## self.Pens = []
237 ## for (LineColor,LineStyle,LineWidth) in zip(LineColors,LineStyles,LineWidths):
238 ## if LineColor is None or LineStyle is None:
239 ## self.Pens.append(wx.TRANSPARENT_PEN)
240 ## # what's this for?> self.LineStyle = 'Transparent'
241 ## if not self.PenList.has_key((LineColor,LineStyle,LineWidth)):
242 ## Pen = wx.Pen(LineColor,LineWidth,self.LineStyleList[LineStyle])
243 ## self.Pens.append(Pen)
244 ## else:
245 ## self.Pens.append(self.PenList[(LineColor,LineStyle,LineWidth)])
246 ## if length == 1:
247 ## self.Pens = self.Pens[0]
248
249
250
251 class DrawObject:
252 """
253 This is the base class for all the objects that can be drawn.
254
255 """
256
257 def __init__(self,InForeground = False):
258 self.InForeground = InForeground
259
260 self._Canvas = None
261
262 self.HitColor = None
263 self.CallBackFuncs = {}
264
265 ## these are the defaults
266 self.HitAble = False
267 self.HitLine = True
268 self.HitFill = True
269 self.MinHitLineWidth = 3
270 self.HitLineWidth = 3 ## this gets re-set by the subclasses if necessary
271
272 # I pre-define all these as class variables to provide an easier
273 # interface, and perhaps speed things up by caching all the Pens
274 # and Brushes, although that may not help, as I think wx now
275 # does that on it's own. Send me a note if you know!
276
277 BrushList = {
278 ( None,"Transparent") : wx.TRANSPARENT_BRUSH,
279 ("Blue","Solid") : wx.BLUE_BRUSH,
280 ("Green","Solid") : wx.GREEN_BRUSH,
281 ("White","Solid") : wx.WHITE_BRUSH,
282 ("Black","Solid") : wx.BLACK_BRUSH,
283 ("Grey","Solid") : wx.GREY_BRUSH,
284 ("MediumGrey","Solid") : wx.MEDIUM_GREY_BRUSH,
285 ("LightGrey","Solid") : wx.LIGHT_GREY_BRUSH,
286 ("Cyan","Solid") : wx.CYAN_BRUSH,
287 ("Red","Solid") : wx.RED_BRUSH
288 }
289 PenList = {
290 (None,"Transparent",1) : wx.TRANSPARENT_PEN,
291 ("Green","Solid",1) : wx.GREEN_PEN,
292 ("White","Solid",1) : wx.WHITE_PEN,
293 ("Black","Solid",1) : wx.BLACK_PEN,
294 ("Grey","Solid",1) : wx.GREY_PEN,
295 ("MediumGrey","Solid",1) : wx.MEDIUM_GREY_PEN,
296 ("LightGrey","Solid",1) : wx.LIGHT_GREY_PEN,
297 ("Cyan","Solid",1) : wx.CYAN_PEN,
298 ("Red","Solid",1) : wx.RED_PEN
299 }
300
301 FillStyleList = {
302 "Transparent" : wx.TRANSPARENT,
303 "Solid" : wx.SOLID,
304 "BiDiagonalHatch": wx.BDIAGONAL_HATCH,
305 "CrossDiagHatch" : wx.CROSSDIAG_HATCH,
306 "FDiagonal_Hatch": wx.FDIAGONAL_HATCH,
307 "CrossHatch" : wx.CROSS_HATCH,
308 "HorizontalHatch": wx.HORIZONTAL_HATCH,
309 "VerticalHatch" : wx.VERTICAL_HATCH
310 }
311
312 LineStyleList = {
313 "Solid" : wx.SOLID,
314 "Transparent": wx.TRANSPARENT,
315 "Dot" : wx.DOT,
316 "LongDash" : wx.LONG_DASH,
317 "ShortDash" : wx.SHORT_DASH,
318 "DotDash" : wx.DOT_DASH,
319 }
320
321 def Bind(self, Event, CallBackFun):
322 self.CallBackFuncs[Event] = CallBackFun
323 self.HitAble = True
324 self._Canvas.UseHitTest = True
325 if not self._Canvas._HTdc:
326 self._Canvas.MakeNewHTdc()
327 if not self.HitColor:
328 if not self._Canvas.HitColorGenerator:
329 self._Canvas.HitColorGenerator = colorGenerator()
330 self._Canvas.HitColorGenerator.next() # first call to prevent the background color from being used.
331 self.HitColor = self._Canvas.HitColorGenerator.next()
332 self.SetHitPen(self.HitColor,self.HitLineWidth)
333 self.SetHitBrush(self.HitColor)
334 # put the object in the hit dict, indexed by it's color
335 if not self._Canvas.HitDict:
336 self._Canvas.MakeHitDict()
337 self._Canvas.HitDict[Event][self.HitColor] = (self) # put the object in the hit dict, indexed by it's color
338
339
340 def UnBindAll(self):
341 ## fixme: this only removes one from each list, there could be more.
342 if self._Canvas.HitDict:
343 for List in self._Canvas.HitDict.itervalues():
344 try:
345 List.remove(self)
346 except ValueError:
347 pass
348 self.HitAble = False
349
350 def SetBrush(self,FillColor,FillStyle):
351 if FillColor is None or FillStyle is None:
352 self.Brush = wx.TRANSPARENT_BRUSH
353 self.FillStyle = "Transparent"
354 else:
355 self.Brush = self.BrushList.setdefault( (FillColor,FillStyle), wx.Brush(FillColor,self.FillStyleList[FillStyle] ) )
356
357 def SetPen(self,LineColor,LineStyle,LineWidth):
358 if (LineColor is None) or (LineStyle is None):
359 self.Pen = wx.TRANSPARENT_PEN
360 self.LineStyle = 'Transparent'
361 else:
362 self.Pen = self.PenList.setdefault( (LineColor,LineStyle,LineWidth), wx.Pen(LineColor,LineWidth,self.LineStyleList[LineStyle]) )
363
364 def SetHitBrush(self,HitColor):
365 if not self.HitFill:
366 self.HitBrush = wx.TRANSPARENT_BRUSH
367 else:
368 self.HitBrush = self.BrushList.setdefault( (HitColor,"solid"), wx.Brush(HitColor,self.FillStyleList["Solid"] ) )
369
370 def SetHitPen(self,HitColor,LineWidth):
371 if not self.HitLine:
372 self.HitPen = wx.TRANSPARENT_PEN
373 else:
374 self.HitPen = self.PenList.setdefault( (HitColor, "solid", LineWidth), wx.Pen(HitColor, LineWidth, self.LineStyleList["Solid"]) )
375
376 def PutInBackground(self):
377 if self._Canvas and self.InForeground:
378 self._Canvas._ForeDrawList.remove(self)
379 self._Canvas._DrawList.append(self)
380 self._Canvas._BackgroundDirty = True
381 self.InForeground = False
382
383 def PutInForeground(self):
384 if self._Canvas and (not self.InForeground):
385 self._Canvas._ForeDrawList.append(self)
386 self._Canvas._DrawList.remove(self)
387 self._Canvas._BackgroundDirty = True
388 self.InForeground = True
389
390 class XYObjectMixin:
391 """
392
393 This is a mixin class that provides some methods suitable for use
394 with objects that have a single (x,y) coordinate pair.
395
396 """
397
398 def Move(self, Delta ):
399 """
400
401 Move(Delta): moves the object by delta, where delta is a
402 (dx,dy) pair. Ideally a Numpy array of shape (2,)
403
404 """
405
406 Delta = asarray(Delta, Float)
407 self.XY += Delta
408 self.BoundingBox = self.BoundingBox + Delta
409 if self._Canvas:
410 self._Canvas.BoundingBoxDirty = True
411
412 class PointsObjectMixin:
413 """
414
415 This is a mixin class that provides some methods suitable for use
416 with objects that have a set of (x,y) coordinate pairs.
417
418 """
419
420 ## This is code for the XYMixin object, it needs to be adapeted and tested.
421 ## def Move(self, Delta ):
422 ## """
423
424 ## Move(Delta): moves the object by delta, where delta is an (dx,
425 ## dy) pair. Ideally a Numpy array or shape (2,)
426
427 ## """
428
429 ## Delta = array(Delta, Float)
430 ## self.XY += Delta
431 ## self.BoundingBox = self.BoundingBox + Delta##array((self.XY, (self.XY + self.WH)), Float)
432 ## if self._Canvas:
433 ## self._Canvas.BoundingBoxDirty = True
434
435 def SetPoints(self,Points):
436 self.Points = Points
437 self.BoundingBox = array(((min(self.Points[:,0]),min(self.Points[:,1])),(max(self.Points[:,0]),max(self.Points[:,1]))),Float)
438 if self._Canvas:
439 self._Canvas.BoundingBoxDirty = True
440
441
442
443 class Polygon(DrawObject,PointsObjectMixin):
444
445 """
446
447 The Polygon class takes a list of 2-tuples, or a NX2 NumPy array of
448 point coordinates. so that Points[N][0] is the x-coordinate of
449 point N and Points[N][1] is the y-coordinate or Points[N,0] is the
450 x-coordinate of point N and Points[N,1] is the y-coordinate for
451 arrays.
452
453 """
454 def __init__(self,
455 Points,
456 LineColor = "Black",
457 LineStyle = "Solid",
458 LineWidth = 1,
459 FillColor = None,
460 FillStyle = "Solid",
461 InForeground = False):
462 DrawObject.__init__(self,InForeground)
463 self.Points = array(Points,Float) # this DOES need to make a copy
464 self.BoundingBox = array(((min(self.Points[:,0]),min(self.Points[:,1])),(max(self.Points[:,0]),max(self.Points[:,1]))),Float)
465
466 self.LineColor = LineColor
467 self.LineStyle = LineStyle
468 self.LineWidth = LineWidth
469 self.FillColor = FillColor
470 self.FillStyle = FillStyle
471
472 self.HitLineWidth = max(LineWidth,self.MinHitLineWidth)
473
474 self.SetPen(LineColor,LineStyle,LineWidth)
475 self.SetBrush(FillColor,FillStyle)
476
477 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel = None, HTdc=None):
478 Points = WorldToPixel(self.Points)
479 dc.SetPen(self.Pen)
480 dc.SetBrush(self.Brush)
481 dc.DrawPolygon(Points)
482 if HTdc and self.HitAble:
483 HTdc.SetPen(self.HitPen)
484 HTdc.SetBrush(self.HitBrush)
485 HTdc.DrawPolygon(Points)
486
487 ##class PolygonSet(DrawObject):
488 ## """
489 ## The PolygonSet class takes a Geometry.Polygon object.
490 ## so that Points[N] = (x1,y1) and Points[N+1] = (x2,y2). N must be an even number!
491
492 ## it creates a set of line segments, from (x1,y1) to (x2,y2)
493
494 ## """
495
496 ## def __init__(self,PolySet,LineColors,LineStyles,LineWidths,FillColors,FillStyles,InForeground = False):
497 ## DrawObject.__init__(self, InForeground)
498
499 ## ##fixme: there should be some error checking for everything being the right length.
500
501
502 ## self.Points = array(Points,Float)
503 ## self.BoundingBox = array(((min(self.Points[:,0]),min(self.Points[:,1])),(max(self.Points[:,0]),max(self.Points[:,1]))),Float)
504
505 ## self.LineColors = LineColors
506 ## self.LineStyles = LineStyles
507 ## self.LineWidths = LineWidths
508 ## self.FillColors = FillColors
509 ## self.FillStyles = FillStyles
510
511 ## self.SetPens(LineColors,LineStyles,LineWidths)
512
513 ## #def _Draw(self,dc,WorldToPixel,ScaleWorldToPixel):
514 ## def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
515 ## Points = WorldToPixel(self.Points)
516 ## Points.shape = (-1,4)
517 ## dc.DrawLineList(Points,self.Pens)
518
519
520 class Line(DrawObject,PointsObjectMixin):
521 """
522 The Line class takes a list of 2-tuples, or a NX2 NumPy array of point coordinates.
523 so that Points[N][0] is the x-coordinate of point N and Points[N][1] is the y-coordinate
524 or Points[N,0] is the x-coordinate of point N and Points[N,1] is the y-coordinate for arrays.
525
526 It will draw a straight line if there are two points, and a polyline if there are more than two.
527
528 """
529 def __init__(self,Points,
530 LineColor = "Black",
531 LineStyle = "Solid",
532 LineWidth = 1,
533 InForeground = False):
534 DrawObject.__init__(self, InForeground)
535
536
537 self.Points = array(Points,Float)
538 self.BoundingBox = array(((min(self.Points[:,0]),min(self.Points[:,1])),(max(self.Points[:,0]),max(self.Points[:,1]))),Float)
539
540 self.LineColor = LineColor
541 self.LineStyle = LineStyle
542 self.LineWidth = LineWidth
543
544 self.SetPen(LineColor,LineStyle,LineWidth)
545
546 self.HitLineWidth = max(LineWidth,self.MinHitLineWidth)
547
548
549 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
550 Points = WorldToPixel(self.Points)
551 dc.SetPen(self.Pen)
552 dc.DrawLines(Points)
553 if HTdc and self.HitAble:
554 HTdc.SetPen(self.HitPen)
555 HTdc.DrawLines(Points)
556
557 ##class LineSet(DrawObject, ObjectSetMixin):
558 ## """
559 ## The LineSet class takes a list of 2-tuples, or a NX2 NumPy array of point coordinates.
560 ## so that Points[N] = (x1,y1) and Points[N+1] = (x2,y2). N must be an even number!
561
562 ## it creates a set of line segments, from (x1,y1) to (x2,y2)
563
564 ## """
565
566 ## def __init__(self,Points,LineColors,LineStyles,LineWidths,InForeground = False):
567 ## DrawObject.__init__(self, InForeground)
568
569 ## NumLines = len(Points) / 2
570 ## ##fixme: there should be some error checking for everything being the right length.
571
572
573 ## self.Points = array(Points,Float)
574 ## self.BoundingBox = array(((min(self.Points[:,0]),min(self.Points[:,1])),(max(self.Points[:,0]),max(self.Points[:,1]))),Float)
575
576 ## self.LineColors = LineColors
577 ## self.LineStyles = LineStyles
578 ## self.LineWidths = LineWidths
579
580 ## self.SetPens(LineColors,LineStyles,LineWidths)
581
582 ## #def _Draw(self,dc,WorldToPixel,ScaleWorldToPixel):
583 ## def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
584 ## Points = WorldToPixel(self.Points)
585 ## Points.shape = (-1,4)
586 ## dc.DrawLineList(Points,self.Pens)
587
588 class PointSet(DrawObject):
589 """
590 The PointSet class takes a list of 2-tuples, or a NX2 NumPy array of point coordinates.
591 so that Points[N][0] is the x-coordinate of point N and Points[N][1] is the y-coordinate
592 or Points[N,0] is the x-coordinate of point N and Points[N,1] is the y-coordinate for arrays.
593
594 Each point will be drawn the same color and Diameter. The Diameter is in screen points,
595 not world coordinates.
596
597 At this point, the hit-test code does not distingish between the
598 points, you will only know that one of the poins got hit, not which
599 one.
600
601 In the case of points, the HitLineWidth is used as diameter.
602
603 """
604 def __init__(self, Points, Color = "Black", Diameter = 1, InForeground = False):
605 DrawObject.__init__(self,InForeground)
606
607 self.Points = array(Points,Float)
608 self.Points.shape = (-1,2) # Make sure it is a NX2 array, even if there is only one point
609 self.BoundingBox = array(((min(self.Points[:,0]),
610 min(self.Points[:,1])),
611 (max(self.Points[:,0]),
612 max(self.Points[:,1]))),Float)
613
614 self.Color = Color
615 self.Diameter = Diameter
616
617 self.HitLineWidth = self.MinHitLineWidth
618 self.SetPen(Color,"Solid",1)
619 self.SetBrush(Color,"Solid")
620
621 def SetPoints(self,Points):
622 self.Points = array(Points, Float)
623 self.Points.shape = (-1,2) # Make sure it is a NX2 array, even if there is only one point
624 self.BoundingBox = array(((min(self.Points[:,0]),
625 min(self.Points[:,1]) ),
626 (max(self.Points[:,0]),
627 max(self.Points[:,1]) ) ) )
628 if self._Canvas:
629 self._Canvas.BoundingBoxDirty = True
630
631 def DrawD2(self, dc, Points):
632 # A Little optimization for a diameter2 - point
633 dc.DrawPointList(Points)
634 dc.DrawPointList(Points + (1,0))
635 dc.DrawPointList(Points + (0,1))
636 dc.DrawPointList(Points + (1,1))
637
638 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
639 dc.SetPen(self.Pen)
640 Points = WorldToPixel(self.Points)
641 if self.Diameter <= 1:
642 dc.DrawPointList(Points)
643 elif self.Diameter <= 2:
644 self.DrawD2(dc, Points)
645 else:
646 dc.SetBrush(self.Brush)
647 radius = int(round(self.Diameter/2))
648 for xy in Points:
649 dc.DrawEllipsePointSize( (xy - radius), (self.Diameter, self.Diameter) )
650 if HTdc and self.HitAble:
651 HTdc.SetPen(self.HitPen)
652 if self.Diameter <= 1:
653 HTdc.DrawPointList(Points)
654 elif self.Diameter <= 2:
655 self.DrawD2(HTdc, Points)
656 else:
657 HTdc.SetBrush(self.HitBrush)
658 radius = int(round(self.Diameter/2))
659 for xy in Points:
660 HTdc.DrawEllipsePointSize( (xy - radius), (self.Diameter, self.Diameter) )
661
662 #### Does anyone need this?
663 ##class Dot(DrawObject):
664 ## """
665 ## The Dot class takes an x.y coordinate pair, and the Diameter of the circle.
666 ## The Diameter is in pixels, so it won't change with zoom.
667
668 ## Also Fill and line data
669
670 ## """
671 ## def __init__(self,x,y,Diameter,LineColor,LineStyle,LineWidth,FillColor,FillStyle,InForeground = False):
672 ## DrawObject.__init__(self,InForeground)
673
674 ## self.X = x
675 ## self.Y = y
676 ## self.Diameter = Diameter
677 ## # NOTE: the bounding box does not include the diameter of the dot, as that is in pixel coords.
678 ## # If this is a problem, perhaps you should use a circle, instead!
679 ## self.BoundingBox = array(((x,y),(x,y)),Float)
680
681 ## self.LineColor = LineColor
682 ## self.LineStyle = LineStyle
683 ## self.LineWidth = LineWidth
684 ## self.FillColor = FillColor
685 ## self.FillStyle = FillStyle
686
687 ## self.SetPen(LineColor,LineStyle,LineWidth)
688 ## self.SetBrush(FillColor,FillStyle)
689
690 ## def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
691 ## #def _Draw(self,dc,WorldToPixel,ScaleWorldToPixel):
692 ## dc.SetPen(self.Pen)
693 ## dc.SetBrush(self.Brush)
694 ## radius = int(round(self.Diameter/2))
695 ## (X,Y) = WorldToPixel((self.X,self.Y))
696 ## dc.DrawEllipse((X - radius), (Y - radius), self.Diameter, self.Diameter)
697
698 class RectEllipse(DrawObject, XYObjectMixin):
699 def __init__(self,x,y,width,height,
700 LineColor = "Black",
701 LineStyle = "Solid",
702 LineWidth = 1,
703 FillColor = None,
704 FillStyle = "Solid",
705 InForeground = False):
706
707 DrawObject.__init__(self,InForeground)
708
709 self.XY = array( (x, y), Float)
710 self.WH = array( (width, height), Float )
711 self.BoundingBox = array(((x,y), (self.XY + self.WH)), Float)
712 self.LineColor = LineColor
713 self.LineStyle = LineStyle
714 self.LineWidth = LineWidth
715 self.FillColor = FillColor
716 self.FillStyle = FillStyle
717
718 self.HitLineWidth = max(LineWidth,self.MinHitLineWidth)
719
720 self.SetPen(LineColor,LineStyle,LineWidth)
721 self.SetBrush(FillColor,FillStyle)
722
723
724 def SetUpDraw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc):
725 dc.SetPen(self.Pen)
726 dc.SetBrush(self.Brush)
727 if HTdc and self.HitAble:
728 HTdc.SetPen(self.HitPen)
729 HTdc.SetBrush(self.HitBrush)
730 return ( WorldToPixel(self.XY),
731 ScaleWorldToPixel(self.WH) )
732
733 def SetXY(self, x, y):
734 self.XY = array( (x, y), Float)
735 self.BoundingBox = array((self.XY, (self.XY + self.WH) ), Float)
736 if self._Canvas:
737 self._Canvas.BoundingBoxDirty = True
738
739
740 class Rectangle(RectEllipse):
741 # def __init__(*args, **kwargs):
742 # RectEllipse.__init__(*args, **kwargs)
743 # raise "an error"
744
745 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
746 ( XY, WH ) = self.SetUpDraw(dc,
747 WorldToPixel,
748 ScaleWorldToPixel,
749 HTdc)
750 dc.DrawRectanglePointSize(XY, WH)
751 if HTdc and self.HitAble:
752 HTdc.DrawRectanglePointSize(XY, WH)
753
754 class Ellipse(RectEllipse):
755 # def __init__(*args, **kwargs):
756 # RectEllipse.__init__(*args, **kwargs)
757
758 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
759 ( XY, WH ) = self.SetUpDraw(dc,
760 WorldToPixel,
761 ScaleWorldToPixel,
762 HTdc)
763 dc.DrawEllipsePointSize(XY, WH)
764 if HTdc and self.HitAble:
765 HTdc.DrawEllipsePointSize(XY, WH)
766
767 class Circle(Ellipse):
768 def __init__(self, x ,y, Diameter, **kwargs):
769 RectEllipse.__init__(self ,
770 x-Diameter/2.,
771 y-Diameter/2.,
772 Diameter,
773 Diameter,
774 **kwargs)
775
776 class TextObjectMixin:
777 """
778
779 A mix in class that holds attributes and methods that are needed by
780 the Text objects
781
782 """
783
784 ## I'm caching fonts, because on GTK, getting a new font can take a
785 ## while. However, it gets cleared after every full draw as hanging
786 ## on to a bunch of large fonts takes a massive amount of memory.
787
788 FontList = {}
789
790 def SetFont(self, Size, Family, Style, Weight, Underline, FaceName):
791 self.Font = self.FontList.setdefault( (Size,
792 Family,
793 Style,
794 Weight,
795 Underline,
796 FaceName),
797 wx.Font(Size,
798 Family,
799 Style,
800 Weight,
801 Underline,
802 FaceName) )
803 return self.Font
804
805 ## store the function that shift the coords for drawing text. The
806 ## "c" parameter is the correction for world coordinates, rather
807 ## than pixel coords as the y axis is reversed
808 ShiftFunDict = {'tl': lambda x, y, w, h, world=0: (x, y) ,
809 'tc': lambda x, y, w, h, world=0: (x - w/2, y) ,
810 'tr': lambda x, y, w, h, world=0: (x - w, y) ,
811 'cl': lambda x, y, w, h, world=0: (x, y - h/2 + world*h) ,
812 'cc': lambda x, y, w, h, world=0: (x - w/2, y - h/2 + world*h) ,
813 'cr': lambda x, y, w, h, world=0: (x - w, y - h/2 + world*h) ,
814 'bl': lambda x, y, w, h, world=0: (x, y - h + 2*world*h) ,
815 'bc': lambda x, y, w, h, world=0: (x - w/2, y - h + 2*world*h) ,
816 'br': lambda x, y, w, h, world=0: (x - w, y - h + 2*world*h)}
817
818 class Text(DrawObject, TextObjectMixin):
819 """
820 This class creates a text object, placed at the coordinates,
821 x,y. the "Position" argument is a two charactor string, indicating
822 where in relation to the coordinates the string should be oriented.
823
824 The first letter is: t, c, or b, for top, center and bottom The
825 second letter is: l, c, or r, for left, center and right The
826 position refers to the position relative to the text itself. It
827 defaults to "tl" (top left).
828
829 Size is the size of the font in pixels, or in points for printing
830 (if it ever gets implimented). Those will be the same, If you assume
831 72 PPI.
832
833 Family:
834 Font family, a generic way of referring to fonts without
835 specifying actual facename. One of:
836 wx.DEFAULT: Chooses a default font.
837 wx.DECORATIVE: A decorative font.
838 wx.ROMAN: A formal, serif font.
839 wx.SCRIPT: A handwriting font.
840 wx.SWISS: A sans-serif font.
841 wx.MODERN: A fixed pitch font.
842 NOTE: these are only as good as the wxWindows defaults, which aren't so good.
843 Style:
844 One of wx.NORMAL, wx.SLANT and wx.ITALIC.
845 Weight:
846 One of wx.NORMAL, wx.LIGHT and wx.BOLD.
847 Underline:
848 The value can be True or False. At present this may have an an
849 effect on Windows only.
850
851 Alternatively, you can set the kw arg: Font, to a wx.Font, and the above will be ignored.
852
853 The size is fixed, and does not scale with the drawing.
854
855 The hit-test is done on the entire text extent
856
857 """
858
859 def __init__(self,String,x,y,
860 Size = 12,
861 Color = "Black",
862 BackgroundColor = None,
863 Family = wx.MODERN,
864 Style = wx.NORMAL,
865 Weight = wx.NORMAL,
866 Underline = False,
867 Position = 'tl',
868 InForeground = False,
869 Font = None):
870
871 DrawObject.__init__(self,InForeground)
872
873 self.String = String
874 # Input size in in Pixels, compute points size from PPI info.
875 # fixme: for printing, we'll have to do something a little different
876 self.Size = int(round(72.0 * Size / ScreenPPI))
877
878 self.Color = Color
879 self.BackgroundColor = BackgroundColor
880
881 if not Font:
882 FaceName = ''
883 else:
884 FaceName = Font.GetFaceName()
885 Family = Font.GetFamily()
886 Size = Font.GetPointSize()
887 Style = Font.GetStyle()
888 Underlined = Font.GetUnderlined()
889 Weight = Font.GetWeight()
890 self.SetFont(Size, Family, Style, Weight, Underline, FaceName)
891
892 self.BoundingBox = array(((x,y),(x,y)),Float)
893
894 self.XY = ( x,y )
895
896 # use a memDC -- ScreenDC doesn't work with 2.5.1 and GTK2
897 #dc = wx.MemoryDC()
898 #bitmap = wx.EmptyBitmap(1, 1)
899 #dc.SelectObject(bitmap)
900 #dc.SetFont(self.Font)
901 #(self.TextWidth, self.TextHeight) = dc.GetTextExtent(self.String)
902 (self.TextWidth, self.TextHeight) = (None, None)
903 self.ShiftFun = self.ShiftFunDict[Position]
904
905 def SetXY(self, x, y):
906 self.XY = ( x,y )
907 self.BoundingBox = array((self.XY, self.XY),Float)
908 if self._Canvas:
909 self._Canvas.BoundingBoxDirty = True
910
911 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
912 XY = WorldToPixel(self.XY)
913 dc.SetFont(self.Font)
914 dc.SetTextForeground(self.Color)
915 if self.BackgroundColor:
916 dc.SetBackgroundMode(wx.SOLID)
917 dc.SetTextBackground(self.BackgroundColor)
918 else:
919 dc.SetBackgroundMode(wx.TRANSPARENT)
920 if self.TextWidth is None or self.TextHeight is None:
921 (self.TextWidth, self.TextHeight) = dc.GetTextExtent(self.String)
922 XY = self.ShiftFun(XY[0], XY[1], self.TextWidth, self.TextHeight)
923 dc.DrawTextPoint(self.String, XY)
924 if HTdc and self.HitAble:
925 HTdc.SetPen(self.HitPen)
926 HTdc.SetBrush(self.HitBrush)
927 HTdc.DrawRectanglePointSize(XY, (self.TextWidth, self.TextHeight) )
928
929 class ScaledText(DrawObject, TextObjectMixin, XYObjectMixin):
930 """
931 This class creates a text object that is scaled when zoomed. It is
932 placed at the coordinates, x,y. the "Position" argument is a two
933 charactor string, indicating where in relation to the coordinates
934 the string should be oriented.
935
936 The first letter is: t, c, or b, for top, center and bottom The
937 second letter is: l, c, or r, for left, center and right The
938 position refers to the position relative to the text itself. It
939 defaults to "tl" (top left).
940
941 Size is the size of the font in world coordinates.
942
943 Family:
944 Font family, a generic way of referring to fonts without
945 specifying actual facename. One of::
946 wx.DEFAULT: Chooses a default font.
947 wx.DECORATI: A decorative font.
948 wx.ROMAN: A formal, serif font.
949 wx.SCRIPT: A handwriting font.
950 wx.SWISS: A sans-serif font.
951 wx.MODERN: A fixed pitch font.
952
953 NOTE: these are only as good as the wxWindows defaults, which aren't so good.
954
955 Style:
956 One of wx.NORMAL, wx.SLANT and wx.ITALIC.
957
958 Weight:
959 One of wx.NORMAL, wx.LIGHT and wx.BOLD.
960
961 Underline:
962 The value can be True or False. At present this may have an an
963 effect on Windows only.
964
965 Alternatively, you can set the kw arg: Font, to a wx.Font, and the
966 above will be ignored. The size of the font you specify will be
967 ignored, but the rest of it's attributes will be preserved.
968
969 The size will scale as the drawing is zoomed.
970
971 Bugs/Limitations:
972
973 As fonts are scaled, the do end up a little different, so you don't
974 get exactly the same picture as you scale up and doen, but it's
975 pretty darn close.
976
977 On wxGTK1 on my Linux system, at least, using a font of over about
978 3000 pts. brings the system to a halt. It's the Font Server using
979 huge amounts of memory. My work around is to max the font size to
980 3000 points, so it won't scale past there. GTK2 uses smarter font
981 drawing, so that may not be an issue in future versions, so feel
982 free to test. Another smarter way to do it would be to set a global
983 zoom limit at that point.
984
985 The hit-test is done on the entire text extent. This could be made
986 optional, but I havn't gotten around to it.
987
988 """
989
990 def __init__(self, String, x, y , Size,
991 Color = "Black",
992 BackgroundColor = None,
993 Family = wx.MODERN,
994 Style = wx.NORMAL,
995 Weight = wx.NORMAL,
996 Underline = False,
997 Position = 'tl',
998 Font = None,
999 InForeground = False):
1000
1001 DrawObject.__init__(self,InForeground)
1002
1003 self.String = String
1004 self.XY = array( (x, y), Float)
1005 self.Size = Size
1006 self.Color = Color
1007 self.BackgroundColor = BackgroundColor
1008 self.Family = Family
1009 self.Style = Style
1010 self.Weight = Weight
1011 self.Underline = Underline
1012 if not Font:
1013 self.FaceName = ''
1014 else:
1015 self.FaceName = Font.GetFaceName()
1016 self.Family = Font.GetFamily()
1017 self.Style = Font.GetStyle()
1018 self.Underlined = Font.GetUnderlined()
1019 self.Weight = Font.GetWeight()
1020
1021 # Experimental max font size value on wxGTK2: this works OK on
1022 # my system If it's any larger, there is a crash, with the
1023 # message: The application 'FloatCanvasDemo.py' lost its
1024 # connection to the display :0.0; most likely the X server was
1025 # shut down or you killed/destroyed the application.
1026 self.MaxSize = 2750
1027
1028 self.ShiftFun = self.ShiftFunDict[Position]
1029
1030 ## Compute the BB
1031 ## this isn't exact, as fonts don't scale exactly.
1032 dc = wx.MemoryDC()
1033 bitmap = wx.EmptyBitmap(1, 1)
1034 dc.SelectObject(bitmap) #wxMac needs a Bitmap selected for GetTextExtent to work.
1035 DrawingSize = 40 # pts This effectively determines the resolution that the BB is computed to.
1036 ScaleFactor = float(Size) / DrawingSize
1037 dc.SetFont(self.SetFont(DrawingSize, self.Family, self.Style, self.Weight, self.Underline, self.FaceName) )
1038 (w,h) = dc.GetTextExtent(self.String)
1039 w = w * ScaleFactor
1040 h = h * ScaleFactor
1041 x, y = self.ShiftFun(x, y, w, h, world = 1)
1042 self.BoundingBox = array(((x, y-h ),(x + w, y)),Float)
1043
1044 # the new coords are set to the corner of the BB:
1045 #self.X = self.BoundingBox[0,0]
1046 #self.Y = self.BoundingBox[1,1]
1047 def _Draw(self, dc , WorldToPixel, ScaleWorldToPixel, HTdc=None):
1048 (X,Y) = WorldToPixel( (self.XY) )
1049
1050 # compute the font size:
1051 Size = abs( ScaleWorldToPixel( (self.Size, self.Size) )[1] ) # only need a y coordinate length
1052 ## Check to see if the font size is large enough to blow up the X font server
1053 ## If so, limit it. Would it be better just to not draw it?
1054 ## note that this limit is dependent on how much memory you have, etc.
1055 if Size > self.MaxSize:
1056 Size = self.MaxSize
1057 dc.SetFont(self.SetFont(Size, self.Family, self.Style, self.Weight, self.Underline, self.FaceName))
1058 dc.SetTextForeground(self.Color)
1059 if self.BackgroundColor:
1060 dc.SetBackgroundMode(wx.SOLID)
1061 dc.SetTextBackground(self.BackgroundColor)
1062 else:
1063 dc.SetBackgroundMode(wx.TRANSPARENT)
1064 (w,h) = dc.GetTextExtent(self.String)
1065 # compute the shift, and adjust the coordinates, if neccesary
1066 # This had to be put in here, because it changes with Zoom, as
1067 # fonts don't scale exactly.
1068 xy = self.ShiftFun(X, Y, w, h)
1069
1070 dc.DrawTextPoint(self.String, xy)
1071 if HTdc and self.HitAble:
1072 HTdc.SetPen(self.HitPen)
1073 HTdc.SetBrush(self.HitBrush)
1074 HTdc.DrawRectanglePointSize(xy, (w, h) )
1075
1076
1077 #---------------------------------------------------------------------------
1078 class FloatCanvas(wx.Panel):
1079 """
1080 FloatCanvas.py
1081
1082 This is a high level window for drawing maps and anything else in an
1083 arbitrary coordinate system.
1084
1085 The goal is to provide a convenient way to draw stuff on the screen
1086 without having to deal with handling OnPaint events, converting to pixel
1087 coordinates, knowing about wxWindows brushes, pens, and colors, etc. It
1088 also provides virtually unlimited zooming and scrolling
1089
1090 I am using it for two things:
1091 1) general purpose drawing in floating point coordinates
1092 2) displaying map data in Lat-long coordinates
1093
1094 If the projection is set to None, it will draw in general purpose
1095 floating point coordinates. If the projection is set to 'FlatEarth', it
1096 will draw a FlatEarth projection, centered on the part of the map that
1097 you are viewing. You can also pass in your own projection function.
1098
1099 It is double buffered, so re-draws after the window is uncovered by something
1100 else are very quick.
1101
1102 It relies on NumPy, which is needed for speed (maybe, I havn't profiled it)
1103
1104 Bugs and Limitations:
1105 Lots: patches, fixes welcome
1106
1107 For Map drawing: It ignores the fact that the world is, in fact, a
1108 sphere, so it will do strange things if you are looking at stuff near
1109 the poles or the date line. so far I don't have a need to do that, so I
1110 havn't bothered to add any checks for that yet.
1111
1112 Zooming:
1113 I have set no zoom limits. What this means is that if you zoom in really
1114 far, you can get integer overflows, and get wierd results. It
1115 doesn't seem to actually cause any problems other than wierd output, at
1116 least when I have run it.
1117
1118 Speed:
1119 I have done a couple of things to improve speed in this app. The one
1120 thing I have done is used NumPy Arrays to store the coordinates of the
1121 points of the objects. This allowed me to use array oriented functions
1122 when doing transformations, and should provide some speed improvement
1123 for objects with a lot of points (big polygons, polylines, pointsets).
1124
1125 The real slowdown comes when you have to draw a lot of objects, because
1126 you have to call the wx.DC.DrawSomething call each time. This is plenty
1127 fast for tens of objects, OK for hundreds of objects, but pretty darn
1128 slow for thousands of objects.
1129
1130 The solution is to be able to pass some sort of object set to the DC
1131 directly. I've used DC.DrawPointList(Points), and it helped a lot with
1132 drawing lots of points. I havn't got a LineSet type object, so I havn't
1133 used DC.DrawLineList yet. I'd like to get a full set of DrawStuffList()
1134 methods implimented, and then I'd also have a full set of Object sets
1135 that could take advantage of them. I hope to get to it some day.
1136
1137 Mouse Events:
1138
1139 At this point, there are a full set of custom mouse events. They are
1140 just like the rebulsr mouse events, but include an extra attribute:
1141 Event.GetCoords(), that returns the (x,y) position in world
1142 coordinates, as a length-2 NumPy vector of Floats.
1143
1144 Copyright: Christopher Barker
1145
1146 License: Same as the version of wxPython you are using it with
1147
1148 Please let me know if you're using this!!!
1149
1150 Contact me at:
1151
1152 Chris.Barker@noaa.gov
1153
1154 """
1155
1156 def __init__(self, parent, id = -1,
1157 size = wx.DefaultSize,
1158 ProjectionFun = None,
1159 BackgroundColor = "WHITE",
1160 Debug = False):
1161
1162 wx.Panel.__init__( self, parent, id, wx.DefaultPosition, size)
1163
1164 global ScreenPPI ## A global variable to hold the Pixels per inch that wxWindows thinks is in use.
1165 dc = wx.ScreenDC()
1166 ScreenPPI = dc.GetPPI()[0] # Assume square pixels
1167 del dc
1168
1169 self.HitColorGenerator = None
1170 self.UseHitTest = None
1171
1172 self.NumBetweenBlits = 500
1173
1174 self.BackgroundBrush = wx.Brush(BackgroundColor,wx.SOLID)
1175
1176 self.Debug = Debug
1177
1178 wx.EVT_PAINT(self, self.OnPaint)
1179 wx.EVT_SIZE(self, self.OnSize)
1180
1181 wx.EVT_LEFT_DOWN(self, self.LeftDownEvent )
1182 wx.EVT_LEFT_UP(self, self.LeftUpEvent )
1183 wx.EVT_LEFT_DCLICK(self, self.LeftDoubleClickEvent )
1184 wx.EVT_MIDDLE_DOWN(self, self.MiddleDownEvent )
1185 wx.EVT_MIDDLE_UP(self, self.MiddleUpEvent )
1186 wx.EVT_MIDDLE_DCLICK(self, self.MiddleDoubleClickEvent )
1187 wx.EVT_RIGHT_DOWN(self, self.RightDownEvent)
1188 wx.EVT_RIGHT_UP(self, self.RightUpEvent )
1189 wx.EVT_RIGHT_DCLICK(self, self.RightDoubleCLickEvent )
1190 wx.EVT_MOTION(self, self.MotionEvent )
1191 wx.EVT_MOUSEWHEEL(self, self.WheelEvent )
1192
1193 ## CHB: I'm leaving these out for now.
1194 #wx.EVT_ENTER_WINDOW(self, self. )
1195 #wx.EVT_LEAVE_WINDOW(self, self. )
1196
1197 ## create the Hit Test Dicts:
1198 self.HitDict = None
1199
1200
1201 self._DrawList = []
1202 self._ForeDrawList = []
1203 self._ForegroundBuffer = None
1204 self.BoundingBox = None
1205 self.BoundingBoxDirty = False
1206 self.ViewPortCenter= array( (0,0), Float)
1207
1208 self.SetProjectionFun(ProjectionFun)
1209
1210 self.MapProjectionVector = array( (1,1), Float) # No Projection to start!
1211 self.TransformVector = array( (1,-1), Float) # default Transformation
1212
1213 self.Scale = 1
1214
1215 self.GUIMode = None
1216 self.StartRBBox = None
1217 self.PrevRBBox = None
1218 self.StartMove = None
1219 self.PrevMoveXY = None
1220 self.ObjectUnderMouse = None
1221
1222 # called just to make sure everything is initialized
1223 self.OnSize(None)
1224
1225 self.InHereNum = 0
1226
1227 def SetProjectionFun(self,ProjectionFun):
1228 if ProjectionFun == 'FlatEarth':
1229 self.ProjectionFun = self.FlatEarthProjection
1230 elif type(ProjectionFun) == types.FunctionType:
1231 self.ProjectionFun = ProjectionFun
1232 elif ProjectionFun is None:
1233 self.ProjectionFun = lambda x=None: array( (1,1), Float)
1234 else:
1235 raise FloatCanvasException('Projectionfun must be either: "FlatEarth", None, or a function that takes the ViewPortCenter and returns a MapProjectionVector')
1236
1237 def FlatEarthProjection(self,CenterPoint):
1238 return array((cos(pi*CenterPoint[1]/180),1),Float)
1239
1240 def SetMode(self,Mode):
1241 if Mode in ["ZoomIn","ZoomOut","Move","Mouse",None]:
1242 self.GUIMode = Mode
1243 else:
1244 raise FloatCanvasException('"%s" is Not a valid Mode'%Mode)
1245
1246 def MakeHitDict(self):
1247 ##fixme: Should this just be None if nothing has been bound?
1248 self.HitDict = {EVT_FC_LEFT_DOWN: {},
1249 EVT_FC_LEFT_UP: {},
1250 EVT_FC_LEFT_DCLICK: {},
1251 EVT_FC_MIDDLE_DOWN: {},
1252 EVT_FC_MIDDLE_UP: {},
1253 EVT_FC_MIDDLE_DCLICK: {},
1254 EVT_FC_RIGHT_DOWN: {},
1255 EVT_FC_RIGHT_UP: {},
1256 EVT_FC_RIGHT_DCLICK: {},
1257 EVT_FC_ENTER_OBJECT: {},
1258 EVT_FC_LEAVE_OBJECT: {},
1259 }
1260
1261 def RaiseMouseEvent(self, Event, EventType):
1262 """
1263 This is called in various other places to raise a Mouse Event
1264 """
1265 #print "in Raise Mouse Event", Event
1266 pt = self.PixelToWorld( Event.GetPosition() )
1267 evt = MouseEvent(EventType, Event, self.GetId(), pt)
1268 self.GetEventHandler().ProcessEvent(evt)
1269
1270 def HitTest(self, event, HitEvent):
1271 if self.HitDict:
1272 # check if there are any objects in the dict for this event
1273 if self.HitDict[ HitEvent ]:
1274 xy = event.GetPosition()
1275 if self._ForegroundHTdc:
1276 hitcolor = self._ForegroundHTdc.GetPixelPoint( xy )
1277 else:
1278 hitcolor = self._HTdc.GetPixelPoint( xy )
1279 color = ( hitcolor.Red(), hitcolor.Green(), hitcolor.Blue() )
1280 if color in self.HitDict[ HitEvent ]:
1281 Object = self.HitDict[ HitEvent ][color]
1282 ## Add the hit coords to the Object
1283 Object.HitCoords = self.PixelToWorld( xy )
1284 Object.CallBackFuncs[HitEvent](Object)
1285 return True
1286 return False
1287
1288 def MouseOverTest(self, event):
1289 ##fixme: Can this be cleaned up?
1290 if self.HitDict:
1291 xy = event.GetPosition()
1292 if self._ForegroundHTdc:
1293 hitcolor = self._ForegroundHTdc.GetPixelPoint( xy )
1294 else:
1295 hitcolor = self._HTdc.GetPixelPoint( xy )
1296 color = ( hitcolor.Red(), hitcolor.Green(), hitcolor.Blue() )
1297 OldObject = self.ObjectUnderMouse
1298 ObjectCallbackCalled = False
1299 if color in self.HitDict[ EVT_FC_ENTER_OBJECT ]:
1300 Object = self.HitDict[ EVT_FC_ENTER_OBJECT][color]
1301 if (OldObject is None):
1302 try:
1303 Object.CallBackFuncs[EVT_FC_ENTER_OBJECT](Object)
1304 ObjectCallbackCalled = True
1305 except KeyError:
1306 pass # this means the enter event isn't bound for that object
1307 elif OldObject == Object: # the mouse is still on the same object
1308 pass
1309 ## Is the mouse on a differnt object as it was...
1310 elif not (Object == OldObject):
1311 # call the leave object callback
1312 try:
1313 OldObject.CallBackFuncs[EVT_FC_LEAVE_OBJECT](OldObject)
1314 ObjectCallbackCalled = True
1315 except KeyError:
1316 pass # this means the leave event isn't bound for that object
1317 try:
1318 Object.CallBackFuncs[EVT_FC_ENTER_OBJECT](Object)
1319 ObjectCallbackCalled = True
1320 except KeyError:
1321 pass # this means the enter event isn't bound for that object
1322 ## set the new object under mouse
1323 self.ObjectUnderMouse = Object
1324 elif color in self.HitDict[ EVT_FC_LEAVE_OBJECT ]:
1325 Object = self.HitDict[ EVT_FC_LEAVE_OBJECT][color]
1326 self.ObjectUnderMouse = Object
1327 else:
1328 # no objects under mouse bound to mouse-over events
1329 self.ObjectUnderMouse = None
1330 if OldObject:
1331 try:
1332 OldObject.CallBackFuncs[EVT_FC_LEAVE_OBJECT](OldObject)
1333 ObjectCallbackCalled = True
1334 except KeyError:
1335 pass # this means the leave event isn't bound for that object
1336 return ObjectCallbackCalled
1337
1338
1339 ## fixme: There is a lot of repeated code here
1340 ## Is there a better way?
1341 def LeftDoubleClickEvent(self,event):
1342 if self.GUIMode == "Mouse":
1343 EventType = EVT_FC_LEFT_DCLICK
1344 if not self.HitTest(event, EventType):
1345 self.RaiseMouseEvent(event, EventType)
1346
1347
1348 def MiddleDownEvent(self,event):
1349 if self.GUIMode == "Mouse":
1350 EventType = EVT_FC_MIDDLE_DOWN
1351 if not self.HitTest(event, EventType):
1352 self.RaiseMouseEvent(event, EventType)
1353
1354 def MiddleUpEvent(self,event):
1355 if self.GUIMode == "Mouse":
1356 EventType = EVT_FC_MIDDLE_UP
1357 if not self.HitTest(event, EventType):
1358 self.RaiseMouseEvent(event, EventType)
1359
1360 def MiddleDoubleClickEvent(self,event):
1361 if self.GUIMode == "Mouse":
1362 EventType = EVT_FC_MIDDLE_DCLICK
1363 if not self.HitTest(event, EventType):
1364 self.RaiseMouseEvent(event, EventType)
1365
1366 def RightUpEvent(self,event):
1367 if self.GUIMode == "Mouse":
1368 EventType = EVT_FC_RIGHT_UP
1369 if not self.HitTest(event, EventType):
1370 self.RaiseMouseEvent(event, EventType)
1371
1372 def RightDoubleCLickEvent(self,event):
1373 if self.GUIMode == "Mouse":
1374 EventType = EVT_FC_RIGHT_DCLICK
1375 if not self.HitTest(event, EventType):
1376 self.RaiseMouseEvent(event, EventType)
1377
1378 def WheelEvent(self,event):
1379 if self.GUIMode == "Mouse":
1380 self.RaiseMouseEvent(event, EVT_FC_MOUSEWHEEL)
1381
1382
1383 def LeftDownEvent(self,event):
1384 if self.GUIMode:
1385 if self.GUIMode == "ZoomIn":
1386 self.StartRBBox = array( event.GetPosition() )
1387 self.PrevRBBox = None
1388 self.CaptureMouse()
1389 elif self.GUIMode == "ZoomOut":
1390 Center = self.PixelToWorld( event.GetPosition() )
1391 self.Zoom(1/1.5,Center)
1392 elif self.GUIMode == "Move":
1393 self.StartMove = array( event.GetPosition() )
1394 self.PrevMoveXY = (0,0)
1395 elif self.GUIMode == "Mouse":
1396 ## check for a hit
1397 if not self.HitTest(event, EVT_FC_LEFT_DOWN):
1398 self.RaiseMouseEvent(event,EVT_FC_LEFT_DOWN)
1399 else:
1400 pass
1401
1402 def LeftUpEvent(self,event):
1403 if self.HasCapture():
1404 self.ReleaseMouse()
1405 if self.GUIMode:
1406 if self.GUIMode == "ZoomIn":
1407 if event.LeftUp() and not self.StartRBBox is None:
1408 self.PrevRBBox = None
1409 EndRBBox = event.GetPosition()
1410 StartRBBox = self.StartRBBox
1411 # if mouse has moved less that ten pixels, don't use the box.
1412 if ( abs(StartRBBox[0] - EndRBBox[0]) > 10
1413 and abs(StartRBBox[1] - EndRBBox[1]) > 10 ):
1414 EndRBBox = self.PixelToWorld(EndRBBox)
1415 StartRBBox = self.PixelToWorld(StartRBBox)
1416 BB = array(((min(EndRBBox[0],StartRBBox[0]),
1417 min(EndRBBox[1],StartRBBox[1])),
1418 (max(EndRBBox[0],StartRBBox[0]),
1419 max(EndRBBox[1],StartRBBox[1]))),Float)
1420 self.ZoomToBB(BB)
1421 else:
1422 Center = self.PixelToWorld(StartRBBox)
1423 self.Zoom(1.5,Center)
1424 self.StartRBBox = None
1425 elif self.GUIMode == "Move":
1426 if not self.StartMove is None:
1427 StartMove = self.StartMove
1428 EndMove = array((event.GetX(),event.GetY()))
1429 if sum((StartMove-EndMove)**2) > 16:
1430 self.Move(StartMove-EndMove,'Pixel')
1431 self.StartMove = None
1432 elif self.GUIMode == "Mouse":
1433 EventType = EVT_FC_LEFT_UP
1434 if not self.HitTest(event, EventType):
1435 self.RaiseMouseEvent(event, EventType)
1436 else:
1437 pass
1438
1439 def MotionEvent(self,event):
1440 if self.GUIMode:
1441 if self.GUIMode == "ZoomIn":
1442 if event.Dragging() and event.LeftIsDown() and not (self.StartRBBox is None):
1443 xy0 = self.StartRBBox
1444 xy1 = array( event.GetPosition() )
1445 wh = abs(xy1 - xy0)
1446 wh[0] = max(wh[0], int(wh[1]*self.AspectRatio))
1447 wh[1] = int(wh[0] / self.AspectRatio)
1448 xy_c = (xy0 + xy1) / 2
1449 dc = wx.ClientDC(self)
1450 dc.BeginDrawing()
1451 dc.SetPen(wx.Pen('WHITE', 2, wx.SHORT_DASH))
1452 dc.SetBrush(wx.TRANSPARENT_BRUSH)
1453 dc.SetLogicalFunction(wx.XOR)
1454 if self.PrevRBBox:
1455 dc.DrawRectanglePointSize(*self.PrevRBBox)
1456 self.PrevRBBox = ( xy_c - wh/2, wh )
1457 dc.DrawRectanglePointSize( *self.PrevRBBox )
1458 dc.EndDrawing()
1459 elif self.GUIMode == "Move":
1460 if event.Dragging() and event.LeftIsDown() and not self.StartMove is None:
1461 xy1 = array( event.GetPosition() )
1462 wh = self.PanelSize
1463 xy_tl = xy1 - self.StartMove
1464 dc = wx.ClientDC(self)
1465 dc.BeginDrawing()
1466 x1,y1 = self.PrevMoveXY
1467 x2,y2 = xy_tl
1468 w,h = self.PanelSize
1469 if x2 > x1 and y2 > y1:
1470 xa = xb = x1
1471 ya = yb = y1
1472 wa = w
1473 ha = y2 - y1
1474 wb = x2- x1
1475 hb = h
1476 elif x2 > x1 and y2 <= y1:
1477 xa = x1
1478 ya = y1
1479 wa = x2 - x1
1480 ha = h
1481 xb = x1
1482 yb = y2 + h
1483 wb = w
1484 hb = y1 - y2
1485 elif x2 <= x1 and y2 > y1:
1486 xa = x1
1487 ya = y1
1488 wa = w
1489 ha = y2 - y1
1490 xb = x2 + w
1491 yb = y1
1492 wb = x1 - x2
1493 hb = h - y2 + y1
1494 elif x2 <= x1 and y2 <= y1:
1495 xa = x2 + w
1496 ya = y1
1497 wa = x1 - x2
1498 ha = h
1499 xb = x1
1500 yb = y2 + h
1501 wb = w
1502 hb = y1 - y2
1503
1504 dc.SetPen(wx.TRANSPARENT_PEN)
1505 dc.SetBrush(self.BackgroundBrush)
1506 dc.DrawRectangle(xa, ya, wa, ha)
1507 dc.DrawRectangle(xb, yb, wb, hb)
1508 self.PrevMoveXY = xy_tl
1509 if self._ForegroundBuffer:
1510 dc.DrawBitmapPoint(self._ForegroundBuffer,xy_tl)
1511 else:
1512 dc.DrawBitmapPoint(self._Buffer,xy_tl)
1513 dc.EndDrawing()
1514 elif self.GUIMode == "Mouse":
1515 ## Only do something if there are mouse over events bound
1516 if self.HitDict and (self.HitDict[ EVT_FC_ENTER_OBJECT ] or self.HitDict[ EVT_FC_LEAVE_OBJECT ] ):
1517 if not self.MouseOverTest(event):
1518 self.RaiseMouseEvent(event,EVT_FC_MOTION)
1519 else:
1520 pass
1521 self.RaiseMouseEvent(event,EVT_FC_MOTION)
1522 else:
1523 pass
1524
1525 def RightDownEvent(self,event):
1526 if self.GUIMode:
1527 if self.GUIMode == "ZoomIn":
1528 Center = self.PixelToWorld((event.GetX(),event.GetY()))
1529 self.Zoom(1/1.5,Center)
1530 elif self.GUIMode == "ZoomOut":
1531 Center = self.PixelToWorld((event.GetX(),event.GetY()))
1532 self.Zoom(1.5,Center)
1533 elif self.GUIMode == "Mouse":
1534 EventType = EVT_FC_RIGHT_DOWN
1535 if not self.HitTest(event, EventType):
1536 self.RaiseMouseEvent(event, EventType)
1537 else:
1538 pass
1539
1540 def MakeNewBuffers(self):
1541 self._BackgroundDirty = True
1542 # Make new offscreen bitmap:
1543 self._Buffer = wx.EmptyBitmap(*self.PanelSize)
1544 #dc = wx.MemoryDC()
1545 #dc.SelectObject(self._Buffer)
1546 #dc.Clear()
1547 if self._ForeDrawList:
1548 self._ForegroundBuffer = wx.EmptyBitmap(*self.PanelSize)
1549 else:
1550 self._ForegroundBuffer = None
1551 if self.UseHitTest:
1552 self.MakeNewHTdc()
1553 else:
1554 self._HTdc = None
1555 self._ForegroundHTdc = None
1556
1557 def MakeNewHTdc(self):
1558 ## Note: While it's considered a "bad idea" to keep a
1559 ## MemoryDC around I'm doing it here because a wx.Bitmap
1560 ## doesn't have a GetPixel method so a DC is needed to do
1561 ## the hit-test. It didn't seem like a good idea to re-create
1562 ## a wx.MemoryDC on every single mouse event, so I keep it
1563 ## around instead
1564 self._HTdc = wx.MemoryDC()
1565 self._HTBitmap = wx.EmptyBitmap(*self.PanelSize)
1566 self._HTdc.SelectObject( self._HTBitmap )
1567 self._HTdc.SetBackground(wx.BLACK_BRUSH)
1568 if self._ForeDrawList:
1569 self._ForegroundHTdc = wx.MemoryDC()
1570 self._ForegroundHTBitmap = wx.EmptyBitmap(*self.PanelSize)
1571 self._ForegroundHTdc.SelectObject( self._ForegroundHTBitmap )
1572 self._ForegroundHTdc.SetBackground(wx.BLACK_BRUSH)
1573 else:
1574 self._ForegroundHTdc = None
1575
1576 def OnSize(self,event):
1577 self.PanelSize = array(self.GetClientSizeTuple(),Int32)
1578 self.HalfPanelSize = self.PanelSize / 2 # lrk: added for speed in WorldToPixel
1579 if self.PanelSize[0] == 0 or self.PanelSize[1] == 0:
1580 self.AspectRatio = 1.0
1581 else:
1582 self.AspectRatio = float(self.PanelSize[0]) / self.PanelSize[1]
1583 self.MakeNewBuffers()
1584 self.Draw()
1585
1586 def OnPaint(self, event):
1587 dc = wx.PaintDC(self)
1588 if self._ForegroundBuffer:
1589 dc.DrawBitmap(self._ForegroundBuffer,0,0)
1590 else:
1591 dc.DrawBitmap(self._Buffer,0,0)
1592
1593 def Draw(self, Force=False):
1594 """
1595 There is a main buffer set up to double buffer the screen, so
1596 you can get quick re-draws when the window gets uncovered.
1597
1598 If there are any objects in self._ForeDrawList, then the
1599 background gets drawn to a new buffer, and the foreground
1600 objects get drawn on top of it. The final result if blitted to
1601 the screen, and stored for future Paint events. This is done so
1602 that you can have a complicated background, but have something
1603 changing on the foreground, without having to wait for the
1604 background to get re-drawn. This can be used to support simple
1605 animation, for instance.
1606
1607 """
1608 #print "In Draw"
1609 if self.PanelSize < (1,1):
1610 return
1611 if self.Debug: start = clock()
1612 ScreenDC = wx.ClientDC(self)
1613 ViewPortWorld = ( self.PixelToWorld((0,0)),
1614 self.PixelToWorld(self.PanelSize) )
1615 ViewPortBB = array( ( minimum.reduce(ViewPortWorld),
1616 maximum.reduce(ViewPortWorld) ) )
1617 dc = wx.MemoryDC()
1618 dc.SelectObject(self._Buffer)
1619 if self._BackgroundDirty or Force:
1620 #print "Background is Dirty"
1621 dc.SetBackground(self.BackgroundBrush)
1622 dc.Clear()
1623 if self._HTdc:
1624 self._HTdc.Clear()
1625 self._DrawObjects(dc, self._DrawList, ScreenDC, ViewPortBB, self._HTdc)
1626 self._BackgroundDirty = False
1627
1628 if self._ForeDrawList:
1629 ## If an object was just added to the Foreground, there might not yet be a buffer
1630 if self._ForegroundBuffer is None:
1631 self._ForegroundBuffer = wx.EmptyBitmap(self.PanelSize[0],
1632 self.PanelSize[1])
1633
1634 dc = wx.MemoryDC() ## I got some strange errors (linewidths wrong) if I didn't make a new DC here
1635 dc.SelectObject(self._ForegroundBuffer)
1636 dc.DrawBitmap(self._Buffer,0,0)
1637 if self._ForegroundHTdc is None:
1638 self._ForegroundHTdc = wx.MemoryDC()
1639 self._ForegroundHTdc.SelectObject( wx.EmptyBitmap(
1640 self.PanelSize[0],
1641 self.PanelSize[1]) )
1642 if self._HTdc:
1643 ## blit the background HT buffer to the foreground HT buffer
1644 self._ForegroundHTdc.Blit(0, 0,
1645 self.PanelSize[0], self.PanelSize[1],
1646 self._HTdc, 0, 0)
1647 self._DrawObjects(dc,
1648 self._ForeDrawList,
1649 ScreenDC,
1650 ViewPortBB,
1651 self._ForegroundHTdc)
1652 ScreenDC.Blit(0, 0, self.PanelSize[0],self.PanelSize[1], dc, 0, 0)
1653 # If the canvas is in the middle of a zoom or move, the Rubber Band box needs to be re-drawn
1654 # This seeems out of place, but it works.
1655 if self.PrevRBBox:
1656 ScreenDC.SetPen(wx.Pen('WHITE', 2,wx.SHORT_DASH))
1657 ScreenDC.SetBrush(wx.TRANSPARENT_BRUSH)
1658 ScreenDC.SetLogicalFunction(wx.XOR)
1659 ScreenDC.DrawRectanglePointSize(*self.PrevRBBox)
1660 if self.Debug: print "Drawing took %f seconds of CPU time"%(clock()-start)
1661
1662 ## Clear the font cache
1663 ## IF you don't do this, the X font server starts to take up Massive amounts of memory
1664 ## This is mostly a problem with very large fonts, that you get with scaled text when zoomed in.
1665 DrawObject.FontList = {}
1666
1667 def _ShouldRedraw(DrawList, ViewPortBB): # lrk: adapted code from BBCheck
1668 # lrk: Returns the objects that should be redrawn
1669
1670 BB2 = ViewPortBB
1671 redrawlist = []
1672 for Object in DrawList:
1673 BB1 = Object.BoundingBox
1674 if (BB1[1,0] > BB2[0,0] and BB1[0,0] < BB2[1,0] and
1675 BB1[1,1] > BB2[0,1] and BB1[0,1] < BB2[1,1]):
1676 redrawlist.append(Object)
1677 return redrawlist
1678 _ShouldRedraw = staticmethod(_ShouldRedraw)
1679
1680
1681 ## def BBCheck(self, BB1, BB2):
1682 ## """
1683
1684 ## BBCheck(BB1, BB2) returns True is the Bounding boxes intesect, False otherwise
1685
1686 ## """
1687 ## if ( (BB1[1,0] > BB2[0,0]) and (BB1[0,0] < BB2[1,0]) and
1688 ## (BB1[1,1] > BB2[0,1]) and (BB1[0,1] < BB2[1,1]) ):
1689 ## return True
1690 ## else:
1691 ## return False
1692
1693 def Move(self,shift,CoordType):
1694 """
1695 move the image in the window.
1696
1697 shift is an (x,y) tuple, specifying the amount to shift in each direction
1698
1699 It can be in any of three coordinates: Panel, Pixel, World,
1700 specified by the CoordType parameter
1701
1702 Panel coordinates means you want to shift the image by some
1703 fraction of the size of the displaed image
1704
1705 Pixel coordinates means you want to shift the image by some number of pixels
1706
1707 World coordinates mean you want to shift the image by an amount
1708 in Floating point world coordinates
1709
1710 """
1711
1712 shift = array(shift,Float)
1713 if CoordType == 'Panel':# convert from panel coordinates
1714 shift = shift * array((-1,1),Float) *self.PanelSize/self.TransformVector
1715 elif CoordType == 'Pixel': # convert from pixel coordinates
1716 shift = shift/self.TransformVector
1717 elif CoordType == 'World': # No conversion
1718 pass
1719 else:
1720 raise FloatCanvasException('CoordType must be either "Panel", "Pixel", or "World"')
1721
1722 self.ViewPortCenter = self.ViewPortCenter + shift
1723 self.MapProjectionVector = self.ProjectionFun(self.ViewPortCenter)
1724 self.TransformVector = array((self.Scale,-self.Scale),Float) * self.MapProjectionVector
1725 self._BackgroundDirty = True
1726 self.Draw()
1727
1728 def Zoom(self,factor,center = None):
1729
1730 """
1731 Zoom(factor, center) changes the amount of zoom of the image by factor.
1732 If factor is greater than one, the image gets larger.
1733 If factor is less than one, the image gets smaller.
1734
1735 Center is a tuple of (x,y) coordinates of the center of the viewport, after zooming.
1736 If center is not given, the center will stay the same.
1737
1738 """
1739 self.Scale = self.Scale*factor
1740 if not center is None:
1741 self.ViewPortCenter = array(center,Float)
1742 self.MapProjectionVector = self.ProjectionFun(self.ViewPortCenter)
1743 self.TransformVector = array((self.Scale,-self.Scale),Float) * self.MapProjectionVector
1744 self._BackgroundDirty = True
1745 self.Draw()
1746
1747 def ZoomToBB(self, NewBB = None, DrawFlag = True):
1748
1749 """
1750
1751 Zooms the image to the bounding box given, or to the bounding
1752 box of all the objects on the canvas, if none is given.
1753
1754 """
1755
1756 if not NewBB is None:
1757 BoundingBox = NewBB
1758 else:
1759 if self.BoundingBoxDirty:
1760 self._ResetBoundingBox()
1761 BoundingBox = self.BoundingBox
1762 if not BoundingBox is None:
1763 self.ViewPortCenter = array(((BoundingBox[0,0]+BoundingBox[1,0])/2,
1764 (BoundingBox[0,1]+BoundingBox[1,1])/2 ),Float)
1765 self.MapProjectionVector = self.ProjectionFun(self.ViewPortCenter)
1766 # Compute the new Scale
1767 BoundingBox = BoundingBox * self.MapProjectionVector
1768 try:
1769 self.Scale = min(abs(self.PanelSize[0] / (BoundingBox[1,0]-BoundingBox[0,0])),
1770 abs(self.PanelSize[1] / (BoundingBox[1,1]-BoundingBox[0,1])) )*0.95
1771 except ZeroDivisionError: # this will happen if the BB has zero width or height
1772 try: #width == 0
1773 self.Scale = (self.PanelSize[0] / (BoundingBox[1,0]-BoundingBox[0,0]))*0.95
1774 except ZeroDivisionError:
1775 try: # height == 0
1776 self.Scale = (self.PanelSize[1] / (BoundingBox[1,1]-BoundingBox[0,1]))*0.95
1777 except ZeroDivisionError: #zero size! (must be a single point)
1778 self.Scale = 1
1779
1780 self.TransformVector = array((self.Scale,-self.Scale),Float)* self.MapProjectionVector
1781 if DrawFlag:
1782 self._BackgroundDirty = True
1783 self.Draw()
1784 else:
1785 # Reset the shifting and scaling to defaults when there is no BB
1786 self.ViewPortCenter= array( (0,0), Float)
1787 self.MapProjectionVector = array( (1,1), Float) # No Projection to start!
1788 self.TransformVector = array( (1,-1), Float) # default Transformation
1789 self.Scale = 1
1790
1791 def RemoveObjects(self, Objects):
1792 for Object in Objects:
1793 self.RemoveObject(Object, ResetBB = False)
1794 self.BoundingBoxDirty = True
1795
1796 def RemoveObject(self, Object, ResetBB = True):
1797 ##fixme: Using the list.remove method is kind of slow
1798 if Object.InForeground:
1799 self._ForeDrawList.remove(Object)
1800 else:
1801 self._DrawList.remove(Object)
1802 self._BackgroundDirty = True
1803 if ResetBB:
1804 self.BoundingBoxDirty = True
1805
1806 def ClearAll(self, ResetBB = True):
1807 self._DrawList = []
1808 self._ForeDrawList = []
1809 self._BackgroundDirty = True
1810 self.HitColorGenerator = None
1811 self.UseHitTest = False
1812 if ResetBB:
1813 self._ResetBoundingBox()
1814 self.MakeNewBuffers()
1815 self.HitDict = None
1816
1817 ## No longer called
1818 ## def _AddBoundingBox(self,NewBB):
1819 ## if self.BoundingBox is None:
1820 ## self.BoundingBox = NewBB
1821 ## self.ZoomToBB(NewBB,DrawFlag = False)
1822 ## else:
1823 ## self.BoundingBox = array( ( (min(self.BoundingBox[0,0],NewBB[0,0]),
1824 ## min(self.BoundingBox[0,1],NewBB[0,1])),
1825 ## (max(self.BoundingBox[1,0],NewBB[1,0]),
1826 ## max(self.BoundingBox[1,1],NewBB[1,1]))),
1827 ## Float)
1828
1829 def _getboundingbox(bboxarray): # lrk: added this
1830
1831 upperleft = minimum.reduce(bboxarray[:,0])
1832 lowerright = maximum.reduce(bboxarray[:,1])
1833 return array((upperleft, lowerright), Float)
1834
1835 _getboundingbox = staticmethod(_getboundingbox)
1836
1837 def _ResetBoundingBox(self):
1838 if self._DrawList or self._ForeDrawList:
1839 bboxarray = zeros((len(self._DrawList)+len(self._ForeDrawList), 2, 2),Float)
1840 i = -1 # just in case _DrawList is empty
1841 for (i, BB) in enumerate(self._DrawList):
1842 bboxarray[i] = BB.BoundingBox
1843 for (j, BB) in enumerate(self._ForeDrawList):
1844 bboxarray[i+j+1] = BB.BoundingBox
1845 self.BoundingBox = self._getboundingbox(bboxarray)
1846 else:
1847 self.BoundingBox = None
1848 self.ViewPortCenter= array( (0,0), Float)
1849 self.TransformVector = array( (1,-1), Float)
1850 self.MapProjectionVector = array( (1,1), Float)
1851 self.Scale = 1
1852 self.BoundingBoxDirty = False
1853
1854 def PixelToWorld(self,Points):
1855 """
1856 Converts coordinates from Pixel coordinates to world coordinates.
1857
1858 Points is a tuple of (x,y) coordinates, or a list of such tuples, or a NX2 Numpy array of x,y coordinates.
1859
1860 """
1861 return (((asarray(Points,Float) - (self.PanelSize/2))/self.TransformVector) + self.ViewPortCenter)
1862
1863 def WorldToPixel(self,Coordinates):
1864 """
1865 This function will get passed to the drawing functions of the objects,
1866 to transform from world to pixel coordinates.
1867 Coordinates should be a NX2 array of (x,y) coordinates, or
1868 a 2-tuple, or sequence of 2-tuples.
1869 """
1870 #Note: this can be called by users code for various reasons, so asarray is needed.
1871 return (((asarray(Coordinates,Float) - self.ViewPortCenter)*self.TransformVector)+(self.HalfPanelSize)).astype('i')
1872
1873 def ScaleWorldToPixel(self,Lengths):
1874 """
1875 This function will get passed to the drawing functions of the objects,
1876 to Change a length from world to pixel coordinates.
1877
1878 Lengths should be a NX2 array of (x,y) coordinates, or
1879 a 2-tuple, or sequence of 2-tuples.
1880 """
1881 return ( (asarray(Lengths,Float)*self.TransformVector) ).astype('i')
1882
1883 def ScalePixelToWorld(self,Lengths):
1884 """
1885 This function computes a pair of x.y lengths,
1886 to change then from pixel to world coordinates.
1887
1888 Lengths should be a NX2 array of (x,y) coordinates, or
1889 a 2-tuple, or sequence of 2-tuples.
1890 """
1891
1892 return (asarray(Lengths,Float) / self.TransformVector)
1893
1894 def AddObject(self,obj):
1895 # put in a reference to the Canvas, so remove and other stuff can work
1896 obj._Canvas = self
1897 if obj.InForeground:
1898 self._ForeDrawList.append(obj)
1899 self.UseForeground = True
1900 else:
1901 self._DrawList.append(obj)
1902 self._BackgroundDirty = True
1903 self.BoundingBoxDirty = True
1904 return True
1905
1906 def _DrawObjects(self, dc, DrawList, ScreenDC, ViewPortBB, HTdc = None):
1907 """
1908 This is a convenience function;
1909 This function takes the list of objects and draws them to specified
1910 device context.
1911 """
1912 dc.SetBackground(self.BackgroundBrush)
1913 dc.BeginDrawing()
1914 #i = 0
1915 PanelSize0, PanelSize1 = self.PanelSize # for speed
1916 WorldToPixel = self.WorldToPixel # for speed
1917 ScaleWorldToPixel = self.ScaleWorldToPixel # for speed
1918 Blit = ScreenDC.Blit # for speed
1919 NumBetweenBlits = self.NumBetweenBlits # for speed
1920 for i, Object in enumerate(self._ShouldRedraw(DrawList, ViewPortBB)):
1921 Object._Draw(dc, WorldToPixel, ScaleWorldToPixel, HTdc)
1922 if i % NumBetweenBlits == 0:
1923 Blit(0, 0, PanelSize0, PanelSize1, dc, 0, 0)
1924 dc.EndDrawing()
1925
1926 ## ## This is a way to automatically add a AddObject method for each
1927 ## ## object type This code has been replaced by Leo's code above, so
1928 ## ## that it happens at module init, rather than as needed. The
1929 ## ## primary advantage of this is that dir(FloatCanvas) will have
1930 ## ## them, and docstrings are preserved. Probably more useful
1931 ## ## exceptions if there is a problem, as well.
1932 ## def __getattr__(self, name):
1933 ## if name[:3] == "Add":
1934 ## func=globals()[name[3:]]
1935 ## def AddFun(*args, **kwargs):
1936 ## Object = func(*args, **kwargs)
1937 ## self.AddObject(Object)
1938 ## return Object
1939 ## ## add it to FloatCanvas' dict for future calls.
1940 ## self.__dict__[name] = AddFun
1941 ## return AddFun
1942 ## else:
1943 ## raise AttributeError("FloatCanvas has no attribute '%s'"%name)
1944
1945 def _makeFloatCanvasAddMethods(): ## lrk's code for doing this in module __init__
1946 classnames = ["Circle", "Ellipse", "Rectangle", "ScaledText", "Polygon",
1947 "Line", "Text", "PointSet"]
1948 for classname in classnames:
1949 klass = globals()[classname]
1950 def getaddshapemethod(klass=klass):
1951 def addshape(self, *args, **kwargs):
1952 Object = klass(*args, **kwargs)
1953 self.AddObject(Object)
1954 return Object
1955 return addshape
1956 addshapemethod = getaddshapemethod()
1957 methodname = "Add" + classname
1958 setattr(FloatCanvas, methodname, addshapemethod)
1959 docstring = " Creates %s and adds its reference to the canvas.\n" % classname
1960 docstring += " Argument protocol same as %s class" % classname
1961 if klass.__doc__:
1962 docstring += ", whose docstring is:\n%s" % klass.__doc__
1963 FloatCanvas.__dict__[methodname].__doc__ = docstring
1964
1965 _makeFloatCanvasAddMethods()
1966
1967