summaryrefslogtreecommitdiffstats
path: root/Mac/Demo/waste
diff options
context:
space:
mode:
authorJack Jansen <jack.jansen@cwi.nl>1996-05-07 15:25:36 (GMT)
committerJack Jansen <jack.jansen@cwi.nl>1996-05-07 15:25:36 (GMT)
commit1df16d095e16e3f77906826ddeedbd6199e4f652 (patch)
tree9ec471a3ee082f4259c93bbcd87cb95e2fcac4f9 /Mac/Demo/waste
parent42bc4b245a80b9508ab15c60fbf5e373bc935554 (diff)
downloadcpython-1df16d095e16e3f77906826ddeedbd6199e4f652.zip
cpython-1df16d095e16e3f77906826ddeedbd6199e4f652.tar.gz
cpython-1df16d095e16e3f77906826ddeedbd6199e4f652.tar.bz2
A version of swed that can import html files. Also demonstrates
implementing object handlers in Python.
Diffstat (limited to 'Mac/Demo/waste')
-rw-r--r--Mac/Demo/waste/htmled.py826
1 files changed, 826 insertions, 0 deletions
diff --git a/Mac/Demo/waste/htmled.py b/Mac/Demo/waste/htmled.py
new file mode 100644
index 0000000..15b9035
--- /dev/null
+++ b/Mac/Demo/waste/htmled.py
@@ -0,0 +1,826 @@
+# A minimal text editor.
+#
+# To be done:
+# - Functionality: find, etc.
+
+from Menu import DrawMenuBar
+from FrameWork import *
+import Win
+import Qd
+import Res
+import Fm
+import waste
+import WASTEconst
+import Scrap
+import os
+import macfs
+import regsub
+import string
+import htmllib
+
+WATCH = Qd.GetCursor(4).data
+
+LEFTMARGIN=0
+
+UNDOLABELS = [ # Indexed by WEGetUndoInfo() value
+ None, "", "typing", "Cut", "Paste", "Clear", "Drag", "Style"]
+
+# Style and size menu. Note that style order is important (tied to bit values)
+STYLES = [
+ ("Bold", "B"), ("Italic", "I"), ("Underline", "U"), ("Outline", "O"),
+ ("Shadow", ""), ("Condensed", ""), ("Extended", "")
+ ]
+SIZES = [ 9, 10, 12, 14, 18, 24]
+
+# Sizes for HTML tag types
+HTML_SIZE={
+ 'h1': 18,
+ 'h2': 14
+}
+
+BIGREGION=Qd.NewRgn()
+Qd.SetRectRgn(BIGREGION, -16000, -16000, 16000, 16000)
+
+class WasteWindow(ScrolledWindow):
+ def open(self, path, name, data):
+ self.path = path
+ self.name = name
+ r = windowbounds(400, 400)
+ w = Win.NewWindow(r, name, 1, 0, -1, 1, 0x55555555)
+ self.wid = w
+ vr = LEFTMARGIN, 0, r[2]-r[0]-15, r[3]-r[1]-15
+ dr = (0, 0, vr[2], 0)
+ Qd.SetPort(w)
+ Qd.TextFont(4)
+ Qd.TextSize(9)
+ flags = WASTEconst.weDoAutoScroll | WASTEconst.weDoOutlineHilite | \
+ WASTEconst.weDoMonoStyled | WASTEconst.weDoUndo
+ self.ted = waste.WENew(dr, vr, flags)
+ style, soup = self.getstylesoup(self.path)
+ self.ted.WEInsert(data, style, soup)
+ self.ted.WESetSelection(0,0)
+ self.ted.WECalText()
+ self.ted.WEResetModCount()
+ w.DrawGrowIcon()
+ self.scrollbars()
+ self.do_postopen()
+ self.do_activate(1, None)
+
+ def getstylesoup(self, pathname):
+ if not pathname:
+ return None, None
+ oldrf = Res.CurResFile()
+ try:
+ rf = Res.OpenResFile(self.path)
+ except Res.Error:
+ return None, None
+ try:
+ hstyle = Res.Get1Resource('styl', 128)
+ hstyle.DetachResource()
+ except Res.Error:
+ hstyle = None
+ try:
+ hsoup = Res.Get1Resource('SOUP', 128)
+ hsoup.DetachResource()
+ except Res.Error:
+ hsoup = None
+ Res.CloseResFile(rf)
+ Res.UseResFile(oldrf)
+ return hstyle, hsoup
+
+ def do_idle(self, event):
+ (what, message, when, where, modifiers) = event
+ Qd.SetPort(self.wid)
+ self.ted.WEIdle()
+ if self.ted.WEAdjustCursor(where, BIGREGION):
+ return
+ Qd.SetCursor(Qd.qd.arrow)
+
+ def getscrollbarvalues(self):
+ dr = self.ted.WEGetDestRect()
+ vr = self.ted.WEGetViewRect()
+ vx = self.scalebarvalue(dr[0], dr[2], vr[0], vr[2])
+ vy = self.scalebarvalue(dr[1], dr[3], vr[1], vr[3])
+ return vx, vy
+
+ def scrollbar_callback(self, which, what, value):
+ if which == 'y':
+ #
+ # "line" size is minimum of top and bottom line size
+ #
+ topline_off,dummy = self.ted.WEGetOffset((1,1))
+ topline_num = self.ted.WEOffsetToLine(topline_off)
+ toplineheight = self.ted.WEGetHeight(topline_num, topline_num+1)
+
+ botlinepos = self.ted.WEGetViewRect()[3]
+ botline_off, dummy = self.ted.WEGetOffset((1, botlinepos-1))
+ botline_num = self.ted.WEOffsetToLine(botline_off)
+ botlineheight = self.ted.WEGetHeight(botline_num, botline_num+1)
+
+ if botlineheight == 0:
+ botlineheight = self.ted.WEGetHeight(botline_num-1, botline_num)
+ if botlineheight < toplineheight:
+ lineheight = botlineheight
+ else:
+ lineheight = toplineheight
+ if lineheight <= 0:
+ lineheight = 1
+ #
+ # Now do the command.
+ #
+ if what == 'set':
+ height = self.ted.WEGetHeight(0, 0x3fffffff)
+ cur = self.getscrollbarvalues()[1]
+ delta = (cur-value)*height/32767
+ if what == '-':
+ delta = lineheight
+ elif what == '--':
+ delta = (self.ted.WEGetViewRect()[3]-lineheight)
+ if delta <= 0:
+ delta = lineheight
+ elif what == '+':
+ delta = -lineheight
+ elif what == '++':
+ delta = -(self.ted.WEGetViewRect()[3]-lineheight)
+ if delta >= 0:
+ delta = -lineheight
+ self.ted.WEScroll(0, delta)
+ else:
+ if what == 'set':
+ return # XXXX
+ vr = self.ted.WEGetViewRect()
+ winwidth = vr[2]-vr[0]
+ if what == '-':
+ delta = winwidth/10
+ elif what == '--':
+ delta = winwidth/2
+ elif what == '+':
+ delta = -winwidth/10
+ elif what == '++':
+ delta = -winwidth/2
+ self.ted.WEScroll(delta, 0)
+ # Pin the scroll
+ l, t, r, b = self.ted.WEGetDestRect()
+ vl, vt, vr, vb = self.ted.WEGetViewRect()
+ if t > 0 or l > 0:
+ dx = dy = 0
+ if t > 0: dy = -t
+ if l > 0: dx = -l
+ self.ted.WEScroll(dx, dy)
+ elif b < vb:
+ self.ted.WEScroll(0, vb-b)
+
+
+ def do_activate(self, onoff, evt):
+ Qd.SetPort(self.wid)
+ ScrolledWindow.do_activate(self, onoff, evt)
+ if onoff:
+ self.ted.WEActivate()
+ self.parent.active = self
+ self.parent.updatemenubar()
+ else:
+ self.ted.WEDeactivate()
+
+ def do_update(self, wid, event):
+ region = wid.GetWindowPort().visRgn
+ if Qd.EmptyRgn(region):
+ return
+ Qd.EraseRgn(region)
+ self.ted.WEUpdate(region)
+ self.updatescrollbars()
+
+ def do_postresize(self, width, height, window):
+ l, t, r, b = self.ted.WEGetViewRect()
+ vr = (l, t, l+width-15, t+height-15)
+ self.ted.WESetViewRect(vr)
+ Win.InvalRect(vr)
+ ScrolledWindow.do_postresize(self, width, height, window)
+
+ def do_contentclick(self, local, modifiers, evt):
+ (what, message, when, where, modifiers) = evt
+ self.ted.WEClick(local, modifiers, when)
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def do_char(self, ch, event):
+ self.ted.WESelView()
+ (what, message, when, where, modifiers) = event
+ self.ted.WEKey(ord(ch), modifiers)
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def close(self):
+ if self.ted.WEGetModCount():
+ save = EasyDialogs.AskYesNoCancel('Save window "%s" before closing?'%self.name, 1)
+ if save > 0:
+ self.menu_save()
+ elif save < 0:
+ return
+ if self.parent.active == self:
+ self.parent.active = None
+ self.parent.updatemenubar()
+ del self.ted
+ self.do_postclose()
+
+ def menu_save(self):
+ if not self.path:
+ self.menu_save_as()
+ return # Will call us recursively
+ #
+ # First save data
+ #
+ dhandle = self.ted.WEGetText()
+ data = dhandle.data
+ fp = open(self.path, 'wb') # NOTE: wb, because data has CR for end-of-line
+ fp.write(data)
+ if data[-1] <> '\r': fp.write('\r')
+ fp.close()
+ #
+ # Now save style and soup
+ #
+ oldresfile = Res.CurResFile()
+ try:
+ rf = Res.OpenResFile(self.path)
+ except Res.Error:
+ Res.CreateResFile(self.path)
+ rf = Res.OpenResFile(self.path)
+ styles = Res.Resource('')
+ soup = Res.Resource('')
+ self.ted.WECopyRange(0, 0x3fffffff, None, styles, soup)
+ styles.AddResource('styl', 128, '')
+ soup.AddResource('SOUP', 128, '')
+ Res.CloseResFile(rf)
+ Res.UseResFile(oldresfile)
+
+ self.ted.WEResetModCount()
+
+ def menu_save_as(self):
+ fss, ok = macfs.StandardPutFile('Save as:')
+ if not ok: return
+ self.path = fss.as_pathname()
+ self.name = os.path.split(self.path)[-1]
+ self.wid.SetWTitle(self.name)
+ self.menu_save()
+
+ def menu_insert(self, fp):
+ self.ted.WESelView()
+ data = fp.read()
+ self.ted.WEInsert(data, None, None)
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_insert_html(self, fp):
+ import htmllib
+ import formatter
+ f = formatter.AbstractFormatter(self)
+
+ # Remember where we are, and don't update
+ Qd.SetCursor(WATCH)
+ start, dummy = self.ted.WEGetSelection()
+ self.ted.WEFeatureFlag(WASTEconst.weFInhibitRecal, 1)
+
+ self.html_init()
+ p = MyHTMLParser(f)
+ p.feed(fp.read())
+
+ # Restore updating, recalc, set focus
+ dummy, end = self.ted.WEGetSelection()
+ self.ted.WECalText()
+ self.ted.WESetSelection(start, end)
+ self.ted.WESelView()
+ self.ted.WEFeatureFlag(WASTEconst.weFInhibitRecal, 0)
+ Win.InvalRect(self.ted.WEGetViewRect())
+
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_cut(self):
+ self.ted.WESelView()
+ self.ted.WECut()
+ Scrap.ZeroScrap()
+ self.ted.WECut()
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_copy(self):
+ Scrap.ZeroScrap()
+ self.ted.WECopy()
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_paste(self):
+ self.ted.WESelView()
+ self.ted.WEPaste()
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_clear(self):
+ self.ted.WESelView()
+ self.ted.WEDelete()
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_undo(self):
+ self.ted.WEUndo()
+ self.updatescrollbars()
+ self.parent.updatemenubar()
+
+ def menu_setfont(self, font):
+ font = Fm.GetFNum(font)
+ self.mysetstyle(WASTEconst.weDoFont, (font, 0, 0, (0,0,0)))
+ self.parent.updatemenubar()
+
+ def menu_modface(self, face):
+ self.mysetstyle(WASTEconst.weDoFace|WASTEconst.weDoToggleFace,
+ (0, face, 0, (0,0,0)))
+
+ def menu_setface(self, face):
+ self.mysetstyle(WASTEconst.weDoFace|WASTEconst.weDoReplaceFace,
+ (0, face, 0, (0,0,0)))
+
+ def menu_setsize(self, size):
+ self.mysetstyle(WASTEconst.weDoSize, (0, 0, size, (0,0,0)))
+
+ def menu_incsize(self, size):
+ self.mysetstyle(WASTEconst.weDoAddSize, (0, 0, size, (0,0,0)))
+
+ def mysetstyle(self, which, how):
+ self.ted.WESelView()
+ self.ted.WESetStyle(which, how)
+ self.parent.updatemenubar()
+
+ def have_selection(self):
+ start, stop = self.ted.WEGetSelection()
+ return start < stop
+
+ def can_paste(self):
+ return self.ted.WECanPaste()
+
+ def can_undo(self):
+ which, redo = self.ted.WEGetUndoInfo()
+ which = UNDOLABELS[which]
+ if which == None: return None
+ if redo:
+ return "Redo "+which
+ else:
+ return "Undo "+which
+
+ def getruninfo(self):
+ all = (WASTEconst.weDoFont | WASTEconst.weDoFace | WASTEconst.weDoSize)
+ dummy, mode, (font, face, size, color) = self.ted.WEContinuousStyle(all)
+ if not (mode & WASTEconst.weDoFont):
+ font = None
+ else:
+ font = Fm.GetFontName(font)
+ if not (mode & WASTEconst.weDoFace): fact = None
+ if not (mode & WASTEconst.weDoSize): size = None
+ return font, face, size
+
+ #
+ # Methods for writer class for html formatter
+ #
+
+ def html_init(self):
+ self.html_font = [12, 0, 0, 0]
+ self.html_style = 0
+ self.html_color = (0,0,0)
+ self.new_font(self.html_font)
+
+ def new_font(self, font):
+ if font == None:
+ font = (12, 0, 0, 0)
+ font = map(lambda x:x, font)
+ for i in range(len(font)):
+ if font[i] == None:
+ font[i] = self.html_font[i]
+ [size, italic, bold, tt] = font
+ self.html_font = font[:]
+ if tt:
+ font = Fm.GetFNum('Courier')
+ else:
+ font = Fm.GetFNum('Times')
+ if HTML_SIZE.has_key(size):
+ size = HTML_SIZE[size]
+ else:
+ size = 12
+ face = 0
+ if bold: face = face | 1
+ if italic: face = face | 2
+ face = face | self.html_style
+ self.ted.WESetStyle(WASTEconst.weDoFont | WASTEconst.weDoFace |
+ WASTEconst.weDoSize | WASTEconst.weDoColor,
+ (font, face, size, self.html_color))
+
+ def new_margin(self, margin, level):
+ self.ted.WEInsert('[Margin %s %s]'%(margin, level), None, None)
+
+ def new_spacing(self, spacing):
+ self.ted.WEInsert('[spacing %s]'%spacing, None, None)
+
+ def new_styles(self, styles):
+ self.html_style = 0
+ self.html_color = (0,0,0)
+ if 'anchor' in styles:
+ self.html_style = self.html_style | 4
+ self.html_color = (0xffff, 0, 0)
+ self.new_font(self.html_font)
+
+ def send_paragraph(self, blankline):
+ self.ted.WEInsert('\r'*(blankline+1), None, None)
+
+ def send_line_break(self):
+ self.ted.WEInsert('\r', None, None)
+
+ def send_hor_rule(self, *args, **kw):
+ # Ignore ruler options, for now
+ dummydata = Res.Resource('')
+ self.ted.WEInsertObject('rulr', dummydata, (0,0))
+
+ def send_label_data(self, data):
+ self.ted.WEInsert(data, None, None)
+
+ def send_flowing_data(self, data):
+ self.ted.WEInsert(data, None, None)
+
+ def send_literal_data(self, data):
+ data = regsub.gsub('\n', '\r', data)
+ data = string.expandtabs(data)
+ self.ted.WEInsert(data, None, None)
+
+class Wed(Application):
+ def __init__(self):
+ Application.__init__(self)
+ self.num = 0
+ self.active = None
+ self.updatemenubar()
+ waste.STDObjectHandlers()
+ # Handler for horizontal ruler
+ waste.WEInstallObjectHandler('rulr', 'new ', self.newRuler)
+ waste.WEInstallObjectHandler('rulr', 'draw', self.drawRuler)
+
+ def makeusermenus(self):
+ self.filemenu = m = Menu(self.menubar, "File")
+ self.newitem = MenuItem(m, "New window", "N", self.open)
+ self.openitem = MenuItem(m, "Open...", "O", self.openfile)
+ self.closeitem = MenuItem(m, "Close", "W", self.closewin)
+ m.addseparator()
+ self.saveitem = MenuItem(m, "Save", "S", self.save)
+ self.saveasitem = MenuItem(m, "Save as...", "", self.saveas)
+ m.addseparator()
+ self.insertitem = MenuItem(m, "Insert plaintext...", "", self.insertfile)
+ self.htmlitem = MenuItem(m, "Insert HTML...", "", self.inserthtml)
+ m.addseparator()
+ self.quititem = MenuItem(m, "Quit", "Q", self.quit)
+
+ self.editmenu = m = Menu(self.menubar, "Edit")
+ self.undoitem = MenuItem(m, "Undo", "Z", self.undo)
+ self.cutitem = MenuItem(m, "Cut", "X", self.cut)
+ self.copyitem = MenuItem(m, "Copy", "C", self.copy)
+ self.pasteitem = MenuItem(m, "Paste", "V", self.paste)
+ self.clearitem = MenuItem(m, "Clear", "", self.clear)
+
+ self.makefontmenu()
+
+ # Groups of items enabled together:
+ self.windowgroup = [self.closeitem, self.saveitem, self.saveasitem,
+ self.editmenu, self.fontmenu, self.facemenu, self.sizemenu,
+ self.insertitem]
+ self.focusgroup = [self.cutitem, self.copyitem, self.clearitem]
+ self.windowgroup_on = -1
+ self.focusgroup_on = -1
+ self.pastegroup_on = -1
+ self.undo_label = "never"
+ self.ffs_values = ()
+
+ def makefontmenu(self):
+ self.fontmenu = Menu(self.menubar, "Font")
+ self.fontnames = getfontnames()
+ self.fontitems = []
+ for n in self.fontnames:
+ m = MenuItem(self.fontmenu, n, "", self.selfont)
+ self.fontitems.append(m)
+ self.facemenu = Menu(self.menubar, "Style")
+ self.faceitems = []
+ for n, shortcut in STYLES:
+ m = MenuItem(self.facemenu, n, shortcut, self.selface)
+ self.faceitems.append(m)
+ self.facemenu.addseparator()
+ self.faceitem_normal = MenuItem(self.facemenu, "Normal", "N",
+ self.selfacenormal)
+ self.sizemenu = Menu(self.menubar, "Size")
+ self.sizeitems = []
+ for n in SIZES:
+ m = MenuItem(self.sizemenu, `n`, "", self.selsize)
+ self.sizeitems.append(m)
+ self.sizemenu.addseparator()
+ self.sizeitem_bigger = MenuItem(self.sizemenu, "Bigger", "+",
+ self.selsizebigger)
+ self.sizeitem_smaller = MenuItem(self.sizemenu, "Smaller", "-",
+ self.selsizesmaller)
+
+ def selfont(self, id, item, *rest):
+ if self.active:
+ font = self.fontnames[item-1]
+ self.active.menu_setfont(font)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def selface(self, id, item, *rest):
+ if self.active:
+ face = (1<<(item-1))
+ self.active.menu_modface(face)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def selfacenormal(self, *rest):
+ if self.active:
+ self.active.menu_setface(0)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def selsize(self, id, item, *rest):
+ if self.active:
+ size = SIZES[item-1]
+ self.active.menu_setsize(size)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def selsizebigger(self, *rest):
+ if self.active:
+ self.active.menu_incsize(2)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def selsizesmaller(self, *rest):
+ if self.active:
+ self.active.menu_incsize(-2)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def updatemenubar(self):
+ changed = 0
+ on = (self.active <> None)
+ if on <> self.windowgroup_on:
+ for m in self.windowgroup:
+ m.enable(on)
+ self.windowgroup_on = on
+ changed = 1
+ if on:
+ # only if we have an edit menu
+ on = self.active.have_selection()
+ if on <> self.focusgroup_on:
+ for m in self.focusgroup:
+ m.enable(on)
+ self.focusgroup_on = on
+ changed = 1
+ on = self.active.can_paste()
+ if on <> self.pastegroup_on:
+ self.pasteitem.enable(on)
+ self.pastegroup_on = on
+ changed = 1
+ on = self.active.can_undo()
+ if on <> self.undo_label:
+ if on:
+ self.undoitem.enable(1)
+ self.undoitem.settext(on)
+ self.undo_label = on
+ else:
+ self.undoitem.settext("Nothing to undo")
+ self.undoitem.enable(0)
+ changed = 1
+ if self.updatefontmenus():
+ changed = 1
+ if changed:
+ DrawMenuBar()
+
+ def updatefontmenus(self):
+ info = self.active.getruninfo()
+ if info == self.ffs_values:
+ return 0
+ # Remove old checkmarks
+ if self.ffs_values == ():
+ self.ffs_values = (None, None, None)
+ font, face, size = self.ffs_values
+ if font <> None:
+ fnum = self.fontnames.index(font)
+ self.fontitems[fnum].check(0)
+ if face <> None:
+ for i in range(len(self.faceitems)):
+ if face & (1<<i):
+ self.faceitems[i].check(0)
+ if size <> None:
+ for i in range(len(self.sizeitems)):
+ if SIZES[i] == size:
+ self.sizeitems[i].check(0)
+
+ self.ffs_values = info
+ # Set new checkmarks
+ font, face, size = self.ffs_values
+ if font <> None:
+ fnum = self.fontnames.index(font)
+ self.fontitems[fnum].check(1)
+ if face <> None:
+ for i in range(len(self.faceitems)):
+ if face & (1<<i):
+ self.faceitems[i].check(1)
+ if size <> None:
+ for i in range(len(self.sizeitems)):
+ if SIZES[i] == size:
+ self.sizeitems[i].check(1)
+ # Set outline/normal for sizes
+ if font:
+ exists = getfontsizes(font, SIZES)
+ for i in range(len(self.sizeitems)):
+ if exists[i]:
+ self.sizeitems[i].setstyle(0)
+ else:
+ self.sizeitems[i].setstyle(8)
+
+ #
+ # Apple menu
+ #
+
+ def do_about(self, id, item, window, event):
+ EasyDialogs.Message("A simple single-font text editor based on WASTE")
+
+ #
+ # File menu
+ #
+
+ def open(self, *args):
+ self._open(0)
+
+ def openfile(self, *args):
+ self._open(1)
+
+ def _open(self, askfile):
+ if askfile:
+ fss, ok = macfs.StandardGetFile('TEXT')
+ if not ok:
+ return
+ path = fss.as_pathname()
+ name = os.path.split(path)[-1]
+ try:
+ fp = open(path, 'rb') # NOTE binary, we need cr as end-of-line
+ data = fp.read()
+ fp.close()
+ except IOError, arg:
+ EasyDialogs.Message("IOERROR: "+`arg`)
+ return
+ else:
+ path = None
+ name = "Untitled %d"%self.num
+ data = ''
+ w = WasteWindow(self)
+ w.open(path, name, data)
+ self.num = self.num + 1
+
+ def insertfile(self, *args):
+ if self.active:
+ fss, ok = macfs.StandardGetFile('TEXT')
+ if not ok:
+ return
+ path = fss.as_pathname()
+ try:
+ fp = open(path, 'rb') # NOTE binary, we need cr as end-of-line
+ except IOError, arg:
+ EasyDialogs.Message("IOERROR: "+`arg`)
+ return
+ self.active.menu_insert(fp)
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def inserthtml(self, *args):
+ if self.active:
+ fss, ok = macfs.StandardGetFile('TEXT')
+ if not ok:
+ return
+ path = fss.as_pathname()
+ try:
+ fp = open(path, 'r')
+ except IOError, arg:
+ EasyDialogs.Message("IOERROR: "+`arg`)
+ return
+ self.active.menu_insert_html(fp)
+ else:
+ EasyDialogs.Message("No active window?")
+
+
+ def closewin(self, *args):
+ if self.active:
+ self.active.close()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def save(self, *args):
+ if self.active:
+ self.active.menu_save()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def saveas(self, *args):
+ if self.active:
+ self.active.menu_save_as()
+ else:
+ EasyDialogs.Message("No active window?")
+
+
+ def quit(self, *args):
+ for w in self._windows.values():
+ w.close()
+ if self._windows:
+ return
+ raise self
+
+ #
+ # Edit menu
+ #
+
+ def undo(self, *args):
+ if self.active:
+ self.active.menu_undo()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def cut(self, *args):
+ if self.active:
+ self.active.menu_cut()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def copy(self, *args):
+ if self.active:
+ self.active.menu_copy()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def paste(self, *args):
+ if self.active:
+ self.active.menu_paste()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ def clear(self, *args):
+ if self.active:
+ self.active.menu_clear()
+ else:
+ EasyDialogs.Message("No active window?")
+
+ #
+ # Other stuff
+ #
+
+ def idle(self, event):
+ if self.active:
+ self.active.do_idle(event)
+
+ def newRuler(self, obj):
+ """Insert a new ruler. Make it as wide as the window minus 2 pxls"""
+ ted = obj.WEGetObjectOwner()
+ l, t, r, b = ted.WEGetDestRect()
+ return r-l, 4
+
+ def drawRuler(self, (l, t, r, b), obj):
+ y = (t+b)/2
+ Qd.MoveTo(l+2, y)
+ Qd.LineTo(r-2, y)
+ return 0
+
+class MyHTMLParser(htmllib.HTMLParser):
+
+ def anchor_bgn(self, href, name, type):
+ self.anchor = href
+ if self.anchor:
+ self.anchorlist.append(href)
+ self.formatter.push_style('anchor')
+
+ def anchor_end(self):
+ if self.anchor:
+ self.anchor = None
+ self.formatter.pop_style()
+
+
+def getfontnames():
+ names = []
+ for i in range(256):
+ n = Fm.GetFontName(i)
+ if n: names.append(n)
+ return names
+
+def getfontsizes(name, sizes):
+ exist = []
+ num = Fm.GetFNum(name)
+ for sz in sizes:
+ if Fm.RealFont(num, sz):
+ exist.append(1)
+ else:
+ exist.append(0)
+ return exist
+
+def main():
+ App = Wed()
+ App.mainloop()
+
+if __name__ == '__main__':
+ main()
+