]> git.saurik.com Git - wxWidgets.git/blob - wxPython/demo/Joystick.py
03ba8e98c7cbc9b75c768f0f44e7b9cef6aab30f
[wxWidgets.git] / wxPython / demo / Joystick.py
1 #----------------------------------------------------------------------------
2 # Name: Joystick.py
3 # Purpose: Demonstrate use of wx.Joystick
4 #
5 # Author: Jeff Grimmett (grimmtoo@softhome.net), adapted from original
6 # .wdr-derived demo
7 #
8 # Created: 01/02/04
9 # RCS-ID: $Id$
10 # Copyright:
11 # Licence: wxWindows license
12 #----------------------------------------------------------------------------
13 #
14
15 import math
16 import wx
17
18 #----------------------------------------------------------------------------
19
20 # For convenience
21 spacer = (10, 10)
22 MAX_BUTTONS = 16
23
24 #----------------------------------------------------------------------------
25
26 class Label(wx.StaticText):
27 # A derived StaticText that always aligns right and renders
28 # in a bold font.
29 def __init__(self, parent, label):
30 wx.StaticText.__init__(self, parent, -1, label, style=wx.ALIGN_RIGHT)
31
32 self.SetFont(
33 wx.Font(
34 parent.GetFont().GetPointSize(),
35 parent.GetFont().GetFamily(),
36 parent.GetFont().GetStyle(),
37 wx.BOLD
38 ))
39
40 #----------------------------------------------------------------------------
41
42
43 class JoyGauge(wx.Panel):
44 def __init__(self, parent, stick):
45
46 self.stick = stick
47 size = (100,100)
48
49 wx.Panel.__init__(self, parent, -1, size=size)
50
51 self.Bind(wx.EVT_PAINT, self.OnPaint)
52 self.Bind(wx.EVT_SIZE, self.OnSize)
53 self.Bind(wx.EVT_ERASE_BACKGROUND, lambda e: None)
54
55 self.buffer = wx.EmptyBitmap(*size)
56 dc = wx.BufferedDC(None, self.buffer)
57 self.DrawFace(dc)
58 self.DrawJoystick(dc)
59
60
61 def OnSize(self, event):
62 # The face Bitmap init is done here, to make sure the buffer is always
63 # the same size as the Window
64 w, h = self.GetClientSize()
65 self.buffer = wx.EmptyBitmap(w,h)
66 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
67 self.DrawFace(dc)
68 self.DrawJoystick(dc)
69
70
71 def DrawFace(self, dc):
72 dc.SetBackground(wx.Brush(self.GetBackgroundColour()))
73 dc.Clear()
74
75
76 def OnPaint(self, evt):
77 # When dc is destroyed it will blit self.buffer to the window,
78 # since no other drawing is needed we'll just return and let it
79 # do it's thing
80 dc = wx.BufferedPaintDC(self, self.buffer)
81
82
83 def DrawJoystick(self, dc):
84 # draw the guage as a maxed square in the center of this window.
85 w, h = self.GetClientSize()
86 edgeSize = min(w, h)
87
88 xorigin = (w - edgeSize) / 2
89 yorigin = (h - edgeSize) / 2
90 center = edgeSize / 2
91
92 # Restrict our drawing activities to the square defined
93 # above.
94 dc.SetClippingRegion((xorigin, yorigin), (edgeSize, edgeSize))
95
96 # Optimize drawing a bit (for Win)
97 dc.BeginDrawing()
98
99 dc.SetBrush(wx.Brush(wx.Colour(251, 252, 237)))
100 dc.DrawRectangle((xorigin, yorigin), (edgeSize, edgeSize))
101
102 dc.SetPen(wx.Pen(wx.BLACK, 1, wx.DOT_DASH))
103
104 dc.DrawLine((xorigin, yorigin + center), (xorigin + edgeSize, yorigin + center))
105 dc.DrawLine((xorigin + center, yorigin), (xorigin + center, yorigin + edgeSize))
106
107 if self.stick:
108 # Get the joystick position as a float
109 joyx = float(self.stick.GetPosition().x)
110 joyy = float(self.stick.GetPosition().y)
111
112 # Get the joystick range of motion
113 xrange = self.stick.GetXMax() - self.stick.GetXMin()
114 yrange = self.stick.GetYMax() - self.stick.GetYMin()
115
116 # calc a ratio of our range versus the joystick range
117 xratio = float(edgeSize) / xrange
118 yratio = float(edgeSize) / yrange
119
120 # calc the displayable value based on position times ratio
121 xval = int(joyx * xratio)
122 yval = int(joyy * xratio)
123
124 # and normalize the value from our brush's origin
125 x = xval + xorigin
126 y = yval + yorigin
127
128 # Now to draw it.
129 dc.SetPen(wx.Pen(wx.RED, 2))
130 dc.CrossHair((x, y))
131
132 # Turn off drawing optimization
133 dc.EndDrawing()
134
135
136 def Update(self):
137 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
138 self.DrawFace(dc)
139 self.DrawJoystick(dc)
140
141
142 #----------------------------------------------------------------------------
143
144 class JoyPanel(wx.Panel):
145 def __init__(self, parent, stick):
146
147 self.stick = stick
148
149 wx.Panel.__init__(self, parent, -1)
150
151 sizer = wx.BoxSizer(wx.VERTICAL)
152
153 fn = wx.Font(
154 parent.GetFont().GetPointSize() + 3,
155 parent.GetFont().GetFamily(),
156 parent.GetFont().GetStyle(),
157 wx.BOLD
158 )
159
160 t = wx.StaticText(self, -1, "X - Y Axes", style = wx.ALIGN_CENTRE)
161 t.SetFont(fn)
162 sizer.Add(t, 0, wx.ALL | wx.EXPAND | wx.ALIGN_CENTER | wx.ALIGN_CENTER_HORIZONTAL, 1)
163
164 self.control = JoyGauge(self, self.stick)
165 sizer.Add(self.control, 1, wx.ALL | wx.EXPAND | wx.ALIGN_CENTER | wx.ALIGN_CENTER_HORIZONTAL, 1)
166
167 self.SetSizer(sizer)
168 sizer.Fit(self)
169
170 def Update(self):
171 self.control.Update()
172
173
174 #----------------------------------------------------------------------------
175
176 class POVGauge(wx.Panel):
177 #
178 # Display the current postion of the POV control
179 #
180 def __init__(self, parent, stick):
181
182 self.stick = stick
183 self.size = (100, 100)
184 self.avail = False
185 self.fourDir = False
186 self.cts = False
187
188 wx.Panel.__init__(self, parent, -1, size=self.size)
189
190 self.Bind(wx.EVT_PAINT, self.OnPaint)
191 self.Bind(wx.EVT_SIZE, self.OnSize)
192 self.Bind(wx.EVT_ERASE_BACKGROUND, lambda e: None)
193
194 self.buffer = wx.EmptyBitmap(*self.size)
195 dc = wx.BufferedDC(None, self.buffer)
196 self.DrawFace(dc)
197 self.DrawPOV(dc)
198
199
200 def OnSize(self, event):
201 # calculate the size of our display and make a buffer for it.
202 w, h = self.GetClientSize()
203 s = min(w, h)
204 self.size = (s, s)
205 self.buffer = wx.EmptyBitmap(w,h)
206 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
207 self.DrawFace(dc)
208 self.DrawPOV(dc)
209
210
211 def DrawFace(self, dc):
212 dc.SetBackground(wx.Brush(self.GetBackgroundColour()))
213 dc.Clear()
214
215
216 def OnPaint(self, evt):
217 # When dc is destroyed it will blit self.buffer to the window,
218 # since no other drawing is needed we'll just return and let it
219 # do it's thing
220 dc = wx.BufferedPaintDC(self, self.buffer)
221
222
223 def DrawPOV(self, dc):
224 # draw the guage as a maxed circle in the center of this window.
225 w, h = self.GetClientSize()
226 diameter = min(w, h)
227
228 xorigin = (w - diameter) / 2
229 yorigin = (h - diameter) / 2
230 xcenter = xorigin + diameter / 2
231 ycenter = yorigin + diameter / 2
232
233 # Optimize drawing a bit (for Win)
234 dc.BeginDrawing()
235
236 # our 'raster'.
237 dc.SetBrush(wx.Brush(wx.WHITE))
238 dc.DrawCircle((xcenter, ycenter), diameter/2)
239 dc.SetBrush(wx.Brush(wx.BLACK))
240 dc.DrawCircle((xcenter, ycenter), 10)
241
242 # fancy decorations
243 dc.SetPen(wx.Pen(wx.BLACK, 1, wx.DOT_DASH))
244 dc.DrawLine((xorigin, ycenter), (xorigin + diameter, ycenter))
245 dc.DrawLine((xcenter, yorigin), (xcenter, yorigin + diameter))
246
247 if self.stick:
248 if self.avail:
249
250 pos = -1
251
252 # use the appropriate function to get the POV position
253 if self.fourDir:
254 pos = self.stick.GetPOVPosition()
255
256 if self.cts:
257 pos = self.stick.GetPOVCTSPosition()
258
259 # trap invalid values
260 if 0 <= pos <= 36000:
261 vector = 30
262 else:
263 vector = 0
264
265 # rotate CCW by 90 so that 0 is up.
266 pos = (pos / 100) - 90
267
268 # Normalize
269 if pos < 0:
270 pos = pos + 360
271
272 # Stolen from wx.lib.analogclock :-)
273 radiansPerDegree = math.pi / 180
274 pointX = int(round(vector * math.cos(pos * radiansPerDegree)))
275 pointY = int(round(vector * math.sin(pos * radiansPerDegree)))
276
277 # normalise value to match our actual center.
278 nx = pointX + xcenter
279 ny = pointY + ycenter
280
281 # Draw the line
282 dc.SetPen(wx.Pen(wx.BLUE, 2))
283 dc.DrawLine((xcenter, ycenter), (nx, ny))
284
285 # And a little thing to show the endpoint
286 dc.SetBrush(wx.Brush(wx.BLUE))
287 dc.DrawCircle((nx, ny), 8)
288
289 # Turn off drawing optimization
290 dc.EndDrawing()
291
292
293 def Update(self):
294 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
295 self.DrawFace(dc)
296 self.DrawPOV(dc)
297
298
299 def Calibrate(self):
300 s = self.stick
301 self.avail = s.HasPOV()
302 self.fourDir = s.HasPOV4Dir()
303 self.cts = s.HasPOVCTS()
304
305
306 #----------------------------------------------------------------------------
307
308 class POVStatus(wx.Panel):
309 #
310 # Displays static info about the POV control
311 #
312 def __init__(self, parent, stick):
313
314 self.stick = stick
315
316 wx.Panel.__init__(self, parent, -1, size=(100, 100))
317
318 sizer = wx.BoxSizer(wx.VERTICAL)
319 sizer.Add((20,20))
320
321 self.avail = wx.CheckBox(self, -1, "Available")
322 sizer.Add(self.avail, 0, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 2)
323
324 self.fourDir = wx.CheckBox(self, -1, "4-Way Only")
325 sizer.Add(self.fourDir, 0, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 2)
326
327 self.cts = wx.CheckBox(self, -1, "Continuous")
328 sizer.Add(self.cts, 0, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 2)
329
330 self.SetSizer(sizer)
331 sizer.Fit(self)
332
333 # Effectively makes the checkboxes read-only.
334 self.Bind(wx.EVT_CHECKBOX, self.Calibrate)
335
336
337 def Calibrate(self, evt=None):
338 s = self.stick
339 self.avail.SetValue(s.HasPOV())
340 self.fourDir.SetValue(s.HasPOV4Dir())
341 self.cts.SetValue(s.HasPOVCTS())
342
343
344 #----------------------------------------------------------------------------
345
346 class POVPanel(wx.Panel):
347 def __init__(self, parent, stick):
348
349 self.stick = stick
350
351 wx.Panel.__init__(self, parent, -1, size=(100, 100))
352
353 sizer = wx.BoxSizer(wx.HORIZONTAL)
354 gsizer = wx.BoxSizer(wx.VERTICAL)
355
356 sizer.Add((25,25))
357
358 fn = wx.Font(
359 parent.GetFont().GetPointSize() + 3,
360 parent.GetFont().GetFamily(),
361 parent.GetFont().GetStyle(),
362 wx.BOLD
363 )
364 t = wx.StaticText(self, -1, "POV Control", style = wx.ALIGN_CENTER)
365 t.SetFont(fn)
366 gsizer.Add(t, 0, wx.ALL | wx.EXPAND, 1)
367
368 self.display = POVGauge(self, stick)
369 gsizer.Add(self.display, 1, wx.ALL | wx.EXPAND | wx.ALIGN_CENTER, 1)
370 sizer.Add(gsizer, 1, wx.ALL | wx.EXPAND | wx.ALIGN_CENTER, 1)
371
372 self.status = POVStatus(self, stick)
373 sizer.Add(self.status, 1, wx.ALL | wx.EXPAND | wx.ALIGN_CENTER, 1)
374
375 self.SetSizer(sizer)
376 sizer.Fit(self)
377
378
379 def Calibrate(self):
380 self.display.Calibrate()
381 self.status.Calibrate()
382
383
384 def Update(self):
385 self.display.Update()
386
387
388 #----------------------------------------------------------------------------
389
390 class LED(wx.Panel):
391 def __init__(self, parent, number):
392
393 self.state = -1
394 self.size = (20, 20)
395 self.number = number
396
397 self.fn = wx.Font(
398 parent.GetFont().GetPointSize() - 1,
399 parent.GetFont().GetFamily(),
400 parent.GetFont().GetStyle(),
401 wx.BOLD
402 )
403
404 wx.Panel.__init__(self, parent, -1, size=self.size)
405
406 self.Bind(wx.EVT_PAINT, self.OnPaint)
407 self.Bind(wx.EVT_SIZE, self.OnSize)
408 self.Bind(wx.EVT_ERASE_BACKGROUND, lambda e: None)
409
410 self.buffer = wx.EmptyBitmap(*self.size)
411 dc = wx.BufferedDC(None, self.buffer)
412 self.DrawFace(dc)
413 self.DrawLED(dc)
414
415
416 def OnSize(self, event):
417 # calculate the size of our display.
418 w, h = self.GetClientSize()
419 s = min(w, h)
420 self.size = (s, s)
421 self.buffer = wx.EmptyBitmap(*self.size)
422 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
423 self.DrawFace(dc)
424 self.DrawLED(dc)
425
426
427 def DrawFace(self, dc):
428 dc.SetBackground(wx.Brush(self.GetBackgroundColour()))
429 dc.Clear()
430
431
432 def OnPaint(self, evt):
433 # When dc is destroyed it will blit self.buffer to the window,
434 # since no other drawing is needed we'll just return and let it
435 # do it's thing
436 dc = wx.BufferedPaintDC(self, self.buffer)
437
438
439 def DrawLED(self, dc):
440 # bitmap size
441 bw, bh = self.size
442
443 # center of bitmap
444 center = bw / 2
445
446 # calc the 0, 0 origin of the bitmap
447 xorigin = center - (bw / 2)
448 yorigin = center - (bh / 2)
449
450 # Optimize drawing a bit (for Win)
451 dc.BeginDrawing()
452
453 # our 'raster'.
454 if self.state == 0:
455 dc.SetBrush(wx.Brush(wx.RED))
456 elif self.state == 1:
457 dc.SetBrush(wx.Brush(wx.GREEN))
458 else:
459 dc.SetBrush(wx.Brush(wx.BLACK))
460
461 dc.DrawCircle((center, center), bw/2)
462
463 txt = str(self.number)
464
465 # Set the font for the DC ...
466 dc.SetFont(self.fn)
467 # ... and calculate how much space our value
468 # will take up.
469 fw, fh = dc.GetTextExtent(txt)
470
471 # Calc the center of the LED, and from that
472 # derive the origin of our value.
473 tx = center - (fw/2)
474 ty = center - (fh/2)
475
476 # I draw the value twice so as to give it a pseudo-shadow.
477 # This is (mostly) because I'm too lazy to figure out how
478 # to blit my text onto the gauge using one of the logical
479 # functions. The pseudo-shadow gives the text contrast
480 # regardless of whether the bar is under it or not.
481 dc.SetTextForeground(wx.WHITE)
482 dc.DrawText(txt, (tx, ty))
483
484 # Turn off drawing optimization
485 dc.EndDrawing()
486
487
488 def Update(self):
489 dc = wx.BufferedDC(wx.ClientDC(self), self.buffer)
490 self.DrawFace(dc)
491 self.DrawLED(dc)
492
493
494 #----------------------------------------------------------------------------
495
496 class JoyButtons(wx.Panel):
497 def __init__(self, parent, stick):
498
499 self.stick = stick
500 self.leds = {}
501
502 wx.Panel.__init__(self, parent, -1)
503
504 tsizer = wx.BoxSizer(wx.VERTICAL)
505
506 fn = wx.Font(
507 parent.GetFont().GetPointSize() + 3,
508 parent.GetFont().GetFamily(),
509 parent.GetFont().GetStyle(),
510 wx.BOLD
511 )
512
513 t = wx.StaticText(self, -1, "Buttons", style = wx.ALIGN_LEFT)
514 t.SetFont(fn)
515 tsizer.Add(t, 0, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 1)
516
517 sizer = wx.FlexGridSizer(4, 16, 2, 2)
518
519 fn.SetPointSize(parent.GetFont().GetPointSize() + 1)
520
521 for i in range(0, MAX_BUTTONS):
522 t = LED(self, i)
523 self.leds[i] = t
524 sizer.Add(t, 1, wx.ALL|wx.ALIGN_CENTER|wx.ALIGN_CENTER_VERTICAL, 1)
525 sizer.AddGrowableCol(i)
526
527 tsizer.Add(sizer, 1, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 1)
528
529 self.SetSizer(tsizer)
530 tsizer.Fit(self)
531
532 def Calibrate(self):
533 for i in range(0, MAX_BUTTONS):
534 self.leds[i].state = -1
535
536 t = self.stick.GetNumberButtons()
537
538 for i in range(0, t):
539 self.leds[i].state = 0
540
541 def Update(self):
542 t = self.stick.GetButtonState()
543
544 for i in range(0, MAX_BUTTONS):
545 if self.leds[i].state == 1:
546 self.leds[i].state = 0
547
548 if (t & (1<<i)):
549 self.leds[i].state = 1
550
551 self.leds[i].Update()
552
553
554 #----------------------------------------------------------------------------
555
556 class InfoPanel(wx.Panel):
557 def __init__(self, parent, stick):
558
559 self.stick = stick
560
561 wx.Panel.__init__(self, parent, -1)
562
563 sizer = wx.GridBagSizer(1, 1)
564
565 sizer.Add(Label(self, 'Mfr ID: '), (0, 0), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
566 self.MfgID = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
567 sizer.Add(self.MfgID, (0, 1), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
568
569 sizer.Add(Label(self, 'Prod Name: '), (0, 2), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
570 self.ProdName = wx.TextCtrl(self, -1, value='', style=wx.TE_READONLY)
571 sizer.Add(self.ProdName, (0, 3), (1, 3), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
572
573 sizer.Add(Label(self, 'Threshold: '), (0, 6), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
574 self.Threshold = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
575 sizer.Add(self.Threshold, (0, 7), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
576
577 #----------------------------------------------------------------------------
578 b = wx.Button(self, -1, "Calibrate")
579 sizer.Add(b, (1, 0), (2, 2), wx.ALL | wx.ALIGN_CENTER, 2)
580
581 sizer.Add(Label(self, '# of Sticks: '), (1, 2), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
582 self.NumJoysticks = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
583 sizer.Add(self.NumJoysticks, (1, 3), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
584
585 sizer.Add(Label(self, '# of Axes: '), (1, 4), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
586 self.NumAxis = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
587 sizer.Add(self.NumAxis, (1, 5), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
588
589 sizer.Add(Label(self, 'Max # Axes: '), (1, 6), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
590 self.MaxAxis = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
591 sizer.Add(self.MaxAxis, (1, 7), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
592
593 #----------------------------------------------------------------------------
594
595 sizer.Add(Label(self, 'Polling -- '), (2, 3), (1, 1), wx.ALL | wx.GROW, 2)
596
597 sizer.Add(Label(self, 'Min: '), (2, 4), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
598 self.PollMin = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
599 sizer.Add(self.PollMin, (2, 5), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
600
601 sizer.Add(Label(self, 'Max: '), (2, 6), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_RIGHT, 2)
602 self.PollMax = wx.TextCtrl(self, -1, value='', size=(45, -1), style=wx.TE_READONLY)
603 sizer.Add(self.PollMax, (2, 7), (1, 1), wx.ALL | wx.GROW | wx.ALIGN_LEFT, 2)
604
605 #----------------------------------------------------------------------------
606
607 self.SetSizer(sizer)
608 sizer.Fit(self)
609
610
611 def Calibrate(self):
612 if not self.stick:
613 return
614
615 s = self.stick
616
617 self.MfgID.SetValue(str(s.GetManufacturerId()))
618 self.ProdName.SetValue(str(s.GetProductName()))
619 self.Threshold.SetValue(str(s.GetMovementThreshold()))
620 self.NumJoysticks.SetValue(str(s.GetNumberJoysticks()))
621 self.NumAxis.SetValue(str(s.GetNumberAxes()))
622 self.MaxAxis.SetValue(str(s.GetMaxAxes()))
623 self.PollMin.SetValue(str(s.GetPollingMin()))
624 self.PollMax.SetValue(str(s.GetPollingMax()))
625
626
627 #----------------------------------------------------------------------------
628
629 class AxisBar(wx.Gauge):
630 #
631 # This class allows us to use a wx.Gauge to display the axis value
632 # with a fancy label overlayed onto the guage itself. Two values are
633 # used to do things: first of all, since the gauge is limited to
634 # positive numbers, the scale is fixed at 0 to 1000. We will receive
635 # an adjusted value to use to render the gauge itself. The other value
636 # is a raw value and actually reflects the value from the joystick itself,
637 # which is then drawn over the gauge.
638 #
639 def __init__(self, parent):
640 wx.Gauge.__init__(self, parent, -1, 1000, size=(-1, 20), style = wx.GA_HORIZONTAL | wx.GA_SMOOTH )
641
642 # This is the value we will display.
643 self.rawvalue = 0
644
645 self.SetBackgroundColour('light blue')
646 self.SetForegroundColour('orange')
647
648 # Capture paint events for purpose of updating
649 # the displayed value.
650 self.Bind(wx.EVT_PAINT, self.onPaint)
651
652 def Update(self, value, rawvalue):
653 # Updates the gauge itself, sets the raw value for
654 # the next EVT_PAINT
655 self.SetValue(value)
656 self.rawvalue = rawvalue
657
658 def onPaint(self, evt):
659 # Must always create a PaintDC when capturing
660 # an EVT_PAINT event
661 self.ShowValue(wx.PaintDC(self), evt)
662
663 def ShowValue(self, dc, evt):
664 # This method handles actual painting of and drawing
665 # on the gauge.
666
667 # Clear out the gauge
668 dc.Clear()
669 # and then carry out business as usual
670 wx.Gauge.OnPaint(self, evt)
671
672 # This is the size available to us.
673 w, h = dc.GetSize()
674
675 # This is what we will overlay on the gauge.
676 # It reflects the actual value received from the
677 # wx.Joystick.
678 txt = str(self.rawvalue)
679
680 # Copy the default font, make it bold.
681 fn = wx.Font(
682 self.GetFont().GetPointSize(),
683 self.GetFont().GetFamily(),
684 self.GetFont().GetStyle(),
685 wx.BOLD
686 )
687
688 # Set the font for the DC ...
689 dc.SetFont(fn)
690 # ... and calculate how much space our value
691 # will take up.
692 fw, fh = dc.GetTextExtent(txt)
693
694 # Calc the center of the gauge, and from that
695 # derive the origin of our value.
696 center = w / 2
697 tx = center - (fw/2)
698
699 center = h / 2
700 ty = center - (fh/2)
701
702 # I draw the value twice so as to give it a pseudo-shadow.
703 # This is (mostly) because I'm too lazy to figure out how
704 # to blit my text onto the gauge using one of the logical
705 # functions. The pseudo-shadow gives the text contrast
706 # regardless of whether the bar is under it or not.
707 dc.SetTextForeground(wx.BLACK)
708 dc.DrawText(txt, (tx, ty))
709
710 dc.SetTextForeground('white')
711 dc.DrawText(txt, (tx-1, ty-1))
712
713
714 #----------------------------------------------------------------------------
715
716 class Axis(wx.Panel):
717 #
718 # This class is a container for the min, max, and current
719 # values of the joystick axis in question. It contains
720 # also special features to render a 'dummy' if the axis
721 # in question is not available.
722 #
723 def __init__(self, parent, token, stick):
724
725 self.stick = stick
726
727 #
728 # token represents the type of axis we're displaying.
729 #
730 self.token = token
731
732 #
733 # Create a call to the 'Has*()' method for the stick.
734 # X and Y are always there, so we tie the Has* method
735 # to a hardwired True value.
736 #
737 if token not in ['X', 'Y']:
738 self.HasFunc = eval('stick.Has%s' % token)
739 else:
740 self.HasFunc = self.alwaysTrue
741
742 # Now init the panel.
743 wx.Panel.__init__(self, parent, -1)
744
745 sizer = wx.BoxSizer(wx.HORIZONTAL)
746
747 if self.HasFunc():
748 #
749 # Tie our calibration functions to the appropriate
750 # stick method. If we don't have the axis in question,
751 # we won't need them.
752 #
753 self.GetMin = eval('stick.Get%sMin' % token)
754 self.GetMax = eval('stick.Get%sMax' % token)
755
756 # Create our displays and set them up.
757 self.Min = wx.StaticText(self, -1, str(self.GetMin()),
758 size=(40,-1), style=wx.ALIGN_RIGHT | wx.ST_NO_AUTORESIZE)
759 self.Max = wx.StaticText(self, -1, str(self.GetMax()),
760 size=(40,-1), style=wx.ALIGN_LEFT | wx.ST_NO_AUTORESIZE)
761 self.bar = AxisBar(self)
762
763 sizer.Add(self.Min, 0, wx.ALL | wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL, 1)
764 sizer.Add(self.bar, 1, wx.ALL | wx.ALIGN_CENTER | wx.ALIGN_CENTER_VERTICAL, 1)
765 sizer.Add(self.Max, 0, wx.ALL | wx.ALIGN_LEFT | wx.ALIGN_CENTER_VERTICAL, 1)
766
767 else:
768 # We go here if the axis in question is not available.
769 self.control = wx.StaticText(self, -1, ' *** Not Present ***')
770 sizer.Add(self.control, 1, wx.ALL | wx.ALIGN_CENTER | wx.ALIGN_CENTER_VERTICAL, 1)
771
772 #----------------------------------------------------------------------------
773
774 self.SetSizer(sizer)
775 sizer.Fit(self)
776 wx.CallAfter(self.Update)
777
778
779 def Calibrate(self):
780 if not self.HasFunc():
781 return
782
783 self.Min.SetLabel(str(self.GetMin()))
784 self.Max.SetLabel(str(self.GetMax()))
785
786
787 def Update(self):
788 # Don't bother if the axis doesn't exist.
789 if not self.HasFunc():
790 return
791
792 min = int(self.Min.GetLabel())
793 max = int(self.Max.GetLabel())
794
795 #
796 # Not all values are available from a wx.JoystickEvent, so I've elected
797 # to not use it at all. Therefore, we are getting our values direct from
798 # the stick. These values also seem to be more stable and reliable than
799 # those received from the event itself, so maybe it's a good idea to
800 # use the stick directly for your program.
801 #
802 # Here we either select the appropriate member of stick.GetPosition() or
803 # apply the appropriate Get*Position method call.
804 #
805 if self.token == 'X':
806 val = self.stick.GetPosition().x
807 elif self.token == 'Y':
808 val = self.stick.GetPosition().y
809 else:
810 val = eval('self.stick.Get%sPosition()' % self.token)
811
812 #
813 # While we might be able to rely on a range of 0-FFFFFF on Win, that might
814 # not be true of all drivers on all platforms. Thus, calc the actual full
815 # range first.
816 #
817 range = float(max - min)
818
819 #
820 # The relative value is used by the derived wx.Gauge since it is a
821 # positive-only control.
822 #
823 relative = 0
824 if range:
825 relative = int(val / range * 1000)
826
827 #
828 # Pass both the raw and relative values to the derived Gauge
829 #
830 self.bar.Update(relative, val)
831
832
833 def alwaysTrue(self):
834 # a dummy method used for X and Y axis.
835 return True
836
837
838 #----------------------------------------------------------------------------
839
840 class AxisPanel(wx.Panel):
841 #
842 # Contained herein is a panel that offers a graphical display
843 # of the levels for all axes supported by wx.Joystick. If
844 # your system doesn't have a particular axis, it will be
845 # 'dummied' for transparent use.
846 #
847 def __init__(self, parent, stick):
848
849 self.stick = stick
850
851 # Defines labels and 'tokens' to identify each
852 # supporte axis.
853 axesList = [
854 ('X Axis ', 'X'), ('Y Axis ', 'Y'),
855 ('Z Axis ', 'Z'), ('Rudder ', 'Rudder'),
856 ('U Axis ', 'U'), ('V Axis ', 'V')
857 ]
858
859 # Contains a list of all axis initialized.
860 self.axes = []
861
862 wx.Panel.__init__(self, parent, -1)
863
864 sizer = wx.FlexGridSizer(3, 4, 1, 1)
865 sizer.AddGrowableCol(1)
866 sizer.AddGrowableCol(3)
867
868 #----------------------------------------------------------------------------
869
870 # Go through the list of labels and tokens and add a label and
871 # axis display to the sizer for each.
872 for label, token in axesList:
873 sizer.Add(Label(self, label), 0, wx.ALL | wx.ALIGN_RIGHT, 2)
874 t = Axis(self, token, self.stick)
875 self.axes.append(t)
876 sizer.Add(t, 1, wx.ALL | wx.EXPAND | wx.ALIGN_LEFT, 2)
877
878 #----------------------------------------------------------------------------
879
880 self.SetSizer(sizer)
881 sizer.Fit(self)
882 wx.CallAfter(self.Update)
883
884 def Calibrate(self):
885 for i in self.axes:
886 i.Calibrate()
887
888 def Update(self):
889 for i in self.axes:
890 i.Update()
891
892
893 #----------------------------------------------------------------------------
894
895 class JoystickDemoPanel(wx.Panel):
896
897 def __init__(self, parent, log):
898
899 self.log = log
900
901 wx.Panel.__init__(self, parent, -1)
902
903 # Try to grab the control. If we get it, capture the stick.
904 # Otherwise, throw up an exception message and play stupid.
905 try:
906 self.stick = wx.Joystick()
907 self.stick.SetCapture(self)
908 # Calibrate our controls
909 wx.CallAfter(self.Calibrate)
910 wx.CallAfter(self.OnJoystick)
911 except NotImplementedError, v:
912 wx.MessageBox(str(v), "Exception Message")
913 self.stick = None
914
915 # One Sizer to Rule Them All...
916 sizer = wx.GridBagSizer(2,2)
917
918 self.info = InfoPanel(self, self.stick)
919 sizer.Add(self.info, (0, 0), (1, 3), wx.ALL | wx.GROW, 2)
920
921 self.info.Bind(wx.EVT_BUTTON, self.Calibrate)
922
923 self.joy = JoyPanel(self, self.stick)
924 sizer.Add(self.joy, (1, 0), (1, 1), wx.ALL | wx.GROW, 2)
925
926 self.pov = POVPanel(self, self.stick)
927 sizer.Add(self.pov, (1, 1), (1, 2), wx.ALL | wx.GROW, 2)
928
929 self.axes = AxisPanel(self, self.stick)
930 sizer.Add(self.axes, (2, 0), (1, 3), wx.ALL | wx.GROW, 2)
931
932 self.buttons = JoyButtons(self, self.stick)
933 sizer.Add(self.buttons, (3, 0), (1, 3), wx.ALL | wx.EXPAND | wx.ALIGN_CENTER | wx.ALIGN_CENTER_VERTICAL, 1)
934
935 self.SetSizer(sizer)
936 sizer.Fit(self)
937
938 # Capture Joystick events (if they happen)
939 self.Bind(wx.EVT_JOYSTICK_EVENTS, self.OnJoystick)
940
941 self.stick.SetMovementThreshold(10)
942
943 def Calibrate(self, evt=None):
944 # Do not try this without a stick
945 if not self.stick:
946 return
947
948 self.info.Calibrate()
949 self.axes.Calibrate()
950 self.pov.Calibrate()
951 self.buttons.Calibrate()
952
953 def OnJoystick(self, evt=None):
954 if not self.stick:
955 return
956
957 self.axes.Update()
958 self.joy.Update()
959 self.pov.Update()
960 self.buttons.Update()
961
962
963 #----------------------------------------------------------------------------
964
965 def runTest(frame, nb, log):
966 win = JoystickDemoPanel(nb, log)
967 return win
968
969 #----------------------------------------------------------------------------
970
971 overview = """\
972
973
974 """
975
976 #----------------------------------------------------------------------------
977
978 if __name__ == '__main__':
979 import sys,os
980 import run
981 run.main(['', os.path.basename(sys.argv[0])])