sccd_widget.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. '''
  2. Created on 31-jul.-2014
  3. @author: Simon
  4. '''
  5. from PIL import Image, ImageTk
  6. import Tkinter as tk
  7. import atexit
  8. from sccd.runtime.statecharts_core import Event
  9. class SCCDWidget:
  10. controller = None
  11. def __init__(self, configure_later=False):
  12. if not configure_later:
  13. self.set_bindable_and_tagorid(None, None)
  14. def set_bindable_and_tagorid(self, bindable=None, tagorid=None):
  15. if bindable is None:
  16. bindable = self
  17. self.bindable = bindable
  18. self.mytagorid = tagorid
  19. if isinstance(self, tk.Toplevel):
  20. self.protocol("WM_DELETE_WINDOW", self.window_close)
  21. if tagorid is not None:
  22. if not isinstance(tagorid, list):
  23. tagorid = [tagorid]
  24. for t in tagorid:
  25. self.bindable.tag_bind(t, "<Button>", self.on_click)
  26. self.bindable.tag_bind(t, "<ButtonRelease>", self.on_release)
  27. self.bindable.tag_bind(t, "<Motion>", self.on_motion)
  28. self.bindable.tag_bind(t, "<Enter>", self.on_enter)
  29. self.bindable.tag_bind(t, "<Leave>", self.on_leave)
  30. self.bindable.tag_bind(t, "<Key>", self.on_key)
  31. self.bindable.tag_bind(t, "<KeyRelease>", self.on_key_release)
  32. else:
  33. self.bindable.bind("<Button>", self.on_click)
  34. self.bindable.bind("<ButtonRelease>", self.on_release)
  35. self.bindable.bind("<Motion>", self.on_motion)
  36. self.bindable.bind("<Enter>", self.on_enter)
  37. self.bindable.bind("<Leave>", self.on_leave)
  38. self.bindable.bind("<Key>", self.on_key)
  39. self.bindable.bind("<KeyRelease>", self.on_key_release)
  40. self.last_x = 50
  41. self.last_y = 50
  42. self.selected_type = None
  43. def on_click(self, event):
  44. ''' Prevent event propagation from canvas elements to canvas... '''
  45. if isinstance(self.bindable, tk.Canvas) and self.mytagorid is None:
  46. if hasattr(event.widget, 'clicked'):
  47. if event.widget.clicked:
  48. event.widget.clicked = False
  49. return
  50. elif isinstance(self.bindable, tk.Canvas):
  51. event.widget.clicked = True
  52. event_name = None
  53. if event.num == 1:
  54. event_name = "left-click"
  55. elif event.num == 2:
  56. event_name = "middle-click"
  57. elif event.num == 3:
  58. event_name = "right-click"
  59. if event_name:
  60. self.last_x = event.x
  61. self.last_y = event.y
  62. SCCDWidget.controller.addInput(Event(event_name, "input", [id(self)]))
  63. return "break"
  64. def on_release(self, event):
  65. event_name = None
  66. if event.num == 1:
  67. event_name = "left-release"
  68. elif event.num == 2:
  69. event_name = "middle-release"
  70. elif event.num == 3:
  71. event_name = "right-release"
  72. if event_name:
  73. self.last_x = event.x
  74. self.last_y = event.y
  75. SCCDWidget.controller.addInput(Event(event_name, "input", [id(self)]))
  76. return "break"
  77. def on_motion(self, event):
  78. self.last_x = event.x
  79. self.last_y = event.y
  80. SCCDWidget.controller.addInput(Event("motion", "input", [id(self)]))
  81. return "break"
  82. def on_enter(self, event):
  83. SCCDWidget.controller.addInput(Event("enter", "input", [id(self)]))
  84. return "break"
  85. def on_leave(self, event):
  86. SCCDWidget.controller.addInput(Event("leave", "input", [id(self)]))
  87. return "break"
  88. def on_key(self, event):
  89. event_name = None
  90. if event.keysym == 'Escape':
  91. event_name = "escape"
  92. elif event.keysym == 'Return':
  93. event_name = "return"
  94. elif event.keysym == 'Delete':
  95. event_name = "delete"
  96. elif event.keysym == 'Shift_L':
  97. event_name = "shift"
  98. elif event.keysym == 'Control_L':
  99. event_name = "control"
  100. elif event.keysym == 'Left':
  101. event_name = "left_key"
  102. elif event.keysym == 'Right':
  103. event_name = "right_key"
  104. elif event.keysym == 'Up':
  105. event_name = "up_key"
  106. elif event.keysym == 'Down':
  107. event_name = "down_key"
  108. if event_name:
  109. SCCDWidget.controller.addInput(Event(event_name, "input", [id(self)]))
  110. return "break"
  111. def on_key_release(self, event):
  112. event_name = None
  113. if event.keysym == 'Escape':
  114. event_name = "escape-release"
  115. elif event.keysym == 'Return':
  116. event_name = "return-release"
  117. elif event.keysym == 'Delete':
  118. event_name = "delete-release"
  119. elif event.keysym == 'Shift_L':
  120. event_name = "shift-release"
  121. elif event.keysym == 'Control_L':
  122. event_name = "control-release"
  123. if event_name:
  124. SCCDWidget.controller.addInput(Event(event_name, "input", [id(self)]))
  125. return "break"
  126. def window_close(self):
  127. SCCDWidget.controller.addInput(Event("window-close", "input", [id(self)]))
  128. class HorizontalScrolledFrame(tk.Frame):
  129. def __init__(self, parent, *args, **kw):
  130. tk.Frame.__init__(self, parent, *args, **kw)
  131. # create a canvas object and a horizontal scrollbar for scrolling it
  132. hscrollbar = tk.Scrollbar(self, orient=tk.HORIZONTAL)
  133. hscrollbar.pack(fill=tk.X, side=tk.BOTTOM, expand=tk.FALSE)
  134. self.canvas = tk.Canvas(self, bd=0, highlightthickness=0,
  135. xscrollcommand=hscrollbar.set, height=78)
  136. self.canvas.pack(side=tk.LEFT, fill=tk.X, expand=tk.TRUE)
  137. hscrollbar.config(command=self.canvas.xview)
  138. # reset the view
  139. self.canvas.xview_moveto(0)
  140. self.canvas.yview_moveto(0)
  141. # create a frame inside the canvas which will be scrolled with it
  142. self.interior = interior = tk.Frame(self.canvas)
  143. self.canvas.create_window(0, 0, window=interior, anchor=tk.NW)
  144. # track changes to the canvas and frame width and sync them,
  145. # also updating the scrollbar
  146. def _configure_interior(event):
  147. # update the scrollbars to match the size of the inner frame
  148. size = (interior.winfo_reqwidth(), interior.winfo_reqheight())
  149. self.canvas.config(scrollregion="0 0 %s %s" % size)
  150. interior.bind('<Configure>', _configure_interior)
  151. class VerticalScrolledFrame(tk.Frame):
  152. def __init__(self, parent, *args, **kw):
  153. tk.Frame.__init__(self, parent, *args, **kw)
  154. vscrollbar = tk.Scrollbar(self, orient=tk.VERTICAL)
  155. vscrollbar.pack(fill=tk.Y, side=tk.RIGHT, expand=tk.FALSE)
  156. canvas = tk.Canvas(self, bd=0, highlightthickness=0,
  157. yscrollcommand=vscrollbar.set, width=100)
  158. canvas.pack(side=tk.LEFT, fill=tk.BOTH, expand=tk.TRUE)
  159. vscrollbar.config(command=canvas.yview)
  160. canvas.xview_moveto(0)
  161. canvas.yview_moveto(0)
  162. self.interior = interior = tk.Frame(canvas)
  163. interior_id = canvas.create_window(0, 0, window=interior,
  164. anchor=tk.NW)
  165. def _configure_interior(event):
  166. size = (interior.winfo_reqwidth(), interior.winfo_reqheight())
  167. canvas.config(scrollregion="0 0 %s %s" % size)
  168. if interior.winfo_reqwidth() != canvas.winfo_width():
  169. canvas.config(width=interior.winfo_reqwidth())
  170. interior.bind('<Configure>', _configure_interior)
  171. def _configure_canvas(event):
  172. if interior.winfo_reqwidth() != canvas.winfo_width():
  173. canvas.itemconfigure(interior_id, width=canvas.winfo_width())
  174. canvas.bind('<Configure>', _configure_canvas)
  175. return
  176. class VerticallyScrolledWindow(tk.Frame):
  177. def __init__(self, root):
  178. tk.Frame.__init__(self, root)
  179. defaultbg = root.cget('bg')
  180. self.canvas = tk.Canvas(root, borderwidth=0, background=defaultbg)
  181. self.frame = tk.Frame(self.canvas, background=defaultbg)
  182. self.vsb = tk.Scrollbar(root, orient="vertical", command=self.canvas.yview)
  183. self.canvas.configure(yscrollcommand=self.vsb.set)
  184. self.vsb.pack(side="right", fill="y")
  185. self.canvas.pack(side="left", fill="both", expand=True)
  186. self.canvas.create_window((4,4), window=self.frame, anchor="nw",
  187. tags="self.frame")
  188. self.frame.bind("<Configure>", self.onFrameConfigure)
  189. self.frame.bind_all("<MouseWheel>", self.scroll)
  190. def scroll(self, event):
  191. self.canvas.yview_scroll(-1 * (event.delta / 120), "units")
  192. def onFrameConfigure(self, event):
  193. '''Reset the scroll region to encompass the inner frame'''
  194. self.canvas.configure(scrollregion=self.canvas.bbox("all"))
  195. class Visual(object):
  196. def get_params(self):
  197. raise NotImplementedError()
  198. class TextVisual(Visual):
  199. def __init__(self, text):
  200. super(TextVisual, self).__init__()
  201. self.text = text
  202. def get_params(self):
  203. return {'text': self.text}
  204. class ImageVisual(Visual):
  205. def __init__(self, img_loc):
  206. super(ImageVisual, self).__init__()
  207. try:
  208. self.img = ImageTk.PhotoImage(Image.open(img_loc).resize((32, 32), Image.ANTIALIAS))
  209. except:
  210. self.img = ImageTk.PhotoImage(Image.open("icons/todo.png").resize((32, 32), Image.ANTIALIAS))
  211. def get_params(self):
  212. return {'image': self.img}
  213. class ToolTip:
  214. def __init__(self, widget, text):
  215. self.widget = widget
  216. self.tipwindow = None
  217. self.id = None
  218. self.x = self.y = 0
  219. self.text = text
  220. def showtip(self):
  221. "Display text in tooltip window"
  222. if self.tipwindow or not self.text:
  223. return
  224. x, y, cx, cy = self.widget.bbox("insert")
  225. x = x + cx + self.widget.winfo_rootx() + 27
  226. y = y + cy + self.widget.winfo_rooty() - 20
  227. self.tipwindow = tw = tk.Toplevel(self.widget)
  228. tw.wm_overrideredirect(1)
  229. tw.wm_geometry("+%d+%d" % (x, y))
  230. try:
  231. # For Mac OS
  232. tw.tk.call("::tk::unsupported::MacWindowStyle",
  233. "style", tw._w,
  234. "help", "noActivates")
  235. except tk.TclError:
  236. pass
  237. label = tk.Label(tw, text=self.text, justify=tk.LEFT,
  238. background="#ffffe0", relief=tk.SOLID, borderwidth=1,
  239. font=("tahoma", "8", "normal"))
  240. label.pack(ipadx=1)
  241. def hidetip(self):
  242. tw = self.tipwindow
  243. self.tipwindow = None
  244. if tw:
  245. tw.destroy()