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