mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 11:14:33 +00:00 
			
		
		
		
	added html parser and supporting cast
This commit is contained in:
		
							parent
							
								
									eb9e9d2b2a
								
							
						
					
					
						commit
						7c750e1e09
					
				
					 6 changed files with 3014 additions and 0 deletions
				
			
		
							
								
								
									
										621
									
								
								Lib/lib-old/fmt.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										621
									
								
								Lib/lib-old/fmt.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,621 @@ | |||
| # 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) | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Guido van Rossum
						Guido van Rossum