# # Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2000-2007 Donald N. Allingham # Copyright (C) 2010 Michiel D. Nauta # # 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$ """ Provide merge capabilities for places. """ #------------------------------------------------------------------------- # # GTK/Gnome modules # #------------------------------------------------------------------------- import gtk #------------------------------------------------------------------------- # # Gramps modules # #------------------------------------------------------------------------- from gen.lib import Person, Family, Event from gen.db import DbTxn from gen.ggettext import sgettext as _ import const import GrampsDisplay import ManagedWindow from Errors import MergeError #------------------------------------------------------------------------- # # Gramps constants # #------------------------------------------------------------------------- WIKI_HELP_PAGE = '%s_-_Entering_and_Editing_Data:_Detailed_-_part_3' % \ const.URL_MANUAL_PAGE WIKI_HELP_SEC = _('manual|Merge_Places') _GLADE_FILE = 'mergeplace.glade' #------------------------------------------------------------------------- # # Merge Places # #------------------------------------------------------------------------- class MergePlaces(ManagedWindow.ManagedWindow): """ Displays a dialog box that allows the places to be combined into one. """ def __init__(self, dbstate, uistate, handle1, handle2): ManagedWindow.ManagedWindow.__init__(self, uistate, [], self.__class__) self.dbstate = dbstate database = dbstate.db self.pl1 = database.get_place_from_handle(handle1) self.pl2 = database.get_place_from_handle(handle2) self.define_glade('mergeplace', _GLADE_FILE) self.set_window(self._gladeobj.toplevel, self.get_widget('place_title'), _("Merge Places")) # Detailed selection widgets title1 = self.pl1.get_title() title2 = self.pl2.get_title() entry1 = self.get_widget("title1") entry2 = self.get_widget("title2") entry1.set_text(title1) entry2.set_text(title2) if entry1.get_text() == entry2.get_text(): for widget_name in ('title1', 'title2', 'title_btn1', 'title_btn2'): self.get_widget(widget_name).set_sensitive(False) entry1 = self.get_widget("lat1") entry2 = self.get_widget("lat2") entry1.set_text(self.pl1.get_latitude()) entry2.set_text(self.pl2.get_latitude()) if entry1.get_text() == entry2.get_text(): for widget_name in ('lat1', 'lat2', 'lat_btn1', 'lat_btn2'): self.get_widget(widget_name).set_sensitive(False) entry1 = self.get_widget("long1") entry2 = self.get_widget("long2") entry1.set_text(self.pl1.get_longitude()) entry2.set_text(self.pl2.get_longitude()) if entry1.get_text() == entry2.get_text(): for widget_name in ('long1', 'long2', 'long_btn1', 'long_btn2'): self.get_widget(widget_name).set_sensitive(False) loc1 = self.pl1.get_main_location().get_text_data_list() loc2 = self.pl2.get_main_location().get_text_data_list() tv1 = self.get_widget("loc1") tv2 = self.get_widget("loc2") tb1 = gtk.TextBuffer() tb2 = gtk.TextBuffer() tv1.set_buffer(tb1) tv2.set_buffer(tb2) tb1.set_text("\n".join(loc1)) tb2.set_text("\n".join(loc2)) if loc1 == loc2: for widget_name in ('loc1', 'loc2', 'loc_btn1', 'loc_btn2'): self.get_widget(widget_name).set_sensitive(False) gramps1 = self.pl1.get_gramps_id() gramps2 = self.pl2.get_gramps_id() entry1 = self.get_widget("gramps1") entry2 = self.get_widget("gramps2") entry1.set_text(gramps1) entry2.set_text(gramps2) if entry1.get_text() == entry2.get_text(): for widget_name in ('gramps1', 'gramps2', 'gramps_btn1', 'gramps_btn2'): self.get_widget(widget_name).set_sensitive(False) # Main window widgets that determine which handle survives rbutton1 = self.get_widget("handle_btn1") rbutton_label1 = self.get_widget("label_handle_btn1") rbutton_label2 = self.get_widget("label_handle_btn2") rbutton_label1.set_label(title1 + " [" + gramps1 + "]") rbutton_label2.set_label(title2 + " [" + gramps2 + "]") rbutton1.connect("toggled", self.on_handle1_toggled) self.connect_button('place_help', self.cb_help) self.connect_button('place_ok', self.cb_merge) self.connect_button('place_cancel', self.close) self.show() def on_handle1_toggled(self, obj): """first chosen place changes""" if obj.get_active(): self.get_widget("title_btn1").set_active(True) self.get_widget("lat_btn1").set_active(True) self.get_widget("long_btn1").set_active(True) self.get_widget("loc_btn1").set_active(True) self.get_widget("gramps_btn1").set_active(True) else: self.get_widget("title_btn2").set_active(True) self.get_widget("lat_btn2").set_active(True) self.get_widget("long_btn2").set_active(True) self.get_widget("loc_btn2").set_active(True) self.get_widget("gramps_btn2").set_active(True) def cb_help(self, obj): """Display the relevant portion of Gramps manual""" GrampsDisplay.help(webpage = WIKI_HELP_PAGE, section = WIKI_HELP_SEC) def cb_merge(self, obj): """ Performs the merge of the places when the merge button is clicked. """ self.uistate.set_busy_cursor(True) use_handle1 = self.get_widget("handle_btn1").get_active() if use_handle1: phoenix = self.pl1 titanic = self.pl2 else: phoenix = self.pl2 titanic = self.pl1 # Add second handle to history so that when merge is complete, # phoenix is the selected row. self.uistate.viewmanager.active_page.get_history().push( phoenix.get_handle()) if self.get_widget("title_btn1").get_active() ^ use_handle1: phoenix.set_title(titanic.get_title()) if self.get_widget("lat_btn1").get_active() ^ use_handle1: phoenix.set_latitude(titanic.get_latitude()) if self.get_widget("long_btn1").get_active() ^ use_handle1: phoenix.set_longitude(titanic.get_longitude()) if self.get_widget("loc_btn1").get_active() ^ use_handle1: swaploc = phoenix.get_main_location() phoenix.set_main_location(titanic.get_main_location()) titanic.set_main_location(swaploc) if self.get_widget("gramps_btn1").get_active() ^ use_handle1: phoenix.set_gramps_id(titanic.get_gramps_id()) query = MergePlaceQuery(self.dbstate, phoenix, titanic) query.execute() self.uistate.set_busy_cursor(False) self.close() class MergePlaceQuery(object): """ Create database query to merge two places. """ def __init__(self, dbstate, phoenix, titanic): self.database = dbstate.db self.phoenix = phoenix self.titanic = titanic def execute(self): """ Merges to places into a single place. """ new_handle = self.phoenix.get_handle() old_handle = self.titanic.get_handle() self.phoenix.merge(self.titanic) with DbTxn(_("Merge Places"), self.database) as trans: self.database.commit_place(self.phoenix, trans) for (class_name, handle) in self.database.find_backlink_handles( old_handle): if class_name == Person.__name__: person = self.database.get_person_from_handle(handle) assert(person.has_handle_reference('Place', old_handle)) person.replace_handle_reference('Place', old_handle, new_handle) self.database.commit_person(person, trans) elif class_name == Family.__name__: family = self.database.get_family_from_handle(handle) assert(family.has_handle_reference('Place', old_handle)) family.replace_handle_reference('Place', old_handle, new_handle) self.database.commit_family(family, trans) elif class_name == Event.__name__: event = self.database.get_event_from_handle(handle) assert(event.has_handle_reference('Place', old_handle)) event.replace_handle_reference('Place', old_handle, new_handle) self.database.commit_event(event, trans) else: raise MergeError("Encounter an object of type %s that has " "a place reference." % class_name) self.database.remove_place(old_handle, trans)