Newer
Older
<?php
/*********************************************************************
class.i18n.php
Internationalization and localization helpers for osTicket
Peter Rotich <peter@osticket.com>
Jared Hancock <jared@osticket.com>
Copyright (c) 2006-2013 osTicket
http://www.osticket.com
Released under the GNU General Public License WITHOUT ANY WARRANTY.
See LICENSE.TXT for details.
vim: expandtab sw=4 ts=4 sts=4:
**********************************************************************/
require_once INCLUDE_DIR.'class.error.php';
require_once INCLUDE_DIR.'class.yaml.php';
class Internationalization {
// Languages in order of decreasing priority. Always use en_US as a
// fallback
var $langs = array('en_US');
function Internationalization($language=false) {
global $cfg;
if ($cfg && ($lang = $cfg->getSystemLanguage()))
array_unshift($this->langs, $language);
// Detect language filesystem path, case insensitively
if ($language && ($info = self::getLanguageInfo($language))) {
array_unshift($this->langs, $info['code']);
}
}
function getTemplate($path) {
return new DataTemplate($path, $this->langs);
}
/**
* Loads data from the I18N_DIR for the target language into the
* database. This is intended to be done at the time of installation;
* however, care should be taken in this process to ensure that the
* process could be repeated if an administrator wanted to change the
* system language and reload the data.
*/
function loadDefaultData() {
# notrans -- do not translate the contents of this array
'department.yaml' => 'Dept::create',
'sla.yaml' => 'SLA::create',
'form.yaml' => 'DynamicForm::create',
// Note that department, sla, and forms are required for
// help_topic
'help_topic.yaml' => 'Topic::create',
'filter.yaml' => 'Filter::create',
'team.yaml' => 'Team::create',
// Organization
'organization.yaml' => 'Organization::__create',
// Ticket
'ticket_status.yaml' => 'TicketStatus::__create',
'group.yaml' => 'Group::create',
'file.yaml' => 'AttachmentFile::create',
'sequence.yaml' => 'Sequence::__create',
foreach ($models as $yaml=>$m) {
if ($objects = $this->getTemplate($yaml)->getData()) {
foreach ($objects as $o) {
if ($m && is_callable($m))
@call_user_func_array($m, array($o, &$errors));
// TODO: Add a warning to the success page for errors
// found here
$errors = array();
// Priorities
$priorities = $this->getTemplate('priority.yaml')->getData();
foreach ($priorities as $name=>$info) {
$sql = 'INSERT INTO '.PRIORITY_TABLE
.' SET priority='.db_input($name)
.', priority_id='.db_input($info['priority_id'])
.', priority_desc='.db_input($info['priority_desc'])
.', priority_color='.db_input($info['priority_color'])
.', priority_urgency='.db_input($info['priority_urgency']);
db_query($sql);
}
// Configuration
require_once INCLUDE_DIR.'class.config.php';
if (($tpl = $this->getTemplate('config.yaml'))
&& ($data = $tpl->getData())) {
foreach ($data as $section=>$items) {
$_config = new Config($section);
foreach ($items as $key=>$value)
$_config->set($key, $value);
}
}
foreach (array('landing','thank-you','offline',
'registration-staff', 'pwreset-staff', 'banner-staff',
'registration-client', 'pwreset-client', 'banner-client',
'registration-confirm', 'registration-thanks',
'access-link') as $type) {
$tpl = $this->getTemplate("templates/page/{$type}.yaml");
if (!($page = $tpl->getData()))
continue;
$sql = 'INSERT INTO '.PAGE_TABLE.' SET type='.db_input($type)
.', name='.db_input($page['name'])
.', body='.db_input($page['body'])
.', lang='.db_input($tpl->getLang())
.', notes='.db_input($page['notes'])
.', created=NOW(), updated=NOW(), isactive=1';
if (db_query($sql) && ($id = db_insert_id())
&& in_array($type, array('landing', 'thank-you', 'offline')))
$_config->set("{$type}_page_id", $id);
}
// Default Language
$_config->set('system_language', $this->langs[0]);
// content_id defaults to the `id` field value
db_query('UPDATE '.PAGE_TABLE.' SET content_id=id');
// Canned response examples
if (($tpl = $this->getTemplate('templates/premade.yaml'))
&& ($canned = $tpl->getData())) {
foreach ($canned as $c) {
if (($id = Canned::create($c, $errors))
$premade = Canned::lookup($id);
foreach ($c['attachments'] as $a) {
$premade->attachments->save($a, false);
}
}
}
}
// Email templates
// TODO: Lookup tpl_id
if ($objects = $this->getTemplate('email_template_group.yaml')->getData()) {
foreach ($objects as $o) {
$o['lang_id'] = $this->langs[0];
$tpl = EmailTemplateGroup::create($o, $errors);
}
}
// This shouldn't be necessary
foreach ($tpl::$all_names as $name=>$info) {
if (($tp = $this->getTemplate("templates/email/$name.yaml"))
&& ($t = $tp->getData())) {
$t['tpl_id'] = $tpl->getId();
$t['code_name'] = $name;
$id = EmailTemplate::create($t, $errors);
if ($id && ($template = EmailTemplate::lookup($id))
&& ($ids = Draft::getAttachmentIds($t['body'])))
$template->attachments->upload($ids, true);
static function getLanguageDescription($lang) {
global $thisstaff, $thisclient;
$langs = self::availableLanguages();
$lang = strtolower($lang);
if (isset($langs[$lang])) {
$info = &$langs[$lang];
if (!isset($info['desc'])) {
if (extension_loaded('intl')) {
$lang = self::getCurrentLanguage();
list($simple_lang,) = explode('_', $lang);
$info['desc'] = sprintf("%s%s",
// Display the localized name of the language
Locale::getDisplayName($info['code'], $info['code']),
// If the major language differes from the user's,
// display the language in the user's language
(strpos($simple_lang, $info['lang']) === false
? sprintf(' (%s)', Locale::getDisplayName($info['code'], $lang)) : '')
);
}
else {
$info['desc'] = sprintf("%s%s (%s)",
$info['nativeName'],
$info['locale'] ? sprintf(' - %s', $info['locale']) : '',
$info['name']);
}
}
return $info['desc'];
}
else
return $lang;
}
static function getLanguageInfo($lang) {
$langs = self::availableLanguages();
return @$langs[strtolower($lang)] ?: array();
static function availableLanguages($base=I18N_DIR) {
static $cache = false;
if ($cache) return $cache;
$langs = (include I18N_DIR . 'langs.php');
// Consider all subdirectories and .phar files in the base dir
$dirs = glob(I18N_DIR . '*', GLOB_ONLYDIR | GLOB_NOSORT);
$phars = glob(I18N_DIR . '*.phar', GLOB_NOSORT) ?: array();
$installed = array();
foreach (array_merge($dirs, $phars) as $f) {
$base = basename($f, '.phar');
@list($code, $locale) = explode('_', $base);
if (isset($langs[$code])) {
$installed[strtolower($base)] =
$langs[$code] + array(
'lang' => $code,
'locale' => $locale,
'path' => $f,
'phar' => substr($f, -5) == '.phar',
uasort($installed, function($a, $b) { return strcasecmp($a['code'], $b['code']); });
return $cache = $installed;
}
// TODO: Move this to the REQUEST class or some middleware when that
// exists.
// Algorithm borrowed from Drupal 7 (locale.inc)
static function getDefaultLanguage() {
global $cfg;
if (empty($_SERVER["HTTP_ACCEPT_LANGUAGE"]))
return $cfg->getSystemLanguage();
$languages = self::availableLanguages();
// The Accept-Language header contains information about the
// language preferences configured in the user's browser / operating
// system. RFC 2616 (section 14.4) defines the Accept-Language
// header as follows:
// Accept-Language = "Accept-Language" ":"
// 1#( language-range [ ";" "q" "=" qvalue ] )
// language-range = ( ( 1*8ALPHA *( "-" 1*8ALPHA ) ) | "*" )
// Samples: "hu, en-us;q=0.66, en;q=0.33", "hu,en-us;q=0.5"
$browser_langcodes = array();
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
if (preg_match_all('@(?<=[, ]|^)([a-zA-Z-]+|\*)(?:;q=([0-9.]+))?(?:$|\s*,\s*)@',
trim($_SERVER['HTTP_ACCEPT_LANGUAGE']), $matches, PREG_SET_ORDER)) {
foreach ($matches as $match) {
// We can safely use strtolower() here, tags are ASCII.
// RFC2616 mandates that the decimal part is no more than three
// digits, so we multiply the qvalue by 1000 to avoid floating
// point comparisons.
$langcode = strtolower($match[1]);
$qvalue = isset($match[2]) ? (float) $match[2] : 1;
$browser_langcodes[$langcode] = (int) ($qvalue * 1000);
}
}
// We should take pristine values from the HTTP headers, but
// Internet Explorer from version 7 sends only specific language
// tags (eg. fr-CA) without the corresponding generic tag (fr)
// unless explicitly configured. In that case, we assume that the
// lowest value of the specific tags is the value of the generic
// language to be as close to the HTTP 1.1 spec as possible.
//
// References:
// http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4
// http://blogs.msdn.com/b/ie/archive/2006/10/17/accept-language-header-for-internet-explorer-7.aspx
asort($browser_langcodes);
foreach ($browser_langcodes as $langcode => $qvalue) {
$generic_tag = strtok($langcode, '-');
if (!isset($browser_langcodes[$generic_tag])) {
$browser_langcodes[$generic_tag] = $qvalue;
}
}
// Find the enabled language with the greatest qvalue, following the rules
// of RFC 2616 (section 14.4). If several languages have the same qvalue,
// prefer the one with the greatest weight.
$best_match_langcode = FALSE;
$max_qvalue = 0;
foreach ($languages as $langcode => $language) {
// Language tags are case insensitive (RFC2616, sec 3.10).
// We use _ as the location separator
$langcode = str_replace('_','-',strtolower($langcode));
// If nothing matches below, the default qvalue is the one of the wildcard
// language, if set, or is 0 (which will never match).
$qvalue = isset($browser_langcodes['*']) ? $browser_langcodes['*'] : 0;
// Find the longest possible prefix of the browser-supplied language
// ('the language-range') that matches this site language ('the language tag').
$prefix = $langcode;
do {
if (isset($browser_langcodes[$prefix])) {
$qvalue = $browser_langcodes[$prefix];
break;
}
} while ($prefix = substr($prefix, 0, strrpos($prefix, '-')));
// Find the best match.
if ($qvalue > $max_qvalue) {
$best_match_langcode = $language['code'];
$max_qvalue = $qvalue;
}
}
return $best_match_langcode;
}
static function getCurrentLanguage($user=false) {
global $thisstaff, $thisclient;
$user = $user ?: $thisstaff ?: $thisclient;
if ($user && method_exists($user, 'getLanguage'))
return $user->getLanguage();
if (isset($_SESSION['client:lang']))
return $_SESSION['client:lang'];
return self::getDefaultLanguage();
static function bootstrap() {
require_once INCLUDE_DIR . 'class.translation.php';
TextDomain::setDefaultDomain($domain);
TextDomain::lookup()->setPath(I18N_DIR);
// User-specific translations
function _N($msgid, $plural, $n) {
return TextDomain::lookup()->getTranslation()
->ngettext($msgid, $plural, is_numeric($n) ? $n : 1);
}
// System-specific translations
function _S($msgid) {
global $cfg;
function _NS($msgid, $plural, $count) {
// Phrases with separate contexts
function _P($context, $msgid) {
return TextDomain::lookup()->getTranslation()
->pgettext($context, $msgid);
}
function _NP($context, $singular, $plural, $n) {
return TextDomain::lookup()->getTranslation()
->npgettext($context, $singular, $plural, is_numeric($n) ? $n : 1);
// Language-specific translations
function _L($msgid, $locale) {
return TextDomain::lookup()->getTranslation($locale)
->translate($msgid);
function _NL($msgid, $plural, $n, $locale) {
return TextDomain::lookup()->getTranslation($locale)
->ngettext($msgid, $plural, is_numeric($n) ? $n : 1);
}
class DataTemplate {
// Base folder for default data and templates
var $base = I18N_DIR;
var $filepath;
var $data;
/**
* Searches for the files matching the template in the order of the
* received languages. Once matched, the language is captured so that
* template itself does not have to keep track of the language for which
* it is defined.
*/
function DataTemplate($path, $langs=array('en_US')) {
foreach ($langs as $l) {
if (file_exists("{$this->base}/$l/$path")) {
$this->lang = $l;
$this->filepath = Misc::realpath("{$this->base}/$l/$path");
elseif (class_exists('Phar')
&& Phar::isValidPharFilename("{$this->base}/$l.phar")
&& file_exists("phar://{$this->base}/$l.phar/$path")) {
$this->lang = $l;
$this->filepath = "phar://{$this->base}/$l.phar/$path";
break;
}
}
}
function getData() {
if (!isset($this->data) && $this->filepath)
$this->data = YamlDataParser::load($this->filepath);
// TODO: If there was a parsing error, attempt to try the next
// language in the list of requested languages
return $this->data;
}
function getRawData() {
if (!isset($this->data) && $this->filepath)
return file_get_contents($this->filepath);
// TODO: If there was a parsing error, attempt to try the next
// language in the list of requested languages
return false;
}
function getLang() {
return $this->lang;
}
}
?>