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