2002-10-20 14:25:16 +00:00
#
# Gramps - a GTK+/GNOME based genealogy program
#
2005-01-11 22:41:15 +00:00
# Copyright (C) 2000-2005 Donald N. Allingham
2002-10-20 14:25:16 +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
#
2003-11-07 16:29:27 +00:00
# $Id$
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# Standard python modules
#
#-------------------------------------------------------------------------
import os
2003-06-15 00:05:43 +00:00
import locale
2005-06-06 05:03:48 +00:00
from gettext import gettext as _
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# GNOME/GTK
#
#-------------------------------------------------------------------------
import gtk
2005-06-03 13:39:36 +00:00
import gtk . gdk
2004-05-13 22:45:51 +00:00
import gnome
2002-10-20 14:25:16 +00:00
2005-01-27 03:18:21 +00:00
try :
from gnomevfs import get_mime_type , mime_get_description
except :
from gnome . vfs import get_mime_type , mime_get_description
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# Gramps modules
#
#-------------------------------------------------------------------------
import const
2004-04-04 04:39:52 +00:00
import GrampsMime
2005-01-01 04:27:15 +00:00
import NameDisplay
2004-10-10 23:22:12 +00:00
import Date
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# modified flag
#
#-------------------------------------------------------------------------
2003-11-07 16:29:27 +00:00
_history_brokenFlag = 0
2002-10-20 14:25:16 +00:00
2003-11-07 16:29:27 +00:00
def history_broken ( ) :
global _history_brokenFlag
_history_brokenFlag = 1
2005-03-12 00:44:11 +00:00
data_recover_msg = _ ( ' The data can only be recovered by Undo operation '
' or by quitting with abandoning changes. ' )
2003-04-04 05:48:25 +00:00
#-------------------------------------------------------------------------
#
# force_unicode
#
#-------------------------------------------------------------------------
def force_unicode ( n ) :
2005-01-09 02:18:49 +00:00
if type ( n ) != unicode :
2003-04-04 05:48:25 +00:00
return ( unicode ( n ) . lower ( ) , unicode ( n ) )
else :
return ( n . lower ( ) , n )
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# Clears the modified flag. Should be called after data is saved.
#
#-------------------------------------------------------------------------
2003-11-07 16:29:27 +00:00
def clearHistory_broken ( ) :
global _history_brokenFlag
_history_brokenFlag = 0
def wasHistory_broken ( ) :
return _history_brokenFlag
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
# Short hand function to return either the person's name, or an empty
# string if the person is None
#
#-------------------------------------------------------------------------
2003-10-12 04:26:00 +00:00
2004-02-23 04:41:37 +00:00
def family_name ( family , db ) :
2002-10-20 14:25:16 +00:00
""" Builds a name for the family from the parents names """
2004-07-28 02:29:07 +00:00
father_handle = family . get_father_handle ( )
mother_handle = family . get_mother_handle ( )
2004-08-07 05:16:57 +00:00
father = db . get_person_from_handle ( father_handle )
mother = db . get_person_from_handle ( mother_handle )
2002-10-20 14:25:16 +00:00
if father and mother :
2005-01-01 04:27:15 +00:00
fname = NameDisplay . displayer . display ( father )
mname = NameDisplay . displayer . display ( mother )
2005-03-11 21:05:46 +00:00
name = _ ( " %(father)s and %(mother)s " ) % {
" father " : fname ,
" mother " : mname }
2002-10-20 14:25:16 +00:00
elif father :
2005-01-01 04:27:15 +00:00
name = NameDisplay . displayer . display ( father )
2005-03-13 22:10:40 +00:00
elif mother :
2005-01-01 04:27:15 +00:00
name = NameDisplay . displayer . display ( mother )
2005-03-13 22:10:40 +00:00
else :
name = _ ( " unknown " )
2002-10-20 14:25:16 +00:00
return name
2004-02-23 04:41:37 +00:00
def family_upper_name ( family , db ) :
2003-10-12 04:26:00 +00:00
""" Builds a name for the family from the parents names """
2004-07-28 02:29:07 +00:00
father_handle = family . get_father_handle ( )
mother_handle = family . get_mother_handle ( )
2004-08-07 05:16:57 +00:00
father = db . get_person_from_handle ( father_handle )
mother = db . get_person_from_handle ( mother_handle )
2003-10-12 04:26:00 +00:00
if father and mother :
2004-02-14 05:40:30 +00:00
fname = father . get_primary_name ( ) . get_upper_name ( )
mname = mother . get_primary_name ( ) . get_upper_name ( )
2003-10-12 04:26:00 +00:00
name = _ ( " %s and %s " ) % ( fname , mname )
elif father :
2004-02-14 05:40:30 +00:00
name = father . get_primary_name ( ) . get_upper_name ( )
2002-10-20 14:25:16 +00:00
else :
2004-02-14 05:40:30 +00:00
name = mother . get_primary_name ( ) . get_upper_name ( )
2003-10-12 04:26:00 +00:00
return name
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def destroy_passed_object ( obj ) :
obj . destroy ( )
while gtk . events_pending ( ) :
2004-04-28 04:06:25 +00:00
gtk . main_iteration ( )
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def get_detail_text ( obj , priv = 1 ) :
2004-02-14 05:40:30 +00:00
if obj . get_note ( ) != " " :
2002-10-20 14:25:16 +00:00
details = " %s " % _ ( " Note " )
else :
details = " "
2004-02-14 05:40:30 +00:00
if len ( obj . get_source_references ( ) ) > 0 :
2002-10-20 14:25:16 +00:00
if details == " " :
details = _ ( " Source " )
else :
details = " %s , %s " % ( details , _ ( " Source " ) )
2004-02-14 05:40:30 +00:00
if priv and obj . get_privacy ( ) == 1 :
2002-10-20 14:25:16 +00:00
if details == " " :
details = _ ( " Private " )
else :
details = " %s , %s " % ( details , _ ( " Private " ) )
return details
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def redraw_list ( dlist , clist , func ) :
clist . clear ( )
index = 0
2004-08-22 23:16:57 +00:00
for obj in dlist :
2002-10-20 14:25:16 +00:00
col = 0
2004-08-22 23:16:57 +00:00
node = clist . append ( )
for data in func ( obj ) :
clist . set_value ( node , col , data )
2002-10-20 14:25:16 +00:00
col = col + 1
index = index + 1
return index
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def delete_selected ( obj , list ) :
sel = obj . get_selection ( )
2004-08-22 23:16:57 +00:00
model , node = sel . get_selected ( )
if node :
index = model . get_path ( node ) [ 0 ]
2002-10-20 14:25:16 +00:00
del list [ index ]
return 1
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def add_menuitem ( menu , msg , obj , func ) :
item = gtk . MenuItem ( msg )
2004-05-13 22:45:51 +00:00
item . set_data ( ' o ' , obj )
2002-10-20 14:25:16 +00:00
item . connect ( " activate " , func )
item . show ( )
menu . append ( item )
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def view_photo ( photo ) :
2004-08-22 23:16:57 +00:00
mime_type = photo . get_mime_type ( )
2004-04-04 04:39:52 +00:00
try :
2004-08-22 23:16:57 +00:00
data = GrampsMime . get_application ( mime_type )
2004-04-04 04:39:52 +00:00
prog = data [ 0 ]
except :
2002-11-02 21:19:58 +00:00
return
2002-10-20 14:25:16 +00:00
2004-10-29 00:49:40 +00:00
args = prog . split ( )
2004-02-14 05:40:30 +00:00
args . append ( photo . get_path ( ) )
2002-10-20 14:25:16 +00:00
if os . fork ( ) == 0 :
os . execvp ( args [ 0 ] , args )
2005-02-24 00:25:34 +00:00
_icon_theme = gtk . icon_theme_get_default ( )
2002-10-20 14:25:16 +00:00
2005-02-24 00:25:34 +00:00
def find_mime_type_pixbuf ( mime_type ) :
try :
2005-07-15 05:44:06 +00:00
icontmp = mime_type . replace ( ' / ' , ' - ' )
2005-02-24 00:25:34 +00:00
newicon = " gnome-mime- %s " % icontmp
try :
return _icon_theme . load_icon ( newicon , 48 , 0 )
except :
icontmp = mime_type . split ( ' / ' ) [ 0 ]
try :
newicon = " gnome-mime- %s " % icontmp
return _icon_theme . load_icon ( newicon , 48 , 0 )
except :
return gtk . gdk . pixbuf_new_from_file ( const . icon )
except :
return gtk . gdk . pixbuf_new_from_file ( const . icon )
2004-08-22 23:16:57 +00:00
def get_mime_description ( mime_type ) :
2004-04-04 04:39:52 +00:00
try :
2005-01-27 03:18:21 +00:00
value = mime_get_description ( mime_type )
2004-04-04 04:39:52 +00:00
if value :
return value
else :
return ' '
except :
return ' '
2002-10-20 14:25:16 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def build_string_optmenu ( mapping , start_val ) :
index = 0
start_index = 0
keys = mapping . keys ( )
keys . sort ( )
myMenu = gtk . Menu ( )
for key in keys :
if key == " default " :
menuitem = gtk . MenuItem ( _ ( " default " ) )
else :
menuitem = gtk . MenuItem ( key )
menuitem . set_data ( " d " , mapping [ key ] )
2003-10-31 01:20:58 +00:00
menuitem . set_data ( " l " , key )
2002-10-20 14:25:16 +00:00
menuitem . show ( )
myMenu . append ( menuitem )
if key == start_val :
start_index = index
index = index + 1
if start_index :
myMenu . set_active ( start_index )
return myMenu
2002-10-25 04:52:51 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def build_columns ( tree , list ) :
cnum = 0
for name in list :
renderer = gtk . CellRendererText ( )
2003-07-27 03:50:57 +00:00
renderer . set_fixed_height_from_font ( 1 )
2002-10-25 04:52:51 +00:00
column = gtk . TreeViewColumn ( name [ 0 ] , renderer , text = cnum )
column . set_min_width ( name [ 1 ] )
if name [ 2 ] > = 0 :
column . set_sort_column_id ( name [ 2 ] )
if name [ 0 ] == ' ' :
2005-02-24 00:25:34 +00:00
column . set_clickable ( True )
column . set_visible ( False )
2002-10-25 04:52:51 +00:00
cnum = cnum + 1
tree . append_column ( column )
2003-01-10 05:21:32 +00:00
#-------------------------------------------------------------------------
#
# Iterate over ancestors.
#
#-------------------------------------------------------------------------
2005-01-11 22:41:15 +00:00
def for_each_ancestor ( db , start , func , data ) :
2003-01-10 05:21:32 +00:00
"""
Recursively iterate ( breadth - first ) over ancestors of
people listed in start .
Call func ( data , pid ) for the Id of each person encountered .
Exit and return 1 , as soon as func returns true .
Return 0 otherwise .
"""
todo = start
doneIds = { }
while len ( todo ) :
2005-01-11 22:41:15 +00:00
p_handle = todo . pop ( )
p = db . get_person_from_handle ( p_handle )
# Don't process the same handle twice. This can happen
2003-01-10 05:21:32 +00:00
# if there is a cycle in the database, or if the
# initial list contains X and some of X's ancestors.
2005-01-11 22:41:15 +00:00
if doneIds . has_key ( p_handle ) :
2003-01-10 05:21:32 +00:00
continue
2005-01-11 22:41:15 +00:00
doneIds [ p_handle ] = 1
if func ( data , p_handle ) :
2003-01-10 05:21:32 +00:00
return 1
2005-01-11 22:41:15 +00:00
for fam_handle , mrel , frel in p . get_parent_family_handle_list ( ) :
fam = db . get_family_from_handle ( fam_handle )
if fam :
f_handle = fam . get_father_handle ( )
m_handle = fam . get_mother_handle ( )
if f_handle : todo . append ( f_handle )
if m_handle : todo . append ( m_handle )
2003-01-10 05:21:32 +00:00
return 0
2003-03-05 06:01:31 +00:00
def title ( n ) :
return ' <span weight= " bold " size= " larger " > %s </span> ' % n
def set_title_label ( xmlobj , t ) :
title_label = xmlobj . get_widget ( ' title ' )
title_label . set_text ( ' <span weight= " bold " size= " larger " > %s </span> ' % t )
2005-02-24 00:25:34 +00:00
title_label . set_use_markup ( True )
2003-03-06 06:12:51 +00:00
def set_titles ( window , title , t , msg = None ) :
title . set_text ( ' <span weight= " bold " size= " larger " > %s </span> ' % t )
2005-02-24 00:25:34 +00:00
title . set_use_markup ( True )
2003-03-06 06:12:51 +00:00
if msg :
window . set_title ( ' %s - GRAMPS ' % msg )
else :
window . set_title ( ' %s - GRAMPS ' % t )
2003-05-10 06:17:07 +00:00
2003-06-14 17:11:11 +00:00
def gfloat ( val ) :
2003-06-15 00:05:43 +00:00
""" Converts to floating number, taking care of possible locale differences.
Useful for reading float values from text entry fields
while under non - English locale .
"""
2003-06-14 17:11:11 +00:00
try :
return float ( val )
except :
try :
2003-06-14 17:14:46 +00:00
return float ( val . replace ( ' . ' , ' , ' ) )
2003-06-14 17:11:11 +00:00
except :
2003-06-14 17:14:46 +00:00
return float ( val . replace ( ' , ' , ' . ' ) )
2003-06-14 17:11:11 +00:00
return 0.0
2003-06-15 00:05:43 +00:00
def gformat ( val ) :
""" Performs ( " %.3f " % val) formatting with the resulting string always
using dot ( ' . ' ) as a decimal point .
Useful for writing float values into XML when under non - English locale .
"""
decimal_point = locale . localeconv ( ) [ ' decimal_point ' ]
return_val = " %.3f " % val
return return_val . replace ( decimal_point , ' . ' )
2003-11-25 17:45:34 +00:00
def search_for ( name ) :
for i in os . environ [ ' PATH ' ] . split ( ' : ' ) :
fname = os . path . join ( i , name )
2003-11-25 21:34:39 +00:00
if os . access ( fname , os . X_OK ) and not os . path . isdir ( fname ) :
2003-11-25 17:45:34 +00:00
return 1
return 0
2003-06-14 17:11:11 +00:00
2003-11-12 18:45:07 +00:00
#-------------------------------------------------------------------------
#
2005-06-06 05:03:48 +00:00
# Change label appearance
2003-11-12 18:45:07 +00:00
#
#-------------------------------------------------------------------------
2005-06-03 13:39:36 +00:00
def bold_label ( label , widget = None ) :
2003-12-17 05:23:16 +00:00
text = unicode ( label . get_text ( ) )
2005-06-03 13:39:36 +00:00
text = text . replace ( ' <i> ' , ' ' )
text = text . replace ( ' </i> ' , ' ' )
2003-11-12 18:45:07 +00:00
label . set_text ( " <b> %s </b> " % text )
2005-06-03 13:39:36 +00:00
label . set_use_markup ( True )
if widget :
widget . window . set_cursor ( None )
2003-11-12 18:45:07 +00:00
2005-06-03 13:39:36 +00:00
def unbold_label ( label , widget = None ) :
2003-12-17 05:23:16 +00:00
text = unicode ( label . get_text ( ) )
2004-10-29 00:49:40 +00:00
text = text . replace ( ' <b> ' , ' ' )
text = text . replace ( ' </b> ' , ' ' )
2005-06-03 13:39:36 +00:00
text = text . replace ( ' <i> ' , ' ' )
text = text . replace ( ' </i> ' , ' ' )
2003-11-12 18:45:07 +00:00
label . set_text ( text )
2005-06-03 13:39:36 +00:00
label . set_use_markup ( False )
if widget :
widget . window . set_cursor ( None )
def temp_label ( label , widget = None ) :
text = unicode ( label . get_text ( ) )
text = text . replace ( ' <b> ' , ' ' )
text = text . replace ( ' </b> ' , ' ' )
label . set_text ( " <i> %s </i> " % text )
label . set_use_markup ( True )
if widget :
widget . window . set_cursor ( gtk . gdk . Cursor ( gtk . gdk . WATCH ) )
2004-04-25 04:48:02 +00:00
2004-06-30 04:06:10 +00:00
#-------------------------------------------------------------------------
#
# create_id
#
#-------------------------------------------------------------------------
import random
import time
rand = random . Random ( time . time ( ) )
2004-06-27 03:10:06 +00:00
def create_id ( ) :
2004-06-30 04:06:10 +00:00
s = " "
for val in [ int ( time . time ( ) * 10000 ) & 0x7fffffff ,
rand . randint ( 0 , 0x7fffffff ) ,
rand . randint ( 0 , 0x7fffffff ) ] :
while val != 0 :
2004-08-24 03:48:15 +00:00
rem = val % 36
if rem < = 9 :
s + = chr ( 48 + rem )
else :
s + = chr ( rem + 55 )
val = int ( val / 36 )
2004-06-30 04:06:10 +00:00
return s
2004-07-09 18:19:47 +00:00
2004-10-08 03:59:55 +00:00
2005-04-06 15:52:52 +00:00
def probably_alive ( person , db , current_year = None ) :
2005-04-08 08:09:18 +00:00
""" Returns true if the person may be alive.
This works by a process of emlimination . If we can ' t find a good
reason to believe that someone is dead then we assume they must
be alive .
"""
2005-04-08 09:40:15 +00:00
death_year = None
2005-04-08 08:09:18 +00:00
# If the recorded death year is before current year then
# things are simple.
2004-10-08 03:59:55 +00:00
if person . death_handle :
2005-06-01 20:36:23 +00:00
if not current_year :
# no current year and we have a death event -> person died
return False
2005-04-08 08:09:18 +00:00
death = db . get_event_from_handle ( person . death_handle )
if death . get_date_object ( ) . get_start_date ( ) != Date . EMPTY :
2005-04-08 09:40:15 +00:00
death_year = death . get_date_object ( ) . get_year ( )
2005-07-30 03:29:34 +00:00
if death_year < current_year :
2005-04-06 20:56:24 +00:00
return False
2004-10-08 03:59:55 +00:00
# Look for Cause Of Death, Burial or Cremation events.
# These are fairly good indications that someone's not alive.
for ev_handle in person . event_list :
ev = db . get_event_from_handle ( ev_handle )
2004-12-08 22:54:26 +00:00
if ev and ev . name in [ " Cause Of Death " , " Burial " , " Cremation " ] :
2005-06-01 20:36:23 +00:00
if not current_year :
# no current year and we have an event related to death
return False
2005-04-08 09:40:15 +00:00
if not death_year :
death_year = ev . get_date_object ( ) . get_year ( )
2005-04-06 20:56:24 +00:00
if ev . get_date_object ( ) . get_start_date ( ) != Date . EMPTY :
2005-04-08 08:09:18 +00:00
if ev . get_date_object ( ) . get_year ( ) < current_year :
2005-04-06 20:56:24 +00:00
return False
2004-10-08 03:59:55 +00:00
2005-06-01 20:36:23 +00:00
if not current_year :
time_struct = time . localtime ( time . time ( ) )
current_year = time_struct [ 0 ]
2005-04-08 09:40:15 +00:00
birth_year = None
2005-04-08 08:09:18 +00:00
# If they were born within 100 years before current year then
# assume they are alive (we already know they are not dead).
2004-10-08 03:59:55 +00:00
if person . birth_handle :
birth = db . get_event_from_handle ( person . birth_handle )
if birth . get_date_object ( ) . get_start_date ( ) != Date . EMPTY :
2005-04-08 09:40:15 +00:00
if not birth_year :
birth_year = birth . get_date_object ( ) . get_year ( )
if birth . get_date_object ( ) . get_year ( ) > current_year :
# person is not yet born
return False
2005-04-08 08:09:18 +00:00
r = not_too_old ( birth . get_date_object ( ) , current_year )
if r :
#print person.get_primary_name().get_name(), " is alive because they were born late enough."
return True
2005-04-08 09:40:15 +00:00
if not birth_year and death_year :
if death_year > current_year + 110 :
# person died more tha 110 after current year
return False
2005-04-08 08:09:18 +00:00
2004-10-08 03:59:55 +00:00
# Neither birth nor death events are available. Try looking
# for descendants that were born more than a lifespan ago.
min_generation = 13
max_generation = 60
max_age_difference = 60
def descendants_too_old ( person , years ) :
for family_handle in person . get_family_handle_list ( ) :
family = db . get_family_from_handle ( family_handle )
2005-04-08 08:09:18 +00:00
family_list = family . get_child_handle_list ( )
2004-10-08 03:59:55 +00:00
for child_handle in family . get_child_handle_list ( ) :
child = db . get_person_from_handle ( child_handle )
if child . birth_handle :
child_birth = db . get_event_from_handle ( child . birth_handle )
dobj = child_birth . get_date_object ( )
if dobj . get_start_date ( ) != Date . EMPTY :
2004-10-29 00:49:40 +00:00
d = Date . Date ( dobj )
val = d . get_start_date ( )
2004-12-08 22:54:26 +00:00
val = d . get_year ( ) - years
2004-10-29 00:49:40 +00:00
d . set_year ( val )
2005-04-06 15:52:52 +00:00
if not not_too_old ( d , current_year ) :
2004-10-08 03:59:55 +00:00
return True
if child . death_handle :
child_death = db . get_event_from_handle ( child . death_handle )
dobj = child_death . get_date_object ( )
2004-10-29 00:49:40 +00:00
if dobj . get_start_date ( ) != Date . EMPTY :
2005-04-06 15:52:52 +00:00
if not not_too_old ( dobj , current_year ) :
2004-10-08 03:59:55 +00:00
return True
if descendants_too_old ( child , years + min_generation ) :
return True
2005-04-08 08:09:18 +00:00
return False
2004-10-08 03:59:55 +00:00
2005-04-08 08:09:18 +00:00
# If there are descendants that are too old for the person to have
# been alive in the current year then they must be dead.
2004-10-08 03:59:55 +00:00
if descendants_too_old ( person , min_generation ) :
2005-04-08 08:09:18 +00:00
#print person.get_primary_name().get_name(), " is dead because descendants are too old."
return False
average_generation_gap = 20
def ancestors_too_old ( person , year ) :
family_handle = person . get_main_parents_family_handle ( )
if family_handle :
family = db . get_family_from_handle ( family_handle )
father_handle = family . get_father_handle ( )
if father_handle :
father = db . get_person_from_handle ( father_handle )
if father . birth_handle :
father_birth = db . get_event_from_handle ( father . birth_handle )
dobj = father_birth . get_date_object ( )
if dobj . get_start_date ( ) != Date . EMPTY :
if not not_too_old ( dobj , year - average_generation_gap ) :
#print father.get_primary_name().get_name(), " father of ", person.get_primary_name().get_name(), " is too old by birth. birth year ", dobj.get_year(), " test year ", year - average_generation_gap
return True
#else:
#print father.get_primary_name().get_name(), " father of ", person.get_primary_name().get_name(), " is NOT too old by birth. birth year ", dobj.get_year(), " test year ", year - average_generation_gap
if father . death_handle :
father_death = db . get_event_from_handle ( father . death_handle )
dobj = father_death . get_date_object ( )
if dobj . get_start_date ( ) != Date . EMPTY :
if dobj . get_year ( ) < year - average_generation_gap :
#print father.get_primary_name().get_name(), " father of ", person.get_primary_name().get_name(), " is too old by death."
return True
if ancestors_too_old ( father , year - average_generation_gap ) :
return True
mother_handle = family . get_mother_handle ( )
if mother_handle :
mother = db . get_person_from_handle ( mother_handle )
if mother . birth_handle :
mother_birth = db . get_event_from_handle ( mother . birth_handle )
dobj = mother_birth . get_date_object ( )
if dobj . get_start_date ( ) != Date . EMPTY :
if not not_too_old ( dobj , year - average_generation_gap ) :
#print mother.get_primary_name().get_name(), " mother of ", person.get_primary_name().get_name(), " is too old by birth. birth year ", dobj.get_year(), " test year ", year - average_generation_gap
return True
#else:
#print mother.get_primary_name().get_name(), " mother of ", person.get_primary_name().get_name(), " is NOT too old by birth. birth year ", dobj.get_year(), " test year ", year - average_generation_gap
if mother . death_handle :
mother_death = db . get_event_from_handle ( mother . death_handle )
dobj = mother_death . get_date_object ( )
if dobj . get_start_date ( ) != Date . EMPTY :
if dobj . get_year ( ) < year - average_generation_gap :
#print mother.get_primary_name().get_name(), " mother of ", person.get_primary_name().get_name(), " is too old by death."
return True
if ancestors_too_old ( mother , year - average_generation_gap ) :
return True
2004-10-08 03:59:55 +00:00
return False
2005-04-08 08:09:18 +00:00
# If there are ancestors that would be too old in the current year
# then assume our person must be dead too.
if ancestors_too_old ( person , current_year ) :
#print person.get_primary_name().get_name(), " is dead because ancestors are too old."
return False
# If we can't find any reason to believe that they are dead we
# must assume they are alive.
#print person.get_primary_name().get_name(), " is probably alive."
return True
2004-10-08 03:59:55 +00:00
2005-04-06 15:52:52 +00:00
def not_too_old ( date , current_year = None ) :
if not current_year :
time_struct = time . localtime ( time . time ( ) )
current_year = time_struct [ 0 ]
2004-10-10 23:22:12 +00:00
year = date . get_year ( )
2005-04-06 20:56:24 +00:00
if year > current_year :
return False
2004-10-10 23:22:12 +00:00
return not ( year != 0 and current_year - year > 110 )
2005-03-11 21:05:46 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
def get_source_referents ( source_handle , db ) :
"""
Find objects that refer the source .
This function finds all primary objects that refer ( directly or through
secondary child - objects ) to a given source handle in a given database .
"""
# Persons
person_list = [ handle \
for handle in db . get_person_handles ( sort_handles = False ) \
if db . get_person_from_handle ( handle ) . has_source_reference ( source_handle )
]
# Families
family_list = [ handle for handle in db . get_family_handles ( ) \
if db . get_family_from_handle ( handle ) . has_source_reference ( source_handle )
]
# Events
event_list = [ handle for handle in db . get_event_handles ( ) \
if db . get_event_from_handle ( handle ) . has_source_reference ( source_handle )
]
# Places
place_list = [ handle for handle in db . get_place_handles ( ) \
if db . get_place_from_handle ( handle ) . has_source_reference ( source_handle )
]
# Sources
source_list = [ handle for handle in db . get_source_handles ( ) \
if db . get_source_from_handle ( handle ) . has_source_reference ( source_handle )
]
# Media Objects
media_list = [ handle for handle in db . get_media_object_handles ( ) \
if db . get_object_from_handle ( handle ) . has_source_reference ( source_handle )
]
return ( person_list , family_list , event_list ,
place_list , source_list , media_list )
2005-03-12 00:44:11 +00:00
def get_media_referents ( media_handle , db ) :
"""
Find objects that refer the media object .
This function finds all primary objects that refer
to a given media handle in a given database .
"""
# Persons
person_list = [ handle \
for handle in db . get_person_handles ( sort_handles = False ) \
if media_handle in \
[ photo . get_reference_handle ( ) for photo \
in db . get_person_from_handle ( handle ) . get_media_list ( ) ]
]
# Families
family_list = [ handle for handle in db . get_family_handles ( ) \
if media_handle in \
[ photo . get_reference_handle ( ) for photo \
in db . get_family_from_handle ( handle ) . get_media_list ( ) ]
]
# Events
event_list = [ handle for handle in db . get_event_handles ( ) \
if media_handle in \
[ photo . get_reference_handle ( ) for photo \
in db . get_event_from_handle ( handle ) . get_media_list ( ) ]
]
# Places
place_list = [ handle for handle in db . get_place_handles ( ) \
if media_handle in \
[ photo . get_reference_handle ( ) for photo \
in db . get_place_from_handle ( handle ) . get_media_list ( ) ]
]
# Sources
source_list = [ handle for handle in db . get_source_handles ( ) \
if media_handle in \
[ photo . get_reference_handle ( ) for photo \
in db . get_source_from_handle ( handle ) . get_media_list ( ) ]
]
return ( person_list , family_list , event_list , place_list , source_list )
2004-07-09 18:19:47 +00:00
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
* src/Plugins.py: Add native_format flag to import plugin registration.
* src/ReadXML.py, src/ReadGedcom.py: Register as native formats
to prevent loading twice on File->Open.
* src/data/gramps.schemas: Add keys for last import and export dirs.
* src/GrampsCfg.py (get_last_import_dir, save_last_import_dir,
get_last_export_dir, save_last_export_dir): Add functions.
* src/Exportder.py (suggest_filename): Try last export and last
import folders before falling back to Home; (save): Save export folder.
* src/Utils.py (get_new_filename): Add optional folder argument.
* src/DbPrompter.py (ExistingDbPrompter.chooser): Only add
importers for non-native formats, the rest is already taken care of;
Try last file, last import, last export, then home folders;
(ImportDbPrompter.chooser): Save import folder; Try last import,
last file, last export, then home folders.
(NewNativeDbPrompter): Try last file, last import, last export folders,
then fall back to home.
svn: r3493
2004-08-23 22:05:55 +00:00
_NEW_NAME_PATTERN = ' %s Untitled_ %d . %s '
2004-07-09 18:19:47 +00:00
* src/Plugins.py: Add native_format flag to import plugin registration.
* src/ReadXML.py, src/ReadGedcom.py: Register as native formats
to prevent loading twice on File->Open.
* src/data/gramps.schemas: Add keys for last import and export dirs.
* src/GrampsCfg.py (get_last_import_dir, save_last_import_dir,
get_last_export_dir, save_last_export_dir): Add functions.
* src/Exportder.py (suggest_filename): Try last export and last
import folders before falling back to Home; (save): Save export folder.
* src/Utils.py (get_new_filename): Add optional folder argument.
* src/DbPrompter.py (ExistingDbPrompter.chooser): Only add
importers for non-native formats, the rest is already taken care of;
Try last file, last import, last export, then home folders;
(ImportDbPrompter.chooser): Save import folder; Try last import,
last file, last export, then home folders.
(NewNativeDbPrompter): Try last file, last import, last export folders,
then fall back to home.
svn: r3493
2004-08-23 22:05:55 +00:00
def get_new_filename ( ext , folder = ' ~/ ' ) :
2004-07-09 18:19:47 +00:00
ix = 1
* src/Plugins.py: Add native_format flag to import plugin registration.
* src/ReadXML.py, src/ReadGedcom.py: Register as native formats
to prevent loading twice on File->Open.
* src/data/gramps.schemas: Add keys for last import and export dirs.
* src/GrampsCfg.py (get_last_import_dir, save_last_import_dir,
get_last_export_dir, save_last_export_dir): Add functions.
* src/Exportder.py (suggest_filename): Try last export and last
import folders before falling back to Home; (save): Save export folder.
* src/Utils.py (get_new_filename): Add optional folder argument.
* src/DbPrompter.py (ExistingDbPrompter.chooser): Only add
importers for non-native formats, the rest is already taken care of;
Try last file, last import, last export, then home folders;
(ImportDbPrompter.chooser): Save import folder; Try last import,
last file, last export, then home folders.
(NewNativeDbPrompter): Try last file, last import, last export folders,
then fall back to home.
svn: r3493
2004-08-23 22:05:55 +00:00
while os . path . isfile ( os . path . expanduser ( _NEW_NAME_PATTERN % ( folder , ix , ext ) ) ) :
2004-07-09 18:19:47 +00:00
ix = ix + 1
* src/Plugins.py: Add native_format flag to import plugin registration.
* src/ReadXML.py, src/ReadGedcom.py: Register as native formats
to prevent loading twice on File->Open.
* src/data/gramps.schemas: Add keys for last import and export dirs.
* src/GrampsCfg.py (get_last_import_dir, save_last_import_dir,
get_last_export_dir, save_last_export_dir): Add functions.
* src/Exportder.py (suggest_filename): Try last export and last
import folders before falling back to Home; (save): Save export folder.
* src/Utils.py (get_new_filename): Add optional folder argument.
* src/DbPrompter.py (ExistingDbPrompter.chooser): Only add
importers for non-native formats, the rest is already taken care of;
Try last file, last import, last export, then home folders;
(ImportDbPrompter.chooser): Save import folder; Try last import,
last file, last export, then home folders.
(NewNativeDbPrompter): Try last file, last import, last export folders,
then fall back to home.
svn: r3493
2004-08-23 22:05:55 +00:00
return os . path . expanduser ( _NEW_NAME_PATTERN % ( folder , ix , ext ) )
2004-12-22 01:56:37 +00:00
def get_type_converter ( val ) :
"""
Returns function that converts strings into the type of val .
"""
val_type = type ( val )
if val_type in ( str , unicode ) :
return unicode
elif val_type == int :
return int
elif val_type == float :
return float
elif val_type in ( list , tuple ) :
return list
def type_name ( val ) :
"""
Returns the name the type of val .
Only numbers and strings are supported .
The rest becomes strings ( unicode ) .
"""
val_type = type ( val )
if val_type == int :
return ' int '
elif val_type == float :
return ' float '
elif val_type in ( str , unicode ) :
return ' unicode '
return ' unicode '
def get_type_converter_by_name ( val_str ) :
"""
Returns function that converts strings into the type given by val_str .
Only numbers and strings are supported .
The rest becomes strings ( unicode ) .
"""
if val_str == ' int ' :
return int
elif val_str == ' float ' :
return float
elif val_str in ( ' str ' , ' unicode ' ) :
return unicode
return unicode
2005-06-06 05:03:48 +00:00
def strip_context ( msgid , sep = ' | ' ) :
"""
Strip the context used for resolving translation ambiguities .
The translation of msgid is returned unless the translation is
not available and the msgid contains the separator . In that case ,
the returned value is the portion of msgid following the last
separator . Default separator is ' | ' .
@param msgid : The string to translated .
@type msgid : unicode
@param sep : The separator marking the context .
@type sep : unicode
@return : Translation or the original with context stripped .
@rtype : unicode
"""
msgval = _ ( msgid )
sep_idx = msgid . rfind ( sep )
if msgval == msgid and sep_idx != - 1 :
msgval = msgid [ sep_idx + 1 : ]
return msgval
2005-08-03 19:25:03 +00:00
class ProgressMeter :
"""
Progress meter class for GRAMPS .
"""
2005-08-03 21:10:11 +00:00
def __init__ ( self , title , header = ' ' ) :
2005-08-03 19:25:03 +00:00
"""
Specify the title and the current pass header .
"""
self . ptop = gtk . Dialog ( )
self . ptop . set_has_separator ( False )
self . ptop . set_title ( title )
self . ptop . set_border_width ( 12 )
self . ptop . vbox . set_spacing ( 10 )
lbl = gtk . Label ( ' <span size= " larger " weight= " bold " > %s </span> ' % title )
lbl . set_use_markup ( True )
self . lbl = gtk . Label ( header )
self . lbl . set_use_markup ( True )
self . ptop . vbox . add ( lbl )
self . ptop . vbox . add ( self . lbl )
self . ptop . vbox . set_border_width ( 24 )
self . pbar = gtk . ProgressBar ( )
self . ptop . set_size_request ( 350 , 125 )
self . ptop . vbox . add ( self . pbar )
self . ptop . show_all ( )
2005-08-03 21:10:11 +00:00
if header == ' ' :
self . lbl . hide ( )
2005-08-03 19:25:03 +00:00
def set_pass ( self , header , total ) :
"""
Reset for another pass . Provide a new header and define number
of steps to be used .
"""
2005-08-03 21:10:11 +00:00
if header == ' ' :
self . lbl . hide ( )
else :
self . lbl . show ( )
2005-08-03 19:25:03 +00:00
self . pbar_max = total
self . pbar_index = 0.0
self . lbl . set_text ( header )
self . pbar . set_fraction ( 0.0 )
while gtk . events_pending ( ) :
gtk . main_iteration ( )
def step ( self ) :
""" Click the progress bar over to the next value. Be paranoid
and insure that it doesn ' t go over 100 % . " " "
self . pbar_index = self . pbar_index + 1.0
if ( self . pbar_index > self . pbar_max ) :
self . pbar_index = self . pbar_max
val = self . pbar_index / self . pbar_max
self . pbar . set_text ( " %d of %d ( %.1f %% ) " % ( self . pbar_index , self . pbar_max , ( val * 100 ) ) )
self . pbar . set_fraction ( val )
while gtk . events_pending ( ) :
gtk . main_iteration ( )
def close ( self ) :
"""
Close the progress meter
"""
self . ptop . destroy ( )