2006-03-01 05:09:01 +00:00
|
|
|
#
|
|
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
|
|
#
|
2007-06-28 05:41:40 +00:00
|
|
|
# Copyright (C) 2000-2007 Donald N. Allingham
|
2006-03-01 05:09:01 +00:00
|
|
|
#
|
|
|
|
# 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$
|
|
|
|
|
2006-04-06 22:02:46 +00:00
|
|
|
from gettext import gettext as _
|
2007-08-30 19:49:04 +00:00
|
|
|
import gtk
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2006-04-01 03:59:42 +00:00
|
|
|
import ManagedWindow
|
2006-03-01 05:09:01 +00:00
|
|
|
import DateHandler
|
2007-06-28 05:41:40 +00:00
|
|
|
from BasicUtils import name_displayer
|
2006-03-03 00:10:52 +00:00
|
|
|
import Config
|
2006-03-01 05:09:01 +00:00
|
|
|
import GrampsDisplay
|
|
|
|
from QuestionDialog import SaveDialog
|
2008-02-18 10:09:50 +00:00
|
|
|
import gen.lib
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2006-04-01 03:59:42 +00:00
|
|
|
class EditPrimary(ManagedWindow.ManagedWindow):
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2007-08-30 19:49:04 +00:00
|
|
|
QR_CATEGORY = -1
|
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def __init__(self, state, uistate, track, obj, get_from_handle,
|
2008-02-10 21:13:19 +00:00
|
|
|
get_from_gramps_id, callback=None):
|
2008-02-18 10:09:50 +00:00
|
|
|
"""
|
|
|
|
Create an edit window.
|
|
|
|
|
|
|
|
Associate a person with the window.
|
|
|
|
|
|
|
|
"""
|
2006-03-01 05:09:01 +00:00
|
|
|
self.dp = DateHandler.parser
|
|
|
|
self.dd = DateHandler.displayer
|
2008-02-08 13:40:04 +00:00
|
|
|
self.name_displayer = name_displayer
|
2006-03-01 05:09:01 +00:00
|
|
|
self.obj = obj
|
|
|
|
self.dbstate = state
|
|
|
|
self.uistate = uistate
|
|
|
|
self.db = state.db
|
|
|
|
self.callback = callback
|
|
|
|
self.signal_keys = []
|
2007-04-08 04:24:38 +00:00
|
|
|
self.ok_button = None
|
2006-03-01 05:09:01 +00:00
|
|
|
self.get_from_handle = get_from_handle
|
2008-02-10 21:13:19 +00:00
|
|
|
self.get_from_gramps_id = get_from_gramps_id
|
2007-08-30 19:49:04 +00:00
|
|
|
self.contexteventbox = None
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2006-04-01 03:59:42 +00:00
|
|
|
ManagedWindow.ManagedWindow.__init__(self, uistate, track, obj)
|
2006-03-01 05:09:01 +00:00
|
|
|
|
|
|
|
self._local_init()
|
|
|
|
|
|
|
|
self._create_tabbed_pages()
|
|
|
|
self._setup_fields()
|
|
|
|
self._connect_signals()
|
|
|
|
self.show()
|
2006-03-02 04:32:37 +00:00
|
|
|
self._post_init()
|
2006-03-01 05:09:01 +00:00
|
|
|
|
|
|
|
def _local_init(self):
|
|
|
|
"""
|
|
|
|
Derived class should do any pre-window initalization in this task.
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _post_init(self):
|
|
|
|
"""
|
|
|
|
Derived class should do any post-window initalization in this task.
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _add_db_signal(self, name, callback):
|
2008-02-08 15:29:28 +00:00
|
|
|
self.signal_keys.append(self.db.connect(name, callback))
|
2006-03-01 05:09:01 +00:00
|
|
|
|
|
|
|
def _connect_signals(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _setup_fields(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def _create_tabbed_pages(self):
|
|
|
|
pass
|
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def build_window_key(self, obj):
|
2006-04-20 04:59:04 +00:00
|
|
|
if obj and obj.get_handle():
|
2006-03-01 05:09:01 +00:00
|
|
|
return obj.get_handle()
|
|
|
|
else:
|
|
|
|
return id(self)
|
2007-02-16 21:20:36 +00:00
|
|
|
|
2007-01-16 07:12:10 +00:00
|
|
|
def _setup_notebook_tabs(self, notebook):
|
|
|
|
for child in notebook.get_children():
|
|
|
|
label = notebook.get_tab_label(child)
|
|
|
|
page_no = notebook.page_num(child)
|
|
|
|
label.drag_dest_set(0, [], 0)
|
2007-02-16 21:20:36 +00:00
|
|
|
label.connect('drag_motion',
|
|
|
|
self._switch_page_on_dnd,
|
|
|
|
notebook,
|
|
|
|
page_no)
|
2007-12-20 13:49:41 +00:00
|
|
|
child.set_parent_notebook(notebook)
|
2007-02-16 21:20:36 +00:00
|
|
|
|
2008-02-15 22:38:47 +00:00
|
|
|
notebook.connect('key-press-event', self.key_pressed, notebook)
|
|
|
|
|
|
|
|
def key_pressed(self, obj, event, notebook):
|
|
|
|
"""
|
|
|
|
Handles the key being pressed on the notebook, pass to key press of
|
|
|
|
current page.
|
|
|
|
"""
|
|
|
|
pag = notebook.get_current_page()
|
|
|
|
if not pag == -1:
|
|
|
|
notebook.get_nth_page(pag).key_pressed(obj, event)
|
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def _switch_page_on_dnd(self, widget, context, x, y, time, notebook,
|
|
|
|
page_no):
|
2007-01-16 07:12:10 +00:00
|
|
|
if notebook.get_current_page() != page_no:
|
|
|
|
notebook.set_current_page(page_no)
|
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def _add_tab(self, notebook, page):
|
2007-01-16 07:12:10 +00:00
|
|
|
notebook.insert_page(page, page.get_tab_widget())
|
2007-10-21 18:25:44 +00:00
|
|
|
page.add_db_signal_callback(self._add_db_signal)
|
2008-02-15 22:38:47 +00:00
|
|
|
page.label.set_use_underline(True)
|
2006-03-01 05:09:01 +00:00
|
|
|
return page
|
|
|
|
|
|
|
|
def _cleanup_on_exit(self):
|
|
|
|
pass
|
|
|
|
|
2006-03-31 19:46:41 +00:00
|
|
|
def object_is_empty(self):
|
|
|
|
return cmp(self.obj.serialize()[1:],
|
|
|
|
self.empty_object().serialize()[1:]) == 0
|
|
|
|
|
2007-04-08 04:24:38 +00:00
|
|
|
def define_ok_button(self, button, function):
|
|
|
|
self.ok_button = button
|
|
|
|
button.connect('clicked', function)
|
2006-03-01 05:09:01 +00:00
|
|
|
button.set_sensitive(not self.db.readonly)
|
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def define_cancel_button(self, button):
|
|
|
|
button.connect('clicked', self.close)
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2008-02-15 11:40:17 +00:00
|
|
|
def define_help_button(self, button, tag, webpage='', section=''):
|
|
|
|
button.connect('clicked', lambda x: GrampsDisplay.help(tag, webpage,
|
|
|
|
section))
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def _do_close(self, *obj):
|
2006-03-01 05:09:01 +00:00
|
|
|
for key in self.signal_keys:
|
|
|
|
self.db.disconnect(key)
|
|
|
|
self._cleanup_on_exit()
|
2006-04-23 22:43:36 +00:00
|
|
|
ManagedWindow.ManagedWindow.close(self)
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def close(self, *obj):
|
2006-03-01 05:09:01 +00:00
|
|
|
"""If the data has changed, give the user a chance to cancel
|
|
|
|
the close window"""
|
2006-04-24 21:04:01 +00:00
|
|
|
if not Config.get(Config.DONT_ASK) and self.data_has_changed():
|
2006-03-01 05:09:01 +00:00
|
|
|
SaveDialog(
|
|
|
|
_('Save Changes?'),
|
|
|
|
_('If you close without saving, the changes you '
|
|
|
|
'have made will be lost'),
|
2006-04-23 22:43:36 +00:00
|
|
|
self._do_close,
|
2006-03-01 05:09:01 +00:00
|
|
|
self.save)
|
|
|
|
return True
|
|
|
|
else:
|
2006-04-23 22:43:36 +00:00
|
|
|
self._do_close()
|
2006-03-01 05:09:01 +00:00
|
|
|
return False
|
|
|
|
|
2006-03-31 19:46:41 +00:00
|
|
|
def empty_object(self):
|
2008-02-18 10:09:50 +00:00
|
|
|
#empty_object should be overridden in base class and will throw an
|
|
|
|
#exception if it is not because self.empty_object().serialize() is
|
|
|
|
#called and PrimaryObject does not implement serialize(). See
|
|
|
|
#BaseObject.serialize()
|
|
|
|
return gen.lib.PrimaryObject
|
2006-03-31 19:46:41 +00:00
|
|
|
|
2006-03-01 05:09:01 +00:00
|
|
|
def data_has_changed(self):
|
2006-04-21 02:59:15 +00:00
|
|
|
|
2006-03-01 05:09:01 +00:00
|
|
|
if self.db.readonly:
|
|
|
|
return False
|
|
|
|
elif self.obj.handle:
|
|
|
|
orig = self.get_from_handle(self.obj.handle)
|
2006-03-09 03:07:41 +00:00
|
|
|
if orig:
|
2006-03-31 19:46:41 +00:00
|
|
|
cmp_obj = orig
|
2006-03-09 03:07:41 +00:00
|
|
|
else:
|
2006-03-31 19:46:41 +00:00
|
|
|
cmp_obj = self.empty_object()
|
|
|
|
return cmp(cmp_obj.serialize()[1:],
|
|
|
|
self.obj.serialize()[1:]) != 0
|
2006-03-01 05:09:01 +00:00
|
|
|
else:
|
2006-03-31 19:46:41 +00:00
|
|
|
cmp_obj = self.empty_object()
|
|
|
|
return cmp(cmp_obj.serialize()[1:],
|
|
|
|
self.obj.serialize()[1:]) != 0
|
2006-03-01 05:09:01 +00:00
|
|
|
|
2008-02-08 15:29:28 +00:00
|
|
|
def save(self, *obj):
|
2006-03-01 05:09:01 +00:00
|
|
|
pass
|
2007-08-30 19:49:04 +00:00
|
|
|
|
|
|
|
def set_contexteventbox(self, eventbox):
|
2008-02-18 15:52:40 +00:00
|
|
|
"""Set the contextbox that grabs button presses if not grabbed
|
2007-08-30 19:49:04 +00:00
|
|
|
by overlying widgets.
|
2008-02-18 15:52:40 +00:00
|
|
|
"""
|
2007-08-30 19:49:04 +00:00
|
|
|
self.contexteventbox = eventbox
|
|
|
|
self.contexteventbox.connect('button-press-event',
|
|
|
|
self._contextmenu_button_press)
|
|
|
|
|
|
|
|
def _contextmenu_button_press(self, obj, event) :
|
|
|
|
"""
|
|
|
|
Button press event that is caught when a mousebutton has been
|
|
|
|
pressed while on contexteventbox
|
|
|
|
It opens a context menu with possible actions
|
|
|
|
"""
|
|
|
|
if event.type == gtk.gdk.BUTTON_PRESS and event.button == 3 :
|
|
|
|
if self.obj.get_handle() == 0 :
|
|
|
|
return False
|
|
|
|
|
|
|
|
#build the possible popup menu
|
|
|
|
self._build_popup_ui()
|
|
|
|
#set or unset sensitivity in popup
|
|
|
|
self._post_build_popup_ui()
|
|
|
|
|
|
|
|
menu = self.popupmanager.get_widget('/Popup')
|
|
|
|
if menu:
|
|
|
|
menu.popup(None, None, None, event.button, event.time)
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _build_popup_ui(self):
|
|
|
|
"""
|
|
|
|
Create actions and ui of context menu
|
|
|
|
"""
|
|
|
|
from QuickReports import create_quickreport_menu
|
|
|
|
|
|
|
|
self.popupmanager = gtk.UIManager()
|
|
|
|
#add custom actions
|
|
|
|
(ui_top, action_groups) = self._top_contextmenu()
|
|
|
|
for action in action_groups :
|
|
|
|
self.popupmanager.insert_action_group(action, -1)
|
|
|
|
#see which quick reports are available now:
|
|
|
|
ui_qr = ''
|
|
|
|
if self.QR_CATEGORY > -1 :
|
|
|
|
(ui_qr, reportactions) = create_quickreport_menu(self.QR_CATEGORY,
|
2007-12-21 06:22:46 +00:00
|
|
|
self.dbstate, self.uistate,
|
|
|
|
self.obj.get_handle())
|
2007-08-30 19:49:04 +00:00
|
|
|
self.report_action = gtk.ActionGroup("/PersonReport")
|
|
|
|
self.report_action.add_actions(reportactions)
|
|
|
|
self.report_action.set_visible(True)
|
|
|
|
self.popupmanager.insert_action_group(self.report_action, -1)
|
|
|
|
|
|
|
|
popupui = '''
|
|
|
|
<ui>
|
|
|
|
<popup name="Popup">''' + ui_top + '''
|
|
|
|
<separator/>''' + ui_qr + '''
|
|
|
|
</popup>
|
|
|
|
</ui>'''
|
|
|
|
|
|
|
|
self.popupmanager.add_ui_from_string(popupui)
|
|
|
|
|
|
|
|
def _top_contextmenu(self):
|
|
|
|
"""
|
|
|
|
Derived class can create a ui with menuitems and corresponding list of
|
|
|
|
actiongroups
|
|
|
|
"""
|
2008-02-08 15:29:28 +00:00
|
|
|
return "", []
|
2007-08-30 19:49:04 +00:00
|
|
|
|
|
|
|
def _post_build_popup_ui(self):
|
|
|
|
"""
|
|
|
|
Derived class should do extra actions here on the menu
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
2008-02-10 21:24:50 +00:00
|
|
|
def _uses_duplicate_id(self):
|
|
|
|
"""
|
2008-02-18 10:09:50 +00:00
|
|
|
Check whether a changed or added GRAMPS ID already exists in the DB.
|
|
|
|
|
|
|
|
Return True if a duplicate GRAMPS ID has been detected.
|
|
|
|
|
2008-02-10 21:24:50 +00:00
|
|
|
"""
|
2008-02-18 10:09:50 +00:00
|
|
|
original = self.get_from_handle(self.obj.get_handle())
|
|
|
|
if original and original.get_gramps_id() == self.obj.get_gramps_id():
|
|
|
|
return (False, 0)
|
|
|
|
else:
|
|
|
|
idval = self.obj.get_gramps_id()
|
|
|
|
if self.get_from_gramps_id(idval):
|
|
|
|
return (True, idval)
|
|
|
|
return (False, 0)
|
|
|
|
|