604336d764
* src/Editors/_EditFamily.py: reverse surname guess for father svn: r9434
952 lines
35 KiB
Python
952 lines
35 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2007 Donald N. Allingham
|
|
#
|
|
# 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$
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
from gettext import gettext as _
|
|
import cPickle as pickle
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# enable logging for error handling
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import logging
|
|
log = logging.getLogger(".")
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# GTK/Gnome modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import gtk
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# gramps modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import const
|
|
import Utils
|
|
from BasicUtils import name_displayer
|
|
import gen.lib
|
|
import Config
|
|
import Errors
|
|
|
|
from _EditPrimary import EditPrimary
|
|
from ReportBase import ReportUtils
|
|
from DdTargets import DdTargets
|
|
from DisplayTabs import \
|
|
EmbeddedList,EventEmbedList,SourceEmbedList,FamilyAttrEmbedList,\
|
|
NoteTab,GalleryTab,FamilyLdsEmbedList, ChildModel
|
|
from GrampsWidgets import *
|
|
from ReportBase import CATEGORY_QR_FAMILY
|
|
import QuestionDialog
|
|
|
|
#from ObjectSelector import PersonSelector,PersonFilterSpec
|
|
|
|
from Selectors import selector_factory
|
|
SelectPerson = selector_factory('Person')
|
|
|
|
_RETURN = gtk.gdk.keyval_from_name("Return")
|
|
_KP_ENTER = gtk.gdk.keyval_from_name("KP_Enter")
|
|
|
|
class ChildEmbedList(EmbeddedList):
|
|
"""
|
|
The child embed list is specific to the Edit Family dialog, so it
|
|
is contained here instead of in DisplayTabs.
|
|
"""
|
|
|
|
_HANDLE_COL = 10
|
|
_DND_TYPE = DdTargets.PERSON_LINK
|
|
|
|
_MSG = {
|
|
'add' : _('Create a new person and add the child to the family'),
|
|
'del' : _('Remove the child from the family'),
|
|
'edit' : _('Edit the child/family relationship'),
|
|
'share' : _('Add an existing person as a child of the family'),
|
|
'up' : _('Move the child up in the childrens list'),
|
|
'down' : _('Move the child down in the childrens list'),
|
|
}
|
|
|
|
_column_names = [
|
|
(_('#'),0) ,
|
|
(_('ID'),1) ,
|
|
(_('Name'),11),
|
|
(_('Gender'),3),
|
|
(_('Paternal'),4),
|
|
(_('Maternal'),5),
|
|
(_('Birth Date'),12),
|
|
(_('Death Date'),13),
|
|
(_('Birth Place'),8),
|
|
(_('Death Place'),9),
|
|
]
|
|
|
|
def __init__(self, dbstate, uistate, track, family):
|
|
"""
|
|
Create the object, storing the passed family value
|
|
"""
|
|
self.family = family
|
|
EmbeddedList.__init__(self, dbstate, uistate, track,
|
|
_('Children'), ChildModel, True, True)
|
|
|
|
def get_popup_menu_items(self):
|
|
return [
|
|
(True, True, gtk.STOCK_ADD, self.add_button_clicked),
|
|
(True, False, _('Share'), self.edit_button_clicked),
|
|
(False,True, _('Edit relationship'), self.edit_button_clicked),
|
|
(False,True, _('Edit child'), self.edit_child_button_clicked),
|
|
(True, True, gtk.STOCK_REMOVE, self.del_button_clicked),
|
|
]
|
|
|
|
def find_index(self,obj):
|
|
"""
|
|
returns the index of the object within the associated data
|
|
"""
|
|
reflist = [ref.ref for ref in self.family.get_child_ref_list()]
|
|
return reflist.index(obj)
|
|
|
|
def _find_row(self,x,y):
|
|
row = self.tree.get_path_at_pos(x,y)
|
|
if row == None:
|
|
return len(self.family.get_child_ref_list())
|
|
else:
|
|
return row[0][0]
|
|
|
|
def _handle_drag(self, row, obj):
|
|
self.family.get_child_ref_list().insert(row,obj)
|
|
self.changed = True
|
|
self.rebuild()
|
|
|
|
def _move(self, row_from, row_to, obj):
|
|
dlist = self.family.get_child_ref_list()
|
|
if row_from < row_to:
|
|
dlist.insert(row_to,obj)
|
|
del dlist[row_from]
|
|
else:
|
|
del dlist[row_from]
|
|
dlist.insert(row_to-1,obj)
|
|
self.changed = True
|
|
self.rebuild()
|
|
|
|
def build_columns(self):
|
|
"""
|
|
We can't use the default build_columns in the base class, because
|
|
we are using the custom TypeCellRenderer to handle father parent
|
|
relationships. The Paternal and Maternal columns (columns 4 and 5)
|
|
use this.
|
|
"""
|
|
for column in self.columns:
|
|
self.tree.remove_column(column)
|
|
self.columns = []
|
|
|
|
for pair in self.column_order():
|
|
if not pair[0]:
|
|
continue
|
|
name = self._column_names[pair[1]][0]
|
|
render = gtk.CellRendererText()
|
|
column = gtk.TreeViewColumn(name, render, text=pair[1])
|
|
column.set_min_width(50)
|
|
|
|
column.set_resizable(True)
|
|
column.set_sort_column_id(self._column_names[pair[1]][1])
|
|
self.columns.append(column)
|
|
self.tree.append_column(column)
|
|
|
|
def get_icon_name(self):
|
|
return 'gramps-family'
|
|
|
|
def is_empty(self):
|
|
"""
|
|
The list is considered empty if the child list is empty.
|
|
"""
|
|
return len(self.family.get_child_ref_list()) == 0
|
|
|
|
def get_data(self):
|
|
"""
|
|
Normally, get_data returns a list. However, we return family
|
|
object here instead.
|
|
"""
|
|
return self.family
|
|
|
|
def column_order(self):
|
|
return [(1,0),(1,1),(1,2),(1,3),(1,4),(1,5),(1,6),(0,8),(0,9)]
|
|
|
|
def add_button_clicked(self,obj):
|
|
from Editors import EditPerson
|
|
|
|
person = gen.lib.Person()
|
|
autoname = Config.get(Config.SURNAME_GUESSING)
|
|
if autoname == 0:
|
|
name = self.north_american()
|
|
elif autoname == 2:
|
|
name = self.latin_american()
|
|
else:
|
|
name = self.no_name()
|
|
person.get_primary_name().set_surname(name[1])
|
|
person.get_primary_name().set_surname_prefix(name[0])
|
|
|
|
EditPerson(self.dbstate, self.uistate, self.track,person,
|
|
self.new_child_added)
|
|
|
|
def new_child_added(self, person):
|
|
ref = gen.lib.ChildRef()
|
|
ref.ref = person.get_handle()
|
|
self.family.add_child_ref(ref)
|
|
self.rebuild()
|
|
|
|
def child_ref_edited(self, person):
|
|
self.rebuild()
|
|
|
|
def share_button_clicked(self,obj):
|
|
# it only makes sense to skip those who are already in the family
|
|
|
|
skip_list = [self.family.get_father_handle(), \
|
|
self.family.get_mother_handle()] + \
|
|
[x.ref for x in self.family.get_child_ref_list() ]
|
|
|
|
sel = SelectPerson(self.dbstate, self.uistate, self.track,
|
|
_("Select Child"), skip=skip_list)
|
|
person = sel.run()
|
|
|
|
if person:
|
|
ref = gen.lib.ChildRef()
|
|
ref.ref = person.get_handle()
|
|
self.family.add_child_ref(ref)
|
|
self.rebuild()
|
|
|
|
def run(self,skip):
|
|
skip_list = [ x for x in skip if x]
|
|
SelectPerson(self.dbstate, self.uistate, self.track,
|
|
_("Select Child"), skip=skip_list)
|
|
|
|
def del_button_clicked(self,obj):
|
|
handle = self.get_selected()
|
|
if handle:
|
|
for ref in self.family.get_child_ref_list():
|
|
if ref.ref == handle:
|
|
self.family.remove_child_ref(ref)
|
|
self.rebuild()
|
|
|
|
def edit_button_clicked(self,obj):
|
|
handle = self.get_selected()
|
|
if handle:
|
|
from Editors import EditChildRef
|
|
|
|
for ref in self.family.get_child_ref_list():
|
|
if ref.ref == handle:
|
|
p = self.dbstate.db.get_person_from_handle(handle)
|
|
n = name_displayer.display(p)
|
|
try:
|
|
EditChildRef(n, self.dbstate, self.uistate, self.track,
|
|
ref, self.child_ref_edited)
|
|
except Errors.WindowActiveError, msg:
|
|
pass
|
|
break
|
|
|
|
def edit_child_button_clicked(self, obj):
|
|
handle = self.get_selected()
|
|
if handle:
|
|
from Editors import EditPerson
|
|
|
|
for ref in self.family.get_child_ref_list():
|
|
if ref.ref == handle:
|
|
p = self.dbstate.db.get_person_from_handle(handle)
|
|
EditPerson(self.dbstate, self.uistate, self.track,
|
|
p, self.child_ref_edited)
|
|
break
|
|
|
|
def up_button_clicked(self, obj):
|
|
handle = self.get_selected()
|
|
if handle:
|
|
pos = self.find_index(handle)
|
|
if pos > 0 :
|
|
self._move_up(pos,self.family.get_child_ref_list()[pos]
|
|
,selmethod=self.family.get_child_ref_list)
|
|
|
|
def down_button_clicked(self, obj):
|
|
ref = self.get_selected()
|
|
if ref:
|
|
pos = self.find_index(ref)
|
|
if pos >=0 and pos < len(self.family.get_child_ref_list())-1:
|
|
self._move_down(pos,self.family.get_child_ref_list()[pos]
|
|
,selmethod=self.family.get_child_ref_list)
|
|
|
|
|
|
def drag_data_received(self, widget, context, x, y, sel_data, info, time):
|
|
"""
|
|
Handle the standard gtk interface for drag_data_received.
|
|
|
|
If the selection data is define, extract the value from sel_data.data,
|
|
and decide if this is a move or a reorder.
|
|
"""
|
|
if sel_data and sel_data.data:
|
|
(mytype, selfid, obj, row_from) = pickle.loads(sel_data.data)
|
|
|
|
# make sure this is the correct DND type for this object
|
|
if mytype == self._DND_TYPE.drag_type:
|
|
|
|
# determine the destination row
|
|
row = self._find_row(x, y)
|
|
|
|
# if the is same object, we have a move, otherwise,
|
|
# it is a standard drag-n-drop
|
|
|
|
if id(self) == selfid:
|
|
obj = self.get_data().get_child_ref_list()[row_from]
|
|
self._move(row_from, row, obj)
|
|
else:
|
|
handle = obj
|
|
obj = gen.lib.ChildRef()
|
|
obj.ref = handle
|
|
self._handle_drag(row, obj)
|
|
self.rebuild()
|
|
elif self._DND_EXTRA and mytype == self._DND_EXTRA.drag_type:
|
|
self.handle_extra_type(mytype, obj)
|
|
|
|
def north_american(self):
|
|
father_handle = self.family.get_father_handle()
|
|
if father_handle:
|
|
father = self.dbstate.db.get_person_from_handle(father_handle)
|
|
pname = father.get_primary_name()
|
|
return (pname.get_surname_prefix(),pname.get_surname())
|
|
return ("","")
|
|
|
|
def no_name(self):
|
|
return ("","")
|
|
|
|
def latin_american(self):
|
|
if self.family:
|
|
father_handle = self.family.get_father_handle()
|
|
mother_handle = self.family.get_mother_handle()
|
|
if not father_handle or not mother_handle:
|
|
return ("","")
|
|
father = self.dbstate.db.get_person_from_handle(father_handle)
|
|
mother = self.dbstate.db.get_person_from_handle(mother_handle)
|
|
fsn = father.get_primary_name().get_surname()
|
|
msn = mother.get_primary_name().get_surname()
|
|
if not father or not mother:
|
|
return ("","")
|
|
try:
|
|
return ("","%s %s" % (fsn.split()[0],msn.split()[0]))
|
|
except:
|
|
return ("","")
|
|
else:
|
|
return ("","")
|
|
|
|
class FastMaleFilter:
|
|
|
|
def __init__(self,db):
|
|
self.db = db
|
|
|
|
def match(self, handle):
|
|
value = self.db.get_raw_person_data(handle)
|
|
return value[2] == gen.lib.Person.MALE
|
|
|
|
class FastFemaleFilter:
|
|
|
|
def __init__(self,db):
|
|
self.db = db
|
|
|
|
def match(self, handle):
|
|
value = self.db.get_raw_person_data(handle)
|
|
return value[2] == gen.lib.Person.FEMALE
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# EditFamily
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class EditFamily(EditPrimary):
|
|
|
|
QR_CATEGORY = CATEGORY_QR_FAMILY
|
|
|
|
def __init__(self,dbstate, uistate, track, family):
|
|
|
|
self.tooltips = gtk.Tooltips()
|
|
EditPrimary.__init__(self, dbstate, uistate, track,
|
|
family, dbstate.db.get_family_from_handle)
|
|
|
|
# look for the scenerio of a child and no parents on a new
|
|
# family
|
|
|
|
if self.added and self.obj.get_father_handle() == None and \
|
|
self.obj.get_mother_handle() == None and \
|
|
len(self.obj.get_child_ref_list()) == 1:
|
|
self.add_parent = True
|
|
if not Config.get(Config.FAMILY_WARN):
|
|
for i in self.hidden:
|
|
i.set_sensitive(False)
|
|
|
|
QuestionDialog.MessageHideDialog(
|
|
_("Adding parents to a person"),
|
|
_("It is possible to accidentally create multiple "
|
|
"families with the same parents. To help avoid "
|
|
"this problem, only the buttons to select parents "
|
|
"are available when you create a new family. The "
|
|
"remaining fields will become available after you "
|
|
"attempt to select a parent."),
|
|
Config.FAMILY_WARN)
|
|
else:
|
|
self.add_parent = False
|
|
|
|
def empty_object(self):
|
|
return gen.lib.Family()
|
|
|
|
def _local_init(self):
|
|
self.build_interface()
|
|
|
|
self.mname = None
|
|
self.fname = None
|
|
|
|
self._add_db_signal('family-update', self.check_for_family_change)
|
|
self._add_db_signal('family-delete', self.check_for_close)
|
|
|
|
self.added = self.obj.handle == None
|
|
if self.added:
|
|
self.obj.handle = Utils.create_id()
|
|
|
|
self.load_data()
|
|
|
|
def check_for_close(self, handles):
|
|
if self.obj.get_handle() in handles:
|
|
self._do_close()
|
|
|
|
def check_for_family_change(self, handles):
|
|
|
|
# check to see if the handle matches the current object
|
|
if self.obj.get_handle() in handles:
|
|
|
|
self.obj = self.dbstate.db.get_family_from_handle(self.obj.get_handle())
|
|
self.reload_people()
|
|
self.event_embed.rebuild()
|
|
self.source_embed.rebuild()
|
|
self.attr_embed.data = self.obj.get_attribute_list()
|
|
self.attr_embed.rebuild()
|
|
self.lds_embed.data = self.obj.get_lds_ord_list()
|
|
self.lds_embed.rebuild()
|
|
|
|
QuestionDialog.WarningDialog(
|
|
_("Family has changed"),
|
|
_("The family you are editing has changed. To make sure that the "
|
|
"database is not corrupted, GRAMPS has updated the family to "
|
|
"reflect these changes. Any edits you have made may have been lost."))
|
|
|
|
def reload_people(self):
|
|
fhandle = self.obj.get_father_handle()
|
|
self.update_father(fhandle)
|
|
|
|
mhandle = self.obj.get_mother_handle()
|
|
self.update_mother(mhandle)
|
|
self.child_list.rebuild()
|
|
|
|
def get_menu_title(self):
|
|
if self.obj.get_handle():
|
|
dialog_title = Utils.family_name(self.obj, self.db, _("New Family"))
|
|
dialog_title = _("Family") + ': ' + dialog_title
|
|
else:
|
|
dialog_title = _("New Family")
|
|
return dialog_title
|
|
|
|
def build_menu_names(self,family):
|
|
return (_('Edit Family'), self.get_menu_title())
|
|
|
|
def build_interface(self):
|
|
|
|
self.top = gtk.glade.XML(const.GLADE_FILE,"family_editor","gramps")
|
|
|
|
self.set_window(self.top.get_widget("family_editor"), None, self.get_menu_title())
|
|
|
|
# restore window size
|
|
width = Config.get(Config.FAMILY_WIDTH)
|
|
height = Config.get(Config.FAMILY_HEIGHT)
|
|
self.window.set_default_size(width, height)
|
|
|
|
self.fbirth = self.top.get_widget('fbirth')
|
|
self.fdeath = self.top.get_widget('fdeath')
|
|
|
|
self.mbirth = self.top.get_widget('mbirth')
|
|
self.mdeath = self.top.get_widget('mdeath')
|
|
|
|
self.mbutton = self.top.get_widget('mbutton')
|
|
self.mbutton2= self.top.get_widget('mbutton2')
|
|
self.fbutton = self.top.get_widget('fbutton')
|
|
self.fbutton2= self.top.get_widget('fbutton2')
|
|
|
|
self.tooltips.set_tip(self.mbutton2,
|
|
_("Add a new person as the mother"))
|
|
self.tooltips.set_tip(self.fbutton2,
|
|
_("Add a new person as the father"))
|
|
|
|
self.mbox = self.top.get_widget('mbox')
|
|
self.fbox = self.top.get_widget('fbox')
|
|
|
|
#allow for a context menu
|
|
self.set_contexteventbox(self.top.get_widget("eventboxtop"))
|
|
|
|
def _connect_signals(self):
|
|
self.define_ok_button(self.top.get_widget('ok'), self.save)
|
|
self.define_cancel_button(self.top.get_widget('cancel'))
|
|
|
|
def _can_be_replaced(self):
|
|
pass
|
|
|
|
def _setup_fields(self):
|
|
|
|
self.private= PrivacyButton(
|
|
self.top.get_widget('private'),
|
|
self.obj,
|
|
self.db.readonly)
|
|
|
|
self.gid = MonitoredEntry(
|
|
self.top.get_widget('gid'),
|
|
self.obj.set_gramps_id,
|
|
self.obj.get_gramps_id,
|
|
self.db.readonly)
|
|
|
|
self.marker = MonitoredDataType(
|
|
self.top.get_widget('marker'),
|
|
self.obj.set_marker,
|
|
self.obj.get_marker,
|
|
self.db.readonly,
|
|
self.db.get_marker_types(),
|
|
)
|
|
|
|
self.data_type = MonitoredDataType(
|
|
self.top.get_widget('marriage_type'),
|
|
self.obj.set_relationship,
|
|
self.obj.get_relationship,
|
|
self.db.readonly,
|
|
self.db.get_marker_types(),
|
|
)
|
|
|
|
def load_data(self):
|
|
fhandle = self.obj.get_father_handle()
|
|
self.update_father(fhandle)
|
|
|
|
mhandle = self.obj.get_mother_handle()
|
|
self.update_mother(mhandle)
|
|
|
|
self.phandles = [mhandle, fhandle] + \
|
|
[ x.ref for x in self.obj.get_child_ref_list()]
|
|
|
|
self.phandles = [handle for handle in self.phandles if handle]
|
|
|
|
self.mbutton.connect('clicked',self.mother_clicked)
|
|
self.mbutton2.connect('clicked',self.add_mother_clicked)
|
|
self.fbutton.connect('clicked',self.father_clicked)
|
|
self.fbutton2.connect('clicked',self.add_father_clicked)
|
|
|
|
def _create_tabbed_pages(self):
|
|
|
|
notebook = gtk.Notebook()
|
|
|
|
self.child_list = self._add_tab(
|
|
notebook,
|
|
ChildEmbedList(self.dbstate,self.uistate, self.track, self.obj))
|
|
|
|
self.event_embed = EventEmbedList(self.dbstate,self.uistate, self.track,self.obj)
|
|
self.event_list = self._add_tab(notebook, self.event_embed)
|
|
|
|
self.source_embed = SourceEmbedList(self.dbstate,self.uistate,self.track,self.obj)
|
|
self.src_list = self._add_tab(notebook, self.source_embed)
|
|
|
|
self.attr_embed = FamilyAttrEmbedList(self.dbstate, self.uistate, self.track,
|
|
self.obj.get_attribute_list())
|
|
self.attr_list = self._add_tab(notebook, self.attr_embed)
|
|
|
|
self.note_tab = self._add_tab(
|
|
notebook,
|
|
NoteTab(self.dbstate, self.uistate, self.track,
|
|
self.obj.get_note_list(), self.get_menu_title(),
|
|
notetype=gen.lib.NoteType.FAMILY))
|
|
|
|
self.gallery_tab = self._add_tab(
|
|
notebook,
|
|
GalleryTab(self.dbstate, self.uistate, self.track,
|
|
self.obj.get_media_list()))
|
|
|
|
self.lds_embed = FamilyLdsEmbedList(self.dbstate, self.uistate, self.track,
|
|
self.obj.get_lds_ord_list())
|
|
self.lds_list = self._add_tab(notebook, self.lds_embed)
|
|
|
|
self._setup_notebook_tabs( notebook)
|
|
notebook.show_all()
|
|
|
|
self.hidden = (notebook, self.top.get_widget('info'))
|
|
self.top.get_widget('vbox').pack_start(notebook,True)
|
|
|
|
def update_father(self,handle):
|
|
self.load_parent(handle, self.fbox, self.fbirth,
|
|
self.fdeath, self.fbutton, self.fbutton2,
|
|
_("Select a person as the father"),
|
|
_("Remove the person as the father"))
|
|
|
|
def update_mother(self,handle):
|
|
self.load_parent(handle, self.mbox, self.mbirth,
|
|
self.mdeath, self.mbutton, self.mbutton2,
|
|
_("Select a person as the mother"),
|
|
_("Remove the person as the mother"))
|
|
|
|
def add_mother_clicked(self, obj):
|
|
from Editors import EditPerson
|
|
person = gen.lib.Person()
|
|
person.set_gender(gen.lib.Person.FEMALE)
|
|
EditPerson(self.dbstate, self.uistate, self.track, person,
|
|
self.new_mother_added)
|
|
|
|
def add_father_clicked(self, obj):
|
|
from Editors import EditPerson
|
|
person = gen.lib.Person()
|
|
person.set_gender(gen.lib.Person.MALE)
|
|
# if child gets same name as father, father should get
|
|
# same name as child
|
|
autoname = Config.get(Config.SURNAME_GUESSING)
|
|
if autoname == 0:
|
|
name = self.north_american_child()
|
|
# FIXME: Can you work backwards from child with latin_amer name?
|
|
#elif autoname == 2:
|
|
# name = self.latin_american()
|
|
else:
|
|
name = self.no_name()
|
|
person.get_primary_name().set_surname(name[1])
|
|
person.get_primary_name().set_surname_prefix(name[0])
|
|
EditPerson(self.dbstate, self.uistate, self.track,
|
|
person, self.new_father_added)
|
|
|
|
def new_mother_added(self, person):
|
|
for i in self.hidden:
|
|
i.set_sensitive(True)
|
|
self.obj.set_mother_handle(person.handle)
|
|
self.update_mother(person.handle)
|
|
|
|
def new_father_added(self, person):
|
|
for i in self.hidden:
|
|
i.set_sensitive(True)
|
|
self.obj.set_father_handle(person.handle)
|
|
self.update_father(person.handle)
|
|
|
|
def mother_clicked(self, obj):
|
|
for i in self.hidden:
|
|
i.set_sensitive(True)
|
|
|
|
handle = self.obj.get_mother_handle()
|
|
|
|
if handle:
|
|
self.obj.set_mother_handle(None)
|
|
self.update_mother(None)
|
|
else:
|
|
data_filter = FastFemaleFilter(self.dbstate.db)
|
|
sel = SelectPerson(self.dbstate, self.uistate, self.track,
|
|
_("Select Mother"),
|
|
filter=data_filter,
|
|
skip=[x.ref for x in self.obj.get_child_ref_list()])
|
|
person = sel.run()
|
|
|
|
if person:
|
|
self.check_for_existing_family(self.obj.get_father_handle(),
|
|
person.handle,
|
|
self.obj.handle)
|
|
|
|
self.obj.set_mother_handle(person.handle)
|
|
self.update_mother(person.handle)
|
|
|
|
def on_change_father(self, selector_window, obj):
|
|
if obj.__class__ == gen.lib.Person:
|
|
try:
|
|
person = obj
|
|
self.obj.set_father_handle(person.get_handle())
|
|
self.update_father(person.get_handle())
|
|
except:
|
|
log.warn("Failed to update father: \n"
|
|
"obj returned from selector was: %s\n"
|
|
% (repr(obj),))
|
|
raise
|
|
|
|
else:
|
|
log.warn(
|
|
"Object selector returned obj.__class__ = %s, it should "
|
|
"have been of type %s." % (obj.__class__.__name__,
|
|
gen.lib.Person.__name__))
|
|
|
|
selector_window.close()
|
|
|
|
def father_clicked(self, obj):
|
|
for i in self.hidden:
|
|
i.set_sensitive(True)
|
|
|
|
handle = self.obj.get_father_handle()
|
|
if handle:
|
|
self.obj.set_father_handle(None)
|
|
self.update_father(None)
|
|
else:
|
|
data_filter = FastMaleFilter(self.dbstate.db)
|
|
sel = SelectPerson(self.dbstate, self.uistate, self.track,
|
|
_("Select Father"),
|
|
filter=data_filter,
|
|
skip=[x.ref for x in self.obj.get_child_ref_list()])
|
|
person = sel.run()
|
|
|
|
if person:
|
|
|
|
self.check_for_existing_family(person.handle,
|
|
self.obj.get_mother_handle(),
|
|
self.obj.handle)
|
|
|
|
self.obj.set_father_handle(person.handle)
|
|
self.update_father(person.handle)
|
|
|
|
def check_for_existing_family(self, father_handle, mother_handle,
|
|
family_handle):
|
|
|
|
if father_handle:
|
|
father = self.dbstate.db.get_person_from_handle(father_handle)
|
|
ffam = set(father.get_family_handle_list())
|
|
if mother_handle:
|
|
mother = self.dbstate.db.get_person_from_handle(mother_handle)
|
|
mfam = set(mother.get_family_handle_list())
|
|
common = list(mfam.intersection(ffam))
|
|
if len(common) > 0:
|
|
if self.add_parent or self.obj.handle not in common:
|
|
QuestionDialog.WarningDialog(
|
|
_('Duplicate Family'),
|
|
_('A family with these parents already exists '
|
|
'in the database. If you save, you will create '
|
|
'a duplicate family. It is recommended that '
|
|
'you cancel the editing of this window, and '
|
|
'select the existing family'))
|
|
|
|
def edit_person(self,obj,event,handle):
|
|
if event.type == gtk.gdk.BUTTON_PRESS and event.button == 1 \
|
|
or event.keyval in (_RETURN, _KP_ENTER):
|
|
from _EditPerson import EditPerson
|
|
try:
|
|
person = self.db.get_person_from_handle(handle)
|
|
EditPerson(self.dbstate, self.uistate,
|
|
self.track, person)
|
|
except Errors.WindowActiveError:
|
|
pass
|
|
|
|
def load_parent(self, handle, box, birth_obj, death_obj,
|
|
btn_obj, btn2_obj, add_msg, del_msg):
|
|
|
|
is_used = handle != None
|
|
|
|
for i in box.get_children():
|
|
box.remove(i)
|
|
|
|
try:
|
|
btn_obj.remove(btn_obj.get_children()[0])
|
|
except IndexError:
|
|
pass
|
|
|
|
if is_used:
|
|
btn2_obj.hide()
|
|
db = self.db
|
|
person = db.get_person_from_handle(handle)
|
|
name = "%s [%s]" % (name_displayer.display(person),
|
|
person.gramps_id)
|
|
data = ReportUtils.get_birth_death_strings(db,person)
|
|
birth = data[0]
|
|
death = data[4]
|
|
|
|
del_image = gtk.Image()
|
|
del_image.show()
|
|
del_image.set_from_stock(gtk.STOCK_REMOVE,gtk.ICON_SIZE_BUTTON)
|
|
self.tooltips.set_tip(btn_obj, del_msg)
|
|
btn_obj.add(del_image)
|
|
|
|
box.pack_start(LinkBox(
|
|
BasicLabel(name),
|
|
IconButton(self.edit_person,person.handle)
|
|
))
|
|
else:
|
|
btn2_obj.show()
|
|
name = ""
|
|
birth = ""
|
|
death = ""
|
|
|
|
add_image = gtk.Image()
|
|
add_image.show()
|
|
add_image.set_from_stock(gtk.STOCK_INDEX,gtk.ICON_SIZE_BUTTON)
|
|
self.tooltips.set_tip(btn_obj, add_msg)
|
|
btn_obj.add(add_image)
|
|
|
|
birth_obj.set_text(birth)
|
|
death_obj.set_text(death)
|
|
|
|
def fix_parent_handles(self,orig_handle, new_handle, trans):
|
|
if orig_handle != new_handle:
|
|
if orig_handle:
|
|
person = self.db.get_person_from_handle(orig_handle)
|
|
person.family_list.remove(self.obj.handle)
|
|
self.db.commit_person(person,trans)
|
|
if new_handle:
|
|
person = self.db.get_person_from_handle(new_handle)
|
|
if self.obj.handle not in person.family_list:
|
|
person.family_list.append(self.obj.handle)
|
|
self.db.commit_person(person,trans)
|
|
|
|
def object_is_empty(self):
|
|
return self.obj.get_father_handle() == None and \
|
|
self.obj.get_mother_handle() == None and \
|
|
len(self.obj.get_child_ref_list()) == 0
|
|
|
|
def save(self,*obj):
|
|
try:
|
|
self.__do_save()
|
|
except db.DBRunRecoveryError, msg:
|
|
QuestionDialog.RunDatabaseRepair(msg[1])
|
|
|
|
def __do_save(self):
|
|
self.ok_button.set_sensitive(False)
|
|
|
|
if not self.added:
|
|
original = self.db.get_family_from_handle(self.obj.handle)
|
|
else:
|
|
original = None
|
|
|
|
# do some basic checks
|
|
|
|
child_list = [ ref.ref for ref in self.obj.get_child_ref_list() ]
|
|
|
|
if self.obj.get_father_handle() in child_list:
|
|
|
|
father = self.db.get_person_from_handle(self.obj.get_father_handle())
|
|
name = "%s [%s]" % (name_displayer.display(father),
|
|
father.gramps_id)
|
|
QuestionDialog.ErrorDialog(_("A father cannot be his own child"),
|
|
_("%s is listed as both the father and child "
|
|
"of the family.") % name)
|
|
self.ok_button.set_sensitive(True)
|
|
return
|
|
elif self.obj.get_mother_handle() in child_list:
|
|
|
|
mother = self.db.get_person_from_handle(self.obj.get_mother_handle())
|
|
name = "%s [%s]" % (name_displayer.display(mother),
|
|
mother.gramps_id)
|
|
QuestionDialog.ErrorDialog(_("A mother cannot be her own child"),
|
|
_("%s is listed as both the mother and child "
|
|
"of the family.") % name)
|
|
self.ok_button.set_sensitive(True)
|
|
return
|
|
|
|
if not original and self.object_is_empty():
|
|
QuestionDialog.ErrorDialog(
|
|
_("Cannot save family"),
|
|
_("No data exists for this family. "
|
|
"Please enter data or cancel the edit."))
|
|
self.ok_button.set_sensitive(True)
|
|
return
|
|
|
|
# We disconnect the callbacks to all signals we connected earlier.
|
|
# This prevents the signals originating in any of the following
|
|
# commits from being caught by us again.
|
|
for key in self.signal_keys:
|
|
self.db.disconnect(key)
|
|
self.signal_keys = []
|
|
|
|
if not original and not self.object_is_empty():
|
|
trans = self.db.transaction_begin()
|
|
|
|
# find the father, add the family handle to the father
|
|
handle = self.obj.get_father_handle()
|
|
if handle:
|
|
parent = self.db.get_person_from_handle(handle)
|
|
parent.add_family_handle(self.obj.handle)
|
|
self.db.commit_person(parent,trans)
|
|
|
|
# find the mother, add the family handle to the mother
|
|
handle = self.obj.get_mother_handle()
|
|
if handle:
|
|
parent = self.db.get_person_from_handle(handle)
|
|
parent.add_family_handle(self.obj.handle)
|
|
self.db.commit_person(parent,trans)
|
|
|
|
# for each child, add the family handle to the child
|
|
for ref in self.obj.get_child_ref_list():
|
|
child = self.db.get_person_from_handle(ref.ref)
|
|
# fix - relationships need to be extracted from the list
|
|
child.add_parent_family_handle(self.obj.handle)
|
|
self.db.commit_person(child,trans)
|
|
|
|
self.db.add_family(self.obj,trans)
|
|
self.db.transaction_commit(trans,_("Add Family"))
|
|
elif original and self.object_is_empty():
|
|
trans = self.db.transaction_begin()
|
|
self.db.remove_family(self.obj.handle,trans)
|
|
self.db.transaction_commit(trans,_("Remove Family"))
|
|
elif cmp(original.serialize(),self.obj.serialize()):
|
|
|
|
trans = self.db.transaction_begin()
|
|
|
|
self.fix_parent_handles(original.get_father_handle(),
|
|
self.obj.get_father_handle(),trans)
|
|
self.fix_parent_handles(original.get_mother_handle(),
|
|
self.obj.get_mother_handle(),trans)
|
|
|
|
orig_set = set(original.get_child_ref_list())
|
|
new_set = set(self.obj.get_child_ref_list())
|
|
|
|
# remove the family from children which have been removed
|
|
for ref in orig_set.difference(new_set):
|
|
person = self.db.get_person_from_handle(ref.ref)
|
|
person.remove_parent_family_handle(self.obj.handle)
|
|
self.db.commit_person(person,trans)
|
|
|
|
# add the family from children which have been addedna
|
|
for ref in new_set.difference(orig_set):
|
|
person = self.db.get_person_from_handle(ref.ref)
|
|
person.add_parent_family_handle(self.obj.handle)
|
|
self.db.commit_person(person,trans)
|
|
|
|
if self.object_is_empty():
|
|
self.db.remove_family(self.obj.handle,trans)
|
|
else:
|
|
self.db.commit_family(self.obj,trans)
|
|
self.db.transaction_commit(trans,_("Edit Family"))
|
|
|
|
self.close()
|
|
|
|
def _cleanup_on_exit(self):
|
|
(width, height) = self.window.get_size()
|
|
Config.set(Config.FAMILY_WIDTH, width)
|
|
Config.set(Config.FAMILY_HEIGHT, height)
|
|
Config.sync()
|
|
|
|
def north_american_child(self):
|
|
"""
|
|
If SURNAME_GUESSING is north american, then find a child
|
|
and return their name for the father.
|
|
"""
|
|
# for each child, find one with a last name
|
|
for ref in self.obj.get_child_ref_list():
|
|
child = self.db.get_person_from_handle(ref.ref)
|
|
if child:
|
|
pname = child.get_primary_name()
|
|
return (pname.get_surname_prefix(),pname.get_surname())
|
|
return ("","")
|
|
|