Localization: Re-implement localization as a class, GrampsLocale
GrampsLocale is effectively a singleton: An instance is created in const.py and retrieved everywhere. Translations are provided via Translations classes, which are derived from GNUTranslations and NullTranslations to provide extra functions like sgettext. svn: r21143
This commit is contained in:
parent
eecf57a0f6
commit
d3c2a8a490
@ -42,8 +42,6 @@ import uuid
|
|||||||
# Gramps modules
|
# Gramps modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
from .ggettext import sgettext as _
|
|
||||||
from .svn_revision import get_svn_revision
|
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
@ -94,6 +92,12 @@ APP_GRAMPS_PKG = "application/x-gramps-package"
|
|||||||
APP_GENEWEB = "application/x-geneweb"
|
APP_GENEWEB = "application/x-geneweb"
|
||||||
APP_VCARD = ["text/x-vcard", "text/x-vcalendar"]
|
APP_VCARD = ["text/x-vcard", "text/x-vcalendar"]
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# system paths
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
LOCALE_DIR = "@LOCALE_DIR@"
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# Platforms
|
# Platforms
|
||||||
@ -213,6 +217,15 @@ LOGO = os.path.join(IMAGE_DIR, "logo.png")
|
|||||||
SPLASH = os.path.join(IMAGE_DIR, "splash.jpg")
|
SPLASH = os.path.join(IMAGE_DIR, "splash.jpg")
|
||||||
|
|
||||||
LICENSE_FILE = os.path.join(DOC_DIR, 'COPYING')
|
LICENSE_FILE = os.path.join(DOC_DIR, 'COPYING')
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# Init Localization
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
from .utils.grampslocale import GrampsLocale
|
||||||
|
GRAMPS_LOCALE = GrampsLocale()
|
||||||
|
from .ggettext import sgettext as _
|
||||||
|
>>>>>>> GrampsLocale: Replace use of the GNU Gettext API with the Gettext Class API
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
@ -326,6 +339,7 @@ SHORTOPTS = "O:C:i:e:f:a:p:d:c:lLhuv?s"
|
|||||||
|
|
||||||
GRAMPS_UUID = uuid.UUID('516cd010-5a41-470f-99f8-eb22f1098ad6')
|
GRAMPS_UUID = uuid.UUID('516cd010-5a41-470f-99f8-eb22f1098ad6')
|
||||||
|
|
||||||
|
<<<<<<< HEAD
|
||||||
def need_to_update_const():
|
def need_to_update_const():
|
||||||
""" Check to see if this file is older than
|
""" Check to see if this file is older than
|
||||||
setup.py or const.py.in """
|
setup.py or const.py.in """
|
||||||
@ -353,4 +367,9 @@ if need_to_update_const():
|
|||||||
print("Outdated gramps.gen.const; please run 'python setup.py build'")
|
print("Outdated gramps.gen.const; please run 'python setup.py build'")
|
||||||
|
|
||||||
GRAMPS_LOCALE = 0
|
GRAMPS_LOCALE = 0
|
||||||
|
from .utils.grampslocale import GrampsLocale
|
||||||
|
GRAMPS_LOCALE = GrampsLocale()
|
||||||
|
|
||||||
|
|
||||||
|
=======
|
||||||
|
>>>>>>> GrampsLocale: Replace use of the GNU Gettext API with the Gettext Class API
|
||||||
|
@ -30,65 +30,17 @@ This module ("Gramps Gettext") is an extension to the Python gettext module.
|
|||||||
# python modules
|
# python modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
import gettext as pgettext
|
from gramps.gen.const import GRAMPS_LOCALE as _gl
|
||||||
|
_tl = _gl.get_translation()
|
||||||
import sys
|
gettext = _tl.gettext
|
||||||
if sys.version_info[0] < 3:
|
# When in the 'C' locale, get_translation returns a NULLTranslation
|
||||||
cuni = unicode
|
# which doesn't provide sgettext. This traps that case and uses
|
||||||
else:
|
# gettext instead -- which is fine, because there's no translation
|
||||||
cuni = str
|
# file involved and it's just going to return the msgid anyeay.
|
||||||
|
sgettext = None
|
||||||
def gettext(msgid):
|
try:
|
||||||
"""
|
_tl.__getattr__(sgettext)
|
||||||
Obtain translation of gettext, return a unicode object
|
sgettext = _tl.sgettext
|
||||||
:param msgid: The string to translated.
|
except AttributeError:
|
||||||
:type msgid: unicode
|
sgettext = _tl.gettext
|
||||||
:returns: Translation or the original.
|
ngettext = _tl.ngettext
|
||||||
:rtype: unicode
|
|
||||||
"""
|
|
||||||
# If msgid =="" then gettext will return po file header
|
|
||||||
# and that's not what we want.
|
|
||||||
if len(msgid.strip()) == 0:
|
|
||||||
return msgid
|
|
||||||
return cuni(pgettext.gettext(msgid))
|
|
||||||
|
|
||||||
def ngettext(singular, plural, n):
|
|
||||||
"""
|
|
||||||
The translation of singular/plural is returned unless the translation is
|
|
||||||
not available and the singular contains the separator. In that case,
|
|
||||||
the returned value is the singular.
|
|
||||||
|
|
||||||
:param singular: The singular form of the string to be translated.
|
|
||||||
may contain a context seperator
|
|
||||||
:type singular: unicode
|
|
||||||
:param plural: The plural form of the string to be translated.
|
|
||||||
:type plural: unicode
|
|
||||||
:param n: the amount for which to decide the translation
|
|
||||||
:type n: int
|
|
||||||
:returns: Translation or the original.
|
|
||||||
:rtype: unicode
|
|
||||||
"""
|
|
||||||
return cuni(pgettext.ngettext(singular, plural, n))
|
|
||||||
|
|
||||||
def sgettext(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
|
|
||||||
:returns: Translation or the original with context stripped.
|
|
||||||
:rtype: unicode
|
|
||||||
|
|
||||||
"""
|
|
||||||
msgval = pgettext.gettext(msgid)
|
|
||||||
if msgval == msgid:
|
|
||||||
sep_idx = msgid.rfind(sep)
|
|
||||||
msgval = msgid[sep_idx+1:]
|
|
||||||
return cuni(msgval)
|
|
||||||
|
@ -42,9 +42,8 @@ import traceback
|
|||||||
# GRAMPS modules
|
# GRAMPS modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
from ..const import VERSION as GRAMPSVERSION, VERSION_TUPLE
|
from ..const import VERSION as GRAMPSVERSION, VERSION_TUPLE, GRAMPS_LOCALE as glocale
|
||||||
from ..const import IMAGE_DIR
|
from ..const import IMAGE_DIR
|
||||||
from ..utils.grampslocale import get_addon_translator
|
|
||||||
from ..ggettext import gettext as _
|
from ..ggettext import gettext as _
|
||||||
from ..constfunc import STRTYPE
|
from ..constfunc import STRTYPE
|
||||||
|
|
||||||
@ -836,8 +835,9 @@ class PluginData(object):
|
|||||||
def _set_gramplet_title(self, gramplet_title):
|
def _set_gramplet_title(self, gramplet_title):
|
||||||
if not self._ptype == GRAMPLET:
|
if not self._ptype == GRAMPLET:
|
||||||
raise ValueError('gramplet_title may only be set for GRAMPLET plugins')
|
raise ValueError('gramplet_title may only be set for GRAMPLET plugins')
|
||||||
if not isinstance(gramplet_title, str):
|
if not (sys.version_info[0] < 3 and isinstance(gramplet_title, unicode)
|
||||||
raise ValueError('Plugin must have a string as gramplet_title')
|
or isinstance(gramplet_title, str)):
|
||||||
|
raise ValueError('gramplet_title is type %s, string or unicode required' % type(gramplet_title))
|
||||||
self._gramplet_title = gramplet_title
|
self._gramplet_title = gramplet_title
|
||||||
|
|
||||||
def _get_gramplet_title(self):
|
def _get_gramplet_title(self):
|
||||||
@ -1091,7 +1091,7 @@ class PluginRegister(object):
|
|||||||
full_filename = os.path.join(dir, filename)
|
full_filename = os.path.join(dir, filename)
|
||||||
if sys.version_info[0] < 3:
|
if sys.version_info[0] < 3:
|
||||||
full_filename = full_filename.encode(sys.getfilesystemencoding())
|
full_filename = full_filename.encode(sys.getfilesystemencoding())
|
||||||
local_gettext = get_addon_translator(full_filename).gettext
|
local_gettext = glocale.get_addon_translator(full_filename).gettext
|
||||||
try:
|
try:
|
||||||
#execfile(full_filename,
|
#execfile(full_filename,
|
||||||
exec(compile(open(full_filename).read(), full_filename, 'exec'),
|
exec(compile(open(full_filename).read(), full_filename, 'exec'),
|
||||||
|
@ -170,7 +170,7 @@ def get_unicode_path_from_env_var(path):
|
|||||||
"""
|
"""
|
||||||
# make only unicode of path of type 'str'
|
# make only unicode of path of type 'str'
|
||||||
if not (isinstance(path, str)):
|
if not (isinstance(path, str)):
|
||||||
return path
|
raise TypeError("path %s isn't a str" % str(path))
|
||||||
|
|
||||||
if win():
|
if win():
|
||||||
# In Windows path/filename returned from a environment variable is in filesystemencoding
|
# In Windows path/filename returned from a environment variable is in filesystemencoding
|
||||||
|
@ -37,109 +37,45 @@ import logging
|
|||||||
# gramps modules
|
# gramps modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
from ..const import ROOT_DIR
|
from ..const import LOCALE_DIR
|
||||||
from ..constfunc import mac, UNITYPE
|
from ..constfunc import mac, win, UNITYPE
|
||||||
|
|
||||||
class GrampsLocale(locale):
|
#------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# GrampsLocale Class
|
||||||
|
#
|
||||||
|
#------------------------------------------------------------------------
|
||||||
|
class GrampsLocale(object):
|
||||||
"""
|
"""
|
||||||
Encapsulate a locale
|
Encapsulate a locale
|
||||||
"""
|
"""
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
def _get_prefix(self):
|
|
||||||
"""
|
|
||||||
Find the root path for share/locale
|
|
||||||
"""
|
|
||||||
if sys.platform == "win32":
|
|
||||||
if sys.prefix == os.path.dirname(os.getcwd()):
|
|
||||||
return sys.prefix
|
|
||||||
else:
|
|
||||||
return os.path.join(os.path.dirname(__file__), os.pardir)
|
|
||||||
elif sys.platform == "darwin" and sys.prefix != sys.exec_prefix:
|
|
||||||
return sys.prefix
|
|
||||||
else:
|
|
||||||
return os.path.join(os.path.dirname(__file__), os.pardir)
|
|
||||||
|
|
||||||
def _init_gettext(self):
|
|
||||||
"""
|
|
||||||
Set up the gettext domain
|
|
||||||
"""
|
|
||||||
#the order in which bindtextdomain on gettext and on locale is called
|
|
||||||
#appears important, so we refrain from doing first all gettext.
|
|
||||||
#
|
|
||||||
#setup_gettext()
|
|
||||||
gettext.bindtextdomain(self.localedomain, self.localedir)
|
|
||||||
try:
|
|
||||||
locale.setlocale(locale.LC_ALL,'')
|
|
||||||
except:
|
|
||||||
logging.warning(_("WARNING: Setting locale failed. Please fix"
|
|
||||||
" the LC_* and/or the LANG environment "
|
|
||||||
"variables to prevent this error"))
|
|
||||||
try:
|
|
||||||
# It is probably not necessary to set the locale to 'C'
|
|
||||||
# because the locale will just stay at whatever it was,
|
|
||||||
# which at startup is "C".
|
|
||||||
# however this is done here just to make sure that the locale
|
|
||||||
# functions are working
|
|
||||||
locale.setlocale(locale.LC_ALL,'C')
|
|
||||||
except:
|
|
||||||
logging.warning(_("ERROR: Setting the 'C' locale didn't "
|
|
||||||
"work either"))
|
|
||||||
# FIXME: This should propagate the exception,
|
|
||||||
# if that doesn't break Gramps under Windows
|
|
||||||
raise
|
|
||||||
|
|
||||||
gettext.textdomain(slef.localedomain)
|
|
||||||
if sys.version_info[0] < 3:
|
|
||||||
gettext.install(self.localedomain, localedir=None, unicode=1) #None is sys default locale
|
|
||||||
else:
|
|
||||||
gettext.install(self.localedomain, localedir=None) #None is sys default locale
|
|
||||||
|
|
||||||
if hasattr(os, "uname"):
|
|
||||||
operating_system = os.uname()[0]
|
|
||||||
else:
|
|
||||||
operating_system = sys.platform
|
|
||||||
|
|
||||||
if win(): # Windows
|
|
||||||
setup_windows_gettext()
|
|
||||||
elif operating_system == 'FreeBSD':
|
|
||||||
try:
|
|
||||||
gettext.bindtextdomain(self.localedomain, self.localedir)
|
|
||||||
except locale.Error:
|
|
||||||
logging.warning('No translation in some Gtk.Builder strings, ')
|
|
||||||
elif operating_system == 'OpenBSD':
|
|
||||||
pass
|
|
||||||
else: # normal case
|
|
||||||
try:
|
|
||||||
locale.bindtextdomain(self.localedomain, self.localedir)
|
|
||||||
#locale.textdomain(self.localedomain)
|
|
||||||
except locale.Error:
|
|
||||||
logging.warning('No translation in some Gtk.Builder strings, ')
|
|
||||||
|
|
||||||
prefixdir = self._get_prefix()
|
|
||||||
if "GRAMPSI18N" in os.environ:
|
|
||||||
if os.path.exists(os.environ["GRAMPSI18N"]):
|
|
||||||
self.localedir = os.environ["GRAMPSI18N"]
|
|
||||||
else:
|
|
||||||
self.localedir = None
|
self.localedir = None
|
||||||
elif os.path.exists( os.path.join(ROOT_DIR, "lang") ):
|
self.lang = None
|
||||||
self.localedir = os.path.join(ROOT_DIR, "lang")
|
self.language = []
|
||||||
elif os.path.exists(os.path.join(prefixdir, "share/locale")):
|
if ("GRAMPSI18N" in os.environ
|
||||||
self.localedir = os.path.join(prefixdir, "share/locale")
|
and os.path.exists(os.environ["GRAMPSI18N"])):
|
||||||
|
self.localedir = os.environ["GRAMPSI18N"]
|
||||||
|
elif os.path.exists(LOCALE_DIR):
|
||||||
|
self.localedir = LOCALE_DIR
|
||||||
|
elif os.path.exists(os.path.join(sys.prefix, "share", "locale")):
|
||||||
|
self.localedir = os.path.join(sys.prefix, "share", "locale")
|
||||||
else:
|
else:
|
||||||
self.lang = os.environ.get('LANG', 'en')
|
lang = os.environ.get('LANG', 'en')
|
||||||
if self.lang and self.lang[:2] == 'en':
|
if lang and lang[:2] == 'en':
|
||||||
pass # No need to display warning, we're in English
|
pass # No need to display warning, we're in English
|
||||||
else:
|
else:
|
||||||
logging.warning('Locale dir does not exist at ' +
|
logging.warning('Locale dir does not exist at %s', LOCALE_DIR)
|
||||||
os.path.join(prefixdir, "share/locale"))
|
|
||||||
logging.warning('Running python setup.py install --prefix=YourPrefixDir might fix the problem')
|
logging.warning('Running python setup.py install --prefix=YourPrefixDir might fix the problem')
|
||||||
self.localedir = None
|
|
||||||
|
|
||||||
|
if not self.localedir:
|
||||||
|
#No localization files, no point in continuing
|
||||||
|
return
|
||||||
self.localedomain = 'gramps'
|
self.localedomain = 'gramps'
|
||||||
|
|
||||||
if mac():
|
if mac():
|
||||||
from . import maclocale
|
from . import maclocale
|
||||||
maclocale.mac_setup_localization(self.localedir, self.localedomain)
|
(self.lang, self.language) = maclocale.mac_setup_localization(self)
|
||||||
else:
|
else:
|
||||||
self.lang = ' '
|
self.lang = ' '
|
||||||
try:
|
try:
|
||||||
@ -151,11 +87,52 @@ Encapsulate a locale
|
|||||||
self.lang = locale.getdefaultlocale()[0] + '.UTF-8'
|
self.lang = locale.getdefaultlocale()[0] + '.UTF-8'
|
||||||
except TypeError:
|
except TypeError:
|
||||||
logging.warning('Unable to determine your Locale, using English')
|
logging.warning('Unable to determine your Locale, using English')
|
||||||
self.lang = 'en.UTF-8'
|
self.lang = 'C.UTF-8'
|
||||||
|
|
||||||
|
if "LANGUAGE" in os.environ:
|
||||||
|
language = [l for l in os.environ["LANGUAGE"].split(":")
|
||||||
|
if l in self.get_available_translations()]
|
||||||
|
self.language = language
|
||||||
|
else:
|
||||||
|
self.language = [self.lang[0:2]]
|
||||||
|
|
||||||
|
#GtkBuilder depends on reading Glade files as UTF-8 and crashes if it
|
||||||
|
#doesn't, so set $LANG to have a UTF-8 locale. NB: This does *not*
|
||||||
|
#affect locale.getpreferredencoding() or sys.getfilesystemencoding()
|
||||||
|
#which are set by python long before we get here.
|
||||||
|
check_lang = self.lang.split('.')
|
||||||
|
if len(check_lang) < 2 or check_lang[1] not in ["utf-8", "UTF-8"]:
|
||||||
|
self.lang = '.'.join((check_lang[0], 'UTF-8'))
|
||||||
os.environ["LANG"] = self.lang
|
os.environ["LANG"] = self.lang
|
||||||
os.environ["LANGUAGE"] = self.lang
|
# Set Gramps's translations
|
||||||
self._init_gettext()
|
self.translation = self._get_translation(self.localedomain, self.localedir, self.language)
|
||||||
|
# Now set the locale for everything besides translations.
|
||||||
|
|
||||||
|
try:
|
||||||
|
# First try the environment to preserve individual variables
|
||||||
|
locale.setlocale(locale.LC_ALL, '')
|
||||||
|
try:
|
||||||
|
#Then set LC_MESSAGES to self.lang
|
||||||
|
locale.setlocale(locale.LC_MESSAGES, self.lang)
|
||||||
|
except locale.Error:
|
||||||
|
logging.warning("Unable to set translations to %s, locale not found.", self.lang)
|
||||||
|
except locale.Error:
|
||||||
|
# That's not a valid locale -- on Linux, probably not installed.
|
||||||
|
try:
|
||||||
|
# First fallback is self.lang
|
||||||
|
locale.setlocale(locale.LC_ALL, self.lang)
|
||||||
|
logging.warning("Setting locale to individual LC_ variables failed, falling back to %s.", self.lang)
|
||||||
|
|
||||||
|
except locale.Error:
|
||||||
|
# No good, set the default encoding to C.UTF-8. Don't
|
||||||
|
# mess with anything else.
|
||||||
|
locale.setlocale(locale.LC_ALL, 'C.UTF-8')
|
||||||
|
logging.error("Failed to set locale %s, falling back to English", self.lang)
|
||||||
|
# $LANGUAGE is what sets the Gtk+ translations
|
||||||
|
os.environ["LANGUAGE"] = ':'.join(self.language)
|
||||||
|
# GtkBuilder uses GLib's g_dgettext wrapper, which oddly is bound
|
||||||
|
# with locale instead of gettext.
|
||||||
|
locale.bindtextdomain(self.localedomain, self.localedir)
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
@ -166,10 +143,86 @@ Encapsulate a locale
|
|||||||
def get_localedomain(self):
|
def get_localedomain(self):
|
||||||
"""
|
"""
|
||||||
Get the LOCALEDOMAIN used for the Gramps application.
|
Get the LOCALEDOMAIN used for the Gramps application.
|
||||||
|
Required by gui/glade.py to pass to Gtk.Builder
|
||||||
"""
|
"""
|
||||||
return self.localedomain
|
return self.localedomain
|
||||||
|
|
||||||
def get_addon_translator(self, filename=None, domain="addon",
|
def _get_translation(self, domain = None,
|
||||||
|
localedir = None,
|
||||||
|
languages=None):
|
||||||
|
"""
|
||||||
|
Get a translation of one of our classes. Doesn't return the
|
||||||
|
singleton so that it can be used by get_addon_translation()
|
||||||
|
"""
|
||||||
|
if not domain:
|
||||||
|
domain = self.localedomain
|
||||||
|
if not languages:
|
||||||
|
languages = self.language
|
||||||
|
if not localedir:
|
||||||
|
localedir = self.localedir
|
||||||
|
|
||||||
|
if gettext.find(domain, localedir, languages):
|
||||||
|
return gettext.translation(domain, localedir,
|
||||||
|
languages,
|
||||||
|
class_ = GrampsTranslations)
|
||||||
|
else:
|
||||||
|
logging.debug("Unable to find translations for %s and %s in %s"
|
||||||
|
, domain, languages, localedir)
|
||||||
|
return GrampsNullTranslations()
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# Public Functions
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
|
||||||
|
def get_localedomain(self):
|
||||||
|
"""
|
||||||
|
Get the LOCALEDOMAIN used for the Gramps application.
|
||||||
|
Required by gui/glade.py to pass to Gtk.Builder
|
||||||
|
"""
|
||||||
|
return self.localedomain
|
||||||
|
|
||||||
|
def get_language_list(self):
|
||||||
|
"""
|
||||||
|
Return the list of configured languages. Used by
|
||||||
|
ViewManager.check_for_updates to select the language for the
|
||||||
|
addons descriptions.
|
||||||
|
"""
|
||||||
|
return self.language
|
||||||
|
|
||||||
|
def get_translation(self, domain = None, languages = None):
|
||||||
|
"""
|
||||||
|
Get a translation object for a particular language.
|
||||||
|
See the gettext documentation for the available functions
|
||||||
|
>>> glocale = GrampsLocale()
|
||||||
|
>>> _ = glocale.get_translation('foo', 'French')
|
||||||
|
>>> _ = tr.gettext
|
||||||
|
"""
|
||||||
|
|
||||||
|
if ((domain and not domain == self.localedomain)
|
||||||
|
or (languages and not languages == self.language)):
|
||||||
|
if not domain:
|
||||||
|
domain = self.localedomain
|
||||||
|
if not languages:
|
||||||
|
languages = self.language
|
||||||
|
fallback = False
|
||||||
|
if "en" in languages:
|
||||||
|
fallback = True
|
||||||
|
try:
|
||||||
|
# Don't use _get_translation because we want to fall
|
||||||
|
# back on the singleton rather than a NullTranslation
|
||||||
|
return gettext.translation(domain, self.localedir,
|
||||||
|
languages,
|
||||||
|
class_ = GrampsTranslations,
|
||||||
|
fallback = fallback)
|
||||||
|
except IOError:
|
||||||
|
logging.warning("None of the requested languages (%s) were available, using %s instead", ', '.join(languages), self.lang)
|
||||||
|
return self.translation
|
||||||
|
else:
|
||||||
|
return self.translation
|
||||||
|
|
||||||
|
def get_addon_translator(self, filename, domain="addon",
|
||||||
languages=None):
|
languages=None):
|
||||||
"""
|
"""
|
||||||
Get a translator for an addon.
|
Get a translator for an addon.
|
||||||
@ -181,27 +234,19 @@ Encapsulate a locale
|
|||||||
returns - a gettext.translation object
|
returns - a gettext.translation object
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
_ = get_addon_translator(languages=["fr_BE.utf8"]).gettext
|
_ = glocale.get_addon_translator(languages=["fr_BE.utf8"]).gettext
|
||||||
|
|
||||||
The return object has the following properties and methods:
|
|
||||||
.gettext
|
|
||||||
.info
|
|
||||||
.lgettext
|
|
||||||
.lngettext
|
|
||||||
.ngettext
|
|
||||||
.output_charset
|
|
||||||
.plural
|
|
||||||
.set_output_charset
|
|
||||||
.ugettext
|
|
||||||
.ungettext
|
|
||||||
|
|
||||||
|
See the python gettext documentation.
|
||||||
Assumes path/filename
|
Assumes path/filename
|
||||||
path/locale/LANG/LC_MESSAGES/addon.mo.
|
path/locale/LANG/LC_MESSAGES/addon.mo.
|
||||||
"""
|
"""
|
||||||
if filename is None:
|
path = self.localedir
|
||||||
filename = sys._getframe(1).f_code.co_filename
|
# If get the path of the calling module's uncompiled file. This seems a remarkably bad idea.
|
||||||
gramps_translator = gettext.translation(LOCALEDOMAIN, LOCALEDIR,
|
# if filename is None:
|
||||||
fallback=True)
|
# filename = sys._getframe(1).f_code.co_filename
|
||||||
|
|
||||||
|
gramps_translator = self._get_translation()
|
||||||
|
|
||||||
path = os.path.dirname(os.path.abspath(filename))
|
path = os.path.dirname(os.path.abspath(filename))
|
||||||
# Check if path is of type str. Do import and conversion if so.
|
# Check if path is of type str. Do import and conversion if so.
|
||||||
# The import cannot be done at the top as that will conflict with the translation system.
|
# The import cannot be done at the top as that will conflict with the translation system.
|
||||||
@ -210,14 +255,11 @@ Encapsulate a locale
|
|||||||
from .file import get_unicode_path_from_env_var
|
from .file import get_unicode_path_from_env_var
|
||||||
path = get_unicode_path_from_env_var(path)
|
path = get_unicode_path_from_env_var(path)
|
||||||
if languages:
|
if languages:
|
||||||
addon_translator = gettext.translation(domain,
|
addon_translator = self._get_translation(domain,
|
||||||
os.path.join(path, "locale"),
|
path,
|
||||||
languages=languages,
|
languages=languages)
|
||||||
fallback=True)
|
|
||||||
else:
|
else:
|
||||||
addon_translator = gettext.translation(domain,
|
addon_translator = self._get_translation(domain, path)
|
||||||
os.path.join(path, "locale"),
|
|
||||||
fallback=True)
|
|
||||||
gramps_translator.add_fallback(addon_translator)
|
gramps_translator.add_fallback(addon_translator)
|
||||||
return gramps_translator # with a language fallback
|
return gramps_translator # with a language fallback
|
||||||
|
|
||||||
@ -231,12 +273,13 @@ Encapsulate a locale
|
|||||||
"""
|
"""
|
||||||
languages = ["en"]
|
languages = ["en"]
|
||||||
|
|
||||||
if slef.localedir is None:
|
if self.localedir is None:
|
||||||
return languages
|
return languages
|
||||||
|
|
||||||
for langdir in os.listdir(self.localedir):
|
for langdir in os.listdir(self.localedir):
|
||||||
mofilename = os.path.join(self.localedir, langdir,
|
mofilename = os.path.join(self.localedir, langdir,
|
||||||
"LC_MESSAGES", "%s.mo" % self.localedomain )
|
"LC_MESSAGES",
|
||||||
|
"%s.mo" % self.localedomain )
|
||||||
if os.path.exists(mofilename):
|
if os.path.exists(mofilename):
|
||||||
languages.append(langdir)
|
languages.append(langdir)
|
||||||
|
|
||||||
@ -249,7 +292,7 @@ Encapsulate a locale
|
|||||||
Translates objclass_str into "... %s", where objclass_str
|
Translates objclass_str into "... %s", where objclass_str
|
||||||
is 'Person', 'person', 'Family', 'family', etc.
|
is 'Person', 'person', 'Family', 'family', etc.
|
||||||
"""
|
"""
|
||||||
from ..ggettext import gettext as _
|
_ = self.translation.gettext
|
||||||
objclass = objclass_str.lower()
|
objclass = objclass_str.lower()
|
||||||
if objclass == "person":
|
if objclass == "person":
|
||||||
return _("the person")
|
return _("the person")
|
||||||
@ -271,3 +314,184 @@ Encapsulate a locale
|
|||||||
return _("the filter")
|
return _("the filter")
|
||||||
else:
|
else:
|
||||||
return _("See details")
|
return _("See details")
|
||||||
|
|
||||||
|
def getfilesystemencoding(self):
|
||||||
|
"""
|
||||||
|
If the locale isn't configured correctly, this will return
|
||||||
|
'ascii' or 'ANSI_X3.4-1968' or some other unfortunate
|
||||||
|
result. Current unix systems all encode filenames in utf-8,
|
||||||
|
and Microsoft Windows uses utf-16 (which they call mbcs). Make
|
||||||
|
sure we return the right value.
|
||||||
|
"""
|
||||||
|
encoding = sys.getfilesystemencoding()
|
||||||
|
|
||||||
|
if encoding in ("utf-8", "UTF-8", "utf8", "UTF8", "mbcs", "MBCS"):
|
||||||
|
return encoding
|
||||||
|
|
||||||
|
return "utf-8"
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# GrampsTranslation Class
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
class GrampsTranslations(gettext.GNUTranslations):
|
||||||
|
"""
|
||||||
|
Overrides and extends gettext.GNUTranslations. See the Python gettext
|
||||||
|
"Class API" documentation for how to use this.
|
||||||
|
"""
|
||||||
|
def language(self):
|
||||||
|
"""
|
||||||
|
Return the target languge of this translations object.
|
||||||
|
"""
|
||||||
|
return self.info()["language"]
|
||||||
|
|
||||||
|
def gettext(self, msgid):
|
||||||
|
"""
|
||||||
|
Obtain translation of gettext, return a unicode object
|
||||||
|
:param msgid: The string to translated.
|
||||||
|
:type msgid: unicode
|
||||||
|
:returns: Translation or the original.
|
||||||
|
:rtype: unicode
|
||||||
|
"""
|
||||||
|
# If msgid =="" then gettext will return po file header
|
||||||
|
# and that's not what we want.
|
||||||
|
if len(msgid.strip()) == 0:
|
||||||
|
return msgid
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
return gettext.GNUTranslations.ugettext(self, msgid)
|
||||||
|
else:
|
||||||
|
return gettext.GNUTranslations.gettext(self, msgid)
|
||||||
|
|
||||||
|
def ngettext(self, singular, plural, num):
|
||||||
|
"""
|
||||||
|
The translation of singular/plural is returned unless the translation is
|
||||||
|
not available and the singular contains the separator. In that case,
|
||||||
|
the returned value is the singular.
|
||||||
|
|
||||||
|
:param singular: The singular form of the string to be translated.
|
||||||
|
may contain a context seperator
|
||||||
|
:type singular: unicode
|
||||||
|
:param plural: The plural form of the string to be translated.
|
||||||
|
:type plural: unicode
|
||||||
|
:param num: the amount for which to decide the translation
|
||||||
|
:type num: int
|
||||||
|
:returns: Translation or the original.
|
||||||
|
:rtype: unicode
|
||||||
|
"""
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
return gettext.GNUTranslations.ungettext(self, singular,
|
||||||
|
plural, num)
|
||||||
|
else:
|
||||||
|
return gettext.GNUTranslations.ngettext(self, singular,
|
||||||
|
plural, num)
|
||||||
|
|
||||||
|
def sgettext(self, msgid, sep='|'):
|
||||||
|
"""
|
||||||
|
Even with a null translator we need to filter out the translator hint.
|
||||||
|
"""
|
||||||
|
msgval = self.gettext(msgid)
|
||||||
|
if msgval == msgid:
|
||||||
|
sep_idx = msgid.rfind(sep)
|
||||||
|
msgval = msgid[sep_idx+1:]
|
||||||
|
return msgval
|
||||||
|
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# Translations Classes
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
class GrampsTranslations(gettext.GNUTranslations):
|
||||||
|
"""
|
||||||
|
Overrides and extends gettext.GNUTranslations. See the Python gettext
|
||||||
|
"Class API" documentation for how to use this.
|
||||||
|
"""
|
||||||
|
def language(self):
|
||||||
|
"""
|
||||||
|
Return the target languge of this translations object.
|
||||||
|
"""
|
||||||
|
return self.info()["language"]
|
||||||
|
|
||||||
|
def gettext(self, msgid):
|
||||||
|
"""
|
||||||
|
Obtain translation of gettext, return a unicode object
|
||||||
|
:param msgid: The string to translated.
|
||||||
|
:type msgid: unicode
|
||||||
|
:returns: Translation or the original.
|
||||||
|
:rtype: unicode
|
||||||
|
"""
|
||||||
|
# If msgid =="" then gettext will return po file header
|
||||||
|
# and that's not what we want.
|
||||||
|
if len(msgid.strip()) == 0:
|
||||||
|
return msgid
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
return gettext.GNUTranslations.ugettext(self, msgid)
|
||||||
|
else:
|
||||||
|
return gettext.GNUTranslations.gettext(self, msgid)
|
||||||
|
|
||||||
|
def ngettext(self, singular, plural, num):
|
||||||
|
"""
|
||||||
|
The translation of singular/plural is returned unless the translation is
|
||||||
|
not available and the singular contains the separator. In that case,
|
||||||
|
the returned value is the singular.
|
||||||
|
|
||||||
|
:param singular: The singular form of the string to be translated.
|
||||||
|
may contain a context seperator
|
||||||
|
:type singular: unicode
|
||||||
|
:param plural: The plural form of the string to be translated.
|
||||||
|
:type plural: unicode
|
||||||
|
:param num: the amount for which to decide the translation
|
||||||
|
:type num: int
|
||||||
|
:returns: Translation or the original.
|
||||||
|
:rtype: unicode
|
||||||
|
"""
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
return gettext.GNUTranslations.ungettext(self, singular,
|
||||||
|
plural, num)
|
||||||
|
else:
|
||||||
|
return gettext.GNUTranslations.ngettext(self, singular,
|
||||||
|
plural, num)
|
||||||
|
|
||||||
|
def sgettext(self, 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
|
||||||
|
:returns: Translation or the original with context stripped.
|
||||||
|
:rtype: unicode
|
||||||
|
"""
|
||||||
|
msgval = self.gettext(msgid)
|
||||||
|
if msgval == msgid:
|
||||||
|
sep_idx = msgid.rfind(sep)
|
||||||
|
msgval = msgid[sep_idx+1:]
|
||||||
|
return msgval
|
||||||
|
|
||||||
|
|
||||||
|
class GrampsNullTranslations(gettext.NullTranslations):
|
||||||
|
"""
|
||||||
|
Extends gettext.NullTranslations to provide the sgettext method.
|
||||||
|
|
||||||
|
Note that it's necessary for msgid to be unicode. If it's not,
|
||||||
|
neither will be the returned string.
|
||||||
|
"""
|
||||||
|
def sgettext(self, msgid):
|
||||||
|
msgval = self.gettext(msgid)
|
||||||
|
if msgval == msgid:
|
||||||
|
sep_idx = msgid.rfind(sep)
|
||||||
|
msgval = msgid[sep_idx+1:]
|
||||||
|
return msgval
|
||||||
|
|
||||||
|
def language(self):
|
||||||
|
"""
|
||||||
|
The null translation returns the raw msgids, which are in English
|
||||||
|
"""
|
||||||
|
return "en"
|
||||||
|
@ -50,8 +50,8 @@ Keyword translation interface
|
|||||||
# 'n' : nickname = nick name
|
# 'n' : nickname = nick name
|
||||||
# 'g' : familynick = family nick name
|
# 'g' : familynick = family nick name
|
||||||
|
|
||||||
import gettext
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
_ = gettext.gettext
|
_ = glocale.get_translation().gettext
|
||||||
|
|
||||||
KEYWORDS = [("title", "t", _("Person|Title"), _("Person|TITLE")),
|
KEYWORDS = [("title", "t", _("Person|Title"), _("Person|TITLE")),
|
||||||
("given", "f", _("Given"), _("GIVEN")),
|
("given", "f", _("Given"), _("GIVEN")),
|
||||||
|
@ -73,36 +73,23 @@ locale, leaving $LANGUAGE unset (which is the same as setting it to
|
|||||||
|
|
||||||
import sys, os, subprocess
|
import sys, os, subprocess
|
||||||
|
|
||||||
def get_available_translations(dir, domain):
|
def mac_setup_localization(glocale):
|
||||||
"""
|
"""
|
||||||
Get a list of available translations.
|
Set up the localization parameters from OSX's "defaults" system,
|
||||||
|
permitting environment variables to override the settings.
|
||||||
:returns: A list of translation languages.
|
|
||||||
:rtype: unicode[]
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
languages = ["en"]
|
|
||||||
|
|
||||||
if dir is None:
|
|
||||||
return languages
|
|
||||||
|
|
||||||
for langdir in os.listdir(dir):
|
|
||||||
mofilename = os.path.join( dir, langdir,
|
|
||||||
"LC_MESSAGES", "%s.mo" % domain )
|
|
||||||
if os.path.exists(mofilename):
|
|
||||||
languages.append(langdir)
|
|
||||||
|
|
||||||
languages.sort()
|
|
||||||
|
|
||||||
return languages
|
|
||||||
|
|
||||||
def mac_setup_localization(dir, domain):
|
|
||||||
defaults = "/usr/bin/defaults"
|
defaults = "/usr/bin/defaults"
|
||||||
find = "/usr/bin/find"
|
find = "/usr/bin/find"
|
||||||
locale_dir = "/usr/share/locale"
|
locale_dir = "/usr/share/locale"
|
||||||
available = get_available_translations(dir, domain)
|
if glocale:
|
||||||
|
available = glocale.get_available_translations()
|
||||||
|
else:
|
||||||
|
available = ['en']
|
||||||
|
|
||||||
def mac_language_list():
|
def mac_language_list():
|
||||||
|
"""
|
||||||
|
Extract the languages list from defaults.
|
||||||
|
"""
|
||||||
languages = []
|
languages = []
|
||||||
try:
|
try:
|
||||||
languages = subprocess.Popen(
|
languages = subprocess.Popen(
|
||||||
@ -140,6 +127,9 @@ def mac_setup_localization(dir, domain):
|
|||||||
return usable
|
return usable
|
||||||
|
|
||||||
def mac_get_locale():
|
def mac_get_locale():
|
||||||
|
"""
|
||||||
|
Get the locale and specifiers from defaults.
|
||||||
|
"""
|
||||||
locale = ""
|
locale = ""
|
||||||
calendar = ""
|
calendar = ""
|
||||||
currency = ""
|
currency = ""
|
||||||
@ -177,6 +167,9 @@ def mac_setup_localization(dir, domain):
|
|||||||
return (locale, calendar, currency)
|
return (locale, calendar, currency)
|
||||||
|
|
||||||
def mac_get_collation():
|
def mac_get_collation():
|
||||||
|
"""
|
||||||
|
Extract the collation (sort order) locale from the defaults string.
|
||||||
|
"""
|
||||||
collation = ""
|
collation = ""
|
||||||
try:
|
try:
|
||||||
collation = subprocess.Popen(
|
collation = subprocess.Popen(
|
||||||
@ -196,11 +189,13 @@ def mac_setup_localization(dir, domain):
|
|||||||
|
|
||||||
return collation
|
return collation
|
||||||
|
|
||||||
# Locale.setlocale() will throw if any LC_* environment variable isn't
|
|
||||||
# a fully qualified one present in
|
|
||||||
# /usr/share/locale. mac_resolve_locale ensures that a locale meets
|
|
||||||
# that requirement.
|
|
||||||
def mac_resolve_locale(loc):
|
def mac_resolve_locale(loc):
|
||||||
|
"""
|
||||||
|
Locale.setlocale() will throw if any LC_* environment variable
|
||||||
|
isn't a fully qualified one present in
|
||||||
|
/usr/share/locale. mac_resolve_locale ensures that a locale
|
||||||
|
meets that requirement.
|
||||||
|
"""
|
||||||
if len(loc) < 2:
|
if len(loc) < 2:
|
||||||
return None
|
return None
|
||||||
if len(loc) >= 5 and os.path.exists(os.path.join(locale_dir, loc[:5])):
|
if len(loc) >= 5 and os.path.exists(os.path.join(locale_dir, loc[:5])):
|
||||||
@ -214,11 +209,10 @@ def mac_setup_localization(dir, domain):
|
|||||||
else:
|
else:
|
||||||
# OK, no, look through the translation list, but that's not likely
|
# OK, no, look through the translation list, but that's not likely
|
||||||
# to be 5 letters long either
|
# to be 5 letters long either
|
||||||
for l in translations:
|
for _la in translations:
|
||||||
if (l.startswith(loc) and len(l) >= 5
|
if (_la.startswith(loc) and len(_la) >= 5
|
||||||
and os.path.exists(os.path.join(locale_dir, l[:5]))):
|
and os.path.exists(os.path.join(locale_dir, _la[:5]))):
|
||||||
return l[:5]
|
return _la[:5]
|
||||||
break
|
|
||||||
|
|
||||||
else:
|
else:
|
||||||
# so as a last resort, pick the first one for that language.
|
# so as a last resort, pick the first one for that language.
|
||||||
@ -237,20 +231,22 @@ def mac_setup_localization(dir, domain):
|
|||||||
collation = mac_get_collation()
|
collation = mac_get_collation()
|
||||||
translations = mac_language_list()
|
translations = mac_language_list()
|
||||||
|
|
||||||
if "LANGUAGE" not in os.environ:
|
if currency and "LC_MONETARY" not in os.environ:
|
||||||
if len(translations) > 0:
|
os.environ["LC_MONETARY"] = currency
|
||||||
if "MULTI_TRANSLATION" in os.environ:
|
|
||||||
os.environ["LANGUAGE"] = ":".join(translations)
|
|
||||||
else:
|
|
||||||
os.environ["LANGUAGE"] = translations[0]
|
|
||||||
elif (len(loc) > 0 and loc in available
|
|
||||||
and not locale.starts_with("en")):
|
|
||||||
os.environ["LANGUAGE"] = locale
|
|
||||||
elif (len(collation) > 0 and collation in available
|
|
||||||
and not collation.starts_with("en")):
|
|
||||||
os.environ["LANGUAGE"] = collation
|
|
||||||
|
|
||||||
if "LANG" not in os.environ:
|
if calendar and "LC_TIME" not in os.environ:
|
||||||
|
os.environ["LC_TIME"] = calendar
|
||||||
|
|
||||||
|
if currency and "LC_MONETARY" not in os.environ:
|
||||||
|
os.environ["LC_MONETARY"] = currency
|
||||||
|
|
||||||
|
|
||||||
|
if calendar and "LC_TIME" not in os.environ:
|
||||||
|
os.environ["LC_TIME"] = calendar
|
||||||
|
|
||||||
|
if "LANG" in os.environ:
|
||||||
|
lang = os.environ["LANG"]
|
||||||
|
else:
|
||||||
lang = "en_US"
|
lang = "en_US"
|
||||||
loc = mac_resolve_locale(loc)
|
loc = mac_resolve_locale(loc)
|
||||||
if loc != None:
|
if loc != None:
|
||||||
@ -261,6 +257,20 @@ def mac_setup_localization(dir, domain):
|
|||||||
|
|
||||||
elif len(collation) > 0:
|
elif len(collation) > 0:
|
||||||
lang = mac_resolve_locale(collation)
|
lang = mac_resolve_locale(collation)
|
||||||
if lang != None:
|
|
||||||
os.environ["LANG"] = lang
|
if "LANGUAGE" in os.environ:
|
||||||
os.environ["LC_CTYPE"] = lang + ".UTF-8"
|
language = [l for l in os.environ["LANGUAGE"].split(":")
|
||||||
|
if l in available]
|
||||||
|
elif "LANG" in os.environ:
|
||||||
|
language = [lang[0:2]]
|
||||||
|
else:
|
||||||
|
if len(translations) > 0:
|
||||||
|
language = translations
|
||||||
|
elif (len(loc) > 0 and loc in available
|
||||||
|
and not loc.startswith("en")):
|
||||||
|
language = [loc]
|
||||||
|
elif (len(collation) > 0 and collation in available
|
||||||
|
and not collation.startswith("en")):
|
||||||
|
language = [collation]
|
||||||
|
|
||||||
|
return (lang, language)
|
||||||
|
@ -39,8 +39,7 @@ if sys.version_info[0] < 3:
|
|||||||
##
|
##
|
||||||
import os
|
import os
|
||||||
import signal
|
import signal
|
||||||
import gettext
|
|
||||||
_ = gettext.gettext
|
|
||||||
import locale
|
import locale
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
@ -53,8 +52,9 @@ from subprocess import Popen, PIPE
|
|||||||
# GRAMPS modules
|
# GRAMPS modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
from .gen.const import APP_GRAMPS, USER_DIRLIST, HOME_DIR, VERSION_TUPLE, GRAMPS_LOCALE
|
from .gen.const import APP_GRAMPS, USER_DIRLIST, HOME_DIR, VERSION_TUPLE
|
||||||
from .gen.constfunc import win
|
from .gen.constfunc import win
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# Setup logging
|
# Setup logging
|
||||||
@ -113,14 +113,15 @@ def exc_hook(type, value, tb):
|
|||||||
sys.excepthook = exc_hook
|
sys.excepthook = exc_hook
|
||||||
|
|
||||||
from .gen.mime import mime_type_is_defined
|
from .gen.mime import mime_type_is_defined
|
||||||
from .gen.utils.grampslocale import GrampsLocale
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# Load internationalization setup
|
# Instantiate Localization
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
|
|
||||||
const.GRAMPS_LOCALE = GrampsLocale()
|
from .gen.const import GRAMPS_LOCALE as glocale
|
||||||
|
_ = glocale.get_translation().gettext
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
|
@ -50,14 +50,13 @@ from gi.repository import GdkPixbuf
|
|||||||
# gramps modules
|
# gramps modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
from gramps.gen.const import IMAGE_DIR, URL_MANUAL_PAGE
|
from gramps.gen.const import IMAGE_DIR, URL_MANUAL_PAGE, GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.config import config
|
from gramps.gen.config import config
|
||||||
from gramps.gen.lib import NoteType
|
from gramps.gen.lib import NoteType
|
||||||
from gramps.gen.datehandler import get_date
|
from gramps.gen.datehandler import get_date
|
||||||
from .display import display_help
|
from .display import display_help
|
||||||
from .managedwindow import ManagedWindow
|
from .managedwindow import ManagedWindow
|
||||||
from gramps.gen.ggettext import sgettext as _
|
from gramps.gen.ggettext import sgettext as _
|
||||||
from gramps.gen.utils.grampslocale import trans_objclass
|
|
||||||
from gramps.gen.constfunc import mac
|
from gramps.gen.constfunc import mac
|
||||||
from .glade import Glade
|
from .glade import Glade
|
||||||
from .ddtargets import DdTargets
|
from .ddtargets import DdTargets
|
||||||
@ -1470,13 +1469,13 @@ class MultiTreeView(Gtk.TreeView):
|
|||||||
objclass, handle = None, None
|
objclass, handle = None, None
|
||||||
if objclass in ['Person', 'Event', 'Media', 'Source',
|
if objclass in ['Person', 'Event', 'Media', 'Source',
|
||||||
'Repository', 'Family', 'Note', 'Place']:
|
'Repository', 'Family', 'Note', 'Place']:
|
||||||
menu_item = Gtk.MenuItem(label=_("the object|See %s details") % trans_objclass(objclass))
|
menu_item = Gtk.MenuItem(label=_("the object|See %s details") % glocale.trans_objclass(objclass))
|
||||||
menu_item.connect("activate",
|
menu_item.connect("activate",
|
||||||
lambda widget: self.edit_obj(objclass, handle))
|
lambda widget: self.edit_obj(objclass, handle))
|
||||||
popup.append(menu_item)
|
popup.append(menu_item)
|
||||||
menu_item.show()
|
menu_item.show()
|
||||||
# ---------------------------
|
# ---------------------------
|
||||||
menu_item = Gtk.MenuItem(label=_("the object|Make %s active") % trans_objclass(objclass))
|
menu_item = Gtk.MenuItem(label=_("the object|Make %s active") % glocale.trans_objclass(objclass))
|
||||||
menu_item.connect("activate",
|
menu_item.connect("activate",
|
||||||
lambda widget: self.uistate.set_active(handle, objclass))
|
lambda widget: self.uistate.set_active(handle, objclass))
|
||||||
popup.append(menu_item)
|
popup.append(menu_item)
|
||||||
@ -1492,7 +1491,7 @@ class MultiTreeView(Gtk.TreeView):
|
|||||||
obj = self.dbstate.db.get_table_metadata(objclass)["handle_func"](my_handle)
|
obj = self.dbstate.db.get_table_metadata(objclass)["handle_func"](my_handle)
|
||||||
if obj:
|
if obj:
|
||||||
gids.add(obj.gramps_id)
|
gids.add(obj.gramps_id)
|
||||||
menu_item = Gtk.MenuItem(label=_("the object|Create Filter from %s selected...") % trans_objclass(objclass))
|
menu_item = Gtk.MenuItem(label=_("the object|Create Filter from %s selected...") % glocale.trans_objclass(objclass))
|
||||||
menu_item.connect("activate",
|
menu_item.connect("activate",
|
||||||
lambda widget: make_filter(self.dbstate, self.uistate,
|
lambda widget: make_filter(self.dbstate, self.uistate,
|
||||||
objclass, gids, title=self.title))
|
objclass, gids, title=self.title))
|
||||||
|
@ -48,8 +48,7 @@ from gi.repository import Gtk
|
|||||||
# gramps modules
|
# gramps modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
from gramps.gen.const import GLADE_DIR
|
from gramps.gen.const import GLADE_DIR, GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.utils.grampslocale import LOCALEDOMAIN
|
|
||||||
from gramps.gen.constfunc import STRTYPE
|
from gramps.gen.constfunc import STRTYPE
|
||||||
|
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
@ -82,7 +81,7 @@ class Glade(Gtk.Builder):
|
|||||||
:returns: reference to the newly-created Glade instance
|
:returns: reference to the newly-created Glade instance
|
||||||
"""
|
"""
|
||||||
GObject.GObject.__init__(self)
|
GObject.GObject.__init__(self)
|
||||||
self.set_translation_domain(LOCALEDOMAIN)
|
self.set_translation_domain(glocale.get_localedomain())
|
||||||
|
|
||||||
filename_given = filename is not None
|
filename_given = filename is not None
|
||||||
dirname_given = dirname is not None
|
dirname_given = dirname is not None
|
||||||
|
@ -52,9 +52,9 @@ from gi.repository import Gtk
|
|||||||
# Gramps modules
|
# Gramps modules
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.ggettext import sgettext as _
|
from gramps.gen.ggettext import sgettext as _
|
||||||
from gramps.gen.simple import SimpleTable
|
from gramps.gen.simple import SimpleTable
|
||||||
from gramps.gen.utils.grampslocale import trans_objclass
|
|
||||||
from gramps.gen.errors import WindowActiveError
|
from gramps.gen.errors import WindowActiveError
|
||||||
from ...widgets.multitreeview import MultiTreeView
|
from ...widgets.multitreeview import MultiTreeView
|
||||||
from ...ddtargets import DdTargets
|
from ...ddtargets import DdTargets
|
||||||
@ -127,7 +127,7 @@ class QuickTable(SimpleTable):
|
|||||||
if (index is not None and self._link[index]):
|
if (index is not None and self._link[index]):
|
||||||
# See details (edit, etc):
|
# See details (edit, etc):
|
||||||
objclass, handle = self._link[index]
|
objclass, handle = self._link[index]
|
||||||
menu_item = Gtk.MenuItem(label=_("the object|See %s details") % trans_objclass(objclass))
|
menu_item = Gtk.MenuItem(label=_("the object|See %s details") % glocale.trans_objclass(objclass))
|
||||||
menu_item.connect("activate",
|
menu_item.connect("activate",
|
||||||
lambda widget: self.on_table_doubleclick(treeview))
|
lambda widget: self.on_table_doubleclick(treeview))
|
||||||
popup.append(menu_item)
|
popup.append(menu_item)
|
||||||
@ -137,7 +137,7 @@ class QuickTable(SimpleTable):
|
|||||||
(index is not None and self._link[index])):
|
(index is not None and self._link[index])):
|
||||||
objclass, handle = self._link[index]
|
objclass, handle = self._link[index]
|
||||||
if objclass == 'Person':
|
if objclass == 'Person':
|
||||||
menu_item = Gtk.MenuItem(label=_("the object|Make %s active") % trans_objclass('Person'))
|
menu_item = Gtk.MenuItem(label=_("the object|Make %s active") % glocale.trans_objclass('Person'))
|
||||||
menu_item.connect("activate",
|
menu_item.connect("activate",
|
||||||
lambda widget: self.on_table_click(treeview))
|
lambda widget: self.on_table_click(treeview))
|
||||||
popup.append(menu_item)
|
popup.append(menu_item)
|
||||||
|
@ -29,15 +29,15 @@ Translator class for use by plugins.
|
|||||||
# python modules
|
# python modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
import gettext
|
|
||||||
_ = gettext.gettext
|
|
||||||
|
|
||||||
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
|
from gramps.gen.ggettext import gettext as _
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# GRAMPS modules
|
# GRAMPS modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
from gramps.gen.utils.grampslocale import get_localedomain
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.datehandler import displayer, LANG_TO_DISPLAY
|
from gramps.gen.datehandler import displayer, LANG_TO_DISPLAY
|
||||||
from gramps.gen.config import config
|
from gramps.gen.config import config
|
||||||
from gramps.gen.lib.grampstype import GrampsType
|
from gramps.gen.lib.grampstype import GrampsType
|
||||||
@ -122,7 +122,7 @@ def get_language_string(lang_code):
|
|||||||
# Translator
|
# Translator
|
||||||
#
|
#
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
class Translator:
|
class Translator(object):
|
||||||
"""
|
"""
|
||||||
This class provides translated strings for the configured language.
|
This class provides translated strings for the configured language.
|
||||||
"""
|
"""
|
||||||
@ -140,14 +140,12 @@ class Translator:
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
if lang == Translator.DEFAULT_TRANSLATION_STR:
|
if lang == Translator.DEFAULT_TRANSLATION_STR:
|
||||||
self.__trans = None
|
self.__trans = glocale.get_translation()
|
||||||
self.__dd = displayer
|
self.__dd = displayer
|
||||||
else:
|
else:
|
||||||
# fallback=True will cause the translator to use English if
|
# If lang isn't supported, this will fallback to the
|
||||||
# lang = "en" or if something goes wrong.
|
# current global language
|
||||||
self.__trans = gettext.translation(get_localedomain(),
|
self.__trans = glocale.get_translation(languages=[lang])
|
||||||
languages=[lang],
|
|
||||||
fallback=True)
|
|
||||||
val = config.get('preferences.date-format')
|
val = config.get('preferences.date-format')
|
||||||
if lang in LANG_TO_DISPLAY:
|
if lang in LANG_TO_DISPLAY:
|
||||||
self.__dd = LANG_TO_DISPLAY[lang](val)
|
self.__dd = LANG_TO_DISPLAY[lang](val)
|
||||||
@ -164,10 +162,7 @@ class Translator:
|
|||||||
:rtype: unicode
|
:rtype: unicode
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if self.__trans is None:
|
return self.__trans.gettext(message)
|
||||||
return cuni(gettext.gettext(message))
|
|
||||||
else:
|
|
||||||
return self.__trans.ugettext(message)
|
|
||||||
|
|
||||||
def ngettext(self, singular, plural, n):
|
def ngettext(self, singular, plural, n):
|
||||||
"""
|
"""
|
||||||
@ -189,10 +184,7 @@ class Translator:
|
|||||||
:rtype: unicode
|
:rtype: unicode
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if self.__trans is None:
|
return self.__trans.ngettext(singular, plural, n)
|
||||||
return cuni(gettext.ngettext(singular, plural, n))
|
|
||||||
else:
|
|
||||||
return self.__trans.ungettext(singular, plural, n)
|
|
||||||
|
|
||||||
def sgettext(self, msgid, sep='|'):
|
def sgettext(self, msgid, sep='|'):
|
||||||
"""
|
"""
|
||||||
@ -211,11 +203,10 @@ class Translator:
|
|||||||
:rtype: unicode
|
:rtype: unicode
|
||||||
|
|
||||||
"""
|
"""
|
||||||
msgval = self.gettext(msgid)
|
try:
|
||||||
if msgval == msgid:
|
return self.__trans.sgettext(msgid)
|
||||||
sep_idx = msgid.rfind(sep)
|
except AttributeError:
|
||||||
msgval = msgid[sep_idx+1:]
|
return self.__trans.gettext(msgid)
|
||||||
return cuni(msgval)
|
|
||||||
|
|
||||||
def get_date(self, date):
|
def get_date(self, date):
|
||||||
"""
|
"""
|
||||||
|
@ -39,6 +39,7 @@ from gramps.gen.ggettext import gettext as _
|
|||||||
# gramps modules
|
# gramps modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.display.name import displayer as global_name_display
|
from gramps.gen.display.name import displayer as global_name_display
|
||||||
from gramps.gen.errors import ReportError
|
from gramps.gen.errors import ReportError
|
||||||
from gramps.gen.lib import ChildRefType
|
from gramps.gen.lib import ChildRefType
|
||||||
@ -50,7 +51,6 @@ from gramps.gen.plug.docgen import (IndexMark, FontStyle, ParagraphStyle,
|
|||||||
from gramps.gen.plug.report import Report
|
from gramps.gen.plug.report import Report
|
||||||
from gramps.gen.plug.report import utils as ReportUtils
|
from gramps.gen.plug.report import utils as ReportUtils
|
||||||
from gramps.gen.plug.report import MenuReportOptions
|
from gramps.gen.plug.report import MenuReportOptions
|
||||||
from gramps.gen.utils.grampslocale import get_available_translations
|
|
||||||
from gramps.plugins.lib.libnarrate import Narrator
|
from gramps.plugins.lib.libnarrate import Narrator
|
||||||
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
||||||
|
|
||||||
@ -299,7 +299,7 @@ class AncestorOptions(MenuReportOptions):
|
|||||||
trans = EnumeratedListOption(_("Translation"),
|
trans = EnumeratedListOption(_("Translation"),
|
||||||
Translator.DEFAULT_TRANSLATION_STR)
|
Translator.DEFAULT_TRANSLATION_STR)
|
||||||
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
||||||
for language in get_available_translations():
|
for language in glocale.get_available_translations():
|
||||||
trans.add_item(language, get_language_string(language))
|
trans.add_item(language, get_language_string(language))
|
||||||
trans.set_help(_("The translation to be used for the report."))
|
trans.set_help(_("The translation to be used for the report."))
|
||||||
menu.add_option(category_name, "trans", trans)
|
menu.add_option(category_name, "trans", trans)
|
||||||
|
@ -42,6 +42,7 @@ from gramps.gen.ggettext import gettext as _
|
|||||||
# GRAMPS modules
|
# GRAMPS modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.display.name import displayer as global_name_display
|
from gramps.gen.display.name import displayer as global_name_display
|
||||||
from gramps.gen.errors import ReportError
|
from gramps.gen.errors import ReportError
|
||||||
from gramps.gen.lib import EventType, FamilyRelType, Person, NoteType
|
from gramps.gen.lib import EventType, FamilyRelType, Person, NoteType
|
||||||
@ -54,7 +55,6 @@ from gramps.gen.plug.report import endnotes
|
|||||||
from gramps.gen.plug.report import utils as ReportUtils
|
from gramps.gen.plug.report import utils as ReportUtils
|
||||||
from gramps.gen.plug.report import MenuReportOptions
|
from gramps.gen.plug.report import MenuReportOptions
|
||||||
from gramps.plugins.lib.libnarrate import Narrator
|
from gramps.plugins.lib.libnarrate import Narrator
|
||||||
from gramps.gen.utils.grampslocale import get_available_translations
|
|
||||||
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
||||||
|
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
@ -753,7 +753,7 @@ class DetAncestorOptions(MenuReportOptions):
|
|||||||
trans = EnumeratedListOption(_("Translation"),
|
trans = EnumeratedListOption(_("Translation"),
|
||||||
Translator.DEFAULT_TRANSLATION_STR)
|
Translator.DEFAULT_TRANSLATION_STR)
|
||||||
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
||||||
for language in get_available_translations():
|
for language in glocale.get_available_translations():
|
||||||
trans.add_item(language, get_language_string(language))
|
trans.add_item(language, get_language_string(language))
|
||||||
trans.set_help(_("The translation to be used for the report."))
|
trans.set_help(_("The translation to be used for the report."))
|
||||||
addopt("trans", trans)
|
addopt("trans", trans)
|
||||||
|
@ -45,6 +45,7 @@ from functools import partial
|
|||||||
# GRAMPS modules
|
# GRAMPS modules
|
||||||
#
|
#
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
||||||
from gramps.gen.display.name import displayer as global_name_display
|
from gramps.gen.display.name import displayer as global_name_display
|
||||||
from gramps.gen.errors import ReportError
|
from gramps.gen.errors import ReportError
|
||||||
from gramps.gen.lib import FamilyRelType, Person, NoteType
|
from gramps.gen.lib import FamilyRelType, Person, NoteType
|
||||||
@ -58,7 +59,6 @@ from gramps.gen.plug.report import endnotes
|
|||||||
from gramps.gen.plug.report import utils as ReportUtils
|
from gramps.gen.plug.report import utils as ReportUtils
|
||||||
from gramps.gen.plug.report import MenuReportOptions
|
from gramps.gen.plug.report import MenuReportOptions
|
||||||
from gramps.plugins.lib.libnarrate import Narrator
|
from gramps.plugins.lib.libnarrate import Narrator
|
||||||
from gramps.gen.utils.grampslocale import get_available_translations
|
|
||||||
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
from gramps.plugins.lib.libtranslate import Translator, get_language_string
|
||||||
|
|
||||||
#------------------------------------------------------------------------
|
#------------------------------------------------------------------------
|
||||||
@ -928,7 +928,7 @@ class DetDescendantOptions(MenuReportOptions):
|
|||||||
trans = EnumeratedListOption(_("Translation"),
|
trans = EnumeratedListOption(_("Translation"),
|
||||||
Translator.DEFAULT_TRANSLATION_STR)
|
Translator.DEFAULT_TRANSLATION_STR)
|
||||||
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
trans.add_item(Translator.DEFAULT_TRANSLATION_STR, _("Default"))
|
||||||
for language in get_available_translations():
|
for language in glocale.get_available_translations():
|
||||||
trans.add_item(language, get_language_string(language))
|
trans.add_item(language, get_language_string(language))
|
||||||
trans.set_help(_("The translation to be used for the report."))
|
trans.set_help(_("The translation to be used for the report."))
|
||||||
add_option("trans", trans)
|
add_option("trans", trans)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user