| # Text formatting abstractions |
| |
| |
| import string |
| import Para |
| |
| |
| # A formatter back-end object has one method that is called by the formatter: |
| # addpara(p), where p is a paragraph object. For example: |
| |
| |
| # Formatter back-end to do nothing at all with the paragraphs |
| class NullBackEnd: |
| # |
| def __init__(self): |
| pass |
| # |
| def addpara(self, p): |
| pass |
| # |
| def bgn_anchor(self, id): |
| pass |
| # |
| def end_anchor(self, id): |
| pass |
| |
| |
| # Formatter back-end to collect the paragraphs in a list |
| class SavingBackEnd(NullBackEnd): |
| # |
| def __init__(self): |
| self.paralist = [] |
| # |
| def addpara(self, p): |
| self.paralist.append(p) |
| # |
| def hitcheck(self, h, v): |
| hits = [] |
| for p in self.paralist: |
| if p.top <= v <= p.bottom: |
| for id in p.hitcheck(h, v): |
| if id not in hits: |
| hits.append(id) |
| return hits |
| # |
| def extract(self): |
| text = '' |
| for p in self.paralist: |
| text = text + (p.extract()) |
| return text |
| # |
| def extractpart(self, long1, long2): |
| if long1 > long2: long1, long2 = long2, long1 |
| para1, pos1 = long1 |
| para2, pos2 = long2 |
| text = '' |
| while para1 < para2: |
| ptext = self.paralist[para1].extract() |
| text = text + ptext[pos1:] |
| pos1 = 0 |
| para1 = para1 + 1 |
| ptext = self.paralist[para2].extract() |
| return text + ptext[pos1:pos2] |
| # |
| def whereis(self, d, h, v): |
| total = 0 |
| for i in range(len(self.paralist)): |
| p = self.paralist[i] |
| result = p.whereis(d, h, v) |
| if result <> None: |
| return i, result |
| return None |
| # |
| def roundtowords(self, long1, long2): |
| i, offset = long1 |
| text = self.paralist[i].extract() |
| while offset > 0 and text[offset-1] <> ' ': offset = offset-1 |
| long1 = i, offset |
| # |
| i, offset = long2 |
| text = self.paralist[i].extract() |
| n = len(text) |
| while offset < n-1 and text[offset] <> ' ': offset = offset+1 |
| long2 = i, offset |
| # |
| return long1, long2 |
| # |
| def roundtoparagraphs(self, long1, long2): |
| long1 = long1[0], 0 |
| long2 = long2[0], len(self.paralist[long2[0]].extract()) |
| return long1, long2 |
| |
| |
| # Formatter back-end to send the text directly to the drawing object |
| class WritingBackEnd(NullBackEnd): |
| # |
| def __init__(self, d, width): |
| self.d = d |
| self.width = width |
| self.lineno = 0 |
| # |
| def addpara(self, p): |
| self.lineno = p.render(self.d, 0, self.lineno, self.width) |
| |
| |
| # A formatter receives a stream of formatting instructions and assembles |
| # these into a stream of paragraphs on to a back-end. The assembly is |
| # parametrized by a text measurement object, which must match the output |
| # operations of the back-end. The back-end is responsible for splitting |
| # paragraphs up in lines of a given maximum width. (This is done because |
| # in a windowing environment, when the window size changes, there is no |
| # need to redo the assembly into paragraphs, but the splitting into lines |
| # must be done taking the new window size into account.) |
| |
| |
| # Formatter base class. Initialize it with a text measurement object, |
| # which is used for text measurements, and a back-end object, |
| # which receives the completed paragraphs. The formatting methods are: |
| # setfont(font) |
| # setleftindent(nspaces) |
| # setjust(type) where type is 'l', 'c', 'r', or 'lr' |
| # flush() |
| # vspace(nlines) |
| # needvspace(nlines) |
| # addword(word, nspaces) |
| class BaseFormatter: |
| # |
| def __init__(self, d, b): |
| # Drawing object used for text measurements |
| self.d = d |
| # |
| # BackEnd object receiving completed paragraphs |
| self.b = b |
| # |
| # Parameters of the formatting model |
| self.leftindent = 0 |
| self.just = 'l' |
| self.font = None |
| self.blanklines = 0 |
| # |
| # Parameters derived from the current font |
| self.space = d.textwidth(' ') |
| self.line = d.lineheight() |
| self.ascent = d.baseline() |
| self.descent = self.line - self.ascent |
| # |
| # Parameter derived from the default font |
| self.n_space = self.space |
| # |
| # Current paragraph being built |
| self.para = None |
| self.nospace = 1 |
| # |
| # Font to set on the next word |
| self.nextfont = None |
| # |
| def newpara(self): |
| return Para.Para() |
| # |
| def setfont(self, font): |
| if font == None: return |
| self.font = self.nextfont = font |
| d = self.d |
| d.setfont(font) |
| self.space = d.textwidth(' ') |
| self.line = d.lineheight() |
| self.ascent = d.baseline() |
| self.descent = self.line - self.ascent |
| # |
| def setleftindent(self, nspaces): |
| self.leftindent = int(self.n_space * nspaces) |
| if self.para: |
| hang = self.leftindent - self.para.indent_left |
| if hang > 0 and self.para.getlength() <= hang: |
| self.para.makehangingtag(hang) |
| self.nospace = 1 |
| else: |
| self.flush() |
| # |
| def setrightindent(self, nspaces): |
| self.rightindent = int(self.n_space * nspaces) |
| if self.para: |
| self.para.indent_right = self.rightindent |
| self.flush() |
| # |
| def setjust(self, just): |
| self.just = just |
| if self.para: |
| self.para.just = self.just |
| # |
| def flush(self): |
| if self.para: |
| self.b.addpara(self.para) |
| self.para = None |
| if self.font <> None: |
| self.d.setfont(self.font) |
| self.nospace = 1 |
| # |
| def vspace(self, nlines): |
| self.flush() |
| if nlines > 0: |
| self.para = self.newpara() |
| tuple = None, '', 0, 0, 0, int(nlines*self.line), 0 |
| self.para.words.append(tuple) |
| self.flush() |
| self.blanklines = self.blanklines + nlines |
| # |
| def needvspace(self, nlines): |
| self.flush() # Just to be sure |
| if nlines > self.blanklines: |
| self.vspace(nlines - self.blanklines) |
| # |
| def addword(self, text, space): |
| if self.nospace and not text: |
| return |
| self.nospace = 0 |
| self.blanklines = 0 |
| if not self.para: |
| self.para = self.newpara() |
| self.para.indent_left = self.leftindent |
| self.para.just = self.just |
| self.nextfont = self.font |
| space = int(space * self.space) |
| self.para.words.append(self.nextfont, text, \ |
| self.d.textwidth(text), space, space, \ |
| self.ascent, self.descent) |
| self.nextfont = None |
| # |
| def bgn_anchor(self, id): |
| if not self.para: |
| self.nospace = 0 |
| self.addword('', 0) |
| self.para.bgn_anchor(id) |
| # |
| def end_anchor(self, id): |
| if not self.para: |
| self.nospace = 0 |
| self.addword('', 0) |
| self.para.end_anchor(id) |
| |
| |
| # Measuring object for measuring text as viewed on a tty |
| class NullMeasurer: |
| # |
| def __init__(self): |
| pass |
| # |
| def setfont(self, font): |
| pass |
| # |
| def textwidth(self, text): |
| return len(text) |
| # |
| def lineheight(self): |
| return 1 |
| # |
| def baseline(self): |
| return 0 |
| |
| |
| # Drawing object for writing plain ASCII text to a file |
| class FileWriter: |
| # |
| def __init__(self, fp): |
| self.fp = fp |
| self.lineno, self.colno = 0, 0 |
| # |
| def setfont(self, font): |
| pass |
| # |
| def text(self, (h, v), str): |
| if not str: return |
| if '\n' in str: |
| raise ValueError, 'can\'t write \\n' |
| while self.lineno < v: |
| self.fp.write('\n') |
| self.colno, self.lineno = 0, self.lineno + 1 |
| while self.lineno > v: |
| # XXX This should never happen... |
| self.fp.write('\033[A') # ANSI up arrow |
| self.lineno = self.lineno - 1 |
| if self.colno < h: |
| self.fp.write(' ' * (h - self.colno)) |
| elif self.colno > h: |
| self.fp.write('\b' * (self.colno - h)) |
| self.colno = h |
| self.fp.write(str) |
| self.colno = h + len(str) |
| |
| |
| # Formatting class to do nothing at all with the data |
| class NullFormatter(BaseFormatter): |
| # |
| def __init__(self): |
| d = NullMeasurer() |
| b = NullBackEnd() |
| BaseFormatter.__init__(self, d, b) |
| |
| |
| # Formatting class to write directly to a file |
| class WritingFormatter(BaseFormatter): |
| # |
| def __init__(self, fp, width): |
| dm = NullMeasurer() |
| dw = FileWriter(fp) |
| b = WritingBackEnd(dw, width) |
| BaseFormatter.__init__(self, dm, b) |
| self.blanklines = 1 |
| # |
| # Suppress multiple blank lines |
| def needvspace(self, nlines): |
| BaseFormatter.needvspace(self, min(1, nlines)) |
| |
| |
| # A "FunnyFormatter" writes ASCII text with a twist: *bold words*, |
| # _italic text_ and _underlined words_, and `quoted text'. |
| # It assumes that the fonts are 'r', 'i', 'b', 'u', 'q': (roman, |
| # italic, bold, underline, quote). |
| # Moreover, if the font is in upper case, the text is converted to |
| # UPPER CASE. |
| class FunnyFormatter(WritingFormatter): |
| # |
| def flush(self): |
| if self.para: finalize(self.para) |
| WritingFormatter.flush(self) |
| |
| |
| # Surrounds *bold words* and _italic text_ in a paragraph with |
| # appropriate markers, fixing the size (assuming these characters' |
| # width is 1). |
| openchar = \ |
| {'b':'*', 'i':'_', 'u':'_', 'q':'`', 'B':'*', 'I':'_', 'U':'_', 'Q':'`'} |
| closechar = \ |
| {'b':'*', 'i':'_', 'u':'_', 'q':'\'', 'B':'*', 'I':'_', 'U':'_', 'Q':'\''} |
| def finalize(para): |
| oldfont = curfont = 'r' |
| para.words.append('r', '', 0, 0, 0, 0) # temporary, deleted at end |
| for i in range(len(para.words)): |
| fo, te, wi = para.words[i][:3] |
| if fo <> None: curfont = fo |
| if curfont <> oldfont: |
| if closechar.has_key(oldfont): |
| c = closechar[oldfont] |
| j = i-1 |
| while j > 0 and para.words[j][1] == '': j = j-1 |
| fo1, te1, wi1 = para.words[j][:3] |
| te1 = te1 + c |
| wi1 = wi1 + len(c) |
| para.words[j] = (fo1, te1, wi1) + \ |
| para.words[j][3:] |
| if openchar.has_key(curfont) and te: |
| c = openchar[curfont] |
| te = c + te |
| wi = len(c) + wi |
| para.words[i] = (fo, te, wi) + \ |
| para.words[i][3:] |
| if te: oldfont = curfont |
| else: oldfont = 'r' |
| if curfont in string.uppercase: |
| te = string.upper(te) |
| para.words[i] = (fo, te, wi) + para.words[i][3:] |
| del para.words[-1] |
| |
| |
| # Formatter back-end to draw the text in a window. |
| # This has an option to draw while the paragraphs are being added, |
| # to minimize the delay before the user sees anything. |
| # This manages the entire "document" of the window. |
| class StdwinBackEnd(SavingBackEnd): |
| # |
| def __init__(self, window, drawnow): |
| self.window = window |
| self.drawnow = drawnow |
| self.width = window.getwinsize()[0] |
| self.selection = None |
| self.height = 0 |
| window.setorigin(0, 0) |
| window.setdocsize(0, 0) |
| self.d = window.begindrawing() |
| SavingBackEnd.__init__(self) |
| # |
| def finish(self): |
| self.d.close() |
| self.d = None |
| self.window.setdocsize(0, self.height) |
| # |
| def addpara(self, p): |
| self.paralist.append(p) |
| if self.drawnow: |
| self.height = \ |
| p.render(self.d, 0, self.height, self.width) |
| else: |
| p.layout(self.width) |
| p.left = 0 |
| p.top = self.height |
| p.right = self.width |
| p.bottom = self.height + p.height |
| self.height = p.bottom |
| # |
| def resize(self): |
| self.window.change((0, 0), (self.width, self.height)) |
| self.width = self.window.getwinsize()[0] |
| self.height = 0 |
| for p in self.paralist: |
| p.layout(self.width) |
| p.left = 0 |
| p.top = self.height |
| p.right = self.width |
| p.bottom = self.height + p.height |
| self.height = p.bottom |
| self.window.change((0, 0), (self.width, self.height)) |
| self.window.setdocsize(0, self.height) |
| # |
| def redraw(self, area): |
| d = self.window.begindrawing() |
| (left, top), (right, bottom) = area |
| d.erase(area) |
| d.cliprect(area) |
| for p in self.paralist: |
| if top < p.bottom and p.top < bottom: |
| v = p.render(d, p.left, p.top, p.right) |
| if self.selection: |
| self.invert(d, self.selection) |
| d.close() |
| # |
| def setselection(self, new): |
| if new: |
| long1, long2 = new |
| pos1 = long1[:3] |
| pos2 = long2[:3] |
| new = pos1, pos2 |
| if new <> self.selection: |
| d = self.window.begindrawing() |
| if self.selection: |
| self.invert(d, self.selection) |
| if new: |
| self.invert(d, new) |
| d.close() |
| self.selection = new |
| # |
| def getselection(self): |
| return self.selection |
| # |
| def extractselection(self): |
| if self.selection: |
| a, b = self.selection |
| return self.extractpart(a, b) |
| else: |
| return None |
| # |
| def invert(self, d, region): |
| long1, long2 = region |
| if long1 > long2: long1, long2 = long2, long1 |
| para1, pos1 = long1 |
| para2, pos2 = long2 |
| while para1 < para2: |
| self.paralist[para1].invert(d, pos1, None) |
| pos1 = None |
| para1 = para1 + 1 |
| self.paralist[para2].invert(d, pos1, pos2) |
| # |
| def search(self, prog): |
| import regex, string |
| if type(prog) == type(''): |
| prog = regex.compile(string.lower(prog)) |
| if self.selection: |
| iold = self.selection[0][0] |
| else: |
| iold = -1 |
| hit = None |
| for i in range(len(self.paralist)): |
| if i == iold or i < iold and hit: |
| continue |
| p = self.paralist[i] |
| text = string.lower(p.extract()) |
| if prog.search(text) >= 0: |
| a, b = prog.regs[0] |
| long1 = i, a |
| long2 = i, b |
| hit = long1, long2 |
| if i > iold: |
| break |
| if hit: |
| self.setselection(hit) |
| i = hit[0][0] |
| p = self.paralist[i] |
| self.window.show((p.left, p.top), (p.right, p.bottom)) |
| return 1 |
| else: |
| return 0 |
| # |
| def showanchor(self, id): |
| for i in range(len(self.paralist)): |
| p = self.paralist[i] |
| if p.hasanchor(id): |
| long1 = i, 0 |
| long2 = i, len(p.extract()) |
| hit = long1, long2 |
| self.setselection(hit) |
| self.window.show( \ |
| (p.left, p.top), (p.right, p.bottom)) |
| break |
| |
| |
| # GL extensions |
| |
| class GLFontCache: |
| # |
| def __init__(self): |
| self.reset() |
| self.setfont('') |
| # |
| def reset(self): |
| self.fontkey = None |
| self.fonthandle = None |
| self.fontinfo = None |
| self.fontcache = {} |
| # |
| def close(self): |
| self.reset() |
| # |
| def setfont(self, fontkey): |
| if fontkey == '': |
| fontkey = 'Times-Roman 12' |
| elif ' ' not in fontkey: |
| fontkey = fontkey + ' 12' |
| if fontkey == self.fontkey: |
| return |
| if self.fontcache.has_key(fontkey): |
| handle = self.fontcache[fontkey] |
| else: |
| import string |
| i = string.index(fontkey, ' ') |
| name, sizestr = fontkey[:i], fontkey[i:] |
| size = eval(sizestr) |
| key1 = name + ' 1' |
| key = name + ' ' + `size` |
| # NB key may differ from fontkey! |
| if self.fontcache.has_key(key): |
| handle = self.fontcache[key] |
| else: |
| if self.fontcache.has_key(key1): |
| handle = self.fontcache[key1] |
| else: |
| import fm |
| handle = fm.findfont(name) |
| self.fontcache[key1] = handle |
| handle = handle.scalefont(size) |
| self.fontcache[fontkey] = \ |
| self.fontcache[key] = handle |
| self.fontkey = fontkey |
| if self.fonthandle <> handle: |
| self.fonthandle = handle |
| self.fontinfo = handle.getfontinfo() |
| handle.setfont() |
| |
| |
| class GLMeasurer(GLFontCache): |
| # |
| def textwidth(self, text): |
| return self.fonthandle.getstrwidth(text) |
| # |
| def baseline(self): |
| return self.fontinfo[6] - self.fontinfo[3] |
| # |
| def lineheight(self): |
| return self.fontinfo[6] |
| |
| |
| class GLWriter(GLFontCache): |
| # |
| # NOTES: |
| # (1) Use gl.ortho2 to use X pixel coordinates! |
| # |
| def text(self, (h, v), text): |
| import gl, fm |
| gl.cmov2i(h, v + self.fontinfo[6] - self.fontinfo[3]) |
| fm.prstr(text) |
| # |
| def setfont(self, fontkey): |
| oldhandle = self.fonthandle |
| GLFontCache.setfont(fontkey) |
| if self.fonthandle <> oldhandle: |
| handle.setfont() |
| |
| |
| class GLMeasurerWriter(GLMeasurer, GLWriter): |
| pass |
| |
| |
| class GLBackEnd(SavingBackEnd): |
| # |
| def __init__(self, wid): |
| import gl |
| gl.winset(wid) |
| self.wid = wid |
| self.width = gl.getsize()[1] |
| self.height = 0 |
| self.d = GLMeasurerWriter() |
| SavingBackEnd.__init__(self) |
| # |
| def finish(self): |
| pass |
| # |
| def addpara(self, p): |
| self.paralist.append(p) |
| self.height = p.render(self.d, 0, self.height, self.width) |
| # |
| def redraw(self): |
| import gl |
| gl.winset(self.wid) |
| width = gl.getsize()[1] |
| if width <> self.width: |
| setdocsize = 1 |
| self.width = width |
| for p in self.paralist: |
| p.top = p.bottom = None |
| d = self.d |
| v = 0 |
| for p in self.paralist: |
| v = p.render(d, 0, v, width) |