This is a major patch by Michael Nauta. It means all writes happen immediately to bsddb, and the bsddb rollback is used on a crash. Transaction is no longer used to store changes and do them on commit. Undo database is set on end. At the same time with statement is used throughout for transactions At the same time, the original bug in merge code should be fixed Still some issues, that will be ironed out svn: r16523
451 lines
15 KiB
Python
451 lines
15 KiB
Python
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2007 Donald N. Allingham
|
|
# Copyright (C) 2008 Gary Burton
|
|
# Copyright (C) 2009-2010 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, pdata, 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, pdata, 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({
|
|
'<CONTROL>J' : self.jump,
|
|
'<CONTROL>BackSpace' : self.key_delete,
|
|
})
|
|
|
|
uistate.connect('nameformat-changed', self.build_tree)
|
|
|
|
self.additional_uis.append(self.additional_ui())
|
|
|
|
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 additional_ui(self):
|
|
"""
|
|
Defines the UI string for UIManager
|
|
"""
|
|
return '''<ui>
|
|
<menubar name="MenuBar">
|
|
<menu action="FileMenu">
|
|
<placeholder name="LocalExport">
|
|
<menuitem action="ExportTab"/>
|
|
</placeholder>
|
|
</menu>
|
|
<menu action="BookMenu">
|
|
<placeholder name="AddEditBook">
|
|
<menuitem action="AddBook"/>
|
|
<menuitem action="EditBook"/>
|
|
</placeholder>
|
|
</menu>
|
|
<menu action="GoMenu">
|
|
<placeholder name="CommonGo">
|
|
<menuitem action="Back"/>
|
|
<menuitem action="Forward"/>
|
|
<separator/>
|
|
<menuitem action="HomePerson"/>
|
|
<separator/>
|
|
</placeholder>
|
|
</menu>
|
|
<menu action="EditMenu">
|
|
<placeholder name="CommonEdit">
|
|
<menuitem action="Add"/>
|
|
<menuitem action="Edit"/>
|
|
<menuitem action="Remove"/>
|
|
<menuitem action="Merge"/>
|
|
</placeholder>
|
|
<menuitem action="SetActive"/>
|
|
<menuitem action="FilterEdit"/>
|
|
</menu>
|
|
</menubar>
|
|
<toolbar name="ToolBar">
|
|
<placeholder name="CommonNavigation">
|
|
<toolitem action="Back"/>
|
|
<toolitem action="Forward"/>
|
|
<toolitem action="HomePerson"/>
|
|
</placeholder>
|
|
<placeholder name="CommonEdit">
|
|
<toolitem action="Add"/>
|
|
<toolitem action="Edit"/>
|
|
<toolitem action="Remove"/>
|
|
<toolitem action="Merge"/>
|
|
</placeholder>
|
|
</toolbar>
|
|
<popup name="Popup">
|
|
<menuitem action="Back"/>
|
|
<menuitem action="Forward"/>
|
|
<menuitem action="HomePerson"/>
|
|
<separator/>
|
|
<menuitem action="Add"/>
|
|
<menuitem action="Edit"/>
|
|
<menuitem action="Remove"/>
|
|
<menuitem action="Merge"/>
|
|
<separator/>
|
|
<menu name="QuickReport" action="QuickReport">
|
|
<menuitem action="Dummy"/>
|
|
</menu>
|
|
<menu name="WebConnect" action="WebConnect">
|
|
<menuitem action="Dummy"/>
|
|
</menu>
|
|
</popup>
|
|
</ui>'''
|
|
|
|
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()
|
|
#the editor requires a surname
|
|
person.primary_name.add_surname(gen.lib.Surname())
|
|
person.primary_name.set_primary_surname(0)
|
|
|
|
try:
|
|
EditPerson(self.dbstate, self.uistate, [], 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
|
|
with self.dbstate.db.transaction_begin() as trans:
|
|
|
|
# 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)
|
|
trans.set_description(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..."),
|
|
"<control>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..."), "<control>Insert",
|
|
_("Add a new person"), self.add),
|
|
('Remove', gtk.STOCK_REMOVE, _("_Remove"), "<control>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)
|
|
|
|
def get_default_gramplets(self):
|
|
"""
|
|
Define the default gramplets for the sidebar and bottombar.
|
|
"""
|
|
return (("Person Filter Gramplet",),
|
|
("Person Details Gramplet",
|
|
"Person Gallery Gramplet",
|
|
"Person Residence Gramplet",
|
|
"Person Attributes Gramplet"))
|