Sophie

Sophie

distrib > Mandriva > 2010.2 > i586 > media > contrib-backports > by-pkgid > c93db43b3e0e63e53aa645da566a833a > files > 76

python-urwid-1.0.1-1mdv2010.2.i586.rpm

#!/usr/bin/python
#
# Urwid tutorial documentation generation program
#    Copyright (C) 2004-2007  Ian Ward
#
#    This library is free software; you can redistribute it and/or
#    modify it under the terms of the GNU Lesser General Public
#    License as published by the Free Software Foundation; either
#    version 2.1 of the License, or (at your option) any later version.
#
#    This library is distributed in the hope that it will be useful,
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
#    Lesser General Public License for more details.
#
#    You should have received a copy of the GNU Lesser General Public
#    License along with this library; if not, write to the Free Software
#    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
#
# Urwid web site: http://excess.org/urwid/

import sys
import re

import urwid.html_fragment
import urwid
try:
    import templayer
except ImportError:
    templayer = None

examples = {}

interp_line = "#!/usr/bin/python\n\n"
cut_comment = "# CUT HERE"

examples["min"] = ["example_min"]
def example_min():
    import urwid

    txt = urwid.Text(u"Hello World")
    fill = urwid.Filler(txt, 'top')
    loop = urwid.MainLoop(fill)
    loop.run()


examples["input"] = ["example_input"]
def example_input():
    import urwid

    txt = urwid.Text(u"Hello World")
    fill = urwid.Filler(txt, 'top')

    def show_or_exit(input):
        if input in ('q', 'Q'):
            raise urwid.ExitMainLoop()
        txt.set_text(repr(input))

    loop = urwid.MainLoop(fill, unhandled_input=show_or_exit)
    loop.run()


examples["attr"] = ["example_attr"]
def example_attr():
    import urwid

    palette = [
        ('banner', 'black', 'light gray', 'standout,underline'),
        ('streak', 'black', 'dark red', 'standout'),
        ('bg', 'black', 'dark blue'),]

    txt = urwid.Text(('banner', u" Hello World "), align='center')
    map1 = urwid.AttrMap(txt, 'streak')
    fill = urwid.Filler(map1)
    map2 = urwid.AttrMap(fill, 'bg')

    def exit_on_q(input):
        if input in ('q', 'Q'):
            raise urwid.ExitMainLoop()

    loop = urwid.MainLoop(map2, palette, unhandled_input=exit_on_q)
    loop.run()


examples["highcolors"] = ["example_highcolors"]
def example_highcolors():
    import urwid

    palette = [
        ('banner', '', '', '', '#ffa', '#60d'),
        ('streak', '', '', '', 'g50', '#60a'),
        ('inside', '', '', '', 'g38', '#808'),
        ('outside', '', '', '', 'g27', '#a06'),
        ('bg', '', '', '', 'g7', '#d06'),]

    txt = urwid.Text(('banner', u" Hello World "), align='center')
    map1 = urwid.AttrMap(txt, 'streak')
    pile = urwid.Pile([
        urwid.AttrMap(urwid.Divider(), 'outside'),
        urwid.AttrMap(urwid.Divider(), 'inside'),
        map1, 
        urwid.AttrMap(urwid.Divider(), 'inside'),
        urwid.AttrMap(urwid.Divider(), 'outside')])
    fill = urwid.Filler(pile)
    map2 = urwid.AttrMap(fill, 'bg')

    def exit_on_q(input):
        if input in ('q', 'Q'):
            raise urwid.ExitMainLoop()

    loop = urwid.MainLoop(map2, palette, unhandled_input=exit_on_q)
    loop.screen.set_terminal_properties(colors=256)
    loop.run()


examples["edit"] = ["example_edit"]
def example_edit():
    import urwid

    ask = urwid.Edit(u"What is your name?\n")
    fill = urwid.Filler( ask )

    def do_reply(input):
        if input != 'enter':
            return
        if fill.body == ask:
            fill.body = urwid.Text(u"Nice to meet you,\n"+
                ask.edit_text+".")
            return True
        else:
            raise urwid.ExitMainLoop()

    loop = urwid.MainLoop(fill, unhandled_input=do_reply)
    loop.run()


examples["frlb"] = ["example_frlb"]
def example_frlb():
    import urwid

    palette = [('I say', 'default,bold', 'default', 'bold'),]

    ask = urwid.Edit(('I say', u"What is your name?\n"))
    reply = urwid.Text(u"")
    content = urwid.SimpleListWalker([ask, reply])
    listbox = urwid.ListBox(content)

    def on_ask_change(edit, new_edit_text):
        assert edit is ask # we are passed our edit widget
        reply.set_text(('I say',
            u"Nice to meet you, " + new_edit_text))

    urwid.connect_signal(ask, 'change', on_ask_change)

    def exit_on_cr(input):
        if input == 'enter':
            raise urwid.ExitMainLoop()

    loop = urwid.MainLoop(listbox, palette, unhandled_input=exit_on_cr)
    loop.run()


examples["lbcont"]=["example_lbcont"]
def example_lbcont():
    import urwid

    palette = [('I say', 'default,bold', 'default', 'bold'),]

    def new_question():
        return urwid.Edit(('I say', u"What is your name?\n"))

    def new_answer(name):
        return urwid.Text(('I say', u"Nice to meet you, "+name+"\n"))

    content = urwid.SimpleListWalker([new_question()])
    listbox = urwid.ListBox(content)

    def update_on_cr(input):
        if input != 'enter':
            return
        focus_widget, position = listbox.get_focus()
        if not hasattr(focus_widget, 'edit_text'):
            return
        if not focus_widget.edit_text:
            raise urwid.ExitMainLoop()
        content[position+1:position+2] = [
            new_answer(focus_widget.edit_text)]
        if not content[position+2:position + 3]:
            content.append(new_question())
        listbox.set_focus(position + 2)
        return True

    loop = urwid.MainLoop(listbox, palette, unhandled_input=update_on_cr)
    loop.run()


examples["lbscr"] = ["example_lbscr"]
def example_lbscr():
    import urwid

    palette = [('header', 'white', 'black'),
        ('reveal focus', 'black', 'dark cyan', 'standout'),]
    content = urwid.SimpleListWalker([
        urwid.AttrMap(w, None, 'reveal focus') for w in [
        urwid.Text(u"This is a text string that is fairly long"),
        urwid.Divider(u"-"),
        urwid.Text(u"Short one"),
        urwid.Text(u"Another"),
        urwid.Divider(u"-"),
        urwid.Text(u"What could be after this?"),
        urwid.Text(u"The end."),]])
    listbox = urwid.ListBox(content)
    show_key = urwid.Text(u"", wrap='clip')
    head = urwid.AttrMap(show_key, 'header')
    top = urwid.Frame(listbox, head)

    def show_all_input(input, raw):
        show_key.set_text(u"Pressed: " + u" ".join([
            unicode(i) for i in input]))
        return input

    def exit_on_cr(input):
        if input == 'enter':
            raise urwid.ExitMainLoop()
    
    loop = urwid.MainLoop(top, palette, 
        input_filter=show_all_input, unhandled_input=exit_on_cr)
    loop.run()


examples["wmod"] = ["example_wmod"]
def example_wmod():
    class QuestionnaireItem(urwid.WidgetWrap):
        def __init__(self):
            self.options = []
            unsure = urwid.RadioButton(self.options, u"Unsure")
            yes = urwid.RadioButton(self.options, u"Yes")
            no = urwid.RadioButton(self.options, u"No")
            display_widget = urwid.GridFlow([unsure, yes, no],
                15, 3, 1, 'left')
            urwid.WidgetWrap.__init__(self, display_widget)
        
        def get_state(self):
            for o in self.options:
                if o.get_state() is True:
                    return o.get_label()

        
examples["wanat"] = ["example_wanat","example_wanat_new","example_wanat_multi"]
def example_wanat():
    class Pudding(urwid.FlowWidget):
        def selectable(self):
            return False
        def rows(self, size, focus=False):
            return 1
        def render(self, size, focus=False):
            (maxcol,) = size
            num_pudding = maxcol / len("Pudding")
            return urwid.TextCanvas(["Pudding"*num_pudding],
                maxcol=maxcol)

    class BoxPudding(urwid.BoxWidget):
        def selectable(self):
            return False
        def render(self, size, focus=False):
            (maxcol, maxrow) = size
            num_pudding = maxcol / len("Pudding")
            return urwid.TextCanvas(
                ["Pudding"*num_pudding] * maxrow,
                maxcol=maxcol)

def example_wanat_new():
    class NewPudding(urwid.FlowWidget):
        def selectable(self):
            return False
        def rows(self, size, focus=False):
            w = self.display_widget(size, focus)
            return w.rows(size, focus)
        def render(self, size, focus=False):
            w = self.display_widget(size, focus)
            return w.render(size, focus)
        def display_widget(self, size, focus):
            (maxcol,) = size
            num_pudding = maxcol / len("Pudding")
            return urwid.Text("Pudding"*num_pudding)

def example_wanat_multi():
    class MultiPudding(urwid.Widget):
        def selectable(self):
            return False
        def rows(self, size, focus=False):
            return 1
        def render(self, size, focus=False):
            if len(size) == 1:
                (maxcol,) = size
                maxrow = 1
            else:
                (maxcol, maxrow) = size
            num_pudding = maxcol / len("Pudding")
            return urwid.TextCanvas(
                ["Pudding"*num_pudding] * maxrow,
                maxcol=maxcol)

examples["wsel"] = ["example_wsel"]
def example_wsel():
    class SelectablePudding(urwid.FlowWidget):
        def __init__(self):
            self.pudding = "pudding"
        def selectable(self):
            return True
        def rows(self, size, focus=False):
            return 1
        def render(self, size, focus=False):
            (maxcol,) = size
            num_pudding = maxcol / len(self.pudding)
            pudding = self.pudding
            if focus: 
                pudding = pudding.upper()
            return urwid.TextCanvas([pudding*num_pudding],
                maxcol=maxcol)
        def keypress(self, size, key):
            (maxcol,) = size
            if len(key)>1:
                return key
            if key.lower() in self.pudding:
                # remove letter from pudding
                n = self.pudding.index(key.lower())
                self.pudding = self.pudding[:n]+self.pudding[n+1:]
                if not self.pudding:
                    self.pudding = "pudding"
            else:
                return key

examples["wcur"] = ["example_wcur"]
def example_wcur():
    class CursorPudding(urwid.FlowWidget):
        def __init__(self):
            self.cursor_col = 0
        def selectable(self):
            return True
        def rows(self, size, focus=False):
            return 1
        def render(self, size, focus=False):
            (maxcol,) = size
            num_pudding = maxcol / len("Pudding")
            cursor = None
            if focus:
                cursor = self.get_cursor_coords(size)
            return urwid.TextCanvas(
                ["Pudding"*num_pudding], [], cursor, maxcol)
        def get_cursor_coords(self, size):
            (maxcol,) = size
            col = min(self.cursor_col, maxcol-1)
            return col, 0
        def keypress(self, size, key):
            (maxcol,) = size
            if key == 'left':
                col = self.cursor_col -1
            elif key == 'right':
                col = self.cursor_col +1
            else:
                return key
            self.cursor_x = max(0, min(maxcol-1, col))
        # CUT HERE
        def get_pref_col(self, (maxcol,)):
            return self.cursor_x
        def move_cursor_to_coords(self, (maxcol,), col, row):
            assert row == 0
            self.cursor_x = col
            return True

def read_sections(tmpl):
    """Read section tags, section descriptions, and column breaks from
    the Templayer template argument.  Convert the section data into a
    Python data structure called sections.  Each sublist of sections
    contains one column.  Each column contains a list of (tag, desc.)
    pairs.  Return sections."""

    sd = tmpl.layer("section_data")
    col_break = "---"
    sections = [[]]
    for ln in sd.split("\n"):
        if not ln: continue
        if ln == col_break:
            sections.append([])
            continue
        tag, desc = ln.split(",",1)
        sections[-1].append( (tag.rstrip(), desc.lstrip()) )
    return sections

def read_example_code():
    """By the time this function runs, the examples dictionary contains
    a list of function names, all starting with "example_".  Create a
    second dictionary called code_blocks.  Open the file containing this
    function.  For each function name in examples, read the text of that
    function into an entry in the code_blocks dictionary.  Return the
    code_blocks dictionary."""

    # invert the "examples" dictionary
    example_fns = {}
    for tag, l in examples.items():
        for i, fn in zip(range(len(l)), l):
            example_fns[fn] = tag, i
    
    # read our own source code
    # strip trailing spaces and tabs from each line
    code_blocks = {}
    current_block = None
    for ln in open( sys.argv[0], 'r').readlines():
        ln = ln.rstrip()
        if ( ln[:4] == "def " and ln[-3:] == "():" and
            example_fns.has_key( ln[4:-3] ) ):
            current_block = ln[4:-3]
            code_blocks[current_block] = []
            continue
        if ln and not ln.startswith("    "):
            current_block = None
            continue
        if current_block is None:
            continue
        if ln.startswith("    "):
            ln = ln[4:]
        code_blocks[current_block].append(ln+"\n")
            
    # recombine code lines into a single string each
    for name, block in code_blocks.items():
        code_blocks[name] = "".join( block )
    
    return code_blocks

def filename(snum, inum, enum):
    """Generate a filename to write an example to.  Take the section
    number from the table of contents, item number (subsection number),
    and example number.  Numbers start at 1.  Return the filename."""

    assert snum > 0, \
     '%d.%d #%d: Section number should be greater than 0' % \
     (snum, inum, enum)
    assert inum > 0, \
     '%d.%d #%d: Item number should be greater than 0' % \
     (snum, inum, enum)
    assert enum > 0, \
     '%d.%d #%d: Example number should be greater than 0' % \
     (snum, inum, enum)
    assert enum < 28, \
     '%d.%d #%d: Example number should be less than 28' % \
     (snum, inum, enum)

    if enum == 1:    estr = ''
    else:        estr = chr(ord('a') - 2 + enum)

    return "example" + str(snum) + "." + str(inum) + estr + ".py"

def write_example_files(sections, blocks):
    """The sections dictionary gives section tags in the order used by
    the HTML output, and going through the tags in order allows us to
    generate the same section numbers that the HTML uses.  The global
    examples dictionary maps each section tag to a list of examples used
    in that section.  And the blocks dictionary maps each example tag
    to the text of that example.
    
    Generate section numbers and find the examples used in each section.
    Assume each example is ready to run -- replace the "cut here" comments
    with blank lines, but don't remove any code.  Create a file named
    according to the section number (and an optional letter to allow for
    multiple examples per section).  Write the cleaned-up example to the
    file."""

    cut_re = '^\s*' + re.escape(cut_comment) + '\s*$'
    cut_re_comp = re.compile(cut_re, re.MULTILINE)
    valid_re = 'import urwid' # some examples are not runnable
    valid_re_comp = re.compile(valid_re)
    snum = inum = enum = 0

    for col in sections:
        for tag, ignore in col:
            if not tag:
                # new section -- do its first item next time
                snum += 1
                inum = 0
                continue

            # new item inside a section
            inum += 1
            enum = 0
            for ename in examples.get(tag, []):
                etext = blocks[ename]
                if not valid_re_comp.search(etext):
                    continue
                enum += 1
                fname = filename(snum, inum, enum)
                f = open(fname, 'w')
                etext = cut_re_comp.sub("", etext)
                etext = interp_line + etext.rstrip() + "\n"
                f.write(etext)
                f.close()

def cut_example_code(blocks):
    """For brevity, the HTML gives excerpts from some of the examples.
    Convert the examples from the full forms stored in this file to the
    excerpted forms.  Use "cut here" comments in the examples, and the
    rules below, to do the conversion.  Also check that the full forms
    of certain examples contain identical code.  Also strip trailing
    spaces and tabs from the code.  Return a dictionary with the
    converted examples."""

    ## do the conversions and the checks
    blocks["example_wcur"], blocks["example_wcur2"] = (
        blocks["example_wcur"].split(cut_comment+"\n") )
    examples["wcur"].append("example_wcur2")
    
    # strip trailing spaces, tabs, blank lines from each block
    # removes spaces/tabs left by splitting at cut_comment
    for name, block in blocks.items():
        blocks[name] = block.rstrip()
    
    return blocks

def generate_example_results():
    """Create HTML "screen shots" from the example programs defined in
    this file.  Store the results in a dictionary (mapping example tag
    to results) and return the dictionary."""

    results = {}
    
    init = urwid.html_fragment.screenshot_init
    collect = urwid.html_fragment.screenshot_collect

    init([(21,7)],[[" "]])
    example_min()
    results["min"] = collect()[:1]

    init([(12,3)],[["enter"], ["N"], ["O"], ["P"], ["Q"]])
    example_input()
    results["input"] = collect()[:5]

    init([(21,7),(10,9),(30,3),(15,2)],[["window resize"]]*3+[["q"]])
    example_attr()
    results["attr"] = collect()[:4]

    init([(26,9)],[["q"]])
    example_highcolors()
    results["highcolors"] = collect()[:1]

    init([(21,7)],[list("Arthur, King of the Britons"),["enter"],[" "]])
    example_edit()
    results["edit"] = collect()[:3]

    init([(21,7)],[list("Tim t"),list("he Ench"),list("anter"),["enter"]])
    example_frlb()
    results["frlb"] = collect()[:4]

    init([(23,13)],[list("Abe")+["enter"]+list("Bob"),["enter"]+
        list("Carl")+["enter"], list("Dave")+["enter"], ["enter"]])
    example_lbcont()
    results["lbcont"] = collect()[1:4]

    init([(15,7), (20,9), (25,7), (11,13)],
    [["down"],["down"],["down"],["up"],["up"]] +
    [["window resize"]]*3 + [["enter"]])
    example_lbscr()
    results["lbscr"] = collect()[:9]

    return results

def generate_body(tmpl, sections, blocks, results):
    """Assemble most of the HTML output (the body) from the pieces
    of text contained in all the arguments.  The body contains a table
    of contents followed by section headers and section text.  Pieces of
    section text may contain example code, and pieces of code may be
    followed by a "screen shot" of themselves.
    
    The sections dictionary gives the column formatting for the
    table of contents in the HTML and the description shared by the TOC
    entry and the section header.  With the addition of section numbers
    (generated now) sections contains the information needed to write
    the finished table of contents.  We expect the template to use
    two slots, toc_left and toc_right, which we fill with the TOC.
    We also expect the template to define two pieces of HTML,
    toc_section and toc_item, which give the formatting for all items
    in the TOC.

    While handling each entry in the TOC, we use the tag stored in
    sections to assemble the corresponding part of the body.  We expect
    the template to define two pieces of HTML, section_head and
    section_body, which give the formatting for all parts of the body
    (colored boxes for section headers, and the formatting that applies
    to all pieces of section text).  The template also defines one slot
    for each individual piece of section text, distinguished from the
    others by the section tag.
    
    Each individual body slot may use more slots to hold the examples
    and results included in a piece of section text.  These slots are
    numbered.  We use the section tags to look in the examples and
    results dictionaries; the dictionary order must match the numerical
    order used in the template.  We do not use example tags as defined
    in the global examples dictionary.  The blocks and results arguments
    describe which slots are defined; we hope the HTML text will use
    all the slots we have defined."""

    # put TOC columns into the variables used by the template
    # assign section numbers
    # generate HTML form of TOC entries, corresponding document parts
    assert len(sections) == 2, 'sections has %d columns but should have 2!' % len(sections)

    toc_slots = {'toc_left':[], 'toc_right':[]}
    body = []
    
    snum = inum = 0
    for slot, l in zip(['toc_left','toc_right'], sections):
        for tag, name in l:
            if not tag:
                # new section -- do its first item next time
                snum += 1
                inum = 0
                t = tmpl.format('toc_section', snum=repr(snum),
                    name=name )
                toc_slots[slot].append( t )
                b = tmpl.format('section_head', snum=repr(snum),
                    name=name )
                body.append( b )
                continue

            # new item inside a section
            inum += 1
            t = tmpl.format('toc_item', snum=repr(snum), 
                inum=repr(inum), name=name, tag=tag)
            toc_slots[slot].append( t )

            slots = {}
            i = 0
            for fn in examples.get(tag, []):
                slots['example[%d]'%i] = blocks[fn]
                i += 1
            i = 0
            for res in results.get(tag, []):
                slots['result[%d]'%i] = templayer.RawHTML(res)
                i += 1
            b = tmpl.format('body[%s]'%tag, ** slots )
            b = tmpl.format('section_body', snum=repr(snum),
                inum=repr(inum), name=name, tag=tag,
                content = b)
            body.append( b )
            
    return (body, toc_slots)

def parse_options():
    usage = "%s [-h|-?|--help]\n%s [-H|--HTML|--html] [-s|--scripts]" % \
     (sys.argv[0], sys.argv[0])
    help = """%s options:

-h, -?, --help        Print this message to standard error and exit.

-H, --HTML, --html    Write the HTML documentation to standard output.
-s, --scripts        Write runnable scripts to files.""" % sys.argv[0]
    do_html = False
    do_scripts = False

    if len(sys.argv) < 2 or len(sys.argv) > 3:
        sys.exit(usage)

    if len(sys.argv) == 2 and (sys.argv[1] in ('-h', '-?', '--help')):
        sys.exit(help)

    for arg in sys.argv[1:]:
        if arg in ('-H', '--HTML', '--html'):
            if do_html:    sys.exit(usage)
            else:        do_html = True
        elif arg in ('-s', '--scripts'):
            if do_scripts:    sys.exit(usage)
            else:        do_scripts = True
        else:
            sys.exit(usage)

    return (do_html, do_scripts)

def main():
    (do_html, do_scripts) = parse_options()

    if templayer is None:
        sys.stderr.write(
"""Error importing templayer.  Please download and install the Templayer
python module available at:
http://excess.org/templayer/
""")
        sys.exit( 1 )

    tmpl = templayer.HtmlTemplate( "tmpl_tutorial.html" )
    sections = read_sections( tmpl )
    code_blocks = read_example_code()

    if do_scripts:
        write_example_files( sections, code_blocks )

    if do_html:
        code_blocks = cut_example_code( code_blocks )
        results = generate_example_results()
        out_file = tmpl.start_file()
        (body, toc_slots) = generate_body( tmpl, sections,
                                           code_blocks, results )
        bottom = out_file.open(version=urwid.__version__, ** toc_slots)
        bottom.write( body )
        out_file.close()

if __name__=="__main__":
    main()