blob: 8e4d6654368881d2a0a07f4cc2e0fdbb1d216c35 [file] [log] [blame]
Just van Rossum40f9b7b1999-01-30 22:39:17 +00001import Qd
2import Win
3import Evt
4import Fm
5import FrameWork
6import Windows
7import Events
8import Wbase
9import Dlg
10import MacOS
11import Menu
12import struct
13import traceback
14
15from types import *
16
17
18class Window(FrameWork.Window, Wbase.SelectableWidget):
19
20 windowkind = Windows.documentProc
21
22 def __init__(self, possize, title = "", minsize = None, maxsize = None, tabbable = 1, show = 1):
23 import W
24 W.SelectableWidget.__init__(self, possize)
25 self._globalbounds = l, t, r, b = self.getwindowbounds(possize, minsize)
26 self._bounds = (0, 0, r - l, b - t)
27 self._tabchain = []
28 self._currentwidget = None
29 self.title = title
30 self._parentwindow = self
31 self._tabbable = tabbable
32 self._defaultbutton = None
33 self._drawwidgetbounds = 0
34 self._show = show
35 self._lastrollover = None
Just van Rossumfc372aa1999-03-03 23:15:39 +000036 self.hasclosebox = 1
Just van Rossum40f9b7b1999-01-30 22:39:17 +000037 # XXX the following is not really compatible with the
38 # new (system >= 7.5) window procs.
39 if minsize:
40 self._hasgrowbox = 1
41 self.windowkind = self.windowkind | 8
42 l, t = minsize
43 if maxsize:
44 r, b = maxsize[0] + 1, maxsize[1] + 1
45 else:
46 r, b = 32000, 32000
47 self.growlimit = (l, t, r, b)
48 else:
49 self._hasgrowbox = 0
50 if (self.windowkind == 0 or self.windowkind >= 8) and self.windowkind < 1000:
51 self.windowkind = self.windowkind | 4
52 FrameWork.Window.__init__(self, W.getapplication())
53
54 def gettitle(self):
55 return self.title
56
57 def settitle(self, title):
58 self.title = title
59 if self.wid:
60 self.wid.SetWTitle(title)
61
62 def getwindowbounds(self, size, minsize = None):
63 return windowbounds(size, minsize)
64
65 def getcurrentwidget(self):
66 return self._currentwidget
67
68 def show(self, onoff):
69 if onoff:
70 self.wid.ShowWindow()
71 else:
72 self.wid.HideWindow()
73
74 def isvisible(self):
75 return self.wid.IsWindowVisible()
76
77 def getbounds(self):
78 if 0: #self.isvisible():
79 self.wid.GetWindowContentRgn(scratchRegion)
80 self._globalbounds = GetRgnBounds(scratchRegion)
81 return self._globalbounds
82
83 def select(self):
84 self.wid.SelectWindow()
85 # not sure if this is the best place, I need it when
86 # an editor gets selected, and immediately scrolled
87 # to a certain line, waste scroll assumes everything
88 # to be in tact.
89 self.do_rawupdate(self.wid, "DummyEvent")
90
91 def open(self):
92 self.wid = Win.NewCWindow(self._globalbounds, self.title, self._show,
Just van Rossumfc372aa1999-03-03 23:15:39 +000093 self.windowkind, -1, self.hasclosebox, 0)
Just van Rossum40f9b7b1999-01-30 22:39:17 +000094 self.SetPort()
95 fnum = Fm.GetFNum("Python-Sans")
96 if fnum == 0:
97 fnum = Fm.GetFNum("Geneva")
98 Qd.TextFont(fnum) # XXX font&size from a function?
99 Qd.TextSize(9) # XXX font&size from a function?
100 if self._bindings.has_key("<open>"):
101 callback = self._bindings["<open>"]
102 callback()
103 for w in self._widgets:
104 w.forall_frombottom("open")
105 self._maketabchain()
106 if self._tabchain:
107 self._tabchain[0].select(1)
108 if self._tabbable:
109 self.bind('tab', self.nextwidget)
110 self.bind('shifttab', self.previouswidget)
111 self.do_postopen()
112
113 def close(self):
114 if not self.wid:
115 return # we are already closed
116 if self._bindings.has_key("<close>"):
117 callback = self._bindings["<close>"]
118 try:
119 rv = callback()
120 except:
121 print 'error in <close> callback'
122 traceback.print_exc()
123 else:
124 if rv:
125 return rv
126 #for key in self._widgetsdict.keys():
127 # self._removewidget(key)
128 self.forall_butself("close")
129 Wbase.SelectableWidget.close(self)
130 self._tabchain = []
131 self._currentwidget = None
132 self.wid.HideWindow()
133 self.do_postclose()
134
135 def domenu_close(self, *args):
136 self.close()
137
138 def move(self, x, y = None):
139 """absolute move"""
140 if y == None:
141 x, y = x
142 self.wid.MoveWindow(x, y, 0)
143
144 def resize(self, x, y = None):
145 if y == None:
146 x, y = x
147 if self._hasgrowbox:
148 self.SetPort()
149 Win.InvalRect(self.getgrowrect())
150 self.wid.SizeWindow(x, y, 1)
151 self._calcbounds()
152
153 def test(self, point):
154 return 1
155
156 def draw(self, visRgn = None):
157 if self._hasgrowbox:
158 self.tempcliprect(self.getgrowrect())
159 self.wid.DrawGrowIcon()
160 self.restoreclip()
161
162 def idle(self, *args):
163 self.SetPort()
164 point = Evt.GetMouse()
165 widget = self.findwidget(point, 0)
166 if self._bindings.has_key("<idle>"):
167 callback = self._bindings["<idle>"]
168 if callback():
169 return
170 if self._currentwidget is not None and hasattr(self._currentwidget, "idle"):
171 if self._currentwidget._bindings.has_key("<idle>"):
172 callback = self._currentwidget._bindings["<idle>"]
173 if callback():
174 return
175 if self._currentwidget.idle():
176 return
177 if widget is not None and hasattr(widget, "rollover"):
178 if 1: #self._lastrollover <> widget:
179 if self._lastrollover:
180 self._lastrollover.rollover(point, 0)
181 self._lastrollover = widget
182 self._lastrollover.rollover(point, 1)
183 else:
184 if self._lastrollover:
185 self._lastrollover.rollover(point, 0)
186 self._lastrollover = None
187 Wbase.SetCursor("arrow")
188
189 def xxx___select(self, widget):
190 if self._currentwidget == widget:
191 return
192 if self._bindings.has_key("<select>"):
193 callback = self._bindings["<select>"]
194 if callback(widget):
195 return
196 if widget is None:
197 if self._currentwidget is not None:
198 self._currentwidget.select(0)
199 elif type(widget) == InstanceType and widget._selectable:
200 widget.select(1)
201 elif widget == -1 or widget == 1:
202 if len(self._tabchain) <= 1:
203 return
204 temp = self._tabchain[(self._tabchain.index(self._currentwidget) + widget) % len(self._tabchain)]
205 temp.select(1)
206 else:
207 raise TypeError, "Widget is not selectable"
208
209 def setdefaultbutton(self, newdefaultbutton = None, *keys):
210 if newdefaultbutton == self._defaultbutton:
211 return
212 if self._defaultbutton:
213 self._defaultbutton._setdefault(0)
214 if not newdefaultbutton:
215 self.bind("return", None)
216 self.bind("enter", None)
217 return
218 import Wcontrols
219 if not isinstance(newdefaultbutton, Wcontrols.Button):
220 raise TypeError, "widget is not a button"
221 self._defaultbutton = newdefaultbutton
222 self._defaultbutton._setdefault(1)
223 if not keys:
224 self.bind("return", self._defaultbutton.push)
225 self.bind("enter", self._defaultbutton.push)
226 else:
227 for key in keys:
228 self.bind(key, self._defaultbutton.push)
229
230 def nextwidget(self):
231 self.xxx___select(1)
232
233 def previouswidget(self):
234 self.xxx___select(-1)
235
236 def drawwidgetbounds(self, onoff):
237 self._drawwidgetbounds = onoff
238 self.SetPort()
239 Win.InvalRect(self._bounds)
240
241 def _drawbounds(self):
242 pass
243
244 def _maketabchain(self):
245 # XXX This has to change, it's no good when we are adding or deleting widgets.
246 # XXX Perhaps we shouldn't keep a "tabchain" at all.
247 self._hasselframes = 0
248 self._collectselectablewidgets(self._widgets)
249 if self._hasselframes and len(self._tabchain) > 1:
250 self._hasselframes = 1
251 else:
252 self._hasselframes = 0
253
254 def _collectselectablewidgets(self, widgets):
255 import W
256 for w in widgets:
257 if w._selectable:
258 self._tabchain.append(w)
259 if isinstance(w, W.List):
260 self._hasselframes = 1
261 self._collectselectablewidgets(w._widgets)
262
263 def _calcbounds(self):
264 self._possize = self.wid.GetWindowPort().portRect[2:]
265 w, h = self._possize
266 self._bounds = (0, 0, w, h)
267 self.wid.GetWindowContentRgn(scratchRegion)
268 l, t, r, b = GetRgnBounds(scratchRegion)
269 self._globalbounds = l, t, l + w, t + h
270 for w in self._widgets:
271 w._calcbounds()
272
273 # FrameWork override methods
274 def do_inDrag(self, partcode, window, event):
275 where = event[3]
276 self.wid.GetWindowContentRgn(scratchRegion)
277 was_l, was_t, r, b = GetRgnBounds(scratchRegion)
278 window.DragWindow(where, self.draglimit)
279 self.wid.GetWindowContentRgn(scratchRegion)
280 is_l, is_t, r, b = GetRgnBounds(scratchRegion)
281 self._globalbounds = Qd.OffsetRect(self._globalbounds,
282 is_l - was_l, is_t - was_t)
283
284 def do_char(self, char, event):
285 import Wkeys
286 (what, message, when, where, modifiers) = event
287 key = char
288 if Wkeys.keynames.has_key(key):
289 key = Wkeys.keynames[char]
290 if modifiers & Events.shiftKey:
291 key = 'shift' + key
292 if modifiers & Events.cmdKey:
293 key = 'cmd' + key
294 if modifiers & Events.controlKey:
295 key = 'control' + key
296 if self._bindings.has_key("<key>"):
297 callback = self._bindings["<key>"]
298 if Wbase.CallbackCall(callback, 0, char, event):
299 return
300 if self._bindings.has_key(key):
301 callback = self._bindings[key]
302 Wbase.CallbackCall(callback, 0, char, event)
303 elif self._currentwidget is not None:
304 if self._currentwidget._bindings.has_key(key):
305 callback = self._currentwidget._bindings[key]
306 Wbase.CallbackCall(callback, 0, char, event)
307 else:
308 if self._currentwidget._bindings.has_key("<key>"):
309 callback = self._currentwidget._bindings["<key>"]
310 if Wbase.CallbackCall(callback, 0, char, event):
311 return
312 self._currentwidget.key(char, event)
313
314 def do_contentclick(self, point, modifiers, event):
315 widget = self.findwidget(point)
316 if widget is not None:
317 if self._bindings.has_key("<click>"):
318 callback = self._bindings["<click>"]
319 if Wbase.CallbackCall(callback, 0, point, modifiers):
320 return
321 if widget._bindings.has_key("<click>"):
322 callback = widget._bindings["<click>"]
323 if Wbase.CallbackCall(callback, 0, point, modifiers):
324 return
325 if widget._selectable:
326 widget.select(1, 1)
327 widget.click(point, modifiers)
328
329 def do_update(self, window, event):
330 Qd.EraseRgn(window.GetWindowPort().visRgn)
331 self.forall_frombottom("draw", window.GetWindowPort().visRgn)
332 if self._drawwidgetbounds:
333 self.forall_frombottom("_drawbounds")
334
335 def do_activate(self, onoff, event):
336 if not onoff:
337 if self._lastrollover:
338 self._lastrollover.rollover((0, 0), 0)
339 self._lastrollover = None
340 self.SetPort()
341 self.forall("activate", onoff)
342 self.draw()
343
344 def do_postresize(self, width, height, window):
345 Win.InvalRect(self.getgrowrect())
346 self._calcbounds()
347
348 def do_inGoAway(self, partcode, window, event):
349 where = event[3]
350 closeall = event[4] & Events.optionKey
351 if window.TrackGoAway(where):
352 if not closeall:
353 self.close()
354 else:
355 for window in self.parent._windows.values():
356 rv = window.close()
357 if rv and rv > 0:
358 return
359
360 # utilities
361 def tempcliprect(self, tempcliprect):
362 tempclip = Qd.NewRgn()
363 Qd.RectRgn(tempclip, tempcliprect)
364 self.tempclip(tempclip)
365 Qd.DisposeRgn(tempclip)
366
367 def tempclip(self, tempclip):
368 if not hasattr(self, "saveclip"):
369 self.saveclip = []
370 saveclip = Qd.NewRgn()
371 Qd.GetClip(saveclip)
372 self.saveclip.append(saveclip)
373 Qd.SetClip(tempclip)
374
375 def restoreclip(self):
376 Qd.SetClip(self.saveclip[-1])
377 Qd.DisposeRgn(self.saveclip[-1])
378 del self.saveclip[-1]
379
380 def getgrowrect(self):
381 l, t, r, b = self.wid.GetWindowPort().portRect
382 return (r - 15, b - 15, r, b)
383
384 def has_key(self, key):
385 return self._widgetsdict.has_key(key)
386
387 def __getattr__(self, attr):
388 global _successcount, _failcount, _magiccount
389 if self._widgetsdict.has_key(attr):
390 _successcount = _successcount + 1
391 return self._widgetsdict[attr]
392 if self._currentwidget is None or (attr[:7] <> 'domenu_' and
393 attr[:4] <> 'can_' and attr <> 'insert'):
394 _failcount = _failcount + 1
395 raise AttributeError, attr
396 # special case: if a domenu_xxx, can_xxx or insert method is asked for,
397 # see if the active widget supports it
398 _magiccount = _magiccount + 1
399 return getattr(self._currentwidget, attr)
400
401_successcount = 0
402_failcount = 0
403_magiccount = 0
404
405class Dialog(Window):
406
407 windowkind = Windows.movableDBoxProc
408
409 # this __init__ seems redundant, but it's not: it has less args
410 def __init__(self, possize, title = ""):
411 Window.__init__(self, possize, title)
412
413 def can_close(self, *args):
414 return 0
415
416 def getwindowbounds(self, size, minsize = None):
417 screenbounds = sl, st, sr, sb = Qd.qd.screenBits.bounds
418 w, h = size
419 l = sl + (sr - sl - w) / 2
420 t = st + (sb - st - h) / 3
421 return l, t, l + w, t + h
422
423
424class ModalDialog(Dialog):
425
426 def __init__(self, possize, title = ""):
427 Dialog.__init__(self, possize, title)
428 if title:
429 self.windowkind = Windows.movableDBoxProc
430 else:
431 self.windowkind = Windows.dBoxProc
432
433 def open(self):
434 import W
435 Dialog.open(self)
436 self.app = W.getapplication()
437 self.done = 0
438 Menu.HiliteMenu(0)
439 app = self.parent
440 app.enablemenubar(0)
441 try:
442 self.mainloop()
443 finally:
444 app.enablemenubar(1)
445
446 def close(self):
447 if not self.wid:
448 return # we are already closed
449 self.done = 1
450 del self.app
451 Dialog.close(self)
452
453 def mainloop(self):
454 saveyield = MacOS.EnableAppswitch(-1)
455 while not self.done:
456 #self.do1event()
457 self.do1event( Events.keyDownMask +
458 Events.autoKeyMask +
459 Events.activMask +
460 Events.updateMask +
461 Events.mDownMask +
462 Events.mUpMask,
463 10)
464 MacOS.EnableAppswitch(saveyield)
465
466 def do1event(self, mask = Events.everyEvent, wait = 0):
467 ok, event = self.app.getevent(mask, wait)
468 if Dlg.IsDialogEvent(event):
469 if self.app.do_dialogevent(event):
470 return
471 if ok:
472 self.dispatch(event)
473 else:
474 self.app.idle(event)
475
476 def do_keyDown(self, event):
477 self.do_key(event)
478
479 def do_autoKey(self, event):
480 if not event[-1] & Events.cmdKey:
481 self.do_key(event)
482
483 def do_key(self, event):
484 (what, message, when, where, modifiers) = event
485 w = Win.FrontWindow()
486 if w <> self.wid:
487 return
488 c = chr(message & Events.charCodeMask)
489 if modifiers & Events.cmdKey:
490 self.app.checkmenus(self)
491 result = Menu.MenuKey(ord(c))
492 id = (result>>16) & 0xffff # Hi word
493 item = result & 0xffff # Lo word
494 if id:
495 self.app.do_rawmenu(id, item, None, event)
496 return
497 self.do_char(c, event)
498
499 def do_mouseDown(self, event):
500 (what, message, when, where, modifiers) = event
501 partcode, wid = Win.FindWindow(where)
502 #
503 # Find the correct name.
504 #
505 if FrameWork.partname.has_key(partcode):
506 name = "do_" + FrameWork.partname[partcode]
507 else:
508 name = "do_%d" % partcode
509
510 if name == "do_inDesk":
511 MacOS.HandleEvent(event)
512 return
513 if wid == self.wid:
514 try:
515 handler = getattr(self, name)
516 except AttributeError:
517 handler = self.app.do_unknownpartcode
518 else:
519 #MacOS.HandleEvent(event)
520 if name == 'do_inMenuBar':
521 handler = getattr(self.parent, name)
522 else:
523 return
524 handler(partcode, wid, event)
525
526 def dispatch(self, event):
527 (what, message, when, where, modifiers) = event
528 if FrameWork.eventname.has_key(what):
529 name = "do_" + FrameWork.eventname[what]
530 else:
531 name = "do_%d" % what
532 try:
533 handler = getattr(self, name)
534 except AttributeError:
535 try:
536 handler = getattr(self.app, name)
537 except AttributeError:
538 handler = self.app.do_unknownevent
539 handler(event)
540
541
542def FrontWindowInsert(stuff):
543 if not stuff:
544 return
545 if type(stuff) <> StringType:
546 raise TypeError, 'string expected'
547 import W
548 app = W.getapplication()
549 wid = Win.FrontWindow()
550 if wid and app._windows.has_key(wid):
551 window = app._windows[wid]
552 if hasattr(window, "insert"):
553 try:
554 window.insert(stuff)
555 return
556 except:
557 pass
558 import EasyDialogs
559 if EasyDialogs.AskYesNoCancel(
Just van Rossumedab9391999-02-02 22:31:05 +0000560 "Can¹t find window or widget to insert text into; copy to clipboard instead?",
Just van Rossum40f9b7b1999-01-30 22:39:17 +0000561 1) == 1:
562 import Scrap
563 Scrap.ZeroScrap()
564 Scrap.PutScrap('TEXT', stuff)
565
566
567# not quite based on the same function in FrameWork
568_windowcounter = 0
569
570def getnextwindowpos():
571 global _windowcounter
572 rows = 8
573 l = 4 * (rows + 1 - (_windowcounter % rows) + _windowcounter / rows)
574 t = 44 + 20 * (_windowcounter % rows)
575 _windowcounter = _windowcounter + 1
576 return l, t
577
578def windowbounds(preferredsize, minsize = None):
579 "Return sensible window bounds"
580
581 global _windowcounter
582 if len(preferredsize) == 4:
583 bounds = l, t, r, b = preferredsize
584 union = Qd.UnionRect(bounds, Qd.qd.screenBits.bounds)
585 if union == Qd.qd.screenBits.bounds:
586 return bounds
587 else:
588 preferredsize = r - l, b - t
589 if not minsize:
590 minsize = preferredsize
591 minwidth, minheight = minsize
592 width, height = preferredsize
593
594 sl, st, sr, sb = screenbounds = Qd.InsetRect(Qd.qd.screenBits.bounds, 4, 4)
595 l, t = getnextwindowpos()
596 if (l + width) > sr:
597 _windowcounter = 0
598 l, t = getnextwindowpos()
599 r = l + width
600 b = t + height
601 if (t + height) > sb:
602 b = sb
603 if (b - t) < minheight:
604 b = t + minheight
605 return l, t, r, b
606
607scratchRegion = Qd.NewRgn()
608
609# util -- move somewhere convenient???
610def GetRgnBounds(the_Rgn):
611 (t, l, b, r) = struct.unpack("hhhh", the_Rgn.data[2:10])
612 return (l, t, r, b)