Source code for gramps.gui.editors.filtereditor

#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2007  Donald N. Allingham
# Copyright (C) 2011       Tim G L Lyons
#
# 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#

"""
Custom Filter Editor tool.
"""

#-------------------------------------------------------------------------
#
# Python modules
#
#-------------------------------------------------------------------------

#------------------------------------------------------------------------
#
# Set up logging
#
#------------------------------------------------------------------------
import logging
log = logging.getLogger(".filtereditor")

#-------------------------------------------------------------------------
#
# GTK/GNOME 
#
#-------------------------------------------------------------------------
from gi.repository import Gtk
from gi.repository import Gdk
from gi.repository import GObject

#-------------------------------------------------------------------------
#
# GRAMPS modules
#
#-------------------------------------------------------------------------
from gramps.gen.const import GRAMPS_LOCALE as glocale
_ = glocale.translation.sgettext
from gramps.gen.filters import (GenericFilterFactory, FilterList, 
                                reload_custom_filters)
from gramps.gen.filters.rules._matchesfilterbase import MatchesFilterBase
from ..listmodel import ListModel
from ..managedwindow import ManagedWindow
from ..dialog import QuestionDialog
from gramps.gen.const import RULE_GLADE, URL_MANUAL_PAGE
from ..display import display_help
from gramps.gen.errors import WindowActiveError
from gramps.gen.lib import (AttributeType, EventType, FamilyRelType,
                            NameOriginType, NameType, NoteType)
from gramps.gen.filters import rules
from ..autocomp import StandardCustomSelector, fill_entry
from ..selectors import SelectorFactory
from gramps.gen.display.name import displayer as _nd
from gramps.gen.utils.db import family_name
from gramps.gen.utils.string import conf_strings
from gramps.gen.constfunc import cuni
from ..widgets import DateEntry
from gramps.gen.datehandler import displayer

#-------------------------------------------------------------------------
#
# Constants
#
#-------------------------------------------------------------------------
WIKI_HELP_PAGE = WIKI_HELP_PAGE = '%s_-_Filters' % URL_MANUAL_PAGE

# dictionary mapping FILTER_TYPE of views to Filter window name
_TITLES = {
            'Person' : _("Person Filters"),
            'Family' : _('Family Filters'),
            'Event' : _('Event Filters'),
            'Place' : _('Place Filters'),
            'Source' : _('Source Filters'),
            'Media' : _('Media Filters'),
            'Repository' : _('Repository Filters'),
            'Note' : _('Note Filters'),
            'Citation' : _('Citation Filters'),
}

_name2typeclass = {
    _('Personal event:')     : EventType,
    _('Family event:')       : EventType,
    _('Event type:')         : EventType,
    _('Personal attribute:') : AttributeType,
    _('Family attribute:')   : AttributeType,
    _('Event attribute:')    : AttributeType,
    _('Media attribute:')    : AttributeType,
    _('Relationship type:')  : FamilyRelType,
    _('Note type:')          : NoteType,
    _('Name type:')          : NameType,
    _('Surname origin type:'): NameOriginType,
}

#-------------------------------------------------------------------------
#
# MyBoolean - check button with standard interface
#
#-------------------------------------------------------------------------
[docs]class MyBoolean(Gtk.CheckButton): def __init__(self, label=None): GObject.GObject.__init__(self) self.set_label(label) self.show()
[docs] def get_text(self): """ Return the text to save. It should be the same no matter the present locale (English or numeric types). This class sets this to get_display_text, but when localization is an issue (events/attr/etc types) then it has to be overridden. """ return str(int(self.get_active()))
[docs] def set_text(self, val): """ Set the selector state to display the passed value. """ is_active = bool(int(val)) self.set_active(is_active) #------------------------------------------------------------------------- # # MyInteger - spin button with standard interface # #-------------------------------------------------------------------------
[docs]class MyInteger(Gtk.SpinButton): def __init__(self, min, max): GObject.GObject.__init__(self) self.set_adjustment(Gtk.Adjustment(min, min, max, 1)) self.show()
[docs] def get_text(self): return str(self.get_value_as_int())
[docs] def set_text(self, val): self.set_value(int(val)) #------------------------------------------------------------------------- # # MyFilters - Combo box with list of filters with a standard interface # #-------------------------------------------------------------------------
[docs]class MyFilters(Gtk.ComboBox): """ Class to present a combobox of selectable filters. """ def __init__(self, filters, filter_name=None): """ Construct the combobox from the entries of the filters list. Filter_name is name of calling filter. If filter_name is given, it will be excluded from the dropdown box. """ GObject.GObject.__init__(self) store = Gtk.ListStore(GObject.TYPE_STRING) self.set_model(store) cell = Gtk.CellRendererText() self.pack_start(cell, True) self.add_attribute(cell, 'text', 0) #remove own name from the list if given. self.flist = sorted(f.get_name() for f in filters if filter_name is None or f.get_name() != filter_name ) for fname in self.flist: store.append(row=[fname]) self.set_active(0) self.show()
[docs] def get_text(self): active = self.get_active() if active < 0: return "" return self.flist[active]
[docs] def set_text(self, val): if val in self.flist: self.set_active(self.flist.index(val)) #------------------------------------------------------------------------- # # MyList - Combo box to allow entries # #-------------------------------------------------------------------------
[docs]class MyList(Gtk.ComboBox): def __init__(self, clist, clist_trans, default=0): GObject.GObject.__init__(self) store = Gtk.ListStore(GObject.TYPE_STRING) self.set_model(store) cell = Gtk.CellRendererText() self.pack_start(cell, True) self.add_attribute(cell, 'text', 0) self.clist = clist for name in clist_trans: store.append(row=[name]) self.set_active(default) self.show()
[docs] def get_text(self): active = self.get_active() return self.clist[active]
[docs] def set_text(self, val): if val in self.clist: self.set_active(self.clist.index(val)) #------------------------------------------------------------------------- # # MyLesserEqualGreater - Combo box to allow selection of "<", "=", or ">" # #-------------------------------------------------------------------------
[docs]class MyLesserEqualGreater(Gtk.ComboBox): def __init__(self, default=0): GObject.GObject.__init__(self) store = Gtk.ListStore(GObject.TYPE_STRING) self.set_model(store) cell = Gtk.CellRendererText() self.pack_start(cell, True) self.add_attribute(cell, 'text', 0) self.clist = ['lesser than', 'equal to', 'greater than'] self.clist_trans = [_('lesser than'), _('equal to'), _('greater than')] for name in self.clist_trans: store.append(row=[name]) self.set_active(default) self.show()
[docs] def get_text(self): active = self.get_active() if active < 0: return 'equal to' return self.clist[active]
[docs] def set_text(self, val): if val in self.clist: self.set_active(self.clist.index(val)) else: self.set_active(self.clist.index('equal to')) #------------------------------------------------------------------------- # # MyPlaces - AutoCombo text entry with list of places attached. Provides # a standard interface # #-------------------------------------------------------------------------
[docs]class MyPlaces(Gtk.Entry): def __init__(self, places): GObject.GObject.__init__(self) fill_entry(self, places) self.show() #------------------------------------------------------------------------- # # MyID - Person/GRAMPS ID selection box with a standard interface # #-------------------------------------------------------------------------
[docs]class MyID(Gtk.Box): _invalid_id_txt = _('Not a valid ID') _empty_id_txt = _invalid_id_txt obj_name = { 'Person' : _('Person'), 'Family' : _('Family'), 'Event' : _('Event'), 'Place' : _('Place'), 'Source' : _('Source'), 'Media' : _('Media'), 'Repository' : _('Repository'), 'Note' : _('Note'), 'Citation' : _('Citation'), } def __init__(self, dbstate, uistate, track, namespace='Person'): GObject.GObject.__init__(self) self.set_orientation(Gtk.Orientation.HORIZONTAL) self.set_homogeneous(False) self.set_spacing(6) self.dbstate = dbstate self.db = dbstate.db self.uistate = uistate self.track = track self.namespace = namespace self.entry = Gtk.Entry() self.entry.show() self.button = Gtk.Button() self.button.set_label(_('Select...')) self.button.connect('clicked', self.button_press) self.button.show() self.pack_start(self.entry, True, True, 0) self.add(self.button) self.button.set_tooltip_text(_('Select %s from a list') % self.obj_name[namespace]) self.show() self.set_text('')
[docs] def button_press(self, obj): obj_class = self.namespace selector = SelectorFactory(obj_class) inst = selector(self.dbstate, self.uistate, self.track) val = inst.run() if val is None: self.set_text('') else: self.set_text(val.get_gramps_id())
[docs] def get_text(self): return cuni(self.entry.get_text())
[docs] def name_from_gramps_id(self, gramps_id): if self.namespace == 'Person': person = self.db.get_person_from_gramps_id(gramps_id) name = _nd.display_name(person.get_primary_name()) elif self.namespace == 'Family': family = self.db.get_family_from_gramps_id(gramps_id) name = family_name(family, self.db) elif self.namespace == 'Event': event = self.db.get_event_from_gramps_id(gramps_id) name = str(event.get_type) elif self.namespace == 'Place': place = self.db.get_place_from_gramps_id(gramps_id) name = place.get_title() elif self.namespace == 'Source': source = self.db.get_source_from_gramps_id(gramps_id) name = source.get_title() elif self.namespace == 'Citation': citation = self.db.get_citation_from_gramps_id(gramps_id) name = citation.get_page() elif self.namespace == 'Media': obj = self.db.get_object_from_gramps_id(gramps_id) name = obj.get_path() elif self.namespace == 'Repository': repo = self.db.get_repository_from_gramps_id(gramps_id) name = repo.get_name() elif self.namespace == 'Note': note = self.db.get_note_from_gramps_id(gramps_id) name = note.get() return name
[docs] def set_text(self, val): if not val: self.entry.set_tooltip_text(self._empty_id_txt) else: try: name = self.name_from_gramps_id(val) self.entry.set_tooltip_text(name) except AttributeError: self.entry.set_tooltip_text(self._invalid_id_txt) self.entry.set_text(val) #------------------------------------------------------------------------- # # MySource - select ID of sources with a standard interface # #-------------------------------------------------------------------------
[docs]class MySource(MyID): _empty_id_txt = _('Give or select a source ID, leave empty to find objects' ' with no source.') def __init__(self, dbstate, uistate, track): MyID.__init__(self, dbstate, uistate, track, namespace='Source') self.entry.set_tooltip_text(self._empty_id_txt) #------------------------------------------------------------------------- # # MySelect # #-------------------------------------------------------------------------
[docs]class MySelect(Gtk.ComboBox): def __init__(self, type_class, additional): #we need to inherit and have an combobox with an entry Gtk.ComboBox.__init__(self, has_entry=True) self.type_class = type_class self.sel = StandardCustomSelector(type_class._I2SMAP, self, type_class._CUSTOM, type_class._DEFAULT, additional, type_class._MENU) self.show()
[docs] def get_text(self): return self.type_class(self.sel.get_values()).xml_str()
[docs] def set_text(self, val): tc = self.type_class() tc.set_from_xml_str(val) self.sel.set_values((int(tc), str(tc))) #------------------------------------------------------------------------- # # MyEntry # #-------------------------------------------------------------------------
[docs]class MyEntry(Gtk.Entry): def __init__(self): GObject.GObject.__init__(self) self.show() #------------------------------------------------------------------------- # # EditRule # #-------------------------------------------------------------------------
[docs]class EditRule(ManagedWindow): def __init__(self, namespace, dbstate, uistate, track, filterdb, val, label, update, filter_name): ManagedWindow.__init__(self, uistate, track, EditRule) self.width_key = "interface.edit-rule-width" self.height_key = "interface.edit-rule-height" self.namespace = namespace self.dbstate = dbstate self.db = dbstate.db self.filterdb = filterdb self.update_rule = update self.filter_name = filter_name self.active_rule = val self.define_glade('rule_editor', RULE_GLADE) self.set_window(self.get_widget('rule_editor'), self.get_widget('rule_editor_title'),label) self.window.hide() self.valuebox = self.get_widget('valuebox') self.rname = self.get_widget('ruletree') self.rule_name = self.get_widget('rulename') self.notebook = Gtk.Notebook() self.notebook.set_show_tabs(0) self.notebook.set_show_border(0) self.notebook.show() self.valuebox.add(self.notebook) self.page_num = 0 self.page = [] self.class2page = {} the_map = {} if self.namespace == 'Person': class_list = rules.person.editor_rule_list elif self.namespace == 'Family': class_list = rules.family.editor_rule_list elif self.namespace == 'Event': class_list = rules.event.editor_rule_list elif self.namespace == 'Source': class_list = rules.source.editor_rule_list elif self.namespace == 'Citation': class_list = rules.citation.editor_rule_list elif self.namespace == 'Place': class_list = rules.place.editor_rule_list elif self.namespace == 'Media': class_list = rules.media.editor_rule_list elif self.namespace == 'Repository': class_list = rules.repository.editor_rule_list elif self.namespace == 'Note': class_list = rules.note.editor_rule_list for class_obj in class_list: arglist = class_obj.labels vallist = [] tlist = [] pos = 0 l2 = Gtk.Label(label=class_obj.name) l2.set_alignment(0, 0.5) l2.show() c = Gtk.TreeView() #c.set_data('d', pos) c.show() the_map[class_obj] = c # Only add a table with parameters if there are any parameters if arglist: table = Gtk.Table(n_rows=3, n_columns=len(arglist)) else: table = Gtk.Table(n_rows=1, n_columns=1) table.set_border_width(6) table.set_col_spacings(6) table.set_row_spacings(6) table.show() for v in arglist: l = Gtk.Label(label=v) l.set_alignment(1, 0.5) l.show() if v == _('Place:'): t = MyPlaces([]) elif v in [_('Reference count:'), _('Number of instances:') ]: t = MyInteger(0, 999) elif v == _('Reference count must be:'): t = MyLesserEqualGreater() elif v == _('Number must be:'): t = MyLesserEqualGreater(2) elif v == _('Number of generations:'): t = MyInteger(1, 32) elif v == _('ID:'): t = MyID(self.dbstate, self.uistate, self.track, self.namespace) elif v == _('Source ID:'): t = MySource(self.dbstate, self.uistate, self.track) elif v == _('Filter name:'): t = MyFilters(self.filterdb.get_filters(self.namespace), self.filter_name) # filters of another namespace, name may be same as caller! elif v == _('Person filter name:'): t = MyFilters(self.filterdb.get_filters('Person')) elif v == _('Event filter name:'): t = MyFilters(self.filterdb.get_filters('Event')) elif v == _('Source filter name:'): t = MyFilters(self.filterdb.get_filters('Source')) elif v == _('Repository filter name:'): t = MyFilters(self.filterdb.get_filters('Repository')) elif v == _('Place filter name:'): t = MyFilters(self.filterdb.get_filters('Place')) elif v in _name2typeclass: additional = None if v in (_('Event type:'), _('Personal event:'), _('Family event:')): additional = self.db.get_event_types() elif v == _('Personal attribute:'): additional = self.db.get_person_attribute_types() elif v == _('Family attribute:'): additional = self.db.get_family_attribute_types() elif v == _('Event attribute:'): additional = self.db.get_event_attribute_types() elif v == _('Media attribute:'): additional = self.db.get_media_attribute_types() elif v == _('Relationship type:'): additional = self.db.get_family_relation_types() elif v == _('Note type:'): additional = self.db.get_note_types() elif v == _('Name type:'): additional = self.db.get_name_types() elif v == _('Surname origin type:'): additional = self.db.get_origin_types() t = MySelect(_name2typeclass[v], additional) elif v == _('Inclusive:'): t = MyBoolean(_('Include original person')) elif v == _('Case sensitive:'): t = MyBoolean(_('Use exact case of letters')) elif v == _('Regular-Expression matching:'): t = MyBoolean(_('Use regular expression')) elif v == _('Include Family events:'): t = MyBoolean(_('Also family events where person is ' 'wife/husband')) elif v == _('Primary Role:'): t = MyBoolean(_('Only include primary participants')) elif v == _('Tag:'): taglist = [''] taglist = taglist + [tag.get_name() for tag in dbstate.db.iter_tags()] t = MyList(taglist, taglist) elif v == _('Confidence level:'): t = MyList(list(map(str, list(range(5)))), [_(conf_strings[i]) for i in range(5)]) elif v == _('Date:'): t = DateEntry(self.uistate, self.track) elif v == _('Day of Week:'): long_days = displayer.long_days days_of_week = long_days[2:] + long_days[1:2] t = MyList(map(str, range(7)), days_of_week) else: t = MyEntry() tlist.append(t) table.attach(l, 1, 2, pos, pos+1, Gtk.AttachOptions.FILL, 0, 5, 5) table.attach(t, 2, 3, pos, pos+1, Gtk.AttachOptions.EXPAND|Gtk.AttachOptions.FILL, 0, 5, 5) pos += 1 use_regex = None if class_obj.allow_regex: use_regex = Gtk.CheckButton(label=_('Use regular expressions')) tip = _('Interpret the contents of string fields as regular ' 'expressions.\n' 'A decimal point will match any character. ' 'A question mark will match zero or one occurences ' 'of the previous character or group. ' 'An asterisk will match zero or more occurences. ' 'A plus sign will match one or more occurences. ' 'Use parentheses to group expressions. ' 'Specify alternatives using a vertical bar. ' 'A caret will match the start of a line. ' 'A dollar sign will match the end of a line.') use_regex.set_tooltip_text(tip) table.attach(use_regex, 2, 3, pos, pos+1, Gtk.AttachOptions.FILL, 0, 5, 5) self.page.append((class_obj, vallist, tlist, use_regex)) # put the table into a scrollable area: scrolled_win = Gtk.ScrolledWindow() scrolled_win.add_with_viewport(table) scrolled_win.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC) scrolled_win.show() self.notebook.append_page(scrolled_win, Gtk.Label(label=class_obj.name)) self.class2page[class_obj] = self.page_num self.page_num = self.page_num + 1 self.page_num = 0 self.store = Gtk.TreeStore(GObject.TYPE_STRING, GObject.TYPE_PYOBJECT) self.selection = self.rname.get_selection() col = Gtk.TreeViewColumn(_('Rule Name'), Gtk.CellRendererText(), text=0) self.rname.append_column(col) self.rname.set_model(self.store) prev = None last_top = None top_level = {} top_node = {} # # If editing a rule, get the name so that we can select it later # sel_node = None if self.active_rule is not None: self.sel_class = self.active_rule.__class__ else: self.sel_class = None keys = sorted(the_map, key=lambda x: x.name, reverse=True) catlist = sorted(set(class_obj.category for class_obj in keys)) for category in catlist: top_node[category] = self.store.insert_after(None, last_top) top_level[category] = [] last_top = top_node[category] self.store.set(last_top, 0, category, 1, '') for class_obj in keys: category = class_obj.category top_level[category].append(class_obj.name) node = self.store.insert_after(top_node[category], prev) self.store.set(node, 0, class_obj.name, 1, class_obj) # if this is an edit rule, save the node if class_obj == self.sel_class: sel_node = (top_node[category], node) if sel_node: self.select_iter(sel_node) page = self.class2page[self.active_rule.__class__] self.notebook.set_current_page(page) self.display_values(self.active_rule.__class__) (class_obj, vallist, tlist, use_regex) = self.page[page] r = list(self.active_rule.values()) for i in range(0, min(len(tlist), len(r))): tlist[i].set_text(r[i]) if class_obj.allow_regex: use_regex.set_active(self.active_rule.use_regex) self.selection.connect('changed', self.on_node_selected) self.rname.connect('button-press-event', self._button_press) self.rname.connect('key-press-event', self._key_press) self.get_widget('rule_editor_ok').connect('clicked', self.rule_ok) self.get_widget('rule_editor_cancel').connect('clicked', self.close_window) self.get_widget('rule_editor_help').connect('clicked', self.on_help_clicked) self._set_size() self.show()
[docs] def select_iter(self, data): """ Workaround to get self.selection to move to iter row. self.selection.select_iter(iter) did not work, so we first select the top_node iter, expand it, and then select_iter. """ top_node, iter = data self.selection.select_iter(top_node) self.expand_collapse() self.selection.select_iter(iter)
def _button_press(self, obj, event): if event.type == Gdk.EventType._2BUTTON_PRESS and event.button == 1: return self.expand_collapse() def _key_press(self, obj, event): if event.keyval in (Gdk.KEY_Return, Gdk.KEY_KP_Enter): return self.expand_collapse() return False
[docs] def expand_collapse(self): """ Expand or collapse the selected parent name node. Return True if change done, False otherwise """ store, paths = self.selection.get_selected_rows() if paths and len(paths[0].get_indices()) == 1 : if self.rname.row_expanded(paths[0]): self.rname.collapse_row(paths[0]) else: self.rname.expand_row(paths[0], 0) return True return False
[docs] def on_help_clicked(self, obj): """ Display the relevant portion of Gramps manual. """ display_help()
[docs] def close_window(self, obj): self.close()
[docs] def on_node_selected(self, obj): """ Update the informational display on the right hand side of the dialog box with the description of the selected report. """ store, node = self.selection.get_selected() if node: try: class_obj = store.get_value(node, 1) self.display_values(class_obj) except: self.valuebox.set_sensitive(0) self.rule_name.set_text(_('No rule selected')) self.get_widget('description').set_text('')
[docs] def display_values(self, class_obj): page = self.class2page[class_obj] self.notebook.set_current_page(page) self.valuebox.set_sensitive(1) self.rule_name.set_text(class_obj.name) self.get_widget('description').set_text(class_obj.description)
[docs] def rule_ok(self, obj): if self.rule_name.get_text() == _('No rule selected'): return try: page = self.notebook.get_current_page() (class_obj, vallist, tlist, use_regex) = self.page[page] value_list = [cuni(sclass.get_text()) for sclass in tlist] if class_obj.allow_regex: new_rule = class_obj(value_list, use_regex.get_active()) else: new_rule = class_obj(value_list) self.update_rule(self.active_rule, new_rule) self.close() except KeyError: pass #------------------------------------------------------------------------- # # EditFilter # #-------------------------------------------------------------------------
[docs]class EditFilter(ManagedWindow): def __init__(self, namespace, dbstate, uistate, track, gfilter, filterdb, update=None, selection_callback=None): ManagedWindow.__init__(self, uistate, track, self) self.width_key = "interface.edit-filter-width" self.height_key = "interface.edit-filter-height" self.namespace = namespace self.update = update self.dbstate = dbstate self.db = dbstate.db self.filter = gfilter self.filterdb = filterdb self.selection_callback = selection_callback self.define_glade('define_filter', RULE_GLADE) self.set_window( self.get_widget('define_filter'), self.get_widget('definition_title'), _('Define filter')) self.rlist = ListModel( self.get_widget('rule_list'), [(_('Name'),-1,150),(_('Values'),-1,150)], self.select_row, self.on_edit_clicked) self.fname = self.get_widget('filter_name') self.logical = self.get_widget('rule_apply') self.logical_not = self.get_widget('logical_not') self.comment = self.get_widget('comment') self.ok_btn = self.get_widget('definition_ok') self.edit_btn = self.get_widget('definition_edit') self.del_btn = self.get_widget('definition_delete') self.add_btn = self.get_widget('definition_add') self.ok_btn.connect('clicked', self.on_ok_clicked) self.edit_btn.connect('clicked', self.on_edit_clicked) self.del_btn.connect('clicked', self.on_delete_clicked) self.add_btn.connect('clicked', self.on_add_clicked) self.get_widget('definition_help').connect('clicked', self.on_help_clicked) self.get_widget('definition_cancel').connect('clicked', self.close_window) self.fname.connect('changed', self.filter_name_changed) op = self.filter.get_logical_op() # WARNING: must be listed in this order: self.logical.set_active(["and", "or", "one", "sequence"].index(op)) self.logical_not.set_active(self.filter.get_invert()) if self.filter.get_name(): self.fname.set_text(self.filter.get_name()) self.comment.set_text(self.filter.get_comment()) self.draw_rules() self._set_size() self.show()
[docs] def on_help_clicked(self, obj): """Display the relevant portion of Gramps manual""" display_help(webpage=WIKI_HELP_PAGE)
[docs] def close_window(self, obj): self.close()
[docs] def filter_name_changed(self, obj): name = cuni(self.fname.get_text()) # Make sure that the name is not empty # and not in the list of existing filters (excluding this one) names = [filt.get_name() for filt in self.filterdb.get_filters(self.namespace) if filt != self.filter] self.ok_btn.set_sensitive((len(name) != 0) and (name not in names))
[docs] def select_row(self, obj): store, node = self.rlist.get_selected() if node: self.edit_btn.set_sensitive(True) self.del_btn.set_sensitive(True) else: self.edit_btn.set_sensitive(False) self.del_btn.set_sensitive(False)
[docs] def draw_rules(self): self.rlist.clear() for r in self.filter.get_rules(): self.rlist.add([r.name,r.display_values()],r)
[docs] def on_ok_clicked(self, obj): n = cuni(self.fname.get_text()).strip() if n == '': return if n != self.filter.get_name(): self.uistate.emit('filter-name-changed', (self.namespace, cuni(self.filter.get_name()), n)) self.filter.set_name(n) self.filter.set_comment(cuni(self.comment.get_text()).strip()) for f in self.filterdb.get_filters(self.namespace)[:]: if n == f.get_name(): self.filterdb.get_filters(self.namespace).remove(f) break val = self.logical.get_active() # WARNING: must be listed in this order: op = ('and' if val == 0 else 'or' if val == 1 else 'one' if val == 2 else 'sequence') self.logical.set_active(val) self.filter.set_logical_op(op) self.filter.set_invert(self.logical_not.get_active()) self.filterdb.add(self.namespace,self.filter) if self.update: self.update() if self.selection_callback: self.selection_callback(self.filterdb, self.filter.get_name()) self.close()
[docs] def on_add_clicked(self, obj): try: EditRule(self.namespace, self.dbstate, self.uistate, self.track, self.filterdb, None, _('Add Rule'), self.update_rule, self.filter.get_name()) except WindowActiveError: pass
[docs] def on_edit_clicked(self, obj): store, node = self.rlist.get_selected() if node: d = self.rlist.get_object(node) try: EditRule(self.namespace, self.dbstate, self.uistate, self.track, self.filterdb, d, _('Edit Rule'), self.update_rule, self.filter.get_name()) except WindowActiveError: pass
[docs] def update_rule(self, old_rule, new_rule): if old_rule is not None: self.filter.delete_rule(old_rule) self.filter.add_rule(new_rule) self.draw_rules()
[docs] def on_delete_clicked(self, obj): store, node = self.rlist.get_selected() if node: gfilter = self.rlist.get_object(node) self.filter.delete_rule(gfilter) self.draw_rules() #------------------------------------------------------------------------- # # ShowResults # #-------------------------------------------------------------------------
[docs]class ShowResults(ManagedWindow): def __init__(self, db, uistate, track, handle_list, filtname, namespace): ManagedWindow.__init__(self, uistate, track, self) self.db = db self.filtname = filtname self.namespace = namespace self.define_glade('test', RULE_GLADE,) self.set_window( self.get_widget('test'), self.get_widget('test_title'), _('Filter Test')) render = Gtk.CellRendererText() tree = self.get_widget('list') model = Gtk.ListStore(GObject.TYPE_STRING, GObject.TYPE_STRING) tree.set_model(model) column_n = Gtk.TreeViewColumn(_('Name'), render, text=0) tree.append_column(column_n) column_n = Gtk.TreeViewColumn(_('ID'), render, text=1) tree.append_column(column_n) self.get_widget('test_close').connect('clicked', self.close) new_list = sorted( (self.sort_val_from_handle(h) for h in handle_list), key=lambda x: glocale.sort_key(x[0]) ) for s_, handle in new_list: name, gid = self.get_name_id(handle) model.append(row=[name, gid]) self.show()
[docs] def get_name_id(self, handle): if self.namespace == 'Person': person = self.db.get_person_from_handle(handle) name = _nd.sorted(person) gid = person.get_gramps_id() elif self.namespace == 'Family': family = self.db.get_family_from_handle(handle) name = family_name(family, self.db) gid = family.get_gramps_id() elif self.namespace == 'Event': event = self.db.get_event_from_handle(handle) name = event.get_description() gid = event.get_gramps_id() elif self.namespace == 'Source': source = self.db.get_source_from_handle(handle) name = source.get_title() gid = source.get_gramps_id() elif self.namespace == 'Citation': citation = self.db.get_citation_from_handle(handle) name = citation.get_page() gid = citation.get_gramps_id() elif self.namespace == 'Place': place = self.db.get_place_from_handle(handle) name = place.get_title() gid = place.get_gramps_id() elif self.namespace == 'Media': obj = self.db.get_object_from_handle(handle) name = obj.get_description() gid = obj.get_gramps_id() elif self.namespace == 'Repository': repo = self.db.get_repository_from_handle(handle) name = repo.get_name() gid = repo.get_gramps_id() elif self.namespace == 'Note': note = self.db.get_note_from_handle(handle) name = note.get().replace('\n', ' ') if len(name) > 80: name = name[:80]+"..." gid = note.get_gramps_id() return (name, gid)
[docs] def sort_val_from_handle(self, handle): if self.namespace == 'Person': name = self.db.get_person_from_handle(handle).get_primary_name() sortname = _nd.sort_string(name) elif self.namespace == 'Family': sortname = family_name( self.db.get_family_from_handle(handle),self.db) elif self.namespace == 'Event': sortname = self.db.get_event_from_handle(handle).get_description() elif self.namespace == 'Source': sortname = self.db.get_source_from_handle(handle).get_title() elif self.namespace == 'Citation': sortname = self.db.get_citation_from_handle(handle).get_page() elif self.namespace == 'Place': sortname = self.db.get_place_from_handle(handle).get_title() elif self.namespace == 'Media': sortname = self.db.get_object_from_handle(handle).get_description() elif self.namespace == 'Repository': sortname = self.db.get_repository_from_handle(handle).get_name() elif self.namespace == 'Note': gid = self.db.get_note_from_handle(handle).get_gramps_id() sortname = gid return (sortname, handle) #------------------------------------------------------------------------- # # FilterEditor # #-------------------------------------------------------------------------
[docs]class FilterEditor(ManagedWindow): def __init__(self, namespace, filterdb, dbstate, uistate): ManagedWindow.__init__(self, uistate, [], FilterEditor) self.dbstate = dbstate self.db = dbstate.db self.filterdb = FilterList(filterdb) self.filterdb.load() self.width_key = "interface.filter-editor-width" self.height_key = "interface.filter-editor-height" self.namespace = namespace self.define_glade('filter_list', RULE_GLADE) self.filter_list = self.get_widget('filters') self.edit = self.get_widget('filter_list_edit') self.clone = self.get_widget('filter_list_clone') self.delete = self.get_widget('filter_list_delete') self.test = self.get_widget('filter_list_test') self.edit.set_sensitive(False) self.clone.set_sensitive(False) self.delete.set_sensitive(False) self.test.set_sensitive(False) self.set_window(self.get_widget('filter_list'), self.get_widget('filter_list_title'), _TITLES[self.namespace]) self.edit.connect('clicked', self.edit_filter) self.clone.connect('clicked', self.clone_filter) self.test.connect('clicked', self.test_clicked) self.delete.connect('clicked', self.delete_filter) self.connect_button('filter_list_help', self.help_clicked) self.connect_button('filter_list_close', self.close) self.connect_button('filter_list_add', self.add_new_filter) self.uistate.connect('filter-name-changed', self.clean_after_rename) self.clist = ListModel( self.filter_list, [(_('Filter'), 0, 150), (_('Comment'), 1, 150)], self.filter_select_row, self.edit_filter) self.draw_filters() self._set_size() self.show()
[docs] def build_menu_names(self, obj): return (_("Custom Filter Editor"), _("Custom Filter Editor"))
[docs] def help_clicked(self, obj): """Display the relevant portion of Gramps manual""" display_help()
[docs] def filter_select_row(self, obj): store, node = self.clist.get_selected() if node: self.edit.set_sensitive(True) self.clone.set_sensitive(True) self.delete.set_sensitive(True) self.test.set_sensitive(True) else: self.edit.set_sensitive(False) self.clone.set_sensitive(False) self.delete.set_sensitive(False) self.test.set_sensitive(False)
[docs] def close(self, *obj): self.filterdb.save() reload_custom_filters() #reload_system_filters() self.uistate.emit('filters-changed', (self.namespace,)) ManagedWindow.close(self, *obj)
[docs] def draw_filters(self): self.clist.clear() for f in self.filterdb.get_filters(self.namespace): self.clist.add([f.get_name(), f.get_comment()], f)
[docs] def add_new_filter(self, obj): the_filter = GenericFilterFactory(self.namespace)() EditFilter(self.namespace, self.dbstate, self.uistate, self.track, the_filter, self.filterdb, self.draw_filters)
[docs] def edit_filter(self, obj): store, node = self.clist.get_selected() if node: gfilter = self.clist.get_object(node) EditFilter(self.namespace, self.dbstate, self.uistate, self.track, gfilter, self.filterdb, self.draw_filters)
[docs] def clone_filter(self, obj): store, node = self.clist.get_selected() if node: old_filter = self.clist.get_object(node) the_filter = GenericFilterFactory(self.namespace)(old_filter) the_filter.set_name('') EditFilter(self.namespace, self.dbstate, self.uistate, self.track, the_filter, self.filterdb, self.draw_filters)
[docs] def test_clicked(self, obj): store, node = self.clist.get_selected() if node: filt = self.clist.get_object(node) handle_list = filt.apply(self.db, self.get_all_handles()) ShowResults(self.db, self.uistate, self.track, handle_list, filt.get_name(),self.namespace)
[docs] def delete_filter(self, obj): store, node = self.clist.get_selected() if node: gfilter = self.clist.get_object(node) name = gfilter.get_name() if self.check_recursive_filters(self.namespace, name): QuestionDialog( _('Delete Filter?'), _('This filter is currently being used ' 'as the base for other filters. Deleting' 'this filter will result in removing all ' 'other filters that depend on it.'), _('Delete Filter'), self._do_delete_selected_filter, self.window) else: self._do_delete_selected_filter()
def _do_delete_selected_filter(self): store, node = self.clist.get_selected() if node: gfilter = self.clist.get_object(node) self._do_delete_filter(self.namespace, gfilter) self.draw_filters() def _do_delete_filter(self, space, gfilter): # Find everything we need to remove filter_set = set() self._find_dependent_filters(space, gfilter, filter_set) # Remove what we found filters = self.filterdb.get_filters(space) list(map(filters.remove, filter_set)) def _find_dependent_filters(self, space, gfilter, filter_set): """ This method recursively calls itself to find all filters that depend on the given filter, either directly through one of the rules, or through the chain of dependencies. The filter_set is amended with the found filters. """ filters = self.filterdb.get_filters(space) name = gfilter.get_name() for the_filter in filters: if the_filter.get_name() == name: continue for rule in the_filter.get_rules(): values = list(rule.values()) if issubclass(rule.__class__, MatchesFilterBase) \ and (name in values): self._find_dependent_filters(space, the_filter, filter_set) break # Add itself to the filter_set filter_set.add(gfilter)
[docs] def get_all_handles(self): # Why use iter for some and get for others? if self.namespace == 'Person': return self.db.iter_person_handles() elif self.namespace == 'Family': return self.db.iter_family_handles() elif self.namespace == 'Event': return self.db.get_event_handles() elif self.namespace == 'Source': return self.db.get_source_handles() elif self.namespace == 'Citation': return self.db.get_citation_handles() elif self.namespace == 'Place': return self.db.iter_place_handles() elif self.namespace == 'Media': return self.db.get_media_object_handles() elif self.namespace == 'Repository': return self.db.get_repository_handles() elif self.namespace == 'Note': return self.db.get_note_handles()
[docs] def clean_after_rename(self, space, old_name, new_name): if old_name == "": return if old_name == new_name: return for the_filter in self.filterdb.get_filters(space): for rule in the_filter.get_rules(): values = list(rule.values()) if issubclass(rule.__class__, MatchesFilterBase) \ and (old_name in values): ind = values.index(old_name) values[ind] = new_name
[docs] def check_recursive_filters(self, space, name): for the_filter in self.filterdb.get_filters(space): for rule in the_filter.get_rules(): values = list(rule.values()) if issubclass(rule.__class__, MatchesFilterBase) \ and (name in values): return True return False