2012-01-24 14:24:31 +00:00
|
|
|
<?php /*
|
|
|
|
LibreSSL - CAcert web application
|
|
|
|
Copyright (C) 2004-2011 CAcert Inc.
|
|
|
|
|
|
|
|
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; version 2 of the License.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class provides some functions for language handling
|
|
|
|
*/
|
|
|
|
class L10n {
|
|
|
|
/**
|
|
|
|
* These are tranlations we currently support.
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* If another translation is added, it doesn't suffice to have gettext set
|
|
|
|
* up, you also need to add it here, because it acts as a white list.
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @var array("ISO-language code" => "native name of the language")
|
|
|
|
*/
|
|
|
|
public static $translations = array(
|
|
|
|
"ar" => "العربية",
|
|
|
|
"bg" => "Български",
|
|
|
|
"cs" => "Čeština",
|
|
|
|
"da" => "Dansk",
|
|
|
|
"de" => "Deutsch",
|
|
|
|
"el" => "Ελληνικά",
|
|
|
|
"en" => "English",
|
|
|
|
"es" => "Español",
|
|
|
|
"fi" => "Suomi",
|
|
|
|
"fr" => "Français",
|
|
|
|
"hu" => "Magyar",
|
|
|
|
"it" => "Italiano",
|
|
|
|
"ja" => "日本語",
|
|
|
|
"lv" => "Latviešu",
|
|
|
|
"nl" => "Nederlands",
|
|
|
|
"pl" => "Polski",
|
|
|
|
"pt" => "Português",
|
|
|
|
"pt-br" => "Português Brasileiro",
|
|
|
|
"ru" => "Русский",
|
|
|
|
"sv" => "Svenska",
|
|
|
|
"tr" => "Türkçe",
|
|
|
|
"zh-cn" => "中文(简体)",
|
|
|
|
"zh-tw" => "中文(臺灣)",
|
|
|
|
);
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
/**
|
|
|
|
* setlocale needs a language + region code for whatever reason so here's
|
|
|
|
* the mapping from a translation code to locales with the region that
|
|
|
|
* seemed the most common for this language
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* You probably never need this. Use {@link set_translation()} to change the
|
|
|
|
* language instead of manually calling setlocale().
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @var array(string => string)
|
|
|
|
*/
|
|
|
|
private static $locales = array(
|
|
|
|
"ar" => "ar_JO",
|
|
|
|
"bg" => "bg_BG",
|
|
|
|
"cs" => "cs_CZ",
|
|
|
|
"da" => "da_DK",
|
|
|
|
"de" => "de_DE",
|
|
|
|
"el" => "el_GR",
|
|
|
|
"en" => "en_US",
|
|
|
|
"es" => "es_ES",
|
|
|
|
"fa" => "fa_IR",
|
|
|
|
"fi" => "fi_FI",
|
|
|
|
"fr" => "fr_FR",
|
|
|
|
"he" => "he_IL",
|
|
|
|
"hr" => "hr_HR",
|
|
|
|
"hu" => "hu_HU",
|
|
|
|
"id" => "id_ID",
|
|
|
|
"is" => "is_IS",
|
|
|
|
"it" => "it_IT",
|
|
|
|
"ja" => "ja_JP",
|
|
|
|
"ka" => "ka_GE",
|
|
|
|
"ko" => "ko_KR",
|
|
|
|
"lv" => "lv_LV",
|
|
|
|
"nb" => "nb_NO",
|
|
|
|
"nl" => "nl_NL",
|
|
|
|
"pl" => "pl_PL",
|
|
|
|
"pt" => "pt_PT",
|
|
|
|
"pt-br" => "pt_BR",
|
|
|
|
"ro" => "ro_RO",
|
|
|
|
"ru" => "ru_RU",
|
|
|
|
"sl" => "sl_SI",
|
|
|
|
"sv" => "sv_SE",
|
|
|
|
"th" => "th_TH",
|
|
|
|
"tr" => "tr_TR",
|
|
|
|
"uk" => "uk_UA",
|
|
|
|
"zh-cn" => "zh_CN",
|
|
|
|
"zh-tw" => "zh_TW",
|
|
|
|
);
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
/**
|
|
|
|
* Auto-detects the language that should be used and sets it. Only works for
|
|
|
|
* HTTP, not in a command line script.
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* Priority:
|
|
|
|
* <ol>
|
|
|
|
* <li>explicit parameter "lang" passed in HTTP (e.g. via GET)</li>
|
|
|
|
* <li>existing setting in the session (stick to the setting we had before)
|
|
|
|
* </li>
|
|
|
|
* <li>auto-detect via the HTTP Accept-Language header sent by the user
|
|
|
|
* agent</li>
|
|
|
|
* </ol>
|
|
|
|
*/
|
|
|
|
public static function detect_language() {
|
|
|
|
if ( (self::get_translation() != "")
|
|
|
|
// already set in the session?
|
|
|
|
&&
|
|
|
|
!(array_key_exists("lang", $_REQUEST) &&
|
|
|
|
trim($_REQUEST["lang"]) != "")
|
|
|
|
// explicit parameter?
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if ( self::set_translation(self::get_translation()) ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
$languages = array();
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// parse Accept-Language header
|
|
|
|
if (array_key_exists('HTTP_ACCEPT_LANGUAGE', $_SERVER)) {
|
|
|
|
$bits = explode(",", strtolower(
|
|
|
|
str_replace(" ", "", $_SERVER['HTTP_ACCEPT_LANGUAGE'])
|
|
|
|
));
|
|
|
|
foreach($bits as $lang)
|
|
|
|
{
|
|
|
|
$b = explode(";", $lang);
|
|
|
|
if(count($b)>1 && substr($b[1], 0, 2) == "q=")
|
|
|
|
$c = floatval(substr($b[1], 2));
|
|
|
|
else
|
|
|
|
$c = 1;
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
if ($c != 0)
|
|
|
|
{
|
|
|
|
$languages[trim($b[0])] = $c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// check if there is an explicit language given as parameter
|
|
|
|
if(array_key_exists("lang",$_REQUEST) && trim($_REQUEST["lang"]) != "")
|
|
|
|
{
|
|
|
|
// higher priority than those values in the header
|
|
|
|
$languages[strtolower(trim($_REQUEST["lang"]))] = 2.0;
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
arsort($languages, SORT_NUMERIC);
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// this is used to be compatible with browsers like internet
|
|
|
|
// explorer which only provide the language code including the
|
|
|
|
// region not without. Also handles the fallback to English (qvalues
|
|
|
|
// may only have three digits after the .)
|
|
|
|
$fallbacks = array("en" => 0.0005);
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
foreach($languages as $lang => $qvalue)
|
|
|
|
{
|
|
|
|
// ignore any non-conforming values (that's why we don't need to
|
|
|
|
// mysql_real_escape() or escapeshellarg(), but take care of
|
|
|
|
// the '*')
|
|
|
|
// spec: ( ( 1*8ALPHA *( "-" 1*8ALPHA ) ) | "*" )
|
|
|
|
if ( preg_match('/^(?:([a-zA-Z]{1,8})(?:-[a-zA-Z]{1,8})*|\*)$/',
|
|
|
|
$lang, $matches) !== 1 ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$lang_prefix = $matches[1]; // usually two-letter language code
|
|
|
|
$fallbacks[$lang_prefix] = $qvalue;
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
$chosen_translation = "";
|
|
|
|
if ($lang === '*') {
|
|
|
|
// According to the standard '*' matches anything but any
|
|
|
|
// language explicitly specified. So in theory if there
|
|
|
|
// was an explicit mention of "en" with a lower priority
|
|
|
|
// this would be incorrect, but that's too much trouble.
|
|
|
|
$chosen_translation = "en";
|
|
|
|
} else {
|
|
|
|
$lang_length = strlen($lang);
|
|
|
|
foreach (self::$translations as $translation => $ignore)
|
|
|
|
{
|
|
|
|
// May match exactly or on every '-'
|
|
|
|
if ( $translation === $lang ||
|
|
|
|
substr($translation, 0, $lang_length + 1)
|
|
|
|
=== $lang.'-'
|
|
|
|
)
|
|
|
|
{
|
|
|
|
$chosen_translation = $translation;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
if ($chosen_translation !== "")
|
|
|
|
{
|
|
|
|
if (self::set_translation($chosen_translation)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// No translation found yet => try the prefixes
|
|
|
|
arsort($fallbacks, SORT_NUMERIC);
|
|
|
|
foreach ($fallbacks as $lang => $qvalue) {
|
|
|
|
if (self::set_translation($lang)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// should not get here, as the fallback of "en" is provided and that
|
|
|
|
// should always work => log an error
|
|
|
|
trigger_error("L10n::detect_language(): could not set language",
|
|
|
|
E_USER_WARNING);
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Normalise the translation code (e.g. from the old codes to the new)
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
* a translation code or the empty string if it can't be normalised
|
|
|
|
*/
|
|
|
|
public static function normalise_translation($translation_code) {
|
|
|
|
// check $translation_code against whitelist
|
|
|
|
if (array_key_exists($translation_code, self::$translations) ) {
|
|
|
|
return $translation_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
// maybe it's a locale as previously used in the system? e.g. en_AU
|
|
|
|
if (preg_match('/^([a-z][a-z])_([A-Z][A-Z])$/', $translation_code, $matches) !== 1) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
$lang_code = $matches[1];
|
|
|
|
$region_code = strtolower($matches[2]);
|
|
|
|
|
|
|
|
if (array_key_exists("${lang_code}-${region_code}", self::$translations)) {
|
|
|
|
return "${lang_code}-${region_code}";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (array_key_exists($lang_code, self::$translations)) {
|
|
|
|
return $lang_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
/**
|
|
|
|
* Get the set translation
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @return string
|
|
|
|
* a translation code or the empty string if not set
|
|
|
|
*/
|
|
|
|
public static function get_translation() {
|
|
|
|
if (array_key_exists('language', $_SESSION['_config'])) {
|
|
|
|
return $_SESSION['_config']['language'];
|
|
|
|
} else {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
/**
|
|
|
|
* Set the translation to use.
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @param string $translation_code
|
|
|
|
* the translation code as specified in the keys of {@link $translations}
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @return bool
|
|
|
|
* <ul>
|
|
|
|
* <li>true if the translation has been set successfully</li>
|
|
|
|
* <li>false if the $translation_code was not contained in the white
|
|
|
|
* list or could not be set for other reasons (e.g. setlocale()
|
|
|
|
* failed because the locale has not been set up on the system -
|
|
|
|
* details will be logged)</li>
|
|
|
|
* </ul>
|
|
|
|
*/
|
|
|
|
public static function set_translation($translation_code) {
|
2014-07-16 10:32:57 +00:00
|
|
|
$translation_code = self::normalise_translation($translation_code);
|
|
|
|
if (empty($translation_code)) {
|
|
|
|
return false;
|
2012-01-24 14:24:31 +00:00
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// map translation to locale
|
|
|
|
if ( !array_key_exists($translation_code, self::$locales) ) {
|
|
|
|
// weird. maybe you added a translation but haven't added a
|
|
|
|
// translation to locale mapping in self::locales?
|
|
|
|
trigger_error("L10n::set_translation(): could not map the ".
|
|
|
|
"translation $translation_code to a locale", E_USER_WARNING);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$locale = self::$locales[$translation_code];
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// set up locale
|
|
|
|
if ( !putenv("LANG=$locale") ) {
|
|
|
|
trigger_error("L10n::set_translation(): could not set the ".
|
|
|
|
"environment variable LANG to $locale", E_USER_WARNING);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if ( !setlocale(LC_ALL, $locale) ) {
|
|
|
|
trigger_error("L10n::set_translation(): could not setlocale() ".
|
|
|
|
"LC_ALL to $locale", E_USER_WARNING);
|
|
|
|
return false;
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// only set if we're running in a server not in a script
|
|
|
|
if (isset($_SESSION)) {
|
|
|
|
// save the setting
|
|
|
|
$_SESSION['_config']['language'] = $translation_code;
|
2014-07-16 10:32:57 +00:00
|
|
|
|
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
// Set up the recode settings needed e.g. in PDF creation
|
|
|
|
$_SESSION['_config']['recode'] = "html..latin-1";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
if($translation_code === "zh-cn" || $translation_code === "zh-tw")
|
|
|
|
{
|
|
|
|
$_SESSION['_config']['recode'] = "html..gb2312";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
} else if($translation_code === "pl" || $translation_code === "hu") {
|
|
|
|
$_SESSION['_config']['recode'] = "html..ISO-8859-2";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
} else if($translation_code === "ja") {
|
|
|
|
$_SESSION['_config']['recode'] = "html..SHIFT-JIS";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
} else if($translation_code === "ru") {
|
|
|
|
$_SESSION['_config']['recode'] = "html..ISO-8859-5";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
} else if($translation_code == "lt") { // legacy, keep for reference
|
|
|
|
$_SESSION['_config']['recode'] = "html..ISO-8859-13";
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
}
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
return true;
|
|
|
|
}
|
2014-07-16 10:32:57 +00:00
|
|
|
|
2012-01-24 14:24:31 +00:00
|
|
|
/**
|
|
|
|
* Sets up the text domain used by gettext
|
2014-07-16 10:32:57 +00:00
|
|
|
*
|
2012-01-24 14:24:31 +00:00
|
|
|
* @param string $domain
|
|
|
|
* the gettext domain that should be used, defaults to "messages"
|
|
|
|
*/
|
|
|
|
public static function init_gettext($domain = 'messages') {
|
|
|
|
bindtextdomain($domain, $_SESSION['_config']['filepath'].'/locale');
|
|
|
|
textdomain($domain);
|
|
|
|
}
|
|
|
|
}
|