98ec7731e7
svn: r13489
676 lines
26 KiB
Python
676 lines
26 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2006 Donald N. Allingham
|
|
# Copyright (C) 2008 Brian G. Matherly
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program 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 General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
|
|
# $Id$
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import traceback
|
|
from gettext import gettext as _
|
|
import os
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# GTK modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import gtk
|
|
import pango
|
|
import gobject
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# GRAMPS modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import ManagedWindow
|
|
import Errors
|
|
from gen.plug import PluginRegister, PTYPE_STR
|
|
from gui.utils import open_file_with_default_application
|
|
from gui.pluginmanager import GuiPluginManager
|
|
import _Tool as Tool
|
|
from QuestionDialog import InfoDialog
|
|
import config
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# PluginStatus: overview of all plugins
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class PluginStatus(ManagedWindow.ManagedWindow):
|
|
"""Displays a dialog showing the status of loaded plugins"""
|
|
HIDDEN = '<span color="red">%s</span>' % _('Hidden')
|
|
AVAILABLE = '<span weight="bold" color="blue">%s</span>'\
|
|
% _('Visible')
|
|
|
|
def __init__(self, uistate, track=[]):
|
|
self.__uistate = uistate
|
|
self.title = _("Plugin Status")
|
|
ManagedWindow.ManagedWindow.__init__(self, uistate, track,
|
|
self.__class__)
|
|
|
|
self.__pmgr = GuiPluginManager.get_instance()
|
|
self.__preg = PluginRegister.get_instance()
|
|
self.set_window(gtk.Dialog("", uistate.window,
|
|
gtk.DIALOG_DESTROY_WITH_PARENT,
|
|
(gtk.STOCK_CLOSE, gtk.RESPONSE_CLOSE)),
|
|
None, self.title)
|
|
self.window.set_size_request(600, 400)
|
|
self.window.connect('response', self.close)
|
|
|
|
notebook = gtk.Notebook()
|
|
|
|
#first page with all registered plugins
|
|
vbox_reg = gtk.VBox()
|
|
scrolled_window_reg = gtk.ScrolledWindow()
|
|
self.list_reg = gtk.TreeView()
|
|
# model: plugintype, hidden, pluginname, plugindescr, pluginid
|
|
self.model_reg = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING,
|
|
gobject.TYPE_STRING, gobject.TYPE_STRING, gobject.TYPE_STRING)
|
|
self.selection_reg = self.list_reg.get_selection()
|
|
self.list_reg.set_model(self.model_reg)
|
|
self.list_reg.set_rules_hint(True)
|
|
self.list_reg.connect('button-press-event', self.button_press_reg)
|
|
col0_reg = gtk.TreeViewColumn(_('Type'), gtk.CellRendererText(), text=0)
|
|
col0_reg.set_sort_column_id(0)
|
|
self.list_reg.append_column(col0_reg)
|
|
self.list_reg.append_column(
|
|
gtk.TreeViewColumn(_('Hidden'), gtk.CellRendererText(), markup=1))
|
|
col2_reg = gtk.TreeViewColumn(_('Name'), gtk.CellRendererText(), text=2)
|
|
col2_reg.set_sort_column_id(2)
|
|
self.list_reg.append_column(col2_reg)
|
|
self.list_reg.append_column(
|
|
gtk.TreeViewColumn(_('Description'), gtk.CellRendererText(), text=3))
|
|
self.list_reg.set_search_column(2)
|
|
|
|
scrolled_window_reg.add(self.list_reg)
|
|
vbox_reg.pack_start(scrolled_window_reg)
|
|
hbutbox = gtk.HButtonBox()
|
|
hbutbox.set_layout(gtk.BUTTONBOX_SPREAD)
|
|
self.__info_btn = gtk.Button(_("Info"))
|
|
hbutbox.add(self.__info_btn)
|
|
self.__info_btn.connect('clicked', self.__info)
|
|
self.__hide_btn = gtk.Button(_("Hide/Unhide"))
|
|
hbutbox.add(self.__hide_btn)
|
|
self.__hide_btn.connect('clicked', self.__hide)
|
|
if __debug__:
|
|
self.__edit_btn = gtk.Button(_("Edit"))
|
|
hbutbox.add(self.__edit_btn)
|
|
self.__edit_btn.connect('clicked', self.__edit)
|
|
self.__load_btn = gtk.Button(_("Load"))
|
|
hbutbox.add(self.__load_btn)
|
|
self.__load_btn.connect('clicked', self.__load)
|
|
vbox_reg.pack_start(hbutbox, expand=False, padding=5)
|
|
|
|
notebook.append_page(vbox_reg,
|
|
tab_label=gtk.Label(_('Registered plugins')))
|
|
|
|
|
|
#second page with loaded plugins
|
|
scrolled_window = gtk.ScrolledWindow()
|
|
self.list = gtk.TreeView()
|
|
self.model = gtk.ListStore(gobject.TYPE_STRING, gobject.TYPE_STRING,
|
|
gobject.TYPE_STRING, object)
|
|
self.selection = self.list.get_selection()
|
|
|
|
self.list.set_model(self.model)
|
|
self.list.set_rules_hint(True)
|
|
self.list.connect('button-press-event', self.button_press)
|
|
self.list.append_column(
|
|
gtk.TreeViewColumn(_('Status'), gtk.CellRendererText(),
|
|
markup=0))
|
|
col1 = gtk.TreeViewColumn(_('File'), gtk.CellRendererText(), text=1)
|
|
col1.set_sort_column_id(1)
|
|
self.list.append_column(col1)
|
|
col2 = gtk.TreeViewColumn(_('Message'), gtk.CellRendererText(), text=2)
|
|
col2.set_sort_column_id(2)
|
|
self.list.append_column(col2)
|
|
self.list.set_search_column(1)
|
|
|
|
scrolled_window.add(self.list)
|
|
notebook.append_page(scrolled_window,
|
|
tab_label=gtk.Label(_('Loaded plugins')))
|
|
self.window.vbox.add(notebook)
|
|
|
|
if __debug__:
|
|
# Only show the "Reload" button when in debug mode
|
|
# (without -O on the command line)
|
|
self.__reload_btn = gtk.Button(_("Reload"))
|
|
self.window.action_area.add(self.__reload_btn)
|
|
self.__reload_btn.connect('clicked', self.__reload)
|
|
|
|
#obtain hidden plugins from the pluginmanager
|
|
self.hidden = self.__pmgr.get_hidden_plugin_ids()
|
|
|
|
self.window.show_all()
|
|
self.__populate_lists()
|
|
|
|
def __populate_lists(self):
|
|
""" Build the lists of plugins """
|
|
self.__populate_load_list()
|
|
self.__populate_reg_list()
|
|
|
|
def __populate_load_list(self):
|
|
""" Build list of loaded plugins"""
|
|
fail_list = self.__pmgr.get_fail_list()
|
|
|
|
for i in fail_list:
|
|
err = i[1][0]
|
|
|
|
if err == Errors.UnavailableError:
|
|
self.model.append(row=[
|
|
'<span color="blue">%s</span>' % _('Unavailable'),
|
|
i[0], str(i[1][1]), None])
|
|
else:
|
|
self.model.append(row=[
|
|
'<span weight="bold" color="red">%s</span>' % _('Fail'),
|
|
i[0], str(i[1][1]), i[1]])
|
|
|
|
success_list = self.__pmgr.get_success_list()
|
|
for i in success_list:
|
|
modname = i[1].__name__
|
|
descr = self.__pmgr.get_module_description(modname)
|
|
self.model.append(row=[
|
|
'<span weight="bold" color="#267726">%s</span>' % _("OK"),
|
|
i[0], descr, None])
|
|
|
|
def __populate_reg_list(self):
|
|
""" Build list of registered plugins"""
|
|
for (type, typestr) in PTYPE_STR.iteritems():
|
|
for pdata in self.__preg.type_plugins(type):
|
|
# model: plugintype, hidden, pluginname, plugindescr, pluginid
|
|
hidden = pdata.id in self.hidden
|
|
if hidden:
|
|
hiddenstr = self.HIDDEN
|
|
else:
|
|
hiddenstr = self.AVAILABLE
|
|
self.model_reg.append(row=[
|
|
typestr, hiddenstr, pdata.name, pdata.description,
|
|
pdata.id])
|
|
|
|
def __rebuild_load_list(self):
|
|
self.model.clear()
|
|
self.__populate_load_list()
|
|
|
|
def __rebuild_reg_list(self):
|
|
self.model_reg.clear()
|
|
self.__populate_reg_list()
|
|
|
|
def button_press(self, obj, event):
|
|
""" Callback function from the user clicking on a line """
|
|
if event.type == gtk.gdk._2BUTTON_PRESS and event.button == 1:
|
|
model, node = self.selection.get_selected()
|
|
data = model.get_value(node, 3)
|
|
name = model.get_value(node, 1)
|
|
if data:
|
|
PluginTrace(self.uistate, [], data, name)
|
|
|
|
def build_menu_names(self, obj):
|
|
return (self.title, "")
|
|
|
|
def __reload(self, obj):
|
|
""" Callback function from the "Reload" button """
|
|
self.__pmgr.reload_plugins()
|
|
self.__rebuild_load_list()
|
|
self.__rebuild_reg_list()
|
|
|
|
def button_press_reg(self, obj, event):
|
|
""" Callback function from the user clicking on a line in reg plugin
|
|
"""
|
|
if event.type == gtk.gdk._2BUTTON_PRESS and event.button == 1:
|
|
self.__info(None)
|
|
|
|
def __info(self, obj):
|
|
""" Callback function from the "Info" button
|
|
"""
|
|
model, node = self.selection_reg.get_selected()
|
|
if not node:
|
|
return
|
|
id = model.get_value(node, 4)
|
|
typestr = model.get_value(node, 0)
|
|
pdata = self.__preg.get_plugin(id)
|
|
auth = ' - '.join(pdata.authors)
|
|
email = ' - '.join(pdata.authors_email)
|
|
if len(auth) > 60:
|
|
auth = auth[:60] + '...'
|
|
if len(email) > 60:
|
|
email = email[:60] + '...'
|
|
if pdata:
|
|
infotxt = """Plugin name: %(name)s [%(typestr)s]
|
|
|
|
Description: %(descr)s
|
|
Authors: %(authors)s
|
|
Email: %(email)s
|
|
Filename: %(fname)s
|
|
Location: %(fpath)s
|
|
""" % {
|
|
'name': pdata.name,
|
|
'typestr': typestr,
|
|
'descr': pdata.description,
|
|
'authors': auth,
|
|
'email': email,
|
|
'fname': pdata.fname,
|
|
'fpath': pdata.fpath,
|
|
}
|
|
InfoDialog('Detailed Info', infotxt, parent=self.window)
|
|
|
|
def __hide(self, obj):
|
|
""" Callback function from the "Hide" button
|
|
"""
|
|
model, node = self.selection_reg.get_selected()
|
|
if not node:
|
|
return
|
|
id = model.get_value(node, 4)
|
|
if id in self.hidden:
|
|
#unhide
|
|
self.hidden.remove(id)
|
|
model.set_value(node, 1, self.AVAILABLE)
|
|
self.__pmgr.unhide_plugin(id)
|
|
else:
|
|
#hide
|
|
self.hidden.add(id)
|
|
model.set_value(node, 1, self.HIDDEN)
|
|
self.__pmgr.hide_plugin(id)
|
|
|
|
def __load(self, obj):
|
|
""" Callback function from the "Load" button
|
|
"""
|
|
model, node = self.selection_reg.get_selected()
|
|
if not node:
|
|
return
|
|
id = model.get_value(node, 4)
|
|
pdata = self.__preg.get_plugin(id)
|
|
self.__pmgr.load_plugin(pdata)
|
|
self.__rebuild_load_list()
|
|
|
|
def __edit(self, obj):
|
|
""" Callback function from the "Load" button
|
|
"""
|
|
model, node = self.selection_reg.get_selected()
|
|
if not node:
|
|
return
|
|
id = model.get_value(node, 4)
|
|
pdata = self.__preg.get_plugin(id)
|
|
open_file_with_default_application(
|
|
os.path.join(pdata.fpath, pdata.fname)
|
|
)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Details for an individual plugin that failed
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class PluginTrace(ManagedWindow.ManagedWindow):
|
|
"""Displays a dialog showing the status of loaded plugins"""
|
|
|
|
def __init__(self, uistate, track, data, name):
|
|
self.name = name
|
|
title = "%s: %s" % (_("Plugin Status"), name)
|
|
ManagedWindow.ManagedWindow.__init__(self, uistate, track, self)
|
|
|
|
self.set_window(gtk.Dialog("", uistate.window,
|
|
gtk.DIALOG_DESTROY_WITH_PARENT,
|
|
(gtk.STOCK_CLOSE, gtk.RESPONSE_CLOSE)),
|
|
None, title)
|
|
self.window.set_size_request(600, 400)
|
|
self.window.connect('response', self.close)
|
|
|
|
scrolled_window = gtk.ScrolledWindow()
|
|
scrolled_window.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
|
|
self.text = gtk.TextView()
|
|
scrolled_window.add(self.text)
|
|
self.text.get_buffer().set_text(
|
|
"".join(traceback.format_exception(data[0],data[1],data[2])))
|
|
|
|
self.window.vbox.add(scrolled_window)
|
|
self.window.show_all()
|
|
|
|
def build_menu_names(self, obj):
|
|
return (self.name, None)
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Classes for tools
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class LinkTag(gtk.TextTag):
|
|
def __init__(self, link, buffer):
|
|
gtk.TextTag.__init__(self, link)
|
|
tag_table = buffer.get_tag_table()
|
|
self.set_property('foreground', "#0000ff")
|
|
self.set_property('underline', pango.UNDERLINE_SINGLE)
|
|
try:
|
|
tag_table.add(self)
|
|
except ValueError:
|
|
pass # already in table
|
|
|
|
class ToolManagedWindowBase(ManagedWindow.ManagedWindow):
|
|
"""
|
|
Copied from src/ReportBase/_BareReportDialog.py BareReportDialog
|
|
"""
|
|
border_pad = 6
|
|
HELP_TOPIC = None
|
|
def __init__(self, dbstate, uistate, option_class, name, callback=None):
|
|
self.name = name
|
|
ManagedWindow.ManagedWindow.__init__(self, uistate, [], self)
|
|
|
|
self.extra_menu = None
|
|
self.widgets = []
|
|
self.frame_names = []
|
|
self.frames = {}
|
|
self.format_menu = None
|
|
self.style_button = None
|
|
|
|
window = gtk.Dialog('Tool')
|
|
self.set_window(window, None, self.get_title())
|
|
#self.window.set_has_separator(False)
|
|
|
|
#self.window.connect('response', self.close)
|
|
self.cancel = self.window.add_button(gtk.STOCK_CLOSE,
|
|
gtk.RESPONSE_CANCEL)
|
|
self.cancel.connect('clicked', self.close)
|
|
|
|
self.ok = self.window.add_button(gtk.STOCK_APPLY, gtk.RESPONSE_OK)
|
|
self.ok.connect('clicked', self.on_ok_clicked)
|
|
|
|
self.window.set_default_size(600, -1)
|
|
|
|
# Set up and run the dialog. These calls are not in top down
|
|
# order when looking at the dialog box as there is some
|
|
# interaction between the various frames.
|
|
|
|
self.setup_title()
|
|
self.setup_header()
|
|
self.tbl = gtk.Table(4, 4, False)
|
|
self.tbl.set_col_spacings(12)
|
|
self.tbl.set_row_spacings(6)
|
|
self.tbl.set_border_width(6)
|
|
self.col = 0
|
|
self.window.vbox.add(self.tbl)
|
|
|
|
# Build the list of widgets that are used to extend the Options
|
|
# frame and to create other frames
|
|
self.add_user_options()
|
|
|
|
self.notebook = gtk.Notebook()
|
|
self.notebook.set_border_width(6)
|
|
self.window.vbox.add(self.notebook)
|
|
|
|
self.results_text = gtk.TextView()
|
|
self.results_text.connect('button-press-event',
|
|
self.on_button_press)
|
|
self.results_text.connect('motion-notify-event',
|
|
self.on_motion)
|
|
self.tags = []
|
|
self.link_cursor = gtk.gdk.Cursor(gtk.gdk.LEFT_PTR)
|
|
self.standard_cursor = gtk.gdk.Cursor(gtk.gdk.XTERM)
|
|
|
|
self.setup_other_frames()
|
|
self.set_current_frame(self.initial_frame())
|
|
self.show()
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Callback functions from the dialog
|
|
#
|
|
#------------------------------------------------------------------------
|
|
def on_cancel(self, *obj):
|
|
pass # cancel just closes
|
|
|
|
def on_ok_clicked(self, obj):
|
|
"""
|
|
The user is satisfied with the dialog choices. Parse all options
|
|
and run the tool.
|
|
"""
|
|
# Save options
|
|
self.options.parse_user_options(self)
|
|
self.options.handler.save_options()
|
|
self.pre_run()
|
|
self.run() # activate results tab
|
|
self.post_run()
|
|
|
|
def initial_frame(self):
|
|
return None
|
|
|
|
def on_motion(self, view, event):
|
|
buffer_location = view.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT,
|
|
int(event.x),
|
|
int(event.y))
|
|
iter = view.get_iter_at_location(*buffer_location)
|
|
for (tag, person_handle) in self.tags:
|
|
if iter.has_tag(tag):
|
|
_window = view.get_window(gtk.TEXT_WINDOW_TEXT)
|
|
_window.set_cursor(self.link_cursor)
|
|
return False # handle event further, if necessary
|
|
view.get_window(gtk.TEXT_WINDOW_TEXT).set_cursor(self.standard_cursor)
|
|
return False # handle event further, if necessary
|
|
|
|
def on_button_press(self, view, event):
|
|
from Editors import EditPerson
|
|
buffer_location = view.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT,
|
|
int(event.x),
|
|
int(event.y))
|
|
iter = view.get_iter_at_location(*buffer_location)
|
|
for (tag, person_handle) in self.tags:
|
|
if iter.has_tag(tag):
|
|
person = self.db.get_person_from_handle(person_handle)
|
|
if event.button == 1:
|
|
if event.type == gtk.gdk._2BUTTON_PRESS:
|
|
try:
|
|
EditPerson(self.dbstate, self.uistate, [], person)
|
|
except Errors.WindowActiveError:
|
|
pass
|
|
else:
|
|
self.dbstate.change_active_person(person)
|
|
return True # handled event
|
|
return False # did not handle event
|
|
|
|
def results_write_link(self, text, person, person_handle):
|
|
self.results_write(" ")
|
|
buffer = self.results_text.get_buffer()
|
|
iter = buffer.get_end_iter()
|
|
offset = buffer.get_char_count()
|
|
self.results_write(text)
|
|
start = buffer.get_iter_at_offset(offset)
|
|
end = buffer.get_end_iter()
|
|
self.tags.append((LinkTag(person_handle, buffer), person_handle))
|
|
buffer.apply_tag(self.tags[-1][0], start, end)
|
|
|
|
def results_write(self, text):
|
|
buffer = self.results_text.get_buffer()
|
|
mark = buffer.create_mark("end", buffer.get_end_iter())
|
|
self.results_text.scroll_to_mark(mark, 0)
|
|
buffer.insert_at_cursor(text)
|
|
buffer.delete_mark_by_name("end")
|
|
|
|
def results_clear(self):
|
|
# Remove all tags and clear text
|
|
buffer = self.results_text.get_buffer()
|
|
tag_table = buffer.get_tag_table()
|
|
start = buffer.get_start_iter()
|
|
end = buffer.get_end_iter()
|
|
for (tag, handle) in self.tags:
|
|
buffer.remove_tag(tag, start, end)
|
|
tag_table.remove(tag)
|
|
self.tags = []
|
|
buffer.set_text("")
|
|
|
|
def pre_run(self):
|
|
from gui.utils import ProgressMeter
|
|
self.progress = ProgressMeter(self.get_title())
|
|
|
|
def run(self):
|
|
raise NotImplementedError, "tool needs to define a run() method"
|
|
|
|
def post_run(self):
|
|
self.progress.close()
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Functions related to setting up the dialog window.
|
|
#
|
|
#------------------------------------------------------------------------
|
|
def get_title(self):
|
|
"""The window title for this dialog"""
|
|
return "Tool" # self.title
|
|
|
|
def get_header(self, name):
|
|
"""The header line to put at the top of the contents of the
|
|
dialog box. By default this will just be the name of the
|
|
selected person. Most subclasses will customize this to give
|
|
some indication of what the report will be, i.e. 'Descendant
|
|
Report for %s'."""
|
|
return self.get_title()
|
|
|
|
def setup_title(self):
|
|
"""Set up the title bar of the dialog. This function relies
|
|
on the get_title() customization function for what the title
|
|
should be."""
|
|
self.window.set_title(self.get_title())
|
|
|
|
def setup_header(self):
|
|
"""Set up the header line bar of the dialog. This function
|
|
relies on the get_header() customization function for what the
|
|
header line should read. If no customization function is
|
|
supplied by the subclass, the default is to use the full name
|
|
of the currently selected person."""
|
|
|
|
title = self.get_header(self.get_title())
|
|
label = gtk.Label('<span size="larger" weight="bold">%s</span>' % title)
|
|
label.set_use_markup(True)
|
|
self.window.vbox.pack_start(label, True, True, self.border_pad)
|
|
|
|
def add_frame_option(self, frame_name, label_text, widget):
|
|
"""Similar to add_option this method takes a frame_name, a
|
|
text string and a Gtk Widget. When the interface is built,
|
|
all widgets with the same frame_name are grouped into a
|
|
GtkFrame. This allows the subclass to create its own sections,
|
|
filling them with its own widgets. The subclass is reponsible for
|
|
all managing of the widgets, including extracting the final value
|
|
before the report executes. This task should only be called in
|
|
the add_user_options task."""
|
|
|
|
if frame_name in self.frames:
|
|
self.frames[frame_name].append((label_text, widget))
|
|
else:
|
|
self.frames[frame_name] = [(label_text, widget)]
|
|
self.frame_names.append(frame_name)
|
|
|
|
def set_current_frame(self, name):
|
|
if name is None:
|
|
self.notebook.set_current_page(0)
|
|
else:
|
|
for frame_name in self.frame_names:
|
|
if name == frame_name:
|
|
if len(self.frames[frame_name]) > 0:
|
|
fname, child = self.frames[frame_name][0]
|
|
page = self.notebook.page_num(child)
|
|
self.notebook.set_current_page(page)
|
|
return
|
|
|
|
def add_results_frame(self, frame_name="Results"):
|
|
if frame_name not in self.frames:
|
|
window = gtk.ScrolledWindow()
|
|
window.set_policy(gtk.POLICY_AUTOMATIC, gtk.POLICY_AUTOMATIC)
|
|
window.add(self.results_text)
|
|
window.set_shadow_type(gtk.SHADOW_IN)
|
|
self.frames[frame_name] = [[frame_name, window]]
|
|
self.frame_names.append(frame_name)
|
|
l = gtk.Label("<b>%s</b>" % _(frame_name))
|
|
l.set_use_markup(True)
|
|
self.notebook.append_page(window, l)
|
|
self.notebook.show_all()
|
|
else:
|
|
self.results_clear()
|
|
self.set_current_frame(frame_name)
|
|
|
|
def setup_other_frames(self):
|
|
"""Similar to add_option this method takes a frame_name, a
|
|
text string and a Gtk Widget. When the interface is built,
|
|
all widgets with the same frame_name are grouped into a
|
|
GtkFrame. This allows the subclass to create its own sections,
|
|
filling them with its own widgets. The subclass is reponsible for
|
|
all managing of the widgets, including extracting the final value
|
|
before the report executes. This task should only be called in
|
|
the add_user_options task."""
|
|
for key in self.frame_names:
|
|
flist = self.frames[key]
|
|
table = gtk.Table(3, len(flist))
|
|
table.set_col_spacings(12)
|
|
table.set_row_spacings(6)
|
|
table.set_border_width(6)
|
|
l = gtk.Label("<b>%s</b>" % key)
|
|
l.set_use_markup(True)
|
|
self.notebook.append_page(table, l)
|
|
row = 0
|
|
for (text, widget) in flist:
|
|
if text:
|
|
text_widget = gtk.Label('%s:' % text)
|
|
text_widget.set_alignment(0.0, 0.5)
|
|
table.attach(text_widget, 1, 2, row, row+1,
|
|
gtk.SHRINK|gtk.FILL, gtk.SHRINK)
|
|
table.attach(widget, 2, 3, row, row+1,
|
|
yoptions=gtk.SHRINK)
|
|
else:
|
|
table.attach(widget, 2, 3, row, row+1,
|
|
yoptions=gtk.SHRINK)
|
|
row += 1
|
|
self.notebook.show_all()
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Functions related to extending the options
|
|
#
|
|
#------------------------------------------------------------------------
|
|
def add_user_options(self):
|
|
"""Called to allow subclasses add widgets to the dialog form.
|
|
It is called immediately before the window is displayed. All
|
|
calls to add_option or add_frame_option should be called in
|
|
this task."""
|
|
self.options.add_user_options(self)
|
|
|
|
def build_menu_names(self, obj):
|
|
return (_('Main window'), self.get_title())
|
|
|
|
|
|
|
|
class ToolManagedWindowBatch(Tool.BatchTool, ToolManagedWindowBase):
|
|
def __init__(self, dbstate, uistate, options_class, name, callback=None):
|
|
# This constructor will ask a question, set self.fail:
|
|
self.dbstate = dbstate
|
|
self.uistate = uistate
|
|
Tool.BatchTool.__init__(self, dbstate, options_class, name)
|
|
if not self.fail:
|
|
ToolManagedWindowBase.__init__(self, dbstate, uistate,
|
|
options_class, name, callback)
|
|
|
|
class ToolManagedWindow(Tool.Tool, ToolManagedWindowBase):
|
|
def __init__(self, dbstate, uistate, options_class, name, callback=None):
|
|
self.dbstate = dbstate
|
|
self.uistate = uistate
|
|
Tool.Tool.__init__(self, dbstate, options_class, name)
|
|
ToolManagedWindowBase.__init__(self, dbstate, uistate, options_class,
|
|
name, callback)
|