i18n.py
110 lines
| 3.7 KiB
| text/x-python
|
PythonLexer
/ mercurial / i18n.py
Martin Geisler
|
r8226 | # i18n.py - internationalization support for mercurial | ||
# | ||||
# Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Benoit Boissinot
|
r1400 | |||
Gregory Szorc
|
r25955 | from __future__ import absolute_import | ||
import gettext as gettextmod | ||||
import locale | ||||
import os | ||||
import sys | ||||
Pulkit Goyal
|
r30050 | from . import ( | ||
encoding, | ||||
pycompat, | ||||
) | ||||
Martin Geisler
|
r7650 | |||
# modelled after templater.templatepath: | ||||
Augie Fackler
|
r14975 | if getattr(sys, 'frozen', None) is not None: | ||
Pulkit Goyal
|
r30669 | module = pycompat.sysexecutable | ||
Martin Geisler
|
r7650 | else: | ||
Pulkit Goyal
|
r31074 | module = pycompat.fsencode(__file__) | ||
Martin Geisler
|
r7650 | |||
timeless
|
r28674 | try: | ||
unicode | ||||
except NameError: | ||||
unicode = str | ||||
Martin Geisler
|
r7650 | |||
Yuya Nishihara
|
r21987 | _languages = None | ||
Jun Wu
|
r34646 | if (pycompat.iswindows | ||
Yuya Nishihara
|
r30035 | and 'LANGUAGE' not in encoding.environ | ||
and 'LC_ALL' not in encoding.environ | ||||
and 'LC_MESSAGES' not in encoding.environ | ||||
and 'LANG' not in encoding.environ): | ||||
Yuya Nishihara
|
r21987 | # Try to detect UI language by "User Interface Language Management" API | ||
# if no locale variables are set. Note that locale.getdefaultlocale() | ||||
# uses GetLocaleInfo(), which may be different from UI language. | ||||
# (See http://msdn.microsoft.com/en-us/library/dd374098(v=VS.85).aspx ) | ||||
try: | ||||
import ctypes | ||||
langid = ctypes.windll.kernel32.GetUserDefaultUILanguage() | ||||
_languages = [locale.windows_locale[langid]] | ||||
except (ImportError, AttributeError, KeyError): | ||||
# ctypes not found or unknown langid | ||||
pass | ||||
Mads Kiilerich
|
r22638 | _ugettext = None | ||
def setdatapath(datapath): | ||||
Pulkit Goyal
|
r30301 | datapath = pycompat.fsdecode(datapath) | ||
Augie Fackler
|
r30085 | localedir = os.path.join(datapath, pycompat.sysstr('locale')) | ||
Mads Kiilerich
|
r22638 | t = gettextmod.translation('hg', localedir, _languages, fallback=True) | ||
global _ugettext | ||||
timeless
|
r28674 | try: | ||
_ugettext = t.ugettext | ||||
except AttributeError: | ||||
_ugettext = t.gettext | ||||
Martin Geisler
|
r7651 | |||
Yuya Nishihara
|
r34661 | _msgcache = {} # encoding: {message: translation} | ||
Augie Fackler
|
r23031 | |||
Martin Geisler
|
r7651 | def gettext(message): | ||
"""Translate message. | ||||
The message is looked up in the catalog to get a Unicode string, | ||||
which is encoded in the local encoding before being returned. | ||||
Important: message is restricted to characters in the encoding | ||||
given by sys.getdefaultencoding() which is most likely 'ascii'. | ||||
""" | ||||
# If message is None, t.ugettext will return u'None' as the | ||||
# translation whereas our callers expect us to return None. | ||||
Mads Kiilerich
|
r22638 | if message is None or not _ugettext: | ||
Martin Geisler
|
r7651 | return message | ||
Yuya Nishihara
|
r34661 | cache = _msgcache.setdefault(encoding.encoding, {}) | ||
if message not in cache: | ||||
Augie Fackler
|
r23031 | if type(message) is unicode: | ||
# goofy unicode docstrings in test | ||||
paragraphs = message.split(u'\n\n') | ||||
else: | ||||
paragraphs = [p.decode("ascii") for p in message.split('\n\n')] | ||||
# Be careful not to translate the empty string -- it holds the | ||||
# meta data of the .po file. | ||||
Gregory Szorc
|
r29415 | u = u'\n\n'.join([p and _ugettext(p) or u'' for p in paragraphs]) | ||
Augie Fackler
|
r23031 | try: | ||
# encoding.tolocal cannot be used since it will first try to | ||||
# decode the Unicode string. Calling u.decode(enc) really | ||||
# means u.encode(sys.getdefaultencoding()).decode(enc). Since | ||||
# the Python encoding defaults to 'ascii', this fails if the | ||||
# translated string use non-ASCII characters. | ||||
Pulkit Goyal
|
r30050 | encodingstr = pycompat.sysstr(encoding.encoding) | ||
Yuya Nishihara
|
r34661 | cache[message] = u.encode(encodingstr, "replace") | ||
Augie Fackler
|
r23031 | except LookupError: | ||
# An unknown encoding results in a LookupError. | ||||
Yuya Nishihara
|
r34661 | cache[message] = message | ||
return cache[message] | ||||
Martin Geisler
|
r7651 | |||
Brodie Rao
|
r13849 | def _plain(): | ||
Yuya Nishihara
|
r30035 | if ('HGPLAIN' not in encoding.environ | ||
and 'HGPLAINEXCEPT' not in encoding.environ): | ||||
Brodie Rao
|
r13849 | return False | ||
Yuya Nishihara
|
r30035 | exceptions = encoding.environ.get('HGPLAINEXCEPT', '').strip().split(',') | ||
Brodie Rao
|
r13849 | return 'i18n' not in exceptions | ||
if _plain(): | ||||
Brodie Rao
|
r10455 | _ = lambda message: message | ||
else: | ||||
_ = gettext | ||||