752d594571
This patch improves on the context changes: * avoid use of transaction_xx methods * force an abort in case of unclean transaction Backward compatibility is broken to achieve this. svn: r16680
217 lines
7.9 KiB
Python
217 lines
7.9 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2009 Gary Burton
|
|
# Copyright (C) 2010 Jakim Friant
|
|
#
|
|
# 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$
|
|
|
|
"""Tools/Database Processing/Sort Events"""
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
from gen.ggettext import gettext as _
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# gramps modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import Sort
|
|
|
|
from gen.db import DbTxn
|
|
from gui.plug import MenuToolOptions, PluginWindows
|
|
from gen.plug.report import utils as ReportUtils
|
|
from gen.plug.menu import FilterOption, PersonOption, \
|
|
EnumeratedListOption, BooleanOption
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Private Functions
|
|
#
|
|
#------------------------------------------------------------------------
|
|
def _get_sort_functions(sort):
|
|
"""
|
|
Define the types of sorting that is available
|
|
"""
|
|
return [
|
|
(_("Date"), sort.by_date_key),
|
|
(_("Type"), sort.by_event_type_key),
|
|
(_("ID"), sort.by_event_id_key),
|
|
(_("Description"), sort.by_event_description_key),
|
|
(_("Place"), sort.by_event_place_key),]
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# SortEvents
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class SortEvents(PluginWindows.ToolManagedWindowBatch):
|
|
"""
|
|
This tool can be used to sort personal and family events by a variety
|
|
of event attributes. It is primarily intended for re-organising data
|
|
imported from an external source where the events are randomly ordered.
|
|
"""
|
|
|
|
def get_title(self):
|
|
"""
|
|
Return the window title.
|
|
"""
|
|
return _("Sort Events")
|
|
|
|
def initial_frame(self):
|
|
"""
|
|
The name of the initial menu tab.
|
|
"""
|
|
return _("Options")
|
|
|
|
def run(self):
|
|
"""
|
|
Perform the actual extraction of information.
|
|
"""
|
|
menu = self.options.menu
|
|
self.filter = menu.get_option_by_name('filter').get_filter()
|
|
sort_func_num = menu.get_option_by_name('sort_by').get_value()
|
|
self.sort_desc = menu.get_option_by_name('sort_desc').get_value()
|
|
self.fam_events = menu.get_option_by_name('family_events').get_value()
|
|
sort_functions = _get_sort_functions(Sort.Sort(self.db))
|
|
self.sort_name = sort_functions[sort_func_num][0]
|
|
self.sort_func = sort_functions[sort_func_num][1]
|
|
self.sort = Sort.Sort(self.db)
|
|
with DbTxn(_("Sort event changes"), self.db, batch=True) as trans:
|
|
self.db.disable_signals()
|
|
family_handles = self.sort_person_events(trans)
|
|
if len(family_handles) > 0:
|
|
self.sort_family_events(family_handles, trans)
|
|
self.db.enable_signals()
|
|
self.db.request_rebuild()
|
|
|
|
def sort_person_events(self, trans):
|
|
"""
|
|
Sort the personal events associated with the selected people.
|
|
"""
|
|
people_handles = self.filter.apply(self.db,
|
|
self.db.iter_person_handles())
|
|
self.progress.set_pass(_("Sorting personal events..."),
|
|
self.db.get_number_of_people())
|
|
family_handles = []
|
|
for handle in people_handles:
|
|
person = self.db.get_person_from_handle(handle)
|
|
self.progress.step()
|
|
event_ref_list = person.get_event_ref_list()
|
|
event_ref_list.sort(key=lambda x: self.sort_func(x.ref))
|
|
if self.sort_desc:
|
|
event_ref_list.reverse()
|
|
if self.fam_events:
|
|
family_handles.extend(person.get_family_handle_list())
|
|
person.set_event_ref_list(event_ref_list)
|
|
self.db.set_birth_death_index(person)
|
|
self.db.commit_person(person, trans)
|
|
self.change = True
|
|
return family_handles
|
|
|
|
def sort_family_events(self, family_handles, trans):
|
|
"""
|
|
Sort the family events associated with the selected people.
|
|
"""
|
|
self.progress.set_pass(_("Sorting family events..."),
|
|
len(family_handles))
|
|
for handle in family_handles:
|
|
family = self.db.get_family_from_handle(handle)
|
|
self.progress.step()
|
|
event_ref_list = family.get_event_ref_list()
|
|
event_ref_list.sort(key=lambda x: self.sort_func(x.ref))
|
|
if self.sort_desc:
|
|
event_ref_list.reverse()
|
|
family.set_event_ref_list(event_ref_list)
|
|
self.db.commit_family(family, trans)
|
|
self.change = True
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
#
|
|
#
|
|
#------------------------------------------------------------------------
|
|
class SortEventOptions(MenuToolOptions):
|
|
"""
|
|
Define options and provides handling interface.
|
|
"""
|
|
|
|
def __init__(self, name, person_id=None, dbstate=None):
|
|
self.__db = dbstate.get_database()
|
|
MenuToolOptions.__init__(self, name, person_id, dbstate)
|
|
|
|
def add_menu_options(self, menu):
|
|
"""
|
|
Define the options for the menu.
|
|
"""
|
|
category_name = _("Tool Options")
|
|
|
|
self.__filter = FilterOption(_("Filter"), 0)
|
|
self.__filter.set_help(_("Select the people to sort"))
|
|
menu.add_option(category_name, "filter", self.__filter)
|
|
self.__filter.connect('value-changed', self.__filter_changed)
|
|
|
|
self.__pid = PersonOption(_("Filter Person"))
|
|
self.__pid.set_help(_("The center person for the filter"))
|
|
menu.add_option(category_name, "pid", self.__pid)
|
|
self.__pid.connect('value-changed', self.__update_filters)
|
|
|
|
self.__update_filters()
|
|
|
|
sort_by = EnumeratedListOption(_('Sort by'), 0 )
|
|
idx = 0
|
|
for item in _get_sort_functions(Sort.Sort(self.__db)):
|
|
sort_by.add_item(idx, item[0])
|
|
idx += 1
|
|
sort_by.set_help( _("Sorting method to use"))
|
|
menu.add_option(category_name, "sort_by", sort_by)
|
|
|
|
sort_desc = BooleanOption(_("Sort descending"), False)
|
|
sort_desc.set_help(_("Set the sort order"))
|
|
menu.add_option(category_name, "sort_desc", sort_desc)
|
|
|
|
family_events = BooleanOption(_("Include family events"), True)
|
|
family_events.set_help(_("Sort family events of the person"))
|
|
menu.add_option(category_name, "family_events", family_events)
|
|
|
|
def __filter_changed(self):
|
|
"""
|
|
Handle filter change. If the filter is not specific to a person,
|
|
disable the person option
|
|
"""
|
|
filter_value = self.__filter.get_value()
|
|
if filter_value in [1, 2, 3, 4]:
|
|
# Filters 1, 2, 3 and 4 rely on the center person
|
|
self.__pid.set_available(True)
|
|
else:
|
|
# The rest don't
|
|
self.__pid.set_available(False)
|
|
|
|
def __update_filters(self):
|
|
"""
|
|
Update the filter list based on the selected person
|
|
"""
|
|
gid = self.__pid.get_value()
|
|
person = self.__db.get_person_from_gramps_id(gid)
|
|
filter_list = ReportUtils.get_person_filters(person, False)
|
|
self.__filter.set_filters(filter_list)
|