123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359 |
- <?php
- // Copyright (C) 2010-2016 Combodo SARL
- //
- // This file is part of iTop.
- //
- // iTop is free software; you can redistribute it and/or modify
- // it under the terms of the GNU Affero General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // iTop 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 Affero General Public License for more details.
- //
- // You should have received a copy of the GNU Affero General Public License
- // along with iTop. If not, see <http://www.gnu.org/licenses/>
- /**
- * Class Dict
- * Management of localizable strings
- *
- * @copyright Copyright (C) 2010-2012 Combodo SARL
- * @license http://opensource.org/licenses/AGPL-3.0
- */
- class DictException extends CoreException
- {
- }
- class DictExceptionUnknownLanguage extends DictException
- {
- public function __construct($sLanguageCode)
- {
- $aContext = array();
- $aContext['language_code'] = $sLanguageCode;
- parent::__construct('Unknown localization language', $aContext);
- }
- }
- class DictExceptionMissingString extends DictException
- {
- public function __construct($sLanguageCode, $sStringCode)
- {
- $aContext = array();
- $aContext['language_code'] = $sLanguageCode;
- $aContext['string_code'] = $sStringCode;
- parent::__construct('Missing localized string', $aContext);
- }
- }
- define('DICT_ERR_STRING', 1); // when a string is missing, return the identifier
- define('DICT_ERR_EXCEPTION', 2); // when a string is missing, throw an exception
- //define('DICT_ERR_LOG', 3); // when a string is missing, log an error
- class Dict
- {
- protected static $m_iErrorMode = DICT_ERR_STRING;
- protected static $m_sDefaultLanguage = 'EN US';
- protected static $m_sCurrentLanguage = null; // No language selected by default
- protected static $m_aLanguages = array(); // array( code => array( 'description' => '...', 'localized_description' => '...') ...)
- protected static $m_aData = array();
- protected static $m_sApplicationPrefix = null;
- public static function SetDefaultLanguage($sLanguageCode)
- {
- if (!array_key_exists($sLanguageCode, self::$m_aLanguages))
- {
- throw new DictExceptionUnknownLanguage($sLanguageCode);
- }
- self::$m_sDefaultLanguage = $sLanguageCode;
- }
- public static function SetUserLanguage($sLanguageCode)
- {
- if (!array_key_exists($sLanguageCode, self::$m_aLanguages))
- {
- throw new DictExceptionUnknownLanguage($sLanguageCode);
- }
- self::$m_sCurrentLanguage = $sLanguageCode;
- }
- public static function GetUserLanguage()
- {
- if (self::$m_sCurrentLanguage == null) // May happen when no user is logged in (i.e login screen, non authentifed page)
- {
- // In which case let's use the default language
- return self::$m_sDefaultLanguage;
- }
- return self::$m_sCurrentLanguage;
- }
- //returns a hash array( code => array( 'description' => '...', 'localized_description' => '...') ...)
- public static function GetLanguages()
- {
- return self::$m_aLanguages;
- }
- // iErrorMode from {DICT_ERR_STRING, DICT_ERR_EXCEPTION}
- public static function SetErrorMode($iErrorMode)
- {
- self::$m_iErrorMode = $iErrorMode;
- }
- /**
- * Returns a localised string from the dictonary
- * @param string $sStringCode The code identifying the dictionary entry
- * @param string $sDefault Default value if there is no match in the dictionary
- * @param bool $bUserLanguageOnly True to allow the use of the default language as a fallback, false otherwise
- * @throws DictExceptionMissingString
- * @return unknown|Ambigous <>|string
- */
- public static function S($sStringCode, $sDefault = null, $bUserLanguageOnly = false)
- {
- // Attempt to find the string in the user language
- //
- self::InitLangIfNeeded(self::GetUserLanguage());
- if (!array_key_exists(self::GetUserLanguage(), self::$m_aData))
- {
- // It may happen, when something happens before the dictionnaries get loaded
- return $sStringCode;
- }
- $aCurrentDictionary = self::$m_aData[self::GetUserLanguage()];
- if (array_key_exists($sStringCode, $aCurrentDictionary))
- {
- return $aCurrentDictionary[$sStringCode];
- }
- if (!$bUserLanguageOnly)
- {
- // Attempt to find the string in the default language
- //
- self::InitLangIfNeeded(self::$m_sDefaultLanguage);
-
- $aDefaultDictionary = self::$m_aData[self::$m_sDefaultLanguage];
- if (array_key_exists($sStringCode, $aDefaultDictionary))
- {
- return $aDefaultDictionary[$sStringCode];
- }
- // Attempt to find the string in english
- //
- self::InitLangIfNeeded('EN US');
- $aDefaultDictionary = self::$m_aData['EN US'];
- if (array_key_exists($sStringCode, $aDefaultDictionary))
- {
- return $aDefaultDictionary[$sStringCode];
- }
- }
- // Could not find the string...
- //
- switch (self::$m_iErrorMode)
- {
- case DICT_ERR_STRING:
- if (is_null($sDefault))
- {
- return $sStringCode;
- }
- else
- {
- return $sDefault;
- }
- break;
- case DICT_ERR_EXCEPTION:
- default:
- throw new DictExceptionMissingString(self::$m_sCurrentLanguage, $sStringCode);
- break;
- }
- return 'bug!';
- }
- /**
- * Formats a localized string with numbered placeholders (%1$s...) for the additional arguments
- * See vsprintf for more information about the syntax of the placeholders
- * @param string $sFormatCode
- * @return string
- */
- public static function Format($sFormatCode /*, ... arguments ....*/)
- {
- $sLocalizedFormat = self::S($sFormatCode);
- $aArguments = func_get_args();
- array_shift($aArguments);
-
- if ($sLocalizedFormat == $sFormatCode)
- {
- // Make sure the information will be displayed (ex: an error occuring before the dictionary gets loaded)
- return $sFormatCode.' - '.implode(', ', $aArguments);
- }
- return vsprintf($sLocalizedFormat, $aArguments);
- }
-
- /**
- * Initialize a the entries for a given language (replaces the former Add() method)
- * @param string $sLanguageCode Code identifying the language i.e. 'FR-FR', 'EN-US'
- * @param hash $aEntries Hash array of dictionnary entries
- */
- public static function SetEntries($sLanguageCode, $aEntries)
- {
- self::$m_aData[$sLanguageCode] = $aEntries;
- }
-
- /**
- * Set the list of available languages
- * @param hash $aLanguagesList
- */
- public static function SetLanguagesList($aLanguagesList)
- {
- self::$m_aLanguages = $aLanguagesList;
- }
-
- /**
- * Load a language from the language dictionary, if not already loaded
- * @param string $sLangCode Language code
- * @return boolean
- */
- public static function InitLangIfNeeded($sLangCode)
- {
- if (array_key_exists($sLangCode, self::$m_aData)) return true;
-
- $bResult = false;
-
- if (function_exists('apc_fetch') && (self::$m_sApplicationPrefix !== null))
- {
- // Note: For versions of APC older than 3.0.17, fetch() accepts only one parameter
- //
- self::$m_aData[$sLangCode] = apc_fetch(self::$m_sApplicationPrefix.'-dict-'.$sLangCode);
- if (self::$m_aData[$sLangCode] === false)
- {
- unset(self::$m_aData[$sLangCode]);
- }
- else
- {
- $bResult = true;
- }
- }
- if (!$bResult)
- {
- $sDictFile = APPROOT.'env-'.utils::GetCurrentEnvironment().'/dictionaries/'.str_replace(' ', '-', strtolower($sLangCode)).'.dict.php';
- require_once($sDictFile);
-
- if (function_exists('apc_store') && (self::$m_sApplicationPrefix !== null))
- {
- apc_store(self::$m_sApplicationPrefix.'-dict-'.$sLangCode, self::$m_aData[$sLangCode]);
- }
- $bResult = true;
- }
- return $bResult;
- }
-
- /**
- * Enable caching (cached using APC)
- * @param string $sApplicationPrefix The prefix for uniquely identiying this iTop instance
- */
- public static function EnableCache($sApplicationPrefix)
- {
- self::$m_sApplicationPrefix = $sApplicationPrefix;
- }
- /**
- * Reset the cached entries (cached using APC)
- * @param string $sApplicationPrefix The prefix for uniquely identiying this iTop instance
- */
- public static function ResetCache($sApplicationPrefix)
- {
- if (function_exists('apc_delete'))
- {
- foreach(self::$m_aLanguages as $sLang => $void)
- {
- apc_delete($sApplicationPrefix.'-dict-'.$sLang);
- }
- }
- }
- /////////////////////////////////////////////////////////////////////////
- /**
- * Clone a string in every language (if it exists in that language)
- */
- public static function CloneString($sSourceCode, $sDestCode)
- {
- foreach(self::$m_aLanguages as $sLanguageCode => $foo)
- {
- if (isset(self::$m_aData[$sLanguageCode][$sSourceCode]))
- {
- self::$m_aData[$sLanguageCode][$sDestCode] = self::$m_aData[$sLanguageCode][$sSourceCode];
- }
- }
- }
-
- public static function MakeStats($sLanguageCode, $sLanguageRef = 'EN US')
- {
- $aMissing = array(); // Strings missing for the target language
- $aUnexpected = array(); // Strings defined for the target language, but not found in the reference dictionary
- $aNotTranslated = array(); // Strings having the same value in both dictionaries
- $aOK = array(); // Strings having different values in both dictionaries
-
- foreach (self::$m_aData[$sLanguageRef] as $sStringCode => $sValue)
- {
- if (!array_key_exists($sStringCode, self::$m_aData[$sLanguageCode]))
- {
- $aMissing[$sStringCode] = $sValue;
- }
- }
-
- foreach (self::$m_aData[$sLanguageCode] as $sStringCode => $sValue)
- {
- if (!array_key_exists($sStringCode, self::$m_aData[$sLanguageRef]))
- {
- $aUnexpected[$sStringCode] = $sValue;
- }
- else
- {
- // The value exists in the reference
- $sRefValue = self::$m_aData[$sLanguageRef][$sStringCode];
- if ($sValue == $sRefValue)
- {
- $aNotTranslated[$sStringCode] = $sValue;
- }
- else
- {
- $aOK[$sStringCode] = $sValue;
- }
- }
- }
- return array($aMissing, $aUnexpected, $aNotTranslated, $aOK);
- }
-
- public static function Dump()
- {
- MyHelpers::var_dump_html(self::$m_aData);
- }
- // Only used by the setup to determine the list of languages to display in the initial setup screen
- // otherwise replaced by LoadModule by its own handler
- // sLanguageCode: Code identifying the language i.e. FR-FR
- // sEnglishLanguageDesc: Description of the language code, in English. i.e. French (France)
- // sLocalizedLanguageDesc: Description of the language code, in its own language. i.e. Français (France)
- // aEntries: Hash array of dictionnary entries
- // ~~ or ~* can be used to indicate entries still to be translated.
- public static function Add($sLanguageCode, $sEnglishLanguageDesc, $sLocalizedLanguageDesc, $aEntries)
- {
- if (!array_key_exists($sLanguageCode, self::$m_aLanguages))
- {
- self::$m_aLanguages[$sLanguageCode] = array('description' => $sEnglishLanguageDesc, 'localized_description' => $sLocalizedLanguageDesc);
- self::$m_aData[$sLanguageCode] = array();
- }
- // No need to actually load the strings since it's only used to know the list of languages
- // at setup time !!
- }
- }
- ?>
|