2002-10-20 14:25:16 +00:00
|
|
|
#
|
|
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
|
|
#
|
2007-06-28 05:41:40 +00:00
|
|
|
# Copyright (C) 2000-2007 Donald N. Allingham
|
2009-07-12 19:45:17 +00:00
|
|
|
# Copyright (C) 2009 Gary Burton
|
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.
|
|
|
|
#
|
2007-09-10 03:03:46 +00:00
|
|
|
# This program is distributed in the hope that it will be useful,
|
2002-10-20 14:25:16 +00:00
|
|
|
# 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$
|
|
|
|
|
2009-07-03 22:00:58 +00:00
|
|
|
"""
|
|
|
|
Non GUI/GTK related utility functions
|
|
|
|
"""
|
|
|
|
|
2002-10-20 14:25:16 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Standard python modules
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
import os
|
2005-12-06 06:38:09 +00:00
|
|
|
import sys
|
2003-06-15 00:05:43 +00:00
|
|
|
import locale
|
2006-03-22 23:03:57 +00:00
|
|
|
import random
|
|
|
|
import time
|
2009-07-03 22:00:58 +00:00
|
|
|
import shutil
|
|
|
|
import platform
|
2002-10-20 14:25:16 +00:00
|
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Gramps modules
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2007-06-28 05:41:40 +00:00
|
|
|
from BasicUtils import name_displayer
|
2007-10-08 16:41:39 +00:00
|
|
|
import gen.lib
|
2005-12-06 06:38:09 +00:00
|
|
|
import Errors
|
2009-07-03 22:00:58 +00:00
|
|
|
from GrampsLocale import codeset
|
2002-10-20 14:25:16 +00:00
|
|
|
|
2009-07-06 09:54:30 +00:00
|
|
|
from const import TEMP_DIR, USER_HOME, WINDOWS, MACOS, LINUX
|
2009-07-03 22:00:58 +00:00
|
|
|
from TransUtils import sgettext as _
|
2007-10-14 03:29:12 +00:00
|
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Constants from config .ini keys
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#obtain the values once, they do not change!
|
|
|
|
try:
|
|
|
|
import Config
|
|
|
|
_MAX_AGE_PROB_ALIVE = Config.get(Config.MAX_AGE_PROB_ALIVE)
|
|
|
|
_MAX_SIB_AGE_DIFF = Config.get(Config.MAX_SIB_AGE_DIFF)
|
|
|
|
_MIN_GENERATION_YEARS = Config.get(Config.MIN_GENERATION_YEARS)
|
|
|
|
_AVG_GENERATION_GAP = Config.get(Config.AVG_GENERATION_GAP)
|
|
|
|
except ImportError:
|
|
|
|
# Utils used as module not part of GRAMPS
|
|
|
|
_MAX_AGE_PROB_ALIVE = 110
|
|
|
|
_MAX_SIB_AGE_DIFF = 20
|
|
|
|
_MIN_GENERATION_YEARS = 13
|
|
|
|
_AVG_GENERATION_GAP = 20
|
|
|
|
|
2002-10-20 14:25:16 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
2005-05-30 20:41:43 +00:00
|
|
|
# Integer to String mappings for constants
|
2002-10-20 14:25:16 +00:00
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2005-05-30 20:41:43 +00:00
|
|
|
gender = {
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.Person.MALE : _("male"),
|
|
|
|
gen.lib.Person.FEMALE : _("female"),
|
|
|
|
gen.lib.Person.UNKNOWN : _("unknown"),
|
2005-05-30 20:41:43 +00:00
|
|
|
}
|
2008-01-17 20:24:44 +00:00
|
|
|
|
2006-03-22 21:19:32 +00:00
|
|
|
def format_gender( type):
|
2007-09-10 03:03:46 +00:00
|
|
|
return gender.get(type[0], _("Invalid"))
|
2005-05-30 20:41:43 +00:00
|
|
|
|
|
|
|
confidence = {
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.SourceRef.CONF_VERY_HIGH : _("Very High"),
|
|
|
|
gen.lib.SourceRef.CONF_HIGH : _("High"),
|
|
|
|
gen.lib.SourceRef.CONF_NORMAL : _("Normal"),
|
|
|
|
gen.lib.SourceRef.CONF_LOW : _("Low"),
|
|
|
|
gen.lib.SourceRef.CONF_VERY_LOW : _("Very Low"),
|
2005-05-30 20:41:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
family_rel_descriptions = {
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.FamilyRelType.MARRIED : _("A legal or common-law relationship "
|
2007-09-10 03:03:46 +00:00
|
|
|
"between a husband and wife"),
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.FamilyRelType.UNMARRIED : _("No legal or common-law relationship "
|
2007-09-10 03:03:46 +00:00
|
|
|
"between man and woman"),
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.FamilyRelType.CIVIL_UNION : _("An established relationship between "
|
2007-09-10 03:03:46 +00:00
|
|
|
"members of the same sex"),
|
2007-10-08 16:41:39 +00:00
|
|
|
gen.lib.FamilyRelType.UNKNOWN : _("Unknown relationship between a man "
|
2007-09-10 03:03:46 +00:00
|
|
|
"and woman"),
|
2008-10-06 12:36:03 +00:00
|
|
|
gen.lib.FamilyRelType.CUSTOM : _("An unspecified relationship between "
|
2007-09-10 03:03:46 +00:00
|
|
|
"a man and woman"),
|
2005-05-30 20:41:43 +00:00
|
|
|
}
|
|
|
|
|
2006-03-22 21:19:32 +00:00
|
|
|
|
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.')
|
|
|
|
|
2005-12-06 06:38:09 +00:00
|
|
|
def fix_encoding(value):
|
2008-05-25 19:55:47 +00:00
|
|
|
if not isinstance(value, unicode):
|
2005-12-06 06:38:09 +00:00
|
|
|
try:
|
|
|
|
return unicode(value)
|
|
|
|
except:
|
2007-01-27 23:08:08 +00:00
|
|
|
try:
|
|
|
|
codeset = locale.getpreferredencoding()
|
|
|
|
except:
|
|
|
|
codeset = "UTF-8"
|
2007-09-10 03:03:46 +00:00
|
|
|
return unicode(value, codeset)
|
2005-12-06 06:38:09 +00:00
|
|
|
else:
|
|
|
|
return value
|
|
|
|
|
2005-12-13 02:07:16 +00:00
|
|
|
def xml_lang():
|
2007-09-10 03:03:46 +00:00
|
|
|
loc = locale.getlocale()
|
2008-06-16 15:01:46 +00:00
|
|
|
if loc[0] is None:
|
2005-12-13 02:07:16 +00:00
|
|
|
return ""
|
|
|
|
else:
|
2007-09-10 03:03:46 +00:00
|
|
|
return loc[0].replace('_', '-')
|
2005-12-13 02:07:16 +00:00
|
|
|
|
2003-04-04 05:48:25 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# force_unicode
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
def force_unicode(n):
|
2008-05-25 19:55:47 +00:00
|
|
|
if not isinstance(n, unicode):
|
2007-09-10 03:03:46 +00:00
|
|
|
return (unicode(n).lower(), unicode(n))
|
2003-04-04 05:48:25 +00:00
|
|
|
else:
|
2007-09-10 03:03:46 +00:00
|
|
|
return (n.lower(), n)
|
2003-04-04 05:48:25 +00:00
|
|
|
|
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
|
|
|
|
2007-04-01 22:37:10 +00:00
|
|
|
def family_name(family, db, noname=_("unknown")):
|
2002-10-20 14:25:16 +00:00
|
|
|
"""Builds a name for the family from the parents names"""
|
2007-04-01 22:37:10 +00:00
|
|
|
|
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:
|
2007-06-28 05:41:40 +00:00
|
|
|
fname = name_displayer.display(father)
|
|
|
|
mname = name_displayer.display(mother)
|
2005-03-11 21:05:46 +00:00
|
|
|
name = _("%(father)s and %(mother)s") % {
|
2007-09-10 03:03:46 +00:00
|
|
|
"father" : fname,
|
2005-03-11 21:05:46 +00:00
|
|
|
"mother" : mname}
|
2002-10-20 14:25:16 +00:00
|
|
|
elif father:
|
2007-06-28 05:41:40 +00:00
|
|
|
name = name_displayer.display(father)
|
2005-03-13 22:10:40 +00:00
|
|
|
elif mother:
|
2007-06-28 05:41:40 +00:00
|
|
|
name = name_displayer.display(mother)
|
2005-03-13 22:10:40 +00:00
|
|
|
else:
|
2006-04-20 04:59:04 +00:00
|
|
|
name = noname
|
2002-10-20 14:25:16 +00:00
|
|
|
return name
|
|
|
|
|
2007-04-01 22:37:10 +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()
|
2007-12-16 17:35:16 +00:00
|
|
|
name = _("%(father)s and %(mother)s") % {
|
2007-09-10 03:03:46 +00:00
|
|
|
'father' : fname,
|
2007-06-25 04:57:53 +00:00
|
|
|
'mother' : mname
|
|
|
|
}
|
2003-10-12 04:26:00 +00:00
|
|
|
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
|
2009-07-03 22:00:58 +00:00
|
|
|
|
2009-07-06 09:54:30 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Platform determination functions
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
def lin():
|
|
|
|
"""
|
|
|
|
Return True if a linux system
|
|
|
|
Note: Normally do as linux in else statement of a check !
|
|
|
|
"""
|
|
|
|
if platform.system() in LINUX:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def mac():
|
|
|
|
"""
|
|
|
|
Return True if a Macintosh system
|
|
|
|
"""
|
|
|
|
if platform.system() in MACOS:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def win():
|
|
|
|
"""
|
|
|
|
Return True if a windows system
|
|
|
|
"""
|
|
|
|
if platform.system() in WINDOWS:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2009-07-03 22:00:58 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# String Encoding functions
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
def encodingdefs():
|
|
|
|
"""
|
|
|
|
4 functions are defined to obtain a byte string that can be used as
|
|
|
|
sort key and is locale aware. Do not print the sortkey, it is a sortable
|
|
|
|
string, but is not a human readable correct string!
|
|
|
|
When gtk is defined, one can avoid some function calls as then the default
|
|
|
|
python encoding is not ascii but utf-8, so use the gtk functions in those
|
|
|
|
cases.
|
|
|
|
|
|
|
|
conv_utf8_tosrtkey: convert a utf8 encoded string to sortkey usable string
|
|
|
|
|
|
|
|
conv_unicode_tosrtkey: convert a unicode object to sortkey usable string
|
|
|
|
|
|
|
|
conv_utf8_tosrtkey_ongtk: convert a utf8 encoded string to sortkey usable
|
|
|
|
string when gtk is loaded or utf-8 is default python encoding
|
|
|
|
|
|
|
|
conv_unicode_tosrtkey_ongtk: convert a unicode object to sortkey usable
|
|
|
|
string when gtk is loaded or utf-8 is default python encoding
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
2009-07-04 10:21:10 +00:00
|
|
|
if platform.system() in WINDOWS:
|
2009-07-03 22:00:58 +00:00
|
|
|
# python encoding is ascii, but C functions need to recieve the
|
|
|
|
# windows codeset, so convert over to it
|
|
|
|
conv_utf8_tosrtkey = lambda x: locale.strxfrm(x.decode("utf-8").encode(
|
|
|
|
codeset))
|
|
|
|
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x.encode(codeset))
|
|
|
|
#when gtk is imported the python defaultencoding is utf-8,
|
|
|
|
#so no need to specify it
|
|
|
|
conv_utf8_tosrtkey_ongtk = lambda x: locale.strxfrm(unicode(x).encode(
|
|
|
|
codeset))
|
|
|
|
conv_unicode_tosrtkey_ongtk = lambda x: locale.strxfrm(x.encode(codeset))
|
|
|
|
else:
|
|
|
|
# on unix C functions need to recieve utf-8. Default conversion would
|
|
|
|
# use ascii, so it is needed to be explicit about the resulting encoding
|
|
|
|
conv_utf8_tosrtkey = lambda x: locale.strxfrm(x)
|
|
|
|
conv_unicode_tosrtkey = lambda x: locale.strxfrm(x.encode("utf-8"))
|
|
|
|
# when gtk loaded, default encoding (sys.getdefaultencoding ) is utf-8,
|
|
|
|
# so default conversion happens with utf-8
|
|
|
|
conv_utf8_tosrtkey_ongtk = lambda x: locale.strxfrm(x)
|
|
|
|
conv_unicode_tosrtkey_ongtk = lambda x: locale.strxfrm(x)
|
2002-10-20 14:25:16 +00:00
|
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2005-12-06 06:38:09 +00:00
|
|
|
def find_file( filename):
|
|
|
|
# try the filename we got
|
|
|
|
try:
|
|
|
|
fname = filename
|
|
|
|
if os.path.isfile( filename):
|
|
|
|
return( filename)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2006-05-24 05:28:33 +00:00
|
|
|
# Build list of alternate encodings
|
2007-01-27 23:08:08 +00:00
|
|
|
encodings = set()
|
|
|
|
|
|
|
|
for enc in [sys.getfilesystemencoding, locale.getpreferredencoding]:
|
|
|
|
try:
|
|
|
|
encodings.add(enc)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
encodings.add('UTF-8')
|
|
|
|
encodings.add('ISO-8859-1')
|
|
|
|
|
2005-12-06 06:38:09 +00:00
|
|
|
for enc in encodings:
|
|
|
|
try:
|
|
|
|
fname = filename.encode(enc)
|
|
|
|
if os.path.isfile( fname):
|
|
|
|
return fname
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# not found
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def find_folder( filename):
|
|
|
|
# try the filename we got
|
|
|
|
try:
|
|
|
|
fname = filename
|
|
|
|
if os.path.isdir( filename):
|
|
|
|
return( filename)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Build list of elternate encodings
|
2006-03-19 03:25:31 +00:00
|
|
|
try:
|
2007-09-10 03:03:46 +00:00
|
|
|
encodings = [sys.getfilesystemencoding(),
|
|
|
|
locale.getpreferredencoding(),
|
2006-03-19 03:25:31 +00:00
|
|
|
'UTF-8', 'ISO-8859-1']
|
|
|
|
except:
|
|
|
|
encodings = [sys.getfilesystemencoding(), 'UTF-8', 'ISO-8859-1']
|
|
|
|
encodings = list(set(encodings))
|
2005-12-06 06:38:09 +00:00
|
|
|
for enc in encodings:
|
|
|
|
try:
|
|
|
|
fname = filename.encode(enc)
|
|
|
|
if os.path.isdir( fname):
|
|
|
|
return fname
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# not found
|
|
|
|
return ''
|
|
|
|
|
2007-11-16 16:45:45 +00:00
|
|
|
def get_unicode_path(path):
|
|
|
|
"""
|
|
|
|
Return the Unicode version of a path string.
|
|
|
|
|
2009-06-24 21:56:07 +00:00
|
|
|
:type path: str
|
|
|
|
:param path: The path to be converted to Unicode
|
|
|
|
:rtype: unicode
|
|
|
|
:returns: The Unicode version of path.
|
2007-11-16 16:45:45 +00:00
|
|
|
"""
|
|
|
|
if os.sys.platform == "win32":
|
|
|
|
return unicode(path)
|
|
|
|
else:
|
|
|
|
return unicode(path,sys.getfilesystemencoding())
|
|
|
|
|
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.
|
2007-09-10 03:03:46 +00:00
|
|
|
Call func(data, pid) for the Id of each person encountered.
|
2003-01-10 05:21:32 +00:00
|
|
|
Exit and return 1, as soon as func returns true.
|
|
|
|
Return 0 otherwise.
|
|
|
|
"""
|
|
|
|
todo = start
|
2007-09-10 03:03:46 +00:00
|
|
|
done_ids = set()
|
2003-01-10 05:21:32 +00:00
|
|
|
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.
|
2007-09-10 03:03:46 +00:00
|
|
|
if p_handle in done_ids:
|
2003-01-10 05:21:32 +00:00
|
|
|
continue
|
2007-09-10 03:03:46 +00:00
|
|
|
done_ids.add(p_handle)
|
|
|
|
if func(data, p_handle):
|
2003-01-10 05:21:32 +00:00
|
|
|
return 1
|
2006-04-13 19:29:36 +00:00
|
|
|
for fam_handle in p.get_parent_family_handle_list():
|
2005-01-11 22:41:15 +00:00
|
|
|
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
|
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def set_title_label(xmlobj, t):
|
2003-03-05 06:01:31 +00:00
|
|
|
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
|
|
|
|
2006-04-23 22:18:01 +00:00
|
|
|
from warnings import warn
|
2007-09-10 03:03:46 +00:00
|
|
|
def set_titles(window, title, t, msg=None):
|
2006-04-23 22:18:01 +00:00
|
|
|
warn('The Utils.set_titles is deprecated. Use ManagedWindow methods')
|
2003-05-10 06:17:07 +00:00
|
|
|
|
2003-11-25 17:45:34 +00:00
|
|
|
def search_for(name):
|
2006-06-18 20:58:25 +00:00
|
|
|
if name.startswith( '"' ):
|
|
|
|
name = name.split('"')[1]
|
|
|
|
else:
|
|
|
|
name = name.split()[0]
|
2006-07-22 05:52:16 +00:00
|
|
|
if os.sys.platform == "win32":
|
|
|
|
for i in os.environ['PATH'].split(';'):
|
2007-09-10 03:03:46 +00:00
|
|
|
fname = os.path.join(i, name)
|
|
|
|
if os.access(fname, os.X_OK) and not os.path.isdir(fname):
|
2006-07-22 05:52:16 +00:00
|
|
|
return 1
|
|
|
|
else:
|
|
|
|
for i in os.environ['PATH'].split(':'):
|
2007-09-10 03:03:46 +00:00
|
|
|
fname = os.path.join(i, name)
|
|
|
|
if os.access(fname, os.X_OK) and not os.path.isdir(fname):
|
2006-07-22 05:52:16 +00:00
|
|
|
return 1
|
2003-11-25 17:45:34 +00:00
|
|
|
return 0
|
2004-04-25 04:48:02 +00:00
|
|
|
|
2004-06-30 04:06:10 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# create_id
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
rand = random.Random(time.time())
|
|
|
|
|
2004-06-27 03:10:06 +00:00
|
|
|
def create_id():
|
2007-09-10 03:03:46 +00:00
|
|
|
return "%08x%08x" % ( int(time.time()*10000),
|
|
|
|
rand.randint(0, sys.maxint))
|
2004-10-08 03:59:55 +00:00
|
|
|
|
2007-11-21 14:49:50 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# probably_alive
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
def probably_alive(person, db, current_date=None, limit=0):
|
2008-02-24 13:55:55 +00:00
|
|
|
"""Return true if the person may be alive on current_date.
|
2005-04-08 08:09:18 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
|
2009-06-24 21:56:07 +00:00
|
|
|
:param current_date: a date object that is not estimated or modified
|
2007-11-21 14:49:50 +00:00
|
|
|
(defaults to today)
|
2009-06-24 21:56:07 +00:00
|
|
|
:param limit: number of years to check beyond death_date
|
2005-04-08 08:09:18 +00:00
|
|
|
"""
|
2008-06-16 15:01:46 +00:00
|
|
|
if current_date is None:
|
2007-11-21 14:49:50 +00:00
|
|
|
current_date = gen.lib.Date()
|
|
|
|
# yr, mon, day:
|
|
|
|
current_date.set_yr_mon_day(*time.localtime(time.time())[0:3])
|
|
|
|
|
|
|
|
death_date = None
|
2005-04-08 08:09:18 +00:00
|
|
|
# If the recorded death year is before current year then
|
|
|
|
# things are simple.
|
2006-05-23 20:31:39 +00:00
|
|
|
death_ref = person.get_death_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if death_ref and death_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
death = db.get_event_from_handle(death_ref.ref)
|
2007-10-08 16:41:39 +00:00
|
|
|
if death.get_date_object().get_start_date() != gen.lib.Date.EMPTY:
|
2007-11-21 14:49:50 +00:00
|
|
|
death_date = death.get_date_object()
|
2007-11-21 20:05:18 +00:00
|
|
|
if death_date.copy_offset_ymd(year=limit).match(current_date, "<<"):
|
2005-04-06 20:56:24 +00:00
|
|
|
return False
|
2005-12-06 06:38:09 +00:00
|
|
|
|
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.
|
2006-11-21 16:20:35 +00:00
|
|
|
for ev_ref in person.get_primary_event_ref_list():
|
2005-06-20 22:18:22 +00:00
|
|
|
ev = db.get_event_from_handle(ev_ref.ref)
|
2008-06-02 00:59:14 +00:00
|
|
|
if ev and ev.type in [gen.lib.EventType.CAUSE_DEATH,
|
|
|
|
gen.lib.EventType.BURIAL,
|
|
|
|
gen.lib.EventType.CREMATION]:
|
2007-11-21 14:49:50 +00:00
|
|
|
if not death_date:
|
|
|
|
death_date = ev.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if ev.get_date_object().get_start_date() != gen.lib.Date.EMPTY:
|
2007-11-21 20:05:18 +00:00
|
|
|
if ev.get_date_object().copy_offset_ymd(year=limit).match(current_date,"<<"):
|
2005-04-06 20:56:24 +00:00
|
|
|
return False
|
2005-12-13 02:07:16 +00:00
|
|
|
# For any other event of this person, check whether it happened
|
|
|
|
# too long ago. If so then the person is likely dead now.
|
2007-11-21 14:49:50 +00:00
|
|
|
elif ev and too_old(ev.get_date_object(), current_date.get_year()):
|
2005-12-13 02:07:16 +00:00
|
|
|
return False
|
2004-10-08 03:59:55 +00:00
|
|
|
|
2007-11-21 14:49:50 +00:00
|
|
|
birth_date = 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).
|
2006-05-23 20:31:39 +00:00
|
|
|
birth_ref = person.get_birth_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if birth_ref and birth_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
birth = db.get_event_from_handle(birth_ref.ref)
|
2008-04-19 16:48:46 +00:00
|
|
|
if (birth.get_date_object().get_start_date() != gen.lib.Date.EMPTY):
|
2007-11-21 14:49:50 +00:00
|
|
|
if not birth_date:
|
|
|
|
birth_date = birth.get_date_object()
|
2005-12-13 02:07:16 +00:00
|
|
|
# Check whether the birth event is too old because the
|
|
|
|
# code above did not look at birth, only at other events
|
2008-04-19 16:48:46 +00:00
|
|
|
birth_obj = birth.get_date_object()
|
|
|
|
if birth_obj.get_valid():
|
|
|
|
# only if this is a valid birth:
|
|
|
|
if too_old(birth_obj, current_date.get_year()):
|
|
|
|
return False
|
|
|
|
if not_too_old(birth_obj, current_date.get_year()):
|
|
|
|
return True
|
2005-04-08 09:40:15 +00:00
|
|
|
|
2007-11-21 14:49:50 +00:00
|
|
|
if not birth_date and death_date:
|
2007-11-21 20:05:18 +00:00
|
|
|
if death_date.match(current_date.copy_offset_ymd(year=_MAX_AGE_PROB_ALIVE), ">>"):
|
2007-10-14 03:29:12 +00:00
|
|
|
# person died more than MAX after current year
|
2005-04-08 09:40:15 +00:00
|
|
|
return False
|
2007-01-08 01:49:33 +00:00
|
|
|
|
|
|
|
# Neither birth nor death events are available. Try looking
|
2007-09-10 03:03:46 +00:00
|
|
|
# at siblings. If a sibling was born more than 120 years past,
|
2007-10-14 03:29:12 +00:00
|
|
|
# or more than 20 future, then probably this person is
|
|
|
|
# not alive. If the sibling died more than 120 years
|
2007-01-08 01:49:33 +00:00
|
|
|
# past, or more than 120 years future, then probably not alive.
|
|
|
|
|
|
|
|
family_list = person.get_parent_family_handle_list()
|
|
|
|
for family_handle in family_list:
|
|
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
for child_ref in family.get_child_ref_list():
|
|
|
|
child_handle = child_ref.ref
|
|
|
|
child = db.get_person_from_handle(child_handle)
|
|
|
|
child_birth_ref = child.get_birth_ref()
|
|
|
|
if child_birth_ref:
|
|
|
|
child_birth = db.get_event_from_handle(child_birth_ref.ref)
|
|
|
|
dobj = child_birth.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-01-08 01:49:33 +00:00
|
|
|
# if sibling birth date too far away, then not alive:
|
|
|
|
year = dobj.get_year()
|
|
|
|
if year != 0:
|
2007-11-21 20:05:18 +00:00
|
|
|
if not (current_date.copy_offset_ymd(-(_MAX_AGE_PROB_ALIVE + _MAX_SIB_AGE_DIFF)).match(dobj,"<<") and
|
|
|
|
dobj.match(current_date.copy_offset_ymd(_MAX_SIB_AGE_DIFF),"<<")):
|
2007-01-08 01:49:33 +00:00
|
|
|
return False
|
|
|
|
child_death_ref = child.get_death_ref()
|
|
|
|
if child_death_ref:
|
|
|
|
child_death = db.get_event_from_handle(child_death_ref.ref)
|
|
|
|
dobj = child_death.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-01-08 01:49:33 +00:00
|
|
|
# if sibling death date too far away, then not alive:
|
|
|
|
year = dobj.get_year()
|
|
|
|
if year != 0:
|
2007-11-21 20:05:18 +00:00
|
|
|
if not (current_date.copy_offset_ymd(-(_MAX_AGE_PROB_ALIVE + _MAX_SIB_AGE_DIFF)).match(dobj,"<<") and
|
|
|
|
dobj.match(current_date.copy_offset_ymd(_MAX_AGE_PROB_ALIVE),"<<")):
|
2007-01-08 01:49:33 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
# Try looking for descendants that were born more than a lifespan
|
|
|
|
# ago.
|
2004-10-08 03:59:55 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
2006-04-13 19:29:36 +00:00
|
|
|
for child_ref in family.get_child_ref_list():
|
|
|
|
child_handle = child_ref.ref
|
2004-10-08 03:59:55 +00:00
|
|
|
child = db.get_person_from_handle(child_handle)
|
2006-05-23 20:31:39 +00:00
|
|
|
child_birth_ref = child.get_birth_ref()
|
|
|
|
if child_birth_ref:
|
|
|
|
child_birth = db.get_event_from_handle(child_birth_ref.ref)
|
2004-10-08 03:59:55 +00:00
|
|
|
dobj = child_birth.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
|
|
|
d = gen.lib.Date(dobj)
|
2004-10-29 00:49:40 +00:00
|
|
|
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)
|
2007-11-21 14:49:50 +00:00
|
|
|
if not not_too_old (d, current_date.get_year()):
|
2004-10-08 03:59:55 +00:00
|
|
|
return True
|
|
|
|
|
2006-05-23 20:31:39 +00:00
|
|
|
child_death_ref = child.get_death_ref()
|
|
|
|
if child_death_ref:
|
|
|
|
child_death = db.get_event_from_handle(child_death_ref.ref)
|
2004-10-08 03:59:55 +00:00
|
|
|
dobj = child_death.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-11-21 14:49:50 +00:00
|
|
|
if not not_too_old (dobj, current_date.get_year()):
|
2004-10-08 03:59:55 +00:00
|
|
|
return True
|
|
|
|
|
2007-10-14 03:29:12 +00:00
|
|
|
if descendants_too_old (child, years + _MIN_GENERATION_YEARS):
|
2004-10-08 03:59:55 +00:00
|
|
|
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.
|
|
|
|
|
2005-12-06 06:38:09 +00:00
|
|
|
try:
|
2009-07-12 19:45:17 +00:00
|
|
|
age_too_old = descendants_too_old(person, _MIN_GENERATION_YEARS)
|
|
|
|
# Set to None otherwise there is a memory leak as this function
|
|
|
|
# is recursive
|
|
|
|
descendants_too_old = None
|
|
|
|
if age_too_old:
|
2005-12-06 06:38:09 +00:00
|
|
|
return False
|
|
|
|
except RuntimeError:
|
|
|
|
raise Errors.DatabaseError(
|
|
|
|
_("Database error: %s is defined as his or her own ancestor") %
|
2007-06-28 05:41:40 +00:00
|
|
|
name_displayer.display(person))
|
2005-04-08 08:09:18 +00:00
|
|
|
|
2007-10-14 03:29:12 +00:00
|
|
|
def ancestors_too_old(person, year):
|
2005-04-08 08:09:18 +00:00
|
|
|
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)
|
2006-05-23 20:31:39 +00:00
|
|
|
father_birth_ref = father.get_birth_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if father_birth_ref and father_birth_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
father_birth = db.get_event_from_handle(
|
|
|
|
father_birth_ref.ref)
|
2005-04-08 08:09:18 +00:00
|
|
|
dobj = father_birth.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-10-14 03:29:12 +00:00
|
|
|
if not not_too_old (dobj, year - _AVG_GENERATION_GAP):
|
2005-04-08 08:09:18 +00:00
|
|
|
return True
|
|
|
|
|
2006-05-23 20:31:39 +00:00
|
|
|
father_death_ref = father.get_death_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if father_death_ref and father_death_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
father_death = db.get_event_from_handle(
|
|
|
|
father_death_ref.ref)
|
2005-04-08 08:09:18 +00:00
|
|
|
dobj = father_death.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-10-14 03:29:12 +00:00
|
|
|
if dobj.get_year() < year - _AVG_GENERATION_GAP:
|
2005-04-08 08:09:18 +00:00
|
|
|
return True
|
|
|
|
|
2007-10-14 03:29:12 +00:00
|
|
|
if ancestors_too_old (father, year - _AVG_GENERATION_GAP):
|
2005-04-08 08:09:18 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
mother_handle = family.get_mother_handle()
|
|
|
|
if mother_handle:
|
|
|
|
mother = db.get_person_from_handle(mother_handle)
|
2006-05-23 20:31:39 +00:00
|
|
|
mother_birth_ref = mother.get_birth_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if mother_birth_ref and mother_birth_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
mother_birth = db.get_event_from_handle(mother_birth_ref.ref)
|
2005-04-08 08:09:18 +00:00
|
|
|
dobj = mother_birth.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-10-14 03:29:12 +00:00
|
|
|
if not not_too_old (dobj, year - _AVG_GENERATION_GAP):
|
2005-04-08 08:09:18 +00:00
|
|
|
return True
|
|
|
|
|
2006-05-23 20:31:39 +00:00
|
|
|
mother_death_ref = mother.get_death_ref()
|
2007-10-08 16:41:39 +00:00
|
|
|
if mother_death_ref and mother_death_ref.get_role() == gen.lib.EventRoleType.PRIMARY:
|
2006-05-23 20:31:39 +00:00
|
|
|
mother_death = db.get_event_from_handle(
|
|
|
|
mother_death_ref.ref)
|
2005-04-08 08:09:18 +00:00
|
|
|
dobj = mother_death.get_date_object()
|
2007-10-08 16:41:39 +00:00
|
|
|
if dobj.get_start_date() != gen.lib.Date.EMPTY:
|
2007-10-14 03:29:12 +00:00
|
|
|
if dobj.get_year() < year - _AVG_GENERATION_GAP:
|
2005-04-08 08:09:18 +00:00
|
|
|
return True
|
|
|
|
|
2007-10-14 03:29:12 +00:00
|
|
|
if ancestors_too_old (mother, year - _AVG_GENERATION_GAP):
|
2005-04-08 08:09:18 +00:00
|
|
|
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.
|
2009-07-12 19:45:17 +00:00
|
|
|
# Set to None otherwise there is a memory leak as this function
|
|
|
|
# is recursive
|
|
|
|
age_too_old = ancestors_too_old (person, current_date.get_year())
|
|
|
|
ancestors_too_old = None
|
|
|
|
if age_too_old:
|
2005-04-08 08:09:18 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
# If we can't find any reason to believe that they are dead we
|
|
|
|
# must assume they are alive.
|
|
|
|
return True
|
2004-10-08 03:59:55 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def not_too_old(date, current_year=None):
|
2005-04-06 15:52:52 +00:00
|
|
|
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
|
2007-10-14 03:29:12 +00:00
|
|
|
return (year != 0 and current_year - year < _MAX_AGE_PROB_ALIVE)
|
2005-12-13 02:07:16 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def too_old(date, current_year=None):
|
2005-12-13 02:07:16 +00:00
|
|
|
if current_year:
|
|
|
|
the_current_year = current_year
|
|
|
|
else:
|
|
|
|
time_struct = time.localtime(time.time())
|
|
|
|
the_current_year = time_struct[0]
|
|
|
|
year = date.get_year()
|
|
|
|
if year > the_current_year:
|
|
|
|
return True
|
2007-10-14 03:29:12 +00:00
|
|
|
return (year != 0 and the_current_year - year > _MAX_AGE_PROB_ALIVE)
|
2004-10-10 23:22:12 +00:00
|
|
|
|
2005-03-11 21:05:46 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2007-03-27 18:28:26 +00:00
|
|
|
def get_referents(handle, db, primary_objects):
|
|
|
|
""" Find objects that refer to an object.
|
|
|
|
|
|
|
|
This function is the base for other get_<object>_referents finctions.
|
|
|
|
|
2005-03-11 21:05:46 +00:00
|
|
|
"""
|
2007-01-23 03:37:13 +00:00
|
|
|
# Use one pass through the reference map to grab all the references
|
2007-03-27 18:28:26 +00:00
|
|
|
object_list = [item for item in db.find_backlink_handles(handle)]
|
2007-01-23 03:37:13 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
# Then form the object-specific lists
|
|
|
|
the_lists = ()
|
2005-03-11 21:05:46 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
for primary in primary_objects:
|
|
|
|
primary_list = [item[1] for item in object_list if item[0] == primary]
|
2007-09-10 03:03:46 +00:00
|
|
|
the_lists = the_lists + (primary_list, )
|
2005-03-11 21:05:46 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
return the_lists
|
2005-03-11 21:05:46 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
def get_source_referents(source_handle, db):
|
|
|
|
""" Find objects that refer the source.
|
2007-01-23 03:37:13 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
This function finds all primary objects that refer (directly or through
|
|
|
|
secondary child-objects) to a given source handle in a given database.
|
|
|
|
|
|
|
|
"""
|
2007-09-10 03:03:46 +00:00
|
|
|
_primaries = ('Person', 'Family', 'Event', 'Place',
|
2007-03-27 18:28:26 +00:00
|
|
|
'Source', 'MediaObject', 'Repository')
|
|
|
|
|
|
|
|
return (get_referents(source_handle, db, _primaries))
|
2005-03-11 21:05:46 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def get_media_referents(media_handle, db):
|
2007-03-27 18:28:26 +00:00
|
|
|
""" Find objects that refer the media object.
|
2005-03-12 00:44:11 +00:00
|
|
|
|
|
|
|
This function finds all primary objects that refer
|
|
|
|
to a given media handle in a given database.
|
2007-03-27 18:28:26 +00:00
|
|
|
|
2005-03-12 00:44:11 +00:00
|
|
|
"""
|
2007-03-27 18:28:26 +00:00
|
|
|
_primaries = ('Person', 'Family', 'Event', 'Place', 'Source')
|
|
|
|
|
|
|
|
return (get_referents(media_handle, db, _primaries))
|
2005-03-12 00:44:11 +00:00
|
|
|
|
2007-03-27 18:28:26 +00:00
|
|
|
def get_note_referents(note_handle, db):
|
|
|
|
""" Find objects that refer a note object.
|
|
|
|
|
|
|
|
This function finds all primary objects that refer
|
|
|
|
to a given note handle in a given database.
|
|
|
|
|
|
|
|
"""
|
2007-09-10 03:03:46 +00:00
|
|
|
_primaries = ('Person', 'Family', 'Event', 'Place',
|
2007-03-27 18:28:26 +00:00
|
|
|
'Source', 'MediaObject', 'Repository')
|
|
|
|
|
|
|
|
return (get_referents(note_handle, db, _primaries))
|
2005-03-12 00:44:11 +00:00
|
|
|
|
2004-07-09 18:19:47 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
2006-05-09 23:15:38 +00:00
|
|
|
_NEW_NAME_PATTERN = '%s%sUntitled_%d.%s'
|
2004-07-09 18:19:47 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def get_new_filename(ext, folder='~/'):
|
2004-07-09 18:19:47 +00:00
|
|
|
ix = 1
|
2006-05-09 23:15:38 +00:00
|
|
|
while os.path.isfile(os.path.expanduser(_NEW_NAME_PATTERN %
|
2007-09-10 03:03:46 +00:00
|
|
|
(folder, os.path.sep, ix, ext))):
|
2004-07-09 18:19:47 +00:00
|
|
|
ix = ix + 1
|
2007-09-10 03:03:46 +00:00
|
|
|
return os.path.expanduser(_NEW_NAME_PATTERN % (folder, os.path.sep, ix, ext))
|
2004-12-22 01:56:37 +00:00
|
|
|
|
2007-11-26 05:41:04 +00:00
|
|
|
def get_empty_tempdir(dirname):
|
|
|
|
""" Return path to TEMP_DIR/dirname, a guaranteed empty directory
|
|
|
|
|
|
|
|
makes intervening directories if required
|
|
|
|
fails if _file_ by that name already exists,
|
|
|
|
or for inadequate permissions to delete dir/files or create dir(s)
|
|
|
|
|
|
|
|
"""
|
|
|
|
dirpath = os.path.join(TEMP_DIR,dirname)
|
|
|
|
if os.path.isdir(dirpath):
|
|
|
|
shutil.rmtree(dirpath)
|
|
|
|
os.makedirs(dirpath)
|
|
|
|
return dirpath
|
|
|
|
|
|
|
|
def rm_tempdir(path):
|
|
|
|
"""Remove a tempdir created with get_empty_tempdir"""
|
|
|
|
if path.startswith(TEMP_DIR) and os.path.isdir(path):
|
|
|
|
shutil.rmtree(path)
|
|
|
|
|
2007-09-05 03:02:50 +00:00
|
|
|
def cast_to_bool(val):
|
|
|
|
if val == str(True):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2004-12-22 01:56:37 +00:00
|
|
|
def get_type_converter(val):
|
|
|
|
"""
|
2008-02-24 13:55:55 +00:00
|
|
|
Return function that converts strings into the type of val.
|
2004-12-22 01:56:37 +00:00
|
|
|
"""
|
|
|
|
val_type = type(val)
|
2007-09-10 03:03:46 +00:00
|
|
|
if val_type in (str, unicode):
|
2004-12-22 01:56:37 +00:00
|
|
|
return unicode
|
|
|
|
elif val_type == int:
|
|
|
|
return int
|
|
|
|
elif val_type == float:
|
|
|
|
return float
|
2007-09-05 03:02:50 +00:00
|
|
|
elif val_type == bool:
|
|
|
|
return cast_to_bool
|
2007-09-10 03:03:46 +00:00
|
|
|
elif val_type in (list, tuple):
|
2004-12-22 01:56:37 +00:00
|
|
|
return list
|
|
|
|
|
|
|
|
def type_name(val):
|
|
|
|
"""
|
2008-02-24 13:55:55 +00:00
|
|
|
Return the name the type of val.
|
2004-12-22 01:56:37 +00:00
|
|
|
|
|
|
|
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'
|
2008-03-02 23:03:52 +00:00
|
|
|
elif val_type == bool:
|
|
|
|
return 'bool'
|
2007-09-10 03:03:46 +00:00
|
|
|
elif val_type in (str, unicode):
|
2004-12-22 01:56:37 +00:00
|
|
|
return 'unicode'
|
|
|
|
return 'unicode'
|
|
|
|
|
|
|
|
def get_type_converter_by_name(val_str):
|
|
|
|
"""
|
2008-02-24 13:55:55 +00:00
|
|
|
Return function that converts strings into the type given by val_str.
|
2004-12-22 01:56:37 +00:00
|
|
|
|
|
|
|
Only numbers and strings are supported.
|
|
|
|
The rest becomes strings (unicode).
|
|
|
|
"""
|
|
|
|
if val_str == 'int':
|
|
|
|
return int
|
|
|
|
elif val_str == 'float':
|
|
|
|
return float
|
2008-03-02 23:03:52 +00:00
|
|
|
elif val_str == 'bool':
|
2008-04-20 03:03:39 +00:00
|
|
|
return cast_to_bool
|
2007-09-10 03:03:46 +00:00
|
|
|
elif val_str in ('str', 'unicode'):
|
2004-12-22 01:56:37 +00:00
|
|
|
return unicode
|
|
|
|
return unicode
|
2005-07-08 20:24:54 +00:00
|
|
|
|
2006-02-04 23:29:44 +00:00
|
|
|
def relative_path(original, base):
|
2008-02-11 22:27:24 +00:00
|
|
|
"""
|
|
|
|
Calculate the relative path from base to original, with base a directory,
|
|
|
|
and original an absolute path
|
|
|
|
On problems, original is returned unchanged
|
|
|
|
"""
|
|
|
|
if not os.path.isdir(base):
|
|
|
|
return original
|
|
|
|
#original and base must be absolute paths
|
|
|
|
if not os.path.isabs(base):
|
|
|
|
return original
|
|
|
|
if not os.path.isabs(original):
|
|
|
|
return original
|
|
|
|
original = os.path.normpath(original)
|
|
|
|
base = os.path.normpath(base)
|
|
|
|
|
|
|
|
# If the db_dir and obj_dir are on different drives (win only)
|
|
|
|
# then there cannot be a relative path. Return original obj_path
|
|
|
|
(base_drive, base) = os.path.splitdrive(base)
|
|
|
|
(orig_drive, orig_name) = os.path.splitdrive(original)
|
|
|
|
if base_drive.upper() != orig_drive.upper():
|
2006-02-04 23:29:44 +00:00
|
|
|
return original
|
|
|
|
|
|
|
|
# Starting from the filepath root, work out how much of the filepath is
|
|
|
|
# shared by base and target.
|
2008-02-11 22:27:24 +00:00
|
|
|
base_list = (base).split(os.sep)
|
|
|
|
target_list = (orig_name).split(os.sep)
|
|
|
|
# make sure '/home/person' and 'c:/home/person' both give
|
|
|
|
# list ['home', 'person']
|
|
|
|
base_list = [word for word in base_list if word]
|
|
|
|
target_list = [word for word in target_list if word]
|
|
|
|
i = -1
|
2006-02-04 23:29:44 +00:00
|
|
|
for i in range(min(len(base_list), len(target_list))):
|
|
|
|
if base_list[i] <> target_list[i]: break
|
|
|
|
else:
|
2008-02-11 22:27:24 +00:00
|
|
|
#if break did not happen we are here at end, and add 1.
|
2007-09-10 03:03:46 +00:00
|
|
|
i += 1
|
2006-02-04 23:29:44 +00:00
|
|
|
rel_list = [os.pardir] * (len(base_list)-i) + target_list[i:]
|
|
|
|
return os.path.join(*rel_list)
|
|
|
|
|
2008-02-11 22:27:24 +00:00
|
|
|
def media_path(db):
|
|
|
|
"""
|
|
|
|
Given a database, return the mediapath to use as basedir for media
|
|
|
|
"""
|
|
|
|
mpath = db.get_mediapath()
|
|
|
|
if mpath is None:
|
|
|
|
#use home dir
|
|
|
|
mpath = USER_HOME
|
|
|
|
return mpath
|
|
|
|
|
|
|
|
def media_path_full(db, filename):
|
|
|
|
"""
|
|
|
|
Given a database and a filename of a media, return the media filename
|
|
|
|
is full form, eg 'graves/tomb.png' becomes '/home/me/genea/graves/tomb.png
|
|
|
|
"""
|
|
|
|
if os.path.isabs(filename):
|
|
|
|
return filename
|
|
|
|
mpath = media_path(db)
|
|
|
|
return os.path.join(mpath, filename)
|
2006-03-30 03:24:04 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
def profile(func, *args):
|
2008-02-18 20:07:09 +00:00
|
|
|
import hotshot.stats
|
2007-01-19 05:16:41 +00:00
|
|
|
|
2007-09-10 03:03:46 +00:00
|
|
|
prf = hotshot.Profile('mystats.profile')
|
2007-01-19 05:16:41 +00:00
|
|
|
print "Start"
|
2007-09-10 03:03:46 +00:00
|
|
|
prf.runcall(func, *args)
|
2007-01-19 05:16:41 +00:00
|
|
|
print "Finished"
|
2007-09-10 03:03:46 +00:00
|
|
|
prf.close()
|
2007-01-19 05:16:41 +00:00
|
|
|
print "Loading profile"
|
|
|
|
stats = hotshot.stats.load('mystats.profile')
|
|
|
|
print "done"
|
|
|
|
stats.strip_dirs()
|
2007-09-10 03:03:46 +00:00
|
|
|
stats.sort_stats('time', 'calls')
|
2007-01-19 05:16:41 +00:00
|
|
|
stats.print_stats(100)
|
2007-08-29 23:01:16 +00:00
|
|
|
stats.print_callers(100)
|
2007-01-19 05:16:41 +00:00
|
|
|
|
2007-12-12 03:59:18 +00:00
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# Keyword translation interface
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
# keyword, code, translated standard, translated upper
|
2009-02-12 07:44:11 +00:00
|
|
|
KEYWORDS = [("title", "t", _("Person|Title"), _("Person|TITLE")),
|
2007-12-12 03:59:18 +00:00
|
|
|
("given", "f", _("Given"), _("GIVEN")),
|
|
|
|
("prefix", "p", _("Prefix"), _("PREFIX")),
|
|
|
|
("surname", "l", _("Surname"), _("SURNAME")),
|
|
|
|
("suffix", "s", _("Suffix"), _("SUFFIX")),
|
|
|
|
("patronymic","y", _("Patronymic"),_("PATRONYMIC")),
|
|
|
|
("call", "c", _("Call"), _("CALL")),
|
|
|
|
("common", "x", _("Common"), _("COMMON")),
|
|
|
|
("initials", "i", _("Initials"), _("INITIALS"))
|
|
|
|
]
|
|
|
|
KEY_TO_TRANS = {}
|
|
|
|
TRANS_TO_KEY = {}
|
|
|
|
for (key, code, standard, upper) in KEYWORDS:
|
|
|
|
KEY_TO_TRANS[key] = standard
|
|
|
|
KEY_TO_TRANS[key.upper()] = upper
|
|
|
|
KEY_TO_TRANS["%" + ("%s" % code)] = standard
|
|
|
|
KEY_TO_TRANS["%" + ("%s" % code.upper())] = upper
|
2007-12-12 16:24:40 +00:00
|
|
|
TRANS_TO_KEY[standard.lower()] = key
|
2007-12-12 03:59:18 +00:00
|
|
|
TRANS_TO_KEY[standard] = key
|
2007-12-12 16:24:40 +00:00
|
|
|
TRANS_TO_KEY[upper] = key.upper()
|
2007-12-12 03:59:18 +00:00
|
|
|
|
|
|
|
def get_translation_from_keyword(keyword):
|
|
|
|
""" Return the translation of keyword """
|
|
|
|
return KEY_TO_TRANS.get(keyword, keyword)
|
|
|
|
|
|
|
|
def get_keyword_from_translation(word):
|
|
|
|
""" Return the keyword of translation """
|
|
|
|
return TRANS_TO_KEY.get(word, word)
|
|
|
|
|
|
|
|
def get_keywords():
|
|
|
|
""" Get all keywords, longest to shortest """
|
|
|
|
keys = KEY_TO_TRANS.keys()
|
|
|
|
keys.sort(lambda a,b: -cmp(len(a), len(b)))
|
|
|
|
return keys
|
|
|
|
|
|
|
|
def get_translations():
|
|
|
|
""" Get all translations, longest to shortest """
|
|
|
|
trans = TRANS_TO_KEY.keys()
|
|
|
|
trans.sort(lambda a,b: -cmp(len(a), len(b)))
|
|
|
|
return trans
|
2009-06-19 15:23:58 +00:00
|
|
|
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#-------------------------------------------------------------------------
|
|
|
|
def get_researcher():
|
|
|
|
import gen.lib
|
|
|
|
import Config
|
2007-12-12 03:59:18 +00:00
|
|
|
|
2009-06-19 15:23:58 +00:00
|
|
|
n = Config.get(Config.RESEARCHER_NAME)
|
|
|
|
a = Config.get(Config.RESEARCHER_ADDR)
|
|
|
|
c = Config.get(Config.RESEARCHER_CITY)
|
|
|
|
s = Config.get(Config.RESEARCHER_STATE)
|
|
|
|
ct = Config.get(Config.RESEARCHER_COUNTRY)
|
|
|
|
p = Config.get(Config.RESEARCHER_POSTAL)
|
|
|
|
ph = Config.get(Config.RESEARCHER_PHONE)
|
|
|
|
e = Config.get(Config.RESEARCHER_EMAIL)
|
|
|
|
|
|
|
|
owner = gen.lib.Researcher()
|
|
|
|
owner.set_name(n)
|
|
|
|
owner.set_address(a)
|
|
|
|
owner.set_city(c)
|
|
|
|
owner.set_state(s)
|
|
|
|
owner.set_country(ct)
|
|
|
|
owner.set_postal_code(p)
|
|
|
|
owner.set_phone(ph)
|
|
|
|
owner.set_email(e)
|
|
|
|
|
|
|
|
return owner
|