# Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2000-2007 Donald N. Allingham # Copyright (C) 2008 Gary Burton # Copyright (C) 2009 Nick Hall # Copyright (C) 2010 Benny Malengier # # 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 021111307 USA # # $Id$ """ Provide the base for a list person view. """ #------------------------------------------------------------------------- # # GTK/Gnome modules # #------------------------------------------------------------------------- import gtk #------------------------------------------------------------------------- # # set up logging # #------------------------------------------------------------------------- import logging _LOG = logging.getLogger(".gui.personview") #------------------------------------------------------------------------- # # gramps modules # #------------------------------------------------------------------------- import gen.lib from gui.views.listview import ListView import Utils from gen.display.name import displayer as name_displayer from QuestionDialog import ErrorDialog, QuestionDialog import Errors import Bookmarks import config from DdTargets import DdTargets from gui.editors import EditPerson from Filters.SideBar import PersonSidebarFilter from gen.plug import CATEGORY_QR_PERSON #------------------------------------------------------------------------- # # Python modules # #------------------------------------------------------------------------- from gen.ggettext import sgettext as _ #------------------------------------------------------------------------- # # PersonView # #------------------------------------------------------------------------- class BasePersonView(ListView): """ Base view for PersonView listviews ListView, a treeview """ COL_NAME = 0 COL_ID = 1 COL_GEN = 2 COL_BDAT = 3 COL_BPLAC = 4 COL_DDAT = 5 COL_DPLAC = 6 COL_SPOUSE = 7 COL_TAGS = 8 COL_CHAN = 9 #name of the columns COLUMN_NAMES = [ _('Name'), _('ID'), _('Gender'), _('Birth Date'), _('Birth Place'), _('Death Date'), _('Death Place'), _('Spouse'), _('Tags'), _('Last Changed'), ] # columns that contain markup MARKUP_COLS = [COL_BDAT, COL_BPLAC, COL_DDAT, COL_DPLAC] # default setting with visible columns, order of the col, and their size CONFIGSETTINGS = ( ('columns.visible', [COL_NAME, COL_ID, COL_GEN, COL_BDAT, COL_DDAT]), ('columns.rank', [COL_NAME, COL_ID, COL_GEN, COL_BDAT, COL_BPLAC, COL_DDAT, COL_DPLAC, COL_SPOUSE, COL_TAGS, COL_CHAN]), ('columns.size', [250, 75, 75, 100, 175, 100, 175, 100, 100, 100]) ) ADD_MSG = _("Add a new person") EDIT_MSG = _("Edit the selected person") DEL_MSG = _("Delete the selected person") FILTER_TYPE = "Person" QR_CATEGORY = CATEGORY_QR_PERSON def __init__(self, dbstate, uistate, title, model, nav_group=0): """ Create the Person View """ signal_map = { 'person-add' : self.row_add, 'person-update' : self.row_update, 'person-delete' : self.row_delete, 'person-rebuild' : self.object_build, 'person-groupname-rebuild' : self.object_build, 'tag-update' : self.tag_updated } ListView.__init__( self, title, dbstate, uistate, BasePersonView.COLUMN_NAMES, len(BasePersonView.COLUMN_NAMES), model, signal_map, dbstate.db.get_bookmarks(), Bookmarks.PersonBookmarks, nav_group, multiple=True, filter_class=PersonSidebarFilter, markup=BasePersonView.MARKUP_COLS) self.func_list.update({ 'J' : self.jump, 'BackSpace' : self.key_delete, }) config.connect("interface.filter", self.filter_toggle) uistate.connect('nameformat-changed', self.build_tree) def navigation_type(self): """ Return the navigation type of the view. """ return 'Person' def get_bookmarks(self): """ Return the bookmark object """ return self.dbstate.db.get_bookmarks() def drag_info(self): """ Specify the drag type for a single selection """ return DdTargets.PERSON_LINK def exact_search(self): """ Returns a tuple indicating columns requiring an exact search 'female' contains the string 'male' so we need an exact search """ return (BasePersonView.COL_GEN,) def get_stock(self): """ Use the grampsperson stock icon """ return 'gramps-person' def ui_definition(self): """ Defines the UI string for UIManager """ return ''' ''' def get_handle_from_gramps_id(self, gid): """ Return the handle of the person having the given Gramps ID. """ obj = self.dbstate.db.get_person_from_gramps_id(gid) if obj: return obj.get_handle() else: return None def add(self, obj): """ Add a new person to the database. """ person = gen.lib.Person() try: EditPerson(self.dbstate, self.uistate, [], gen.lib.Person()) except Errors.WindowActiveError: pass def edit(self, obj): """ Edit an existing person in the database. """ for handle in self.selected_handles(): person = self.dbstate.db.get_person_from_handle(handle) try: EditPerson(self.dbstate, self.uistate, [], person) except Errors.WindowActiveError: pass def remove(self, obj): """ Remove a person from the database. """ for sel in self.selected_handles(): person = self.dbstate.db.get_person_from_handle(sel) self.active_person = person name = name_displayer.display(person) msg = _('Deleting the person will remove the person ' 'from the database.') msg = "%s %s" % (msg, Utils.data_recover_msg) QuestionDialog(_('Delete %s?') % name, msg, _('_Delete Person'), self.delete_person_response) def delete_person_response(self): """ Deletes the person from the database. """ # set the busy cursor, so the user knows that we are working self.uistate.set_busy_cursor(True) # create the transaction trans = self.dbstate.db.transaction_begin() # create name to save person = self.active_person active_name = _("Delete Person (%s)") % name_displayer.display(person) # delete the person from the database # Above will emit person-delete, which removes the person via # callback to the model, so row delete is signaled self.dbstate.db.delete_person_from_database(person, trans) # commit the transaction self.dbstate.db.transaction_commit(trans, active_name) self.uistate.set_busy_cursor(False) def dummy_report(self, obj): """ For the xml UI definition of popup to work, the submenu Quick Report must have an entry in the xml As this submenu will be dynamically built, we offer a dummy action """ pass def define_actions(self): """ Required define_actions function for PageView. Builds the action group information required. We extend beyond the normal here, since we want to have more than one action group for the PersonView. Most PageViews really won't care about this. Special action groups for Forward and Back are created to allow the handling of navigation buttons. Forward and Back allow the user to advance or retreat throughout the history, and we want to have these be able to toggle these when you are at the end of the history or at the beginning of the history. """ ListView.define_actions(self) self.all_action = gtk.ActionGroup(self.title + "/PersonAll") self.edit_action = gtk.ActionGroup(self.title + "/PersonEdit") self.all_action.add_actions([ ('FilterEdit', None, _('Person Filter Editor'), None, None, self.filter_editor), ('Edit', gtk.STOCK_EDIT, _("action|_Edit..."), "Return", _("Edit the selected person"), self.edit), ('QuickReport', None, _("Quick View"), None, None, None), ('WebConnect', None, _("Web Connection"), None, None, None), ('Dummy', None, ' ', None, None, self.dummy_report), ]) self.edit_action.add_actions( [ ('Add', gtk.STOCK_ADD, _("_Add..."), "Insert", _("Add a new person"), self.add), ('Remove', gtk.STOCK_REMOVE, _("_Remove"), "Delete", _("Remove the Selected Person"), self.remove), ('Merge', 'gramps-merge', _('_Merge...'), None, None, self.merge), ('ExportTab', None, _('Export View...'), None, None, self.export), ]) self._add_action_group(self.edit_action) self._add_action_group(self.all_action) def enable_action_group(self, obj): """ Turns on the visibility of the View's action group. """ ListView.enable_action_group(self, obj) self.all_action.set_visible(True) self.edit_action.set_visible(True) self.edit_action.set_sensitive(not self.dbstate.db.readonly) def disable_action_group(self): """ Turns off the visibility of the View's action group. """ ListView.disable_action_group(self) self.all_action.set_visible(False) self.edit_action.set_visible(False) def set_active(self): """ Called when the page is displayed. """ ListView.set_active(self) self.uistate.viewmanager.tags.tag_enable() def set_inactive(self): """ Called when the page is no longer displayed. """ ListView.set_inactive(self) self.uistate.viewmanager.tags.tag_disable() def merge(self, obj): """ Merge the selected people. """ mlist = self.selected_handles() if len(mlist) != 2: ErrorDialog( _("Cannot merge people"), _("Exactly two people must be selected to perform a merge. " "A second person can be selected by holding down the " "control key while clicking on the desired person.")) else: import Merge Merge.MergePeople(self.dbstate, self.uistate, mlist[0], mlist[1]) def tag_updated(self, handle_list): """ Update tagged rows when a tag color changes. """ all_links = set([]) for tag_handle in handle_list: links = set([link[1] for link in self.dbstate.db.find_backlink_handles(tag_handle, include_classes='Person')]) all_links = all_links.union(links) self.row_update(list(all_links)) def add_tag(self, transaction, person_handle, tag_handle): """ Add the given tag to the given person. """ person = self.dbstate.db.get_person_from_handle(person_handle) person.add_tag(tag_handle) self.dbstate.db.commit_person(person, transaction)