mirror of
https://github.com/python/cpython.git
synced 2025-11-03 03:22:27 +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