Rewrite using custom treemodel
Fixes performance issue.
This commit is contained in:
parent
9c64ded292
commit
1d6e7264fc
248
pyfontview.py
248
pyfontview.py
@ -3,6 +3,7 @@
|
|||||||
import pygtk
|
import pygtk
|
||||||
pygtk.require('2.0')
|
pygtk.require('2.0')
|
||||||
import gtk
|
import gtk
|
||||||
|
import sys
|
||||||
|
|
||||||
import fontTools.ttx
|
import fontTools.ttx
|
||||||
import fontTools.ttLib
|
import fontTools.ttLib
|
||||||
@ -13,50 +14,20 @@ class FontTreeStoreBuilder:
|
|||||||
def __init__(self, treestore):
|
def __init__(self, treestore):
|
||||||
self.ts = treestore
|
self.ts = treestore
|
||||||
|
|
||||||
def add_object(self, parent, key, value):
|
|
||||||
# Make sure item is decompiled
|
|
||||||
try:
|
|
||||||
getattr(value, "asdf")
|
|
||||||
except AttributeError:
|
|
||||||
pass
|
|
||||||
if isinstance(value, fontTools.ttLib.getTableModule('glyf').Glyph):
|
|
||||||
# Glyph type needs explicit expanding to be useful
|
|
||||||
value.expand(self.font['glyf'])
|
|
||||||
if isinstance(value, fontTools.cffLib.Index):
|
|
||||||
# Load all items
|
|
||||||
for i in range(len(value)):
|
|
||||||
print value[i]
|
|
||||||
# Discard offsets as should not be needed anymore
|
|
||||||
if hasattr(value, 'offsets'):
|
|
||||||
del value.offsets
|
|
||||||
|
|
||||||
item = self.ts.append(parent, [key, '%s' % value.__class__.__name__])
|
class Row(object):
|
||||||
for k,v in sorted(value.__dict__.items()):
|
def __init__(self, parent, index, key, value):
|
||||||
if k[0] == '_':
|
self._parent = parent
|
||||||
continue
|
self._index = index
|
||||||
self.add_thing (item, k, v)
|
self._key = key
|
||||||
|
self._value = value
|
||||||
|
|
||||||
def add_dict(self, parent, key, value):
|
if isinstance(value, fontTools.ttx.TTFont):
|
||||||
name = '%s of %d items' % (value.__class__.__name__, len(value))
|
self._add_font(value)
|
||||||
item = self.ts.append(parent, [key, name])
|
|
||||||
for k,v in sorted(value.items()):
|
|
||||||
self.add_thing (item, k, v)
|
|
||||||
|
|
||||||
def add_list(self, parent, key, value):
|
|
||||||
name = '%s of %d items' % (value.__class__.__name__, len(value))
|
|
||||||
if len(value) and len(value) <= 32:
|
|
||||||
name = str(value)
|
|
||||||
item = self.ts.append(parent, [key, name])
|
|
||||||
for k,v in enumerate(value):
|
|
||||||
self.add_thing (item, k, v)
|
|
||||||
|
|
||||||
def add_thing(self, parent, key, value):
|
|
||||||
if value == self.font:
|
|
||||||
return
|
|
||||||
if key in ['reader', 'file', 'tableTag', 'compileStatus', 'recurse']:
|
|
||||||
return
|
return
|
||||||
|
|
||||||
if not isinstance(value, basestring):
|
if not isinstance(value, basestring):
|
||||||
# Sequences
|
# Try sequences
|
||||||
is_sequence = True
|
is_sequence = True
|
||||||
try:
|
try:
|
||||||
len(value)
|
len(value)
|
||||||
@ -73,27 +44,178 @@ class FontTreeStoreBuilder:
|
|||||||
except IndexError:
|
except IndexError:
|
||||||
is_sequence = False
|
is_sequence = False
|
||||||
if is_sequence:
|
if is_sequence:
|
||||||
self.add_list(parent, key, value)
|
self._add_list(key, value)
|
||||||
return
|
return
|
||||||
if hasattr(value, '__dict__'):
|
if hasattr(value, '__dict__'):
|
||||||
self.add_object(parent, key, value)
|
self._add_object(key, value)
|
||||||
return
|
return
|
||||||
if hasattr(value, 'items'):
|
if hasattr(value, 'items'):
|
||||||
self.add_dict(parent, key, value)
|
self._add_dict(key, value)
|
||||||
return
|
return
|
||||||
|
|
||||||
if isinstance(value, basestring):
|
if isinstance(value, basestring):
|
||||||
item = self.ts.append(parent, [key, '"'+value+'"'])
|
self._value_str = '"'+value+'"'
|
||||||
|
self._children = []
|
||||||
return
|
return
|
||||||
|
|
||||||
# Everything else
|
# Everything else
|
||||||
item = self.ts.append(parent, [key, value])
|
self._children = []
|
||||||
|
|
||||||
def add_font(self, font):
|
def _filter_items(self):
|
||||||
|
items = []
|
||||||
|
for k,v in self._items:
|
||||||
|
if isinstance(v, fontTools.ttx.TTFont):
|
||||||
|
continue
|
||||||
|
if k in ['reader', 'file', 'tableTag', 'compileStatus', 'recurse']:
|
||||||
|
continue
|
||||||
|
if isinstance(k, basestring) and k[0] == '_':
|
||||||
|
continue
|
||||||
|
items.append((k,v))
|
||||||
|
self._items = items
|
||||||
|
|
||||||
|
def _add_font(self, font):
|
||||||
|
self._items = [(tag,font[tag]) for tag in font.keys()]
|
||||||
|
|
||||||
|
def _add_object(self, key, value):
|
||||||
|
# Make sure item is decompiled
|
||||||
|
try:
|
||||||
|
getattr(value, "asdf")
|
||||||
|
except AttributeError:
|
||||||
|
pass
|
||||||
|
if isinstance(value, fontTools.ttLib.getTableModule('glyf').Glyph):
|
||||||
|
# Glyph type needs explicit expanding to be useful
|
||||||
|
value.expand(self.font['glyf'])
|
||||||
|
if isinstance(value, fontTools.cffLib.Index):
|
||||||
|
# Load all items
|
||||||
|
for i in range(len(value)):
|
||||||
|
value[i]
|
||||||
|
# Discard offsets as should not be needed anymore
|
||||||
|
if hasattr(value, 'offsets'):
|
||||||
|
del value.offsets
|
||||||
|
|
||||||
|
self._value_str = value.__class__.__name__
|
||||||
|
self._items = sorted(value.__dict__.items())
|
||||||
|
self._filter_items()
|
||||||
|
|
||||||
|
def _add_dict(self, key, value):
|
||||||
|
self._value_str = '%s of %d items' % (value.__class__.__name__, len(value))
|
||||||
|
self._items = sorted(value.items())
|
||||||
|
self._filter_items()
|
||||||
|
|
||||||
|
def _add_list(self, key, value):
|
||||||
|
if len(value) and len(value) <= 32:
|
||||||
|
self._value_str = str(value)
|
||||||
|
else:
|
||||||
|
self._value_str = '%s of %d items' % (value.__class__.__name__,
|
||||||
|
len(value))
|
||||||
|
self._items = enumerate(value)
|
||||||
|
self._filter_items()
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
self._ensure_children()
|
||||||
|
return len(self._children)
|
||||||
|
|
||||||
|
def get_parent(self):
|
||||||
|
return self._parent
|
||||||
|
|
||||||
|
def _ensure_children(self):
|
||||||
|
if hasattr(self, '_children'):
|
||||||
|
return
|
||||||
|
children = []
|
||||||
|
for i,(k,v) in enumerate(self._items):
|
||||||
|
children.append(Row(self, i, k, v))
|
||||||
|
self._children = children
|
||||||
|
del self._items
|
||||||
|
|
||||||
|
def get_children(self):
|
||||||
|
self._ensure_children()
|
||||||
|
return self._children
|
||||||
|
|
||||||
|
def get_children(self):
|
||||||
|
self._ensure_children()
|
||||||
|
return self._children
|
||||||
|
|
||||||
|
def get_nth_child(self, n):
|
||||||
|
self._ensure_children()
|
||||||
|
if n < len(self._children):
|
||||||
|
return self._children[n]
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def get_key(self):
|
||||||
|
return self._key
|
||||||
|
|
||||||
|
def get_value(self):
|
||||||
|
return self._value
|
||||||
|
|
||||||
|
def get_value_str(self):
|
||||||
|
if hasattr(self, '_value_str'):
|
||||||
|
return self._value_str
|
||||||
|
return str(self._value)
|
||||||
|
|
||||||
|
def get_iter(self, path):
|
||||||
|
if not path:
|
||||||
|
return self
|
||||||
|
return self.get_nth_child(path[0]).get_iter(path[1:])
|
||||||
|
|
||||||
|
def iter_next(self):
|
||||||
|
if not self._parent:
|
||||||
|
return None
|
||||||
|
return self._parent.get_nth_child(self._index + 1)
|
||||||
|
|
||||||
|
def get_path(self):
|
||||||
|
if not self._parent: return ()
|
||||||
|
return self._parent.get_path() + (self._index,)
|
||||||
|
|
||||||
|
class FontTreeModel(gtk.GenericTreeModel):
|
||||||
|
|
||||||
|
__gtype_name__ = 'FontTreeModel'
|
||||||
|
|
||||||
|
def __init__(self, font):
|
||||||
|
super(FontTreeModel, self).__init__()
|
||||||
|
self._columns = (str, str)
|
||||||
self.font = font
|
self.font = font
|
||||||
for tag in font.keys():
|
self._root = Row(None, 0, "font", font)
|
||||||
self.add_thing (None, tag, font[tag])
|
|
||||||
self.font = None
|
def on_get_flags(self):
|
||||||
|
return 0
|
||||||
|
|
||||||
|
def on_get_n_columns(self):
|
||||||
|
return len(self._columns)
|
||||||
|
|
||||||
|
def on_get_column_type(self, index):
|
||||||
|
return self._columns[index]
|
||||||
|
|
||||||
|
def on_get_iter(self, path):
|
||||||
|
return self._root.get_iter(path)
|
||||||
|
|
||||||
|
def on_get_path(self, rowref):
|
||||||
|
return rowref.get_path()
|
||||||
|
|
||||||
|
def on_get_value(self, rowref, column):
|
||||||
|
if column == 0:
|
||||||
|
return rowref.get_key()
|
||||||
|
else:
|
||||||
|
return rowref.get_value_str()
|
||||||
|
|
||||||
|
def on_iter_next(self, rowref):
|
||||||
|
return rowref.iter_next()
|
||||||
|
|
||||||
|
def on_iter_children(self, rowref):
|
||||||
|
return rowref.get_nth_child(0)
|
||||||
|
|
||||||
|
def on_iter_has_child(self, rowref):
|
||||||
|
return bool(len(rowref))
|
||||||
|
|
||||||
|
def on_iter_n_children(self, rowref):
|
||||||
|
return len(rowref)
|
||||||
|
|
||||||
|
def on_iter_nth_child(self, rowref, n):
|
||||||
|
if not rowref: rowref = self._root
|
||||||
|
return rowref.get_nth_child(n)
|
||||||
|
|
||||||
|
def on_iter_parent(self, rowref):
|
||||||
|
return rowref.get_parent()
|
||||||
|
|
||||||
class PyFontView:
|
class PyFontView:
|
||||||
|
|
||||||
@ -101,26 +223,19 @@ class PyFontView:
|
|||||||
gtk.main_quit()
|
gtk.main_quit()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self, fontfile):
|
||||||
|
|
||||||
self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
|
self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
|
||||||
self.window.set_title("PyFontView")
|
self.window.set_title("%s - PyFontView" % fontfile)
|
||||||
self.window.connect("delete_event", self.delete_event)
|
self.window.connect("delete_event", self.delete_event)
|
||||||
self.window.set_size_request(400, 600)
|
self.window.set_size_request(400, 600)
|
||||||
|
|
||||||
self.scrolled_window = gtk.ScrolledWindow()
|
self.scrolled_window = gtk.ScrolledWindow()
|
||||||
self.window.add(self.scrolled_window)
|
self.window.add(self.scrolled_window)
|
||||||
|
|
||||||
self.treestore = gtk.TreeStore(str, str)
|
self.font = fontTools.ttx.TTFont(fontfile)
|
||||||
|
self.treemodel = FontTreeModel(self.font)
|
||||||
font = fontTools.ttx.TTFont("abc.woff")
|
self.treeview = gtk.TreeView(self.treemodel)
|
||||||
#font = fontTools.ttx.TTFont("IranNastaliq2.ttf")
|
|
||||||
#font = fontTools.ttx.TTFont("KozGoPr6N-Medium.otf")
|
|
||||||
#font = fontTools.ttx.TTFont("SmaGoSS07262013-Book.otf")
|
|
||||||
builder = FontTreeStoreBuilder(self.treestore)
|
|
||||||
builder.add_font (font)
|
|
||||||
|
|
||||||
self.treeview = gtk.TreeView(self.treestore)
|
|
||||||
#self.treeview.set_reorderable(True)
|
#self.treeview.set_reorderable(True)
|
||||||
|
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
@ -134,15 +249,20 @@ class PyFontView:
|
|||||||
|
|
||||||
col.add_attribute(cell, 'text', i)
|
col.add_attribute(cell, 'text', i)
|
||||||
|
|
||||||
self.treeview.set_search_column(i)
|
|
||||||
col.set_sort_column_id(i)
|
col.set_sort_column_id(i)
|
||||||
|
|
||||||
|
self.treeview.set_search_column(1)
|
||||||
self.scrolled_window.add(self.treeview)
|
self.scrolled_window.add(self.treeview)
|
||||||
self.window.show_all()
|
self.window.show_all()
|
||||||
|
|
||||||
def main():
|
def main(args=None):
|
||||||
|
if args == None: args = sys.argv
|
||||||
|
if len(args) < 2:
|
||||||
|
print >>sys.stderr, "usage: %s font..." % args[0]
|
||||||
|
sys.exit(1)
|
||||||
|
for arg in args[1:]:
|
||||||
|
viewer = PyFontView(arg)
|
||||||
gtk.main()
|
gtk.main()
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
viewer = PyFontView()
|
|
||||||
main()
|
main()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user