mirror of
				https://github.com/python/cpython.git
				synced 2025-10-30 21:21:22 +00:00 
			
		
		
		
	Initial revision
This commit is contained in:
		
							parent
							
								
									5113569151
								
							
						
					
					
						commit
						f09b770f68
					
				
					 2 changed files with 332 additions and 0 deletions
				
			
		
							
								
								
									
										149
									
								
								Demo/tkinter/guido/ManPage.py
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										149
									
								
								Demo/tkinter/guido/ManPage.py
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,149 @@ | ||||||
|  | # Widget to display a man page | ||||||
|  | 
 | ||||||
|  | import regex | ||||||
|  | from Tkinter import * | ||||||
|  | from ScrolledText import ScrolledText | ||||||
|  | 
 | ||||||
|  | # XXX These fonts may have to be changed to match your system | ||||||
|  | BOLDFONT = '*-Courier-Bold-R-Normal-*-120-*' | ||||||
|  | ITALICFONT = '*-Courier-Medium-O-Normal-*-120-*' | ||||||
|  | 
 | ||||||
|  | # XXX Recognizing footers is system dependent | ||||||
|  | # (This one works for IRIX 5.2 and Solaris 2.2) | ||||||
|  | footerprog = regex.compile( | ||||||
|  | 	'^     Page [1-9][0-9]*[ \t]+\|^.*Last change:.*[1-9][0-9]*\n') | ||||||
|  | emptyprog = regex.compile('^[ \t]*\n') | ||||||
|  | ulprog = regex.compile('^[ \t]*[Xv!_][Xv!_ \t]*\n') | ||||||
|  | 
 | ||||||
|  | # Basic Man Page class -- does not disable editing | ||||||
|  | class EditableManPage(ScrolledText): | ||||||
|  | 
 | ||||||
|  | 	def __init__(self, master=None, cnf={}): | ||||||
|  | 		# Initialize base class | ||||||
|  | 		ScrolledText.__init__(self, master, cnf) | ||||||
|  | 
 | ||||||
|  | 		# Define tags for formatting styles | ||||||
|  | 		self.text.tag_config('bold', {'font': BOLDFONT}) | ||||||
|  | 		self.text.tag_config('italic', {'font': ITALICFONT}) | ||||||
|  | 		self.text.tag_config('underline', {'underline': 1}) | ||||||
|  | 
 | ||||||
|  | 		# Create mapping from characters to tags | ||||||
|  | 		self.tagmap = { | ||||||
|  | 			'X': 'bold', | ||||||
|  | 			'_': 'underline', | ||||||
|  | 			'!': 'italic', | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 	# Parse nroff output piped through ul -i and append it to the | ||||||
|  | 	# text widget | ||||||
|  | 	def parsefile(self, fp): | ||||||
|  | 		save_cursor = self.text['cursor'] | ||||||
|  | 		self.text['cursor'] = 'watch' | ||||||
|  | 		self.text.update() | ||||||
|  | 		ok = 0 | ||||||
|  | 		empty = 0 | ||||||
|  | 		nextline = None | ||||||
|  | 		while 1: | ||||||
|  | 			if nextline: | ||||||
|  | 				line = nextline | ||||||
|  | 				nextline = None | ||||||
|  | 			else: | ||||||
|  | 				line = fp.readline() | ||||||
|  | 				if not line: | ||||||
|  | 					break | ||||||
|  | 			if emptyprog.match(line) >= 0: | ||||||
|  | 				empty = 1 | ||||||
|  | 				continue | ||||||
|  | 			nextline = fp.readline() | ||||||
|  | 			if nextline and ulprog.match(nextline) >= 0: | ||||||
|  | 				propline = nextline | ||||||
|  | 				nextline = None | ||||||
|  | 			else: | ||||||
|  | 				propline = '' | ||||||
|  | 			if not ok: | ||||||
|  | 				ok = 1 | ||||||
|  | 				empty = 0 | ||||||
|  | 				continue | ||||||
|  | 			if footerprog.match(line) >= 0: | ||||||
|  | 				ok = 0 | ||||||
|  | 				empty = 0 | ||||||
|  | 				continue | ||||||
|  | 			if empty: | ||||||
|  | 				self.insert_prop('\n') | ||||||
|  | 				empty = 0 | ||||||
|  | 			p = '' | ||||||
|  | 			j = 0 | ||||||
|  | 			for i in range(min(len(propline), len(line))): | ||||||
|  | 				if propline[i] != p: | ||||||
|  | 					if j < i: | ||||||
|  | 						self.insert_prop(line[j:i], p) | ||||||
|  | 						j = i | ||||||
|  | 					p = propline[i] | ||||||
|  | 			self.insert_prop(line[j:]) | ||||||
|  | 		self.text['cursor'] = save_cursor | ||||||
|  | 
 | ||||||
|  | 	def insert_prop(self, str, prop = ' '): | ||||||
|  | 		here = self.text.index(AtInsert()) | ||||||
|  | 		self.text.insert(AtInsert(), str) | ||||||
|  | 		for tag in self.tagmap.values(): | ||||||
|  | 			self.text.tag_remove(tag, here, AtInsert()) | ||||||
|  | 		if self.tagmap.has_key(prop): | ||||||
|  | 			self.text.tag_add(self.tagmap[prop], here, AtInsert()) | ||||||
|  | 
 | ||||||
|  | # Readonly Man Page class -- disables editing, otherwise the same | ||||||
|  | class ReadonlyManPage(EditableManPage): | ||||||
|  | 
 | ||||||
|  | 	def __init__(self, master=None, cnf={}): | ||||||
|  | 		# Initialize base class | ||||||
|  | 		EditableManPage.__init__(self, master, cnf) | ||||||
|  | 
 | ||||||
|  | 		# Make the text readonly | ||||||
|  | 		self.text.bind('<Any-KeyPress>', self.modify_cb) | ||||||
|  | 		self.text.bind('<Return>', self.modify_cb) | ||||||
|  | 		self.text.bind('<BackSpace>', self.modify_cb) | ||||||
|  | 		self.text.bind('<Delete>', self.modify_cb) | ||||||
|  | 		self.text.bind('<Control-h>', self.modify_cb) | ||||||
|  | 		self.text.bind('<Control-d>', self.modify_cb) | ||||||
|  | 		self.text.bind('<Control-v>', self.modify_cb) | ||||||
|  | 
 | ||||||
|  | 	def modify_cb(self, e): | ||||||
|  | 		pass | ||||||
|  | 
 | ||||||
|  | # Alias | ||||||
|  | ManPage = ReadonlyManPage | ||||||
|  | 
 | ||||||
|  | # Test program. | ||||||
|  | # usage: ManPage [manpage]; or ManPage [-f] file | ||||||
|  | # -f means that the file is nroff -man output run through ul -i | ||||||
|  | def test(): | ||||||
|  | 	import os | ||||||
|  | 	import sys | ||||||
|  | 	# XXX This directory may be different on your system | ||||||
|  | 	MANDIR = '/usr/local/man/mann' | ||||||
|  | 	DEFAULTPAGE = 'Tcl' | ||||||
|  | 	formatted = 0 | ||||||
|  | 	if sys.argv[1:] and sys.argv[1] == '-f': | ||||||
|  | 		formatted = 1 | ||||||
|  | 		del sys.argv[1] | ||||||
|  | 	if sys.argv[1:]: | ||||||
|  | 		name = sys.argv[1] | ||||||
|  | 	else: | ||||||
|  | 		name = DEFAULTPAGE | ||||||
|  | 	if not formatted: | ||||||
|  | 		if name[-2:-1] != '.': | ||||||
|  | 			name = name + '.n' | ||||||
|  | 		name = os.path.join(MANDIR, name) | ||||||
|  | 	root = Tk() | ||||||
|  | 	root.minsize(1, 1) | ||||||
|  | 	manpage = ManPage(root, {'relief': 'sunken', 'bd': 2, | ||||||
|  | 				 Pack: {'expand': 1, 'fill': 'both'}}) | ||||||
|  | 	if formatted: | ||||||
|  | 		fp = open(name, 'r') | ||||||
|  | 	else: | ||||||
|  | 		fp = os.popen('nroff -man %s | ul -i' % name, 'r') | ||||||
|  | 	manpage.parsefile(fp) | ||||||
|  | 	root.mainloop() | ||||||
|  | 
 | ||||||
|  | # Run the test program when called as a script | ||||||
|  | if __name__ == '__main__': | ||||||
|  | 	test() | ||||||
							
								
								
									
										183
									
								
								Demo/tkinter/guido/tkman.py
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										183
									
								
								Demo/tkinter/guido/tkman.py
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,183 @@ | ||||||
|  | #! /ufs/guido/bin/sgi/tkpython | ||||||
|  | 
 | ||||||
|  | # Tk man page browser -- currently only shows the Tcl/Tk man pages | ||||||
|  | 
 | ||||||
|  | import sys | ||||||
|  | import os | ||||||
|  | import string | ||||||
|  | import regex | ||||||
|  | from Tkinter import * | ||||||
|  | from ManPage import ManPage | ||||||
|  | 
 | ||||||
|  | MANDIR = '/usr/local/man/mann' | ||||||
|  | 
 | ||||||
|  | def listmanpages(mandir = MANDIR): | ||||||
|  | 	files = os.listdir(mandir) | ||||||
|  | 	names = [] | ||||||
|  | 	for file in files: | ||||||
|  | 		if file[-2:] == '.n': | ||||||
|  | 			names.append(file[:-2]) | ||||||
|  | 	names.sort() | ||||||
|  | 	return names | ||||||
|  | 
 | ||||||
|  | class SelectionBox: | ||||||
|  | 
 | ||||||
|  | 	def __init__(self, master=None): | ||||||
|  | 		self.choices = [] | ||||||
|  | 
 | ||||||
|  | 		self.frame = Frame(master, { | ||||||
|  | 			Pack: {'expand': 1, 'fill': 'both'}}) | ||||||
|  | 		self.master = self.frame.master | ||||||
|  | 		self.subframe = Frame(self.frame, { | ||||||
|  | 			Pack: {'expand': 0, 'fill': 'both'}}) | ||||||
|  | 		self.listbox = Listbox(self.subframe, | ||||||
|  | 				       {'relief': 'sunken', 'bd': 2, | ||||||
|  | 					'geometry': '20x6', | ||||||
|  | 					Pack: {'side': 'right', | ||||||
|  | 					       'expand': 1, 'fill': 'both'}}) | ||||||
|  | 		self.subsubframe = Frame(self.subframe, { | ||||||
|  | 			Pack: {'side': 'left', 'expand': 1, 'fill': 'both'}}) | ||||||
|  | 		self.l1 = Label(self.subsubframe, | ||||||
|  | 				{'text': 'Display manual page named:', | ||||||
|  | 				 Pack: {'side': 'top'}}) | ||||||
|  | 		self.entry = Entry(self.subsubframe, | ||||||
|  | 				   {'relief': 'sunken', 'bd': 2, | ||||||
|  | 				    'width': 20, | ||||||
|  | 				    Pack: {'side': 'top', | ||||||
|  | 					   'expand': 0, 'fill': 'x'}}) | ||||||
|  | 		self.l2 = Label(self.subsubframe, | ||||||
|  | 				{'text': 'Search (regexp, case insensitive):', | ||||||
|  | 				 Pack: {'side': 'top'}}) | ||||||
|  | 		self.search = Entry(self.subsubframe, | ||||||
|  | 				   {'relief': 'sunken', 'bd': 2, | ||||||
|  | 				    'width': 20, | ||||||
|  | 				    Pack: {'side': 'top', | ||||||
|  | 					   'expand': 0, 'fill': 'x'}}) | ||||||
|  | 		self.title = Label(self.subsubframe, | ||||||
|  | 				   {'text': '(none)', | ||||||
|  | 				    Pack: {'side': 'bottom'}}) | ||||||
|  | 		self.text = ManPage(self.frame, | ||||||
|  | 					 {'relief': 'sunken', 'bd': 2, | ||||||
|  | 					  'wrap': 'none', 'width': 72, | ||||||
|  | 					  Pack: {'expand': 1, 'fill': 'both'}}) | ||||||
|  | 
 | ||||||
|  | 		self.entry.bind('<Return>', self.entry_cb) | ||||||
|  | 		self.search.bind('<Return>', self.search_cb) | ||||||
|  | 		self.listbox.bind('<Double-1>', self.listbox_cb) | ||||||
|  | 
 | ||||||
|  | 		self.entry.focus_set() | ||||||
|  | 
 | ||||||
|  | 		self.showing = None | ||||||
|  | 
 | ||||||
|  | 	def addchoice(self, choice): | ||||||
|  | 		if choice not in self.choices: | ||||||
|  | 			self.choices.append(choice) | ||||||
|  | 			self.choices.sort() | ||||||
|  | 		self.update() | ||||||
|  | 
 | ||||||
|  | 	def addlist(self, list): | ||||||
|  | 		self.choices[len(self.choices):] = list | ||||||
|  | 		self.choices.sort() | ||||||
|  | 		self.update() | ||||||
|  | 
 | ||||||
|  | 	def updatelist(self): | ||||||
|  | 		key = self.entry.get() | ||||||
|  | 		ok = filter(lambda name, key=key, n=len(key): name[:n]==key, | ||||||
|  | 			 self.choices) | ||||||
|  | 		self.listbox.delete(0, AtEnd()) | ||||||
|  | 		exactmatch = 0 | ||||||
|  | 		for item in ok: | ||||||
|  | 			if item == key: exactmatch = 1 | ||||||
|  | 			self.listbox.insert(AtEnd(), item) | ||||||
|  | 		if exactmatch: | ||||||
|  | 			return key | ||||||
|  | 		elif self.listbox.size() == 1: | ||||||
|  | 			return self.listbox.get(0) | ||||||
|  | 
 | ||||||
|  | 	def entry_cb(self, e): | ||||||
|  | 		self.update() | ||||||
|  | 
 | ||||||
|  | 	def update(self): | ||||||
|  | 		self.show_page(self.updatelist()) | ||||||
|  | 
 | ||||||
|  | 	def show_page(self, name): | ||||||
|  | 		if not name: | ||||||
|  | 			return | ||||||
|  | 		if name == self.showing: | ||||||
|  | 			print 'show_page: already showing' | ||||||
|  | 			return | ||||||
|  | 		name = '%s/%s.n' % (MANDIR, name) | ||||||
|  | 		fp = os.popen('nroff -man %s | ul -i' % name, 'r') | ||||||
|  | 		self.text.delete('1.0', AtEnd()) | ||||||
|  | 		frame_cursor = self.frame['cursor'] | ||||||
|  | 		entry_cursor = self.entry['cursor'] | ||||||
|  | 		self.entry['cursor'] = 'watch' | ||||||
|  | 		self.search['cursor'] = 'watch' | ||||||
|  | 		self.frame['cursor'] = 'watch' | ||||||
|  | 		self.text.parsefile(fp) | ||||||
|  | 		self.search['cursor'] = entry_cursor | ||||||
|  | 		self.entry['cursor'] = entry_cursor | ||||||
|  | 		self.frame['cursor'] = frame_cursor | ||||||
|  | 		self.entry.delete(0, AtEnd()) | ||||||
|  | 		self.updatelist() | ||||||
|  | 
 | ||||||
|  | 	def listbox_cb(self, e): | ||||||
|  | 		selection = self.listbox.curselection() | ||||||
|  | 		if selection and len(selection) == 1: | ||||||
|  | 			which = self.listbox.get(selection[0]) | ||||||
|  | 			self.show_page(which) | ||||||
|  | 
 | ||||||
|  | 	def search_cb(self, e): | ||||||
|  | 		self.search_string(self.search.get()) | ||||||
|  | 
 | ||||||
|  | 	def search_string(self, search): | ||||||
|  | 		if not search: | ||||||
|  | 			print 'Empty search string' | ||||||
|  | 			return | ||||||
|  | 		try: | ||||||
|  | 			prog = regex.compile(search, regex.casefold) | ||||||
|  | 		except regex.error, msg: | ||||||
|  | 			print 'Regex error:', msg | ||||||
|  | 			return | ||||||
|  | 		here = self.text.index(AtInsert()) | ||||||
|  | 		lineno = string.atoi(here[:string.find(here, '.')]) | ||||||
|  | 		end = self.text.index(AtEnd()) | ||||||
|  | 		endlineno = string.atoi(end[:string.find(end, '.')]) | ||||||
|  | 		wraplineno = lineno | ||||||
|  | 		while 1: | ||||||
|  | 			lineno = lineno + 1 | ||||||
|  | 			if lineno > endlineno: | ||||||
|  | 				if wraplineno <= 0: | ||||||
|  | 					break | ||||||
|  | 				endlineno = wraplineno | ||||||
|  | 				lineno = 0 | ||||||
|  | 				wraplineno = 0 | ||||||
|  | 			line = self.text.get('%d.0 linestart' % lineno, | ||||||
|  | 					     '%d.0 lineend' % lineno) | ||||||
|  | 			i = prog.search(line) | ||||||
|  | 			if i >= 0: | ||||||
|  | 				n = max(1, len(prog.group(0))) | ||||||
|  | 				try: | ||||||
|  | 					self.text.tag_remove('sel', | ||||||
|  | 							     AtSelFirst(), | ||||||
|  | 							     AtSelLast()) | ||||||
|  | 				except TclError: | ||||||
|  | 					pass | ||||||
|  | 				self.text.tag_add('sel', | ||||||
|  | 						  '%d.%d' % (lineno, i), | ||||||
|  | 						  '%d.%d' % (lineno, i+n)) | ||||||
|  | 				self.text.mark_set(AtInsert(), | ||||||
|  | 						   '%d.%d' % (lineno, i)) | ||||||
|  | 				self.text.yview_pickplace(AtInsert()) | ||||||
|  | 				break | ||||||
|  | 
 | ||||||
|  | def main(): | ||||||
|  | 	root = Tk() | ||||||
|  | 	sb = SelectionBox(root) | ||||||
|  | 	sb.addlist(listmanpages()) | ||||||
|  | 	if sys.argv[1:]: | ||||||
|  | 		sb.show_page(sys.argv[1]) | ||||||
|  | 	root.minsize(1, 1) | ||||||
|  | 	root.mainloop() | ||||||
|  | 
 | ||||||
|  | main() | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Guido van Rossum
						Guido van Rossum