2000-02-24 04:03:49 +00:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
|
|
|
import xmllib;
|
|
|
|
import sys;
|
|
|
|
import string
|
|
|
|
import re
|
|
|
|
|
|
|
|
def html_subst(s):
|
|
|
|
if s.group(1) != None:
|
|
|
|
return s.group(0)
|
|
|
|
elif s.group(2) != None:
|
|
|
|
return '<a href="' + s.group(0) + '">' + s.group(0) + '</a>'
|
|
|
|
elif s.group(3) != None:
|
|
|
|
return '<a href="mailto:' + s.group(0) + '">' + s.group(0) + '</a>'
|
|
|
|
|
|
|
|
def htmlify(str):
|
|
|
|
return re.sub ("(<[^>]*>)|(http://[~.:/\w-]+)|([\w._!-]+@[\w_-]+).[\w_-]+", html_subst, str)
|
|
|
|
|
|
|
|
def bug_subst(s):
|
|
|
|
if s.group(1) != None:
|
|
|
|
return s.group(0)
|
|
|
|
else:
|
|
|
|
n = s.group(2)
|
|
|
|
return '<a href="http://bugs.gnome.org/db/%s/%s.html">#%s</a>' % (n[0:2], n, n)
|
|
|
|
|
|
|
|
def bugify(str):
|
|
|
|
str = re.sub ("(<[^>]*>)|#(\d+)", bug_subst, str)
|
|
|
|
return htmlify(str)
|
|
|
|
|
|
|
|
def make_id(str):
|
|
|
|
return re.sub ("[^a-z]","-", string.lower(str))
|
|
|
|
|
|
|
|
class ParseError (Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
class Entry:
|
|
|
|
def __init__(self):
|
|
|
|
self.description = None
|
|
|
|
self.title = None
|
|
|
|
self.url = None
|
|
|
|
self.contact = None
|
|
|
|
self.bugs = None
|
|
|
|
|
|
|
|
def set_size(self, size):
|
|
|
|
size = string.lower(size)
|
|
|
|
if size == "small":
|
|
|
|
self.size = "Small"
|
|
|
|
elif size == "medium":
|
|
|
|
self.size = "Medium"
|
|
|
|
elif size == "big":
|
|
|
|
self.size = "Big"
|
|
|
|
else:
|
|
|
|
raise ParseError, 'size must be "small", "medium", or "big"'
|
|
|
|
|
|
|
|
def output(self):
|
|
|
|
if self.size == "Big":
|
|
|
|
bgcolor = "#88bb88"
|
|
|
|
elif self.size == "Medium":
|
|
|
|
bgcolor = "#b4d4b4"
|
|
|
|
else:
|
|
|
|
bgcolor = "#d0e0d0"
|
|
|
|
|
|
|
|
print '''<table cellspacing="0" cellpadding="2" width="97%%" border="0" bgcolor="#000000">
|
|
|
|
<tbody><tr><td colspan=2>
|
|
|
|
<table cellspacing="0" cellpadding="5" width="100%%" border="0" bgcolor="#ffffff">
|
|
|
|
<tbody>
|
|
|
|
<tr bgcolor="%s">
|
|
|
|
<td align="left"><font size="+1">%s</font></font></td>
|
|
|
|
<td align="left" width="20%%"><b>Size</b>: %s</td>
|
|
|
|
<td align="center" width="20%%"><b>Status</b>: %s</td>
|
|
|
|
<td align="right" width="20%%"><b>Target Version</b>: %s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td colspan=4>
|
|
|
|
%s
|
|
|
|
<table cellspacing="0" cellpadding="0">
|
|
|
|
<tbody>''' % (bgcolor, self.title, self.size, self.status, self.target, htmlify(self.description))
|
|
|
|
|
|
|
|
if self.url != None:
|
|
|
|
print '''<tr><td width="0"><b>More Info</b>:</td>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>''' % htmlify (self.url)
|
|
|
|
|
|
|
|
if self.bugs != None:
|
|
|
|
print '''<tr><td width="0"><b>Bug Reports</b>:</td>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>''' % bugify (self.bugs)
|
|
|
|
|
|
|
|
if self.contact != None:
|
|
|
|
print '''<tr><td width="0"><b>Contact</b>:</td>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>''' % htmlify (self.contact)
|
|
|
|
|
|
|
|
print '''</tbody>
|
|
|
|
</table>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody></table>
|
|
|
|
</td></tr></tbody></table>
|
|
|
|
'''
|
|
|
|
|
|
|
|
class Section:
|
|
|
|
def __init__(self):
|
|
|
|
self.title = None
|
|
|
|
self.entries = []
|
|
|
|
|
|
|
|
def output(self):
|
|
|
|
|
|
|
|
print '<h2><a name="%s">%s</a></h2>' % (make_id(self.title), self.title)
|
|
|
|
|
|
|
|
first = 1
|
|
|
|
for entry in self.entries:
|
|
|
|
if not first:
|
|
|
|
print "<br>"
|
|
|
|
first = 0
|
|
|
|
entry.output()
|
|
|
|
|
|
|
|
class TodoParser (xmllib.XMLParser):
|
|
|
|
def __init__(self):
|
|
|
|
xmllib.XMLParser.__init__(self)
|
|
|
|
|
|
|
|
self.in_todo = 0
|
|
|
|
self.in_data = 0
|
|
|
|
self.data = ""
|
|
|
|
self.section = None
|
|
|
|
self.entry = None
|
2000-02-25 22:56:58 +00:00
|
|
|
self.logourl = None
|
|
|
|
self.title = None
|
2000-02-24 04:03:49 +00:00
|
|
|
self.sections = []
|
|
|
|
|
|
|
|
self.entitydefs = {}
|
|
|
|
|
|
|
|
def start_todo(self,attributes):
|
|
|
|
if self.in_todo:
|
|
|
|
raise ParseError, "<todo> tags may not be nested"
|
2000-02-25 22:56:58 +00:00
|
|
|
if attributes.has_key ("logourl"):
|
|
|
|
self.logourl = attributes["logourl"]
|
2000-02-24 04:03:49 +00:00
|
|
|
self.in_todo = 1
|
|
|
|
|
|
|
|
def end_todo(self):
|
|
|
|
self.in_todo = 0
|
|
|
|
|
|
|
|
def start_section(self,attributes):
|
|
|
|
if self.section:
|
|
|
|
raise ParseError, "<section> tags may not be nested"
|
|
|
|
|
|
|
|
self.section = Section()
|
|
|
|
|
|
|
|
def end_section(self):
|
|
|
|
if self.section.title == None:
|
|
|
|
raise ParseError, "<section> requires <title>"
|
|
|
|
|
|
|
|
self.sections.append(self.section)
|
|
|
|
self.section = None
|
|
|
|
|
|
|
|
def start_title(self,attributes):
|
2000-02-25 22:56:58 +00:00
|
|
|
if not self.in_todo:
|
|
|
|
raise ParseError, "<title> tag must be in <todo>, <section> or <entry>"
|
2000-02-24 04:03:49 +00:00
|
|
|
if self.in_data:
|
|
|
|
raise ParseError, "Unexpected <title> tag in content"
|
|
|
|
self.in_data = 1
|
|
|
|
|
|
|
|
def end_title(self):
|
|
|
|
self.in_data = 0
|
|
|
|
if self.entry:
|
|
|
|
self.entry.title = self.data
|
2000-02-25 22:56:58 +00:00
|
|
|
elif self.section:
|
2000-02-24 04:03:49 +00:00
|
|
|
self.section.title = self.data
|
2000-02-25 22:56:58 +00:00
|
|
|
else:
|
|
|
|
self.title = self.data
|
|
|
|
self.data = ""
|
2000-02-24 04:03:49 +00:00
|
|
|
|
|
|
|
def start_description(self,attributes):
|
|
|
|
if not self.entry:
|
|
|
|
raise ParseError, "<description> tag must be in <entry>"
|
|
|
|
if self.in_data:
|
|
|
|
raise ParseError, "Unexpected <description> tag in content"
|
|
|
|
self.in_data = 1
|
|
|
|
|
|
|
|
def end_description(self):
|
|
|
|
self.in_data = 0
|
|
|
|
self.entry.description = self.data
|
|
|
|
self.data = ""
|
|
|
|
|
|
|
|
def start_url(self,attributes):
|
|
|
|
if not self.entry:
|
|
|
|
raise ParseError, "<url> tag must be in <entry>"
|
|
|
|
if self.in_data:
|
|
|
|
raise ParseError, "Unexpected <url> tag in content"
|
|
|
|
self.in_data = 1
|
|
|
|
|
|
|
|
def end_url(self):
|
|
|
|
self.in_data = 0
|
|
|
|
self.entry.url = self.data
|
|
|
|
self.data = ""
|
|
|
|
|
|
|
|
def start_contact(self,attributes):
|
|
|
|
if not self.entry:
|
Make parent_class static.
Sun Nov 5 04:24:53 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtkcellrenderertextpixbuf.c: Make parent_class
static.
Tue Sep 19 10:54:22 2000 Owen Taylor <otaylor@redhat.com>
* modules/input/{gtkimcontextxim.[ch],imxim.c}: Rip
out support for multiple locales; that simple doesn't
work reliably with current Xlib
* gtk/gtkimcontext*.[ch] gtk/gtkimmulticontext.[ch]
gtk/gtktextlayout.[ch] gtk/gtktextview.c gtk/gtkentry.c:
Add support for positioning the cursor within the preedit string.
Mon Sep 18 23:56:32 2000 Owen Taylor <otaylor@redhat.com>
* modules/input/{imxim.c,gtkimcontextxim.[ch]}: Start
at XIM input method module.
* gtk/gtktextview.c: Check for bindings after passing
events to im context filter.
Mon Sep 18 11:50:51 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtktextlayout.c (add_preedit_attrs): Handle
empty attribute lists properly.
Sun Sep 17 10:08:16 2000 Owen Taylor <otaylor@redhat.com>
* gtk/queryimmodules.c (main): Return non-zero exit
status if errors were encountered querying any
modules.
* modules/input/Makefile.am (moduledir): remove
leftover bin program target.
* docs/make-todo: Fix typo in error message.
Sat Sep 16 14:04:30 2000 Owen Taylor <otaylor@redhat.com>
* configure.in: Add modules/input/Makefile
Sat Sep 16 14:01:52 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtk.h: include gtkmodule.h gtkoldeditable.h,
don't include gtkthemes.h.
* gtk/testgtk.c gtk/testtext.c: Set environment variables
to point
* gtk/Makefile.am: Add new .c and .h files, build
gtk-query-immodules and use it to create a gtk.immodules
file for use of test programs.
* gtk/gtkpreview.c: remove extra blank line.
Sat Sep 16 13:21:04 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtkimcontextsimple.c (gtk_im_context_simple_add_table):
Add the ability to add extra tables beyond the default
one, and also the ability to have compose sequences
that are prefixes of other compose sequences.
* gtk/gtkimcontextsimple.c: Export a preedit string which
consists of possible candidates for keystrokes that have
been entered but not yet committed.
* gtk/gtkimcontext.[ch] gtk/immulticontext.[ch]
gtk/gtkimcontextsimple.[ch]: add gtk_im_context_reset()
* gtk/gtkmulticontext.[ch] (gtk_im_multicontext_append_menuitems):
Add a function to add input-method switching menu items
to a menu.
* gtk/gtkimmulticontext.[ch]: Properly handly set_client_window
when switching input methods.
* gtk/gtkimcontextsimple.[ch]: Change the format of
the compose table to allow compose tables of different
lengths / sequence.
Sat Sep 16 13:05:48 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtkimmodule.[ch]: Support routines for loading
GtkIMContext implementations dynamically at runtime.
* modules/input/imcyrillic-translit.c: A sample input
method (based on GtkIMContextSimple with an extra table),
which demonstrates preedit strings and the module
system for input modules
* gtk/queryimmodules.c: Program to query the available
input modules and write the results into a file.
* gtk/gtkrc.[ch] (gtk_rc_get_im_module_file): Add
extra config options "im_module_file" (cache file for
input method modules), and "im_module_path" - path
to look for modules when generating cache file.
This doesn't scale.
Sat Sep 16 13:09:06 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtkthemes.[ch] gtk/gtkmodule.[ch]: Move most of the
generic code from gtkthemes into a new abstraction
GtkModule which has the logic for implementing
a loadable module which implements a number of
GObject types.
Sat Sep 16 13:07:13 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtkeditable.[ch]: Convert GtkEditable from
a class into an interface
* gtk/gtkoldeditable.[ch]: Move the old editable
implementation into here, so legacy widgets can
still rely on the implemenation. GtkOldEditable
exports GtkEditable. Make selection handling
code use new text conversion functions (and
handle UTF-8 as a side-effect). Use GtkClipboard
for CLIPBOARD.
* gtk/gtktext.[ch] gtk/gtkcombo.c gtk/gtkspinbutton.c:
Adopt to match above changes.
* gtk/gtkentry.[ch]: Implement GtkEditable directly,
avoid GtkOldEditable implementation. Restructure
to reduce number of places that modify state directly.
Move to GtkBindingSet. Display the preedit string.
Queue recomputation of PangoLayout and scroll position
to improve effiency of doing complex changes naively.
Add a menu with cut/copy/paste and input method selection.
Thu Sep 14 22:11:05 2000 Owen Taylor <otaylor@redhat.com>
* gtk/gtktextlayout.[ch]: Add gtk_text_layout_set_preedit_string()
to set preedit string and attributes; display preedit string by
inserting string and attributes at cursor when creating the
GtkTextLineDisplay.
* gtk/gtktextlayout.c: Move all conversions between byte
positions in PangoLayout and GtkTextIter into new functions
line_display_iter_to_index/index_to_iter that properly
handle the preedit string.
* gtk/gtktextmark.[ch]: Restore gtk_text_mark_get_name, modify
it to return const char * (eventually will end up
as GCONST char *, most likely.)
* gtk/gtktextview.[ch]: Handle the preedit string, call
gtk_im_context_reset() as necessary, add a menu to switch
input methods.
* gtk/gtktextlayout.[ch]: Remove useless
gtk_text_layout_get_log_attrs() function.
2000-11-12 03:35:09 +00:00
|
|
|
raise ParseError, "<contact> tag must be in <entry>"
|
2000-02-24 04:03:49 +00:00
|
|
|
if self.in_data:
|
|
|
|
raise ParseError, "Unexpected <contact> tag in content"
|
|
|
|
self.in_data = 1
|
|
|
|
|
|
|
|
def end_contact(self):
|
|
|
|
self.in_data = 0
|
|
|
|
self.entry.contact = self.data
|
|
|
|
self.data = ""
|
|
|
|
|
|
|
|
def start_bugs(self,attributes):
|
|
|
|
if not self.entry:
|
|
|
|
raise ParseError, "<bugs> tag must be in <bugs>"
|
|
|
|
if self.in_data:
|
|
|
|
raise ParseError, "Unexpected <bugs> tag in content"
|
|
|
|
self.in_data = 1
|
|
|
|
|
|
|
|
def end_bugs(self):
|
|
|
|
self.in_data = 0
|
|
|
|
self.entry.bugs = self.data
|
|
|
|
self.data = ""
|
|
|
|
|
|
|
|
def start_entry(self,attributes):
|
|
|
|
if not self.section:
|
|
|
|
raise ParseError, "<entry> tag must be in <section>"
|
|
|
|
if self.entry:
|
|
|
|
raise ParseError, "<entry> tags may not be nested"
|
|
|
|
|
|
|
|
self.entry = Entry()
|
|
|
|
|
|
|
|
if not attributes.has_key("size"):
|
|
|
|
raise ParseError, '"size" attribute required for entry'
|
|
|
|
self.entry.set_size(attributes["size"])
|
|
|
|
|
|
|
|
if not attributes.has_key("status"):
|
|
|
|
raise ParseError, '"status" attribute (completion percentage) required for entry'
|
|
|
|
self.entry.status=attributes["status"]
|
|
|
|
|
|
|
|
if not attributes.has_key("target"):
|
|
|
|
raise ParseError, '"target" attribute (target version) required for entry'
|
|
|
|
self.entry.target=attributes["target"]
|
|
|
|
|
|
|
|
def end_entry(self):
|
|
|
|
if self.entry.title == None:
|
|
|
|
raise ParseError, "<entry> requires <title>"
|
|
|
|
|
|
|
|
if self.entry.description == None:
|
|
|
|
raise ParseError, "<entry> requires <description>"
|
|
|
|
|
|
|
|
self.section.entries.append(self.entry)
|
|
|
|
self.entry = None
|
|
|
|
|
|
|
|
def handle_data(self,data):
|
|
|
|
if self.in_data:
|
|
|
|
self.data = self.data + data
|
|
|
|
|
|
|
|
def unknown_starttag(self,tag,attributes):
|
|
|
|
if not self.in_data:
|
|
|
|
raise ParseError, "Unexpected start tag: " + tag
|
|
|
|
else:
|
|
|
|
self.data = self.data + "<" + tag
|
|
|
|
for (key,val) in attributes.items():
|
|
|
|
self.data = self.data + ' %s="%s"' % (key,val)
|
|
|
|
self.data = self.data + ">"
|
|
|
|
|
|
|
|
def unknown_endtag(self,tag):
|
|
|
|
if not self.in_data:
|
|
|
|
raise ParseError, "Unexpected end tag: " + tag
|
|
|
|
else:
|
|
|
|
self.data = self.data + "</%s>" % tag
|
|
|
|
|
|
|
|
def syntax_error(self, err):
|
|
|
|
if re.match("reference to unknown entity", err):
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
xmllib.XMLParser.syntax_error (self, err)
|
|
|
|
|
|
|
|
def unknown_entityref(self,ref):
|
|
|
|
if not self.in_data:
|
|
|
|
raise ParseError, "Unknown entity &" + ref + ";"
|
|
|
|
else:
|
|
|
|
self.data = self.data + "&" + ref + ";"
|
|
|
|
|
|
|
|
file = open(sys.argv[1])
|
|
|
|
parser = TodoParser()
|
|
|
|
|
|
|
|
lineno = 1
|
|
|
|
while 1:
|
|
|
|
line = file.readline()
|
|
|
|
if line == "":
|
|
|
|
break
|
|
|
|
|
|
|
|
try:
|
|
|
|
parser.feed(line)
|
|
|
|
except ParseError, err:
|
|
|
|
sys.stderr.write("Parse error at line " + `lineno` + ": " + err.__str__() + "\n")
|
|
|
|
sys.exit(1)
|
|
|
|
except RuntimeError, err:
|
|
|
|
sys.stderr.write(err.__str__() + "\n")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
lineno = lineno + 1
|
|
|
|
|
|
|
|
parser.close()
|
2000-02-25 22:56:58 +00:00
|
|
|
if parser.title == None:
|
|
|
|
sys.stderr.write ("<todo> Document must have a <title>\n")
|
|
|
|
sys.exit (1)
|
2000-02-24 04:03:49 +00:00
|
|
|
|
|
|
|
print '''<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN">
|
|
|
|
<html>
|
|
|
|
<head>
|
2000-02-25 22:56:58 +00:00
|
|
|
<title>%s</title>
|
2000-02-24 04:03:49 +00:00
|
|
|
</head>
|
|
|
|
<body bgcolor="#ffffff">
|
2000-02-25 22:56:58 +00:00
|
|
|
<table width="100%%" cellspacing="0" cellpadding="0" border="0">
|
2000-02-24 04:03:49 +00:00
|
|
|
<tbody>
|
|
|
|
<tr valign="top">
|
|
|
|
<td>
|
2000-02-25 22:56:58 +00:00
|
|
|
<h1>%s</h1>''' % (parser.title, parser.title)
|
2000-02-24 04:03:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
for section in parser.sections:
|
|
|
|
ntasks = len(section.entries)
|
|
|
|
id = make_id (section.title)
|
|
|
|
if ntasks == 1:
|
|
|
|
print '<a href="#%s">%s</a> (1 item)<br>' % (id,section.title)
|
|
|
|
else:
|
|
|
|
print '<a href="#%s">%s</a> (%d items)<br>' % (id,section.title,ntasks)
|
|
|
|
|
|
|
|
print '''
|
2000-02-25 22:56:58 +00:00
|
|
|
</td>'''
|
|
|
|
if parser.logourl != None:
|
|
|
|
print ''' <td align="right">
|
|
|
|
<img src="%s" alt="Logo"></img>
|
|
|
|
</td>''' % parser.logourl
|
|
|
|
print '''
|
2000-02-24 04:03:49 +00:00
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
2000-02-25 22:56:58 +00:00
|
|
|
'''
|
2000-02-24 04:03:49 +00:00
|
|
|
|
|
|
|
first = 1
|
|
|
|
for section in parser.sections:
|
|
|
|
if not first:
|
|
|
|
print "<br><br>"
|
|
|
|
first = 0
|
|
|
|
section.output()
|
|
|
|
|
|
|
|
print '''</body>
|
|
|
|
</html>'''
|
|
|
|
|
|
|
|
|