chiark / gitweb /
Add print time and filament usage estimates.
[cura.git] / Cura / gui / util / openglGui.py
1 from __future__ import absolute_import
2 from __future__ import division
3
4 import wx
5 import traceback
6 import sys
7 import os
8 import time
9
10 from wx import glcanvas
11 import OpenGL
12 OpenGL.ERROR_CHECKING = False
13 from OpenGL.GL import *
14
15 from Cura.gui.util import opengl
16
17 class animation(object):
18         def __init__(self, gui, start, end, runTime):
19                 self._start = start
20                 self._end = end
21                 self._startTime = time.time()
22                 self._runTime = runTime
23                 gui._animationList.append(self)
24
25         def isDone(self):
26                 return time.time() > self._startTime + self._runTime
27
28         def getPosition(self):
29                 if self.isDone():
30                         return self._end
31                 f = (time.time() - self._startTime) / self._runTime
32                 ts = f*f
33                 tc = f*f*f
34                 #f = 6*tc*ts + -15*ts*ts + 10*tc
35                 f = tc + -3*ts + 3*f
36                 return self._start + (self._end - self._start) * f
37
38 class glGuiControl(object):
39         def __init__(self, parent, pos):
40                 self._parent = parent
41                 self._base = parent._base
42                 self._pos = pos
43                 self._size = (0,0, 1, 1)
44                 self._parent.add(self)
45
46         def setSize(self, x, y, w, h):
47                 self._size = (x, y, w, h)
48
49         def getSize(self):
50                 return self._size
51
52         def getMinSize(self):
53                 return 1, 1
54
55         def updateLayout(self):
56                 pass
57
58         def focusNext(self):
59                 for n in xrange(self._parent._glGuiControlList.index(self) + 1, len(self._parent._glGuiControlList)):
60                         if self._parent._glGuiControlList[n].setFocus():
61                                 return
62                 for n in xrange(0, self._parent._glGuiControlList.index(self)):
63                         if self._parent._glGuiControlList[n].setFocus():
64                                 return
65
66         def focusPrevious(self):
67                 for n in xrange(self._parent._glGuiControlList.index(self) -1, -1, -1):
68                         if self._parent._glGuiControlList[n].setFocus():
69                                 return
70                 for n in xrange(len(self._parent._glGuiControlList) - 1, self._parent._glGuiControlList.index(self), -1):
71                         if self._parent._glGuiControlList[n].setFocus():
72                                 return
73
74         def setFocus(self):
75                 return False
76
77         def hasFocus(self):
78                 return self._base._focus == self
79
80         def OnMouseUp(self, x, y):
81                 pass
82
83         def OnKeyChar(self, key):
84                 pass
85
86 class glGuiContainer(glGuiControl):
87         def __init__(self, parent, pos):
88                 self._glGuiControlList = []
89                 glGuiLayoutButtons(self)
90                 super(glGuiContainer, self).__init__(parent, pos)
91
92         def add(self, ctrl):
93                 self._glGuiControlList.append(ctrl)
94                 self.updateLayout()
95
96         def OnMouseDown(self, x, y, button):
97                 for ctrl in self._glGuiControlList:
98                         if ctrl.OnMouseDown(x, y, button):
99                                 return True
100                 return False
101
102         def OnMouseUp(self, x, y):
103                 for ctrl in self._glGuiControlList:
104                         if ctrl.OnMouseUp(x, y):
105                                 return True
106                 return False
107
108         def OnMouseMotion(self, x, y):
109                 handled = False
110                 for ctrl in self._glGuiControlList:
111                         if ctrl.OnMouseMotion(x, y):
112                                 handled = True
113                 return handled
114
115         def draw(self):
116                 for ctrl in self._glGuiControlList:
117                         ctrl.draw()
118
119         def updateLayout(self):
120                 self._layout.update()
121                 for ctrl in self._glGuiControlList:
122                         ctrl.updateLayout()
123
124 class glGuiPanel(glcanvas.GLCanvas):
125         def __init__(self, parent):
126                 attribList = (glcanvas.WX_GL_RGBA, glcanvas.WX_GL_DOUBLEBUFFER, glcanvas.WX_GL_DEPTH_SIZE, 32, glcanvas.WX_GL_STENCIL_SIZE, 8)
127                 glcanvas.GLCanvas.__init__(self, parent, style=wx.WANTS_CHARS, attribList = attribList)
128                 self._base = self
129                 self._focus = None
130                 self._container = None
131                 self._container = glGuiContainer(self, (0,0))
132                 self._shownError = False
133
134                 self._context = glcanvas.GLContext(self)
135                 self._glButtonsTexture = None
136                 self._glRobotTexture = None
137                 self._buttonSize = 64
138
139                 self._animationList = []
140                 self.glReleaseList = []
141                 self._refreshQueued = False
142
143                 wx.EVT_PAINT(self, self._OnGuiPaint)
144                 wx.EVT_SIZE(self, self._OnSize)
145                 wx.EVT_ERASE_BACKGROUND(self, self._OnEraseBackground)
146                 wx.EVT_LEFT_DOWN(self, self._OnGuiMouseDown)
147                 wx.EVT_LEFT_DCLICK(self, self._OnGuiMouseDown)
148                 wx.EVT_LEFT_UP(self, self._OnGuiMouseUp)
149                 wx.EVT_RIGHT_DOWN(self, self._OnGuiMouseDown)
150                 wx.EVT_RIGHT_DCLICK(self, self._OnGuiMouseDown)
151                 wx.EVT_RIGHT_UP(self, self._OnGuiMouseUp)
152                 wx.EVT_MIDDLE_DOWN(self, self._OnGuiMouseDown)
153                 wx.EVT_MIDDLE_DCLICK(self, self._OnGuiMouseDown)
154                 wx.EVT_MIDDLE_UP(self, self._OnGuiMouseUp)
155                 wx.EVT_MOTION(self, self._OnGuiMouseMotion)
156                 wx.EVT_CHAR(self, self._OnGuiKeyChar)
157                 wx.EVT_KILL_FOCUS(self, self.OnFocusLost)
158                 wx.EVT_IDLE(self, self._OnIdle)
159
160         def _OnIdle(self, e):
161                 if len(self._animationList) > 0 or self._refreshQueued:
162                         self._refreshQueued = False
163                         for anim in self._animationList:
164                                 if anim.isDone():
165                                         self._animationList.remove(anim)
166                         self.Refresh()
167
168         def _OnGuiKeyChar(self, e):
169                 if self._focus is not None:
170                         self._focus.OnKeyChar(e.GetKeyCode())
171                         self.Refresh()
172                 else:
173                         self.OnKeyChar(e.GetKeyCode())
174
175         def OnFocusLost(self, e):
176                 self._focus = None
177                 self.Refresh()
178
179         def _OnGuiMouseDown(self,e):
180                 self.SetFocus()
181                 if self._container.OnMouseDown(e.GetX(), e.GetY(), e.GetButton()):
182                         self.Refresh()
183                         return
184                 self.OnMouseDown(e)
185
186         def _OnGuiMouseUp(self, e):
187                 if self._container.OnMouseUp(e.GetX(), e.GetY()):
188                         self.Refresh()
189                         return
190                 self.OnMouseUp(e)
191
192         def _OnGuiMouseMotion(self,e):
193                 self.Refresh()
194                 if not self._container.OnMouseMotion(e.GetX(), e.GetY()):
195                         self.OnMouseMotion(e)
196
197         def _OnGuiPaint(self, e):
198                 h = self.GetSize().GetHeight()
199                 w = self.GetSize().GetWidth()
200                 oldButtonSize = self._buttonSize
201                 if h / 3 < w / 4:
202                         w = h * 4 / 3
203                 if w < 64 * 8:
204                         self._buttonSize = 32
205                 elif w < 64 * 10:
206                         self._buttonSize = 48
207                 elif w < 64 * 15:
208                         self._buttonSize = 64
209                 elif w < 64 * 20:
210                         self._buttonSize = 80
211                 else:
212                         self._buttonSize = 96
213                 if self._buttonSize != oldButtonSize:
214                         self._container.updateLayout()
215
216                 dc = wx.PaintDC(self)
217                 try:
218                         self.SetCurrent(self._context)
219                         for obj in self.glReleaseList:
220                                 obj.release()
221                         del self.glReleaseList[:]
222                         self.OnPaint(e)
223                         self._drawGui()
224                         glFlush()
225                         self.SwapBuffers()
226                 except:
227                         errStr = 'An error has occurred during the 3D view drawing.'
228                         tb = traceback.extract_tb(sys.exc_info()[2])
229                         errStr += "\n%s: '%s'" % (str(sys.exc_info()[0].__name__), str(sys.exc_info()[1]))
230                         for n in xrange(len(tb)-1, -1, -1):
231                                 locationInfo = tb[n]
232                                 errStr += "\n @ %s:%s:%d" % (os.path.basename(locationInfo[0]), locationInfo[2], locationInfo[1])
233                         if not self._shownError:
234                                 wx.CallAfter(wx.MessageBox, errStr, '3D window error', wx.OK | wx.ICON_EXCLAMATION)
235                                 self._shownError = True
236
237         def _drawGui(self):
238                 if self._glButtonsTexture is None:
239                         self._glButtonsTexture = opengl.loadGLTexture('glButtons.png')
240                         self._glRobotTexture = opengl.loadGLTexture('UltimakerRobot.png')
241
242                 glDisable(GL_DEPTH_TEST)
243                 glEnable(GL_BLEND)
244                 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA)
245                 glDisable(GL_LIGHTING)
246                 glColor4ub(255,255,255,255)
247
248                 glMatrixMode(GL_PROJECTION)
249                 glLoadIdentity()
250                 size = self.GetSize()
251                 glOrtho(0, size.GetWidth()-1, size.GetHeight()-1, 0, -1000.0, 1000.0)
252                 glMatrixMode(GL_MODELVIEW)
253                 glLoadIdentity()
254
255                 self._container.draw()
256
257                 glBindTexture(GL_TEXTURE_2D, self._glRobotTexture)
258                 glEnable(GL_TEXTURE_2D)
259                 glPushMatrix()
260                 glColor4f(1,1,1,1)
261                 glTranslate(size.GetWidth() - 8,size.GetHeight() - 16,0)
262                 s = self._buttonSize * 1.4
263                 glScale(s,s,s)
264                 glBegin(GL_QUADS)
265                 glTexCoord2f(1, 0)
266                 glVertex2f(0,-1)
267                 glTexCoord2f(0, 0)
268                 glVertex2f(-1,-1)
269                 glTexCoord2f(0, 1)
270                 glVertex2f(-1, 0)
271                 glTexCoord2f(1, 1)
272                 glVertex2f(0, 0)
273                 glEnd()
274                 glDisable(GL_TEXTURE_2D)
275                 glPopMatrix()
276
277         def _OnEraseBackground(self,event):
278                 #Workaround for windows background redraw flicker.
279                 pass
280
281         def _OnSize(self,e):
282                 self._container.setSize(0, 0, self.GetSize().GetWidth(), self.GetSize().GetHeight())
283                 self._container.updateLayout()
284                 self.Refresh()
285
286         def OnMouseDown(self,e):
287                 pass
288         def OnMouseUp(self,e):
289                 pass
290         def OnMouseMotion(self, e):
291                 pass
292         def OnKeyChar(self, keyCode):
293                 pass
294         def OnPaint(self, e):
295                 pass
296         def OnKeyChar(self, keycode):
297                 pass
298
299         def QueueRefresh(self):
300                 wx.CallAfter(self._queueRefresh)
301
302         def _queueRefresh(self):
303                 self._refreshQueued = True
304
305         def add(self, ctrl):
306                 if self._container is not None:
307                         self._container.add(ctrl)
308
309 class glGuiLayoutButtons(object):
310         def __init__(self, parent):
311                 self._parent = parent
312                 self._parent._layout = self
313
314         def update(self):
315                 bs = self._parent._base._buttonSize
316                 x0, y0, w, h = self._parent.getSize()
317                 gridSize = bs * 1.0
318                 for ctrl in self._parent._glGuiControlList:
319                         pos = ctrl._pos
320                         if pos[0] < 0:
321                                 x = w + pos[0] * gridSize - bs * 0.2
322                         else:
323                                 x = pos[0] * gridSize + bs * 0.2
324                         if pos[1] < 0:
325                                 y = h + pos[1] * gridSize * 1.2 - bs * 0.2
326                         else:
327                                 y = pos[1] * gridSize * 1.2 + bs * 0.2
328                         ctrl.setSize(x, y, gridSize, gridSize)
329
330         def getLayoutSize(self):
331                 _, _, w, h = self._parent.getSize()
332                 return w, h
333
334 class glGuiLayoutGrid(object):
335         def __init__(self, parent):
336                 self._parent = parent
337                 self._parent._layout = self
338                 self._size = 0,0
339                 self._alignBottom = True
340
341         def update(self):
342                 borderSize = self._parent._base._buttonSize * 0.2
343                 x0, y0, w, h = self._parent.getSize()
344                 x0 += borderSize
345                 y0 += borderSize
346                 widths = {}
347                 heights = {}
348                 for ctrl in self._parent._glGuiControlList:
349                         x, y = ctrl._pos
350                         w, h = ctrl.getMinSize()
351                         if not x in widths:
352                                 widths[x] = w
353                         else:
354                                 widths[x] = max(widths[x], w)
355                         if not y in heights:
356                                 heights[y] = h
357                         else:
358                                 heights[y] = max(heights[y], h)
359                 self._size = sum(widths.values()) + borderSize * 2, sum(heights.values()) + borderSize * 2
360                 if self._alignBottom:
361                         y0 -= self._size[1] - self._parent.getSize()[3]
362                         self._parent.setSize(x0 - borderSize, y0 - borderSize, self._size[0], self._size[1])
363                 for ctrl in self._parent._glGuiControlList:
364                         x, y = ctrl._pos
365                         x1 = x0
366                         y1 = y0
367                         for n in xrange(0, x):
368                                 if not n in widths:
369                                         widths[n] = 3
370                                 x1 += widths[n]
371                         for n in xrange(0, y):
372                                 if not n in heights:
373                                         heights[n] = 3
374                                 y1 += heights[n]
375                         ctrl.setSize(x1, y1, widths[x], heights[y])
376
377         def getLayoutSize(self):
378                 return self._size
379
380 class glButton(glGuiControl):
381         def __init__(self, parent, imageID, tooltip, pos, callback, size = None):
382                 self._buttonSize = size
383                 super(glButton, self).__init__(parent, pos)
384                 self._tooltip = tooltip
385                 self._parent = parent
386                 self._imageID = imageID
387                 self._callback = callback
388                 self._selected = False
389                 self._focus = False
390                 self._hidden = False
391                 self._disabled = False
392                 self._showExpandArrow = False
393                 self._progressBar = None
394                 self._altTooltip = ''
395
396         def setSelected(self, value):
397                 self._selected = value
398
399         def setExpandArrow(self, value):
400                 self._showExpandArrow = value
401
402         def setHidden(self, value):
403                 self._hidden = value
404
405         def setDisabled(self, value):
406                 self._disabled = value
407
408         def setProgressBar(self, value):
409                 self._progressBar = value
410
411         def setBottomText(self, value):
412                 self._altTooltip = value
413
414         def getSelected(self):
415                 return self._selected
416
417         def getMinSize(self):
418                 if self._buttonSize is not None:
419                         return self._buttonSize, self._buttonSize
420                 return self._base._buttonSize, self._base._buttonSize
421
422         def _getPixelPos(self):
423                 x0, y0, w, h = self.getSize()
424                 return x0 + w / 2, y0 + h / 2
425
426         def draw(self):
427                 if self._hidden:
428                         return
429
430                 cx = (self._imageID % 4) / 4
431                 cy = int(self._imageID / 4) / 4
432                 bs = self.getMinSize()[0]
433                 pos = self._getPixelPos()
434
435                 glBindTexture(GL_TEXTURE_2D, self._base._glButtonsTexture)
436                 scale = 0.8
437                 if self._selected:
438                         scale = 1.0
439                 elif self._focus:
440                         scale = 0.9
441                 if self._disabled:
442                         glColor4ub(128,128,128,128)
443                 else:
444                         glColor4ub(255,255,255,255)
445                 opengl.glDrawTexturedQuad(pos[0]-bs*scale/2, pos[1]-bs*scale/2, bs*scale, bs*scale, 0)
446                 opengl.glDrawTexturedQuad(pos[0]-bs*scale/2, pos[1]-bs*scale/2, bs*scale, bs*scale, self._imageID)
447                 if self._showExpandArrow:
448                         if self._selected:
449                                 opengl.glDrawTexturedQuad(pos[0]+bs*scale/2-bs*scale/4*1.2, pos[1]-bs*scale/2*1.2, bs*scale/4, bs*scale/4, 1)
450                         else:
451                                 opengl.glDrawTexturedQuad(pos[0]+bs*scale/2-bs*scale/4*1.2, pos[1]-bs*scale/2*1.2, bs*scale/4, bs*scale/4, 1, 2)
452                 glPushMatrix()
453                 glTranslatef(pos[0], pos[1], 0)
454                 glDisable(GL_TEXTURE_2D)
455                 if self._focus:
456                         glTranslatef(0, -0.55*bs*scale, 0)
457
458                         glPushMatrix()
459                         glColor4ub(60,60,60,255)
460                         glTranslatef(-1, -1, 0)
461                         opengl.glDrawStringCenter(self._tooltip)
462                         glTranslatef(0, 2, 0)
463                         opengl.glDrawStringCenter(self._tooltip)
464                         glTranslatef(2, 0, 0)
465                         opengl.glDrawStringCenter(self._tooltip)
466                         glTranslatef(0, -2, 0)
467                         opengl.glDrawStringCenter(self._tooltip)
468                         glPopMatrix()
469
470                         glColor4ub(255,255,255,255)
471                         opengl.glDrawStringCenter(self._tooltip)
472                 glPopMatrix()
473                 if self._progressBar is not None:
474                         glColor4ub(255,255,255,192)
475                         opengl.glDrawTexturedQuad(pos[0]-bs/2, pos[1]+bs/2, bs, bs / 4, 0)
476                         opengl.glDrawTexturedQuad(pos[0]-bs/2, pos[1]+bs/2, bs * self._progressBar, bs / 4, 0)
477                 elif len(self._altTooltip) > 0:
478                         glPushMatrix()
479                         glTranslatef(pos[0], pos[1], 0)
480                         glTranslatef(0.6*bs*scale, 0, 0)
481
482                         glPushMatrix()
483                         glColor4ub(60,60,60,255)
484                         glTranslatef(-1, -1, 0)
485                         opengl.glDrawStringLeft(self._altTooltip)
486                         glTranslatef(0, 2, 0)
487                         opengl.glDrawStringLeft(self._altTooltip)
488                         glTranslatef(2, 0, 0)
489                         opengl.glDrawStringLeft(self._altTooltip)
490                         glTranslatef(0, -2, 0)
491                         opengl.glDrawStringLeft(self._altTooltip)
492                         glPopMatrix()
493
494                         glColor4ub(255,255,255,255)
495                         opengl.glDrawStringLeft(self._altTooltip)
496                         glPopMatrix()
497
498         def _checkHit(self, x, y):
499                 if self._hidden or self._disabled:
500                         return False
501                 bs = self.getMinSize()[0]
502                 pos = self._getPixelPos()
503                 return -bs * 0.5 <= x - pos[0] <= bs * 0.5 and -bs * 0.5 <= y - pos[1] <= bs * 0.5
504
505         def OnMouseMotion(self, x, y):
506                 if self._checkHit(x, y):
507                         self._focus = True
508                         return True
509                 self._focus = False
510                 return False
511
512         def OnMouseDown(self, x, y, button):
513                 if self._checkHit(x, y):
514                         self._callback(button)
515                         return True
516                 return False
517
518 class glRadioButton(glButton):
519         def __init__(self, parent, imageID, tooltip, pos, group, callback):
520                 super(glRadioButton, self).__init__(parent, imageID, tooltip, pos, self._onRadioSelect)
521                 self._group = group
522                 self._radioCallback = callback
523                 self._group.append(self)
524
525         def setSelected(self, value):
526                 self._selected = value
527
528         def _onRadioSelect(self, button):
529                 self._base._focus = None
530                 for ctrl in self._group:
531                         if ctrl != self:
532                                 ctrl.setSelected(False)
533                 if self.getSelected():
534                         self.setSelected(False)
535                 else:
536                         self.setSelected(True)
537                 self._radioCallback(button)
538
539 class glComboButton(glButton):
540         def __init__(self, parent, tooltip, imageIDs, tooltips, pos, callback):
541                 super(glComboButton, self).__init__(parent, imageIDs[0], tooltip, pos, self._onComboOpenSelect)
542                 self._imageIDs = imageIDs
543                 self._tooltips = tooltips
544                 self._comboCallback = callback
545                 self._selection = 0
546
547         def _onComboOpenSelect(self, button):
548                 if self.hasFocus():
549                         self._base._focus = None
550                 else:
551                         self._base._focus = self
552
553         def draw(self):
554                 if self._hidden:
555                         return
556                 self._selected = self.hasFocus()
557                 super(glComboButton, self).draw()
558
559                 bs = self._base._buttonSize / 2
560                 pos = self._getPixelPos()
561
562                 if not self._selected:
563                         return
564
565                 glPushMatrix()
566                 glTranslatef(pos[0]+bs*0.5, pos[1] + bs*0.5, 0)
567                 glBindTexture(GL_TEXTURE_2D, self._base._glButtonsTexture)
568                 for n in xrange(0, len(self._imageIDs)):
569                         glTranslatef(0, bs, 0)
570                         glColor4ub(255,255,255,255)
571                         opengl.glDrawTexturedQuad(-0.5*bs,-0.5*bs,bs,bs, 0)
572                         opengl.glDrawTexturedQuad(-0.5*bs,-0.5*bs,bs,bs, self._imageIDs[n])
573                         glDisable(GL_TEXTURE_2D)
574
575                         glPushMatrix()
576                         glTranslatef(-0.55*bs, 0.1*bs, 0)
577
578                         glPushMatrix()
579                         glColor4ub(60,60,60,255)
580                         glTranslatef(-1, -1, 0)
581                         opengl.glDrawStringRight(self._tooltips[n])
582                         glTranslatef(0, 2, 0)
583                         opengl.glDrawStringRight(self._tooltips[n])
584                         glTranslatef(2, 0, 0)
585                         opengl.glDrawStringRight(self._tooltips[n])
586                         glTranslatef(0, -2, 0)
587                         opengl.glDrawStringRight(self._tooltips[n])
588                         glPopMatrix()
589
590                         glColor4ub(255,255,255,255)
591                         opengl.glDrawStringRight(self._tooltips[n])
592                         glPopMatrix()
593                 glPopMatrix()
594
595         def getValue(self):
596                 return self._selection
597
598         def setValue(self, value):
599                 self._selection = value
600                 self._imageID = self._imageIDs[self._selection]
601                 self._comboCallback()
602
603         def OnMouseDown(self, x, y, button):
604                 if self._hidden or self._disabled:
605                         return False
606                 if self.hasFocus():
607                         bs = self._base._buttonSize / 2
608                         pos = self._getPixelPos()
609                         if 0 <= x - pos[0] <= bs and 0 <= y - pos[1] - bs <= bs * len(self._imageIDs):
610                                 self._selection = int((y - pos[1] - bs) / bs)
611                                 self._imageID = self._imageIDs[self._selection]
612                                 self._base._focus = None
613                                 self._comboCallback()
614                                 return True
615                 return super(glComboButton, self).OnMouseDown(x, y, button)
616
617 class glFrame(glGuiContainer):
618         def __init__(self, parent, pos):
619                 super(glFrame, self).__init__(parent, pos)
620                 self._selected = False
621                 self._focus = False
622                 self._hidden = False
623
624         def setSelected(self, value):
625                 self._selected = value
626
627         def setHidden(self, value):
628                 self._hidden = value
629                 for child in self._glGuiControlList:
630                         if self._base._focus == child:
631                                 self._base._focus = None
632
633         def getSelected(self):
634                 return self._selected
635
636         def getMinSize(self):
637                 return self._base._buttonSize, self._base._buttonSize
638
639         def _getPixelPos(self):
640                 x0, y0, w, h = self.getSize()
641                 return x0, y0
642
643         def draw(self):
644                 if self._hidden:
645                         return
646
647                 bs = self._parent._buttonSize
648                 pos = self._getPixelPos()
649
650                 size = self._layout.getLayoutSize()
651                 glColor4ub(255,255,255,255)
652                 opengl.glDrawStretchedQuad(pos[0], pos[1], size[0], size[1], bs*0.75, 0)
653                 #Draw the controls on the frame
654                 super(glFrame, self).draw()
655
656         def _checkHit(self, x, y):
657                 if self._hidden:
658                         return False
659                 pos = self._getPixelPos()
660                 w, h = self._layout.getLayoutSize()
661                 return 0 <= x - pos[0] <= w and 0 <= y - pos[1] <= h
662
663         def OnMouseMotion(self, x, y):
664                 super(glFrame, self).OnMouseMotion(x, y)
665                 if self._checkHit(x, y):
666                         self._focus = True
667                         return True
668                 self._focus = False
669                 return False
670
671         def OnMouseDown(self, x, y, button):
672                 if self._checkHit(x, y):
673                         super(glFrame, self).OnMouseDown(x, y, button)
674                         return True
675                 return False
676
677 class glNotification(glFrame):
678         def __init__(self, parent, pos):
679                 self._anim = None
680                 super(glNotification, self).__init__(parent, pos)
681                 glGuiLayoutGrid(self)._alignBottom = False
682                 self._label = glLabel(self, "Notification", (0, 0))
683                 self._button = glButton(self, 30, "", (1, 0), self.onClose, 25)
684                 self._padding = glLabel(self, "", (0, 1))
685                 self.setHidden(True)
686
687         def setSize(self, x, y, w, h):
688                 w, h = self._layout.getLayoutSize()
689                 baseSize = self._base.GetSizeTuple()
690                 if self._anim is not None:
691                         super(glNotification, self).setSize(baseSize[0] / 2 - w / 2, baseSize[1] - self._anim.getPosition() - self._base._buttonSize * 0.2, 1, 1)
692                 else:
693                         super(glNotification, self).setSize(baseSize[0] / 2 - w / 2, baseSize[1] - self._base._buttonSize * 0.2, 1, 1)
694
695         def draw(self):
696                 self.setSize(0,0,0,0)
697                 self.updateLayout()
698                 super(glNotification, self).draw()
699
700         def message(self, text):
701                 if self._anim is not None:
702                         self._anim = animation(self._base, self._anim.getPosition(), 25, 1)
703                 else:
704                         self._anim = animation(self._base, -20, 25, 1)
705                 self.setHidden(False)
706                 self._label.setLabel(text)
707                 self.updateLayout()
708
709         def onClose(self, button):
710                 if self._anim is not None:
711                         self._anim = animation(self._base, self._anim.getPosition(), -20, 1)
712                 else:
713                         self._anim = animation(self._base, 25, -20, 1)
714
715 class glLabel(glGuiControl):
716         def __init__(self, parent, label, pos):
717                 self._label = label
718                 super(glLabel, self).__init__(parent, pos)
719
720         def setLabel(self, label):
721                 self._label = label
722
723         def getMinSize(self):
724                 w, h = opengl.glGetStringSize(self._label)
725                 return w + 10, h + 4
726
727         def _getPixelPos(self):
728                 x0, y0, w, h = self.getSize()
729                 return x0, y0
730
731         def draw(self):
732                 x, y, w, h = self.getSize()
733
734                 glPushMatrix()
735                 glTranslatef(x, y, 0)
736
737 #               glColor4ub(255,255,255,128)
738 #               glBegin(GL_QUADS)
739 #               glTexCoord2f(1, 0)
740 #               glVertex2f( w, 0)
741 #               glTexCoord2f(0, 0)
742 #               glVertex2f( 0, 0)
743 #               glTexCoord2f(0, 1)
744 #               glVertex2f( 0, h)
745 #               glTexCoord2f(1, 1)
746 #               glVertex2f( w, h)
747 #               glEnd()
748
749                 glTranslate(5, h - 5, 0)
750                 glColor4ub(255,255,255,255)
751                 opengl.glDrawStringLeft(self._label)
752                 glPopMatrix()
753
754         def _checkHit(self, x, y):
755                 return False
756
757         def OnMouseMotion(self, x, y):
758                 return False
759
760         def OnMouseDown(self, x, y, button):
761                 return False
762
763 class glNumberCtrl(glGuiControl):
764         def __init__(self, parent, value, pos, callback):
765                 self._callback = callback
766                 self._value = str(value)
767                 self._selectPos = 0
768                 self._maxLen = 6
769                 self._inCallback = False
770                 super(glNumberCtrl, self).__init__(parent, pos)
771
772         def setValue(self, value):
773                 if self._inCallback:
774                         return
775                 self._value = str(value)
776
777         def getMinSize(self):
778                 w, h = opengl.glGetStringSize("VALUES")
779                 return w + 10, h + 4
780
781         def _getPixelPos(self):
782                 x0, y0, w, h = self.getSize()
783                 return x0, y0
784
785         def draw(self):
786                 x, y, w, h = self.getSize()
787
788                 glPushMatrix()
789                 glTranslatef(x, y, 0)
790
791                 if self.hasFocus():
792                         glColor4ub(255,255,255,255)
793                 else:
794                         glColor4ub(255,255,255,192)
795                 glBegin(GL_QUADS)
796                 glTexCoord2f(1, 0)
797                 glVertex2f( w, 0)
798                 glTexCoord2f(0, 0)
799                 glVertex2f( 0, 0)
800                 glTexCoord2f(0, 1)
801                 glVertex2f( 0, h-1)
802                 glTexCoord2f(1, 1)
803                 glVertex2f( w, h-1)
804                 glEnd()
805
806                 glTranslate(5, h - 5, 0)
807                 glColor4ub(0,0,0,255)
808                 opengl.glDrawStringLeft(self._value)
809                 if self.hasFocus():
810                         glTranslate(opengl.glGetStringSize(self._value[0:self._selectPos])[0] - 2, -1, 0)
811                         opengl.glDrawStringLeft('|')
812                 glPopMatrix()
813
814         def _checkHit(self, x, y):
815                 x1, y1, w, h = self.getSize()
816                 return 0 <= x - x1 <= w and 0 <= y - y1 <= h
817
818         def OnMouseMotion(self, x, y):
819                 return False
820
821         def OnMouseDown(self, x, y, button):
822                 if self._checkHit(x, y):
823                         self.setFocus()
824                         return True
825                 return False
826
827         def OnKeyChar(self, c):
828                 self._inCallback = True
829                 if c == wx.WXK_LEFT:
830                         self._selectPos -= 1
831                         self._selectPos = max(0, self._selectPos)
832                 if c == wx.WXK_RIGHT:
833                         self._selectPos += 1
834                         self._selectPos = min(self._selectPos, len(self._value))
835                 if c == wx.WXK_UP:
836                         try:
837                                 value = float(self._value)
838                         except:
839                                 pass
840                         else:
841                                 value += 0.1
842                                 self._value = str(value)
843                                 self._callback(self._value)
844                 if c == wx.WXK_DOWN:
845                         try:
846                                 value = float(self._value)
847                         except:
848                                 pass
849                         else:
850                                 value -= 0.1
851                                 if value > 0:
852                                         self._value = str(value)
853                                         self._callback(self._value)
854                 if c == wx.WXK_BACK and self._selectPos > 0:
855                         self._value = self._value[0:self._selectPos - 1] + self._value[self._selectPos:]
856                         self._selectPos -= 1
857                         self._callback(self._value)
858                 if c == wx.WXK_DELETE:
859                         self._value = self._value[0:self._selectPos] + self._value[self._selectPos + 1:]
860                         self._callback(self._value)
861                 if c == wx.WXK_TAB or c == wx.WXK_NUMPAD_ENTER or c == wx.WXK_RETURN:
862                         if wx.GetKeyState(wx.WXK_SHIFT):
863                                 self.focusPrevious()
864                         else:
865                                 self.focusNext()
866                 if (ord('0') <= c <= ord('9') or c == ord('.')) and len(self._value) < self._maxLen:
867                         self._value = self._value[0:self._selectPos] + chr(c) + self._value[self._selectPos:]
868                         self._selectPos += 1
869                         self._callback(self._value)
870                 self._inCallback = False
871
872         def setFocus(self):
873                 self._base._focus = self
874                 self._selectPos = len(self._value)
875                 return True
876
877 class glCheckbox(glGuiControl):
878         def __init__(self, parent, value, pos, callback):
879                 self._callback = callback
880                 self._value = value
881                 self._selectPos = 0
882                 self._maxLen = 6
883                 self._inCallback = False
884                 super(glCheckbox, self).__init__(parent, pos)
885
886         def setValue(self, value):
887                 if self._inCallback:
888                         return
889                 self._value = str(value)
890
891         def getValue(self):
892                 return self._value
893
894         def getMinSize(self):
895                 return 20, 20
896
897         def _getPixelPos(self):
898                 x0, y0, w, h = self.getSize()
899                 return x0, y0
900
901         def draw(self):
902                 x, y, w, h = self.getSize()
903
904                 glPushMatrix()
905                 glTranslatef(x, y, 0)
906
907                 glColor3ub(255,255,255)
908                 if self._value:
909                         opengl.glDrawTexturedQuad(w/2-h/2,0, h, h, 28)
910                 else:
911                         opengl.glDrawTexturedQuad(w/2-h/2,0, h, h, 29)
912
913                 glPopMatrix()
914
915         def _checkHit(self, x, y):
916                 x1, y1, w, h = self.getSize()
917                 return 0 <= x - x1 <= w and 0 <= y - y1 <= h
918
919         def OnMouseMotion(self, x, y):
920                 return False
921
922         def OnMouseDown(self, x, y, button):
923                 if self._checkHit(x, y):
924                         self._value = not self._value
925                         return True
926                 return False
927
928 class glSlider(glGuiControl):
929         def __init__(self, parent, value, minValue, maxValue, pos, callback):
930                 super(glSlider, self).__init__(parent, pos)
931                 self._callback = callback
932                 self._focus = False
933                 self._hidden = False
934                 self._value = value
935                 self._minValue = minValue
936                 self._maxValue = maxValue
937
938         def setValue(self, value):
939                 self._value = value
940                 self._value = max(self._minValue, self._value)
941                 self._value = min(self._maxValue, self._value)
942
943         def getValue(self):
944                 return self._value
945
946         def setRange(self, minValue, maxValue):
947                 if maxValue < minValue:
948                         maxValue = minValue
949                 self._minValue = minValue
950                 self._maxValue = maxValue
951                 self._value = max(minValue, self._value)
952                 self._value = min(maxValue, self._value)
953
954         def getMinValue(self):
955                 return self._minValue
956
957         def getMaxValue(self):
958                 return self._maxValue
959
960         def setHidden(self, value):
961                 self._hidden = value
962
963         def getMinSize(self):
964                 return self._base._buttonSize * 0.2, self._base._buttonSize * 4
965
966         def _getPixelPos(self):
967                 x0, y0, w, h = self.getSize()
968                 minSize = self.getMinSize()
969                 return x0 + w / 2 - minSize[0] / 2, y0 + h / 2 - minSize[1] / 2
970
971         def draw(self):
972                 if self._hidden:
973                         return
974
975                 w, h = self.getMinSize()
976                 pos = self._getPixelPos()
977
978                 glPushMatrix()
979                 glTranslatef(pos[0], pos[1], 0)
980                 glDisable(GL_TEXTURE_2D)
981                 if self.hasFocus():
982                         glColor4ub(60,60,60,255)
983                 else:
984                         glColor4ub(60,60,60,192)
985                 glBegin(GL_QUADS)
986                 glVertex2f( w/2,-h/2)
987                 glVertex2f(-w/2,-h/2)
988                 glVertex2f(-w/2, h/2)
989                 glVertex2f( w/2, h/2)
990                 glEnd()
991                 scrollLength = h - w
992                 glTranslate(0.0,scrollLength/2,0)
993                 if self._focus:
994                         glColor4ub(0,0,0,255)
995                         glPushMatrix()
996                         glTranslate(-w/2,opengl.glGetStringSize(str(self._minValue))[1]/2,0)
997                         opengl.glDrawStringRight(str(self._minValue))
998                         glTranslate(0,-scrollLength,0)
999                         opengl.glDrawStringRight(str(self._maxValue))
1000                         if self._maxValue-self._minValue > 0:
1001                                 glTranslate(w,scrollLength-scrollLength*((self._value-self._minValue)/(self._maxValue-self._minValue)),0)
1002                         opengl.glDrawStringLeft(str(self._value))
1003                         glPopMatrix()
1004                 glColor4ub(255,255,255,240)
1005                 if self._maxValue - self._minValue != 0:
1006                         glTranslate(0.0,-scrollLength*((self._value-self._minValue)/(self._maxValue-self._minValue)),0)
1007                 glBegin(GL_QUADS)
1008                 glVertex2f( w/2,-w/2)
1009                 glVertex2f(-w/2,-w/2)
1010                 glVertex2f(-w/2, w/2)
1011                 glVertex2f( w/2, w/2)
1012                 glEnd()
1013                 glPopMatrix()
1014
1015         def _checkHit(self, x, y):
1016                 if self._hidden:
1017                         return False
1018                 pos = self._getPixelPos()
1019                 w, h = self.getMinSize()
1020                 return -w/2 <= x - pos[0] <= w/2 and -h/2 <= y - pos[1] <= h/2
1021
1022         def setFocus(self):
1023                 self._base._focus = self
1024                 return True
1025
1026         def OnMouseMotion(self, x, y):
1027                 if self.hasFocus():
1028                         w, h = self.getMinSize()
1029                         scrollLength = h - w
1030                         pos = self._getPixelPos()
1031                         self.setValue(int(self._minValue + (self._maxValue - self._minValue) * -(y - pos[1] - scrollLength/2) / scrollLength))
1032                         self._callback()
1033                         return True
1034                 if self._checkHit(x, y):
1035                         self._focus = True
1036                         return True
1037                 self._focus = False
1038                 return False
1039
1040         def OnMouseDown(self, x, y, button):
1041                 if self._checkHit(x, y):
1042                         self.setFocus()
1043                         self.OnMouseMotion(x, y)
1044                         return True
1045                 return False
1046
1047         def OnMouseUp(self, x, y):
1048                 if self.hasFocus():
1049                         self._base._focus = None
1050                         return True
1051                 return False