File indexing completed on 2025-01-05 05:24:27

0001 <?php
0002 
0003 /**
0004  * Represents a language and defines localizable string formatting and
0005  * other functions, as well as the localized messages for HTML Purifier.
0006  */
0007 class HTMLPurifier_Language
0008 {
0009 
0010     /**
0011      * ISO 639 language code of language. Prefers shortest possible version.
0012      * @type string
0013      */
0014     public $code = 'en';
0015 
0016     /**
0017      * Fallback language code.
0018      * @type bool|string
0019      */
0020     public $fallback = false;
0021 
0022     /**
0023      * Array of localizable messages.
0024      * @type array
0025      */
0026     public $messages = array();
0027 
0028     /**
0029      * Array of localizable error codes.
0030      * @type array
0031      */
0032     public $errorNames = array();
0033 
0034     /**
0035      * True if no message file was found for this language, so English
0036      * is being used instead. Check this if you'd like to notify the
0037      * user that they've used a non-supported language.
0038      * @type bool
0039      */
0040     public $error = false;
0041 
0042     /**
0043      * Has the language object been loaded yet?
0044      * @type bool
0045      * @todo Make it private, fix usage in HTMLPurifier_LanguageTest
0046      */
0047     public $_loaded = false;
0048 
0049     /**
0050      * @type HTMLPurifier_Config
0051      */
0052     protected $config;
0053 
0054     /**
0055      * @type HTMLPurifier_Context
0056      */
0057     protected $context;
0058 
0059     /**
0060      * @param HTMLPurifier_Config $config
0061      * @param HTMLPurifier_Context $context
0062      */
0063     public function __construct($config, $context)
0064     {
0065         $this->config  = $config;
0066         $this->context = $context;
0067     }
0068 
0069     /**
0070      * Loads language object with necessary info from factory cache
0071      * @note This is a lazy loader
0072      */
0073     public function load()
0074     {
0075         if ($this->_loaded) {
0076             return;
0077         }
0078         $factory = HTMLPurifier_LanguageFactory::instance();
0079         $factory->loadLanguage($this->code);
0080         foreach ($factory->keys as $key) {
0081             $this->$key = $factory->cache[$this->code][$key];
0082         }
0083         $this->_loaded = true;
0084     }
0085 
0086     /**
0087      * Retrieves a localised message.
0088      * @param string $key string identifier of message
0089      * @return string localised message
0090      */
0091     public function getMessage($key)
0092     {
0093         if (!$this->_loaded) {
0094             $this->load();
0095         }
0096         if (!isset($this->messages[$key])) {
0097             return "[$key]";
0098         }
0099         return $this->messages[$key];
0100     }
0101 
0102     /**
0103      * Retrieves a localised error name.
0104      * @param int $int error number, corresponding to PHP's error reporting
0105      * @return string localised message
0106      */
0107     public function getErrorName($int)
0108     {
0109         if (!$this->_loaded) {
0110             $this->load();
0111         }
0112         if (!isset($this->errorNames[$int])) {
0113             return "[Error: $int]";
0114         }
0115         return $this->errorNames[$int];
0116     }
0117 
0118     /**
0119      * Converts an array list into a string readable representation
0120      * @param array $array
0121      * @return string
0122      */
0123     public function listify($array)
0124     {
0125         $sep      = $this->getMessage('Item separator');
0126         $sep_last = $this->getMessage('Item separator last');
0127         $ret = '';
0128         for ($i = 0, $c = count($array); $i < $c; $i++) {
0129             if ($i == 0) {
0130             } elseif ($i + 1 < $c) {
0131                 $ret .= $sep;
0132             } else {
0133                 $ret .= $sep_last;
0134             }
0135             $ret .= $array[$i];
0136         }
0137         return $ret;
0138     }
0139 
0140     /**
0141      * Formats a localised message with passed parameters
0142      * @param string $key string identifier of message
0143      * @param array $args Parameters to substitute in
0144      * @return string localised message
0145      * @todo Implement conditionals? Right now, some messages make
0146      *     reference to line numbers, but those aren't always available
0147      */
0148     public function formatMessage($key, $args = array())
0149     {
0150         if (!$this->_loaded) {
0151             $this->load();
0152         }
0153         if (!isset($this->messages[$key])) {
0154             return "[$key]";
0155         }
0156         $raw = $this->messages[$key];
0157         $subst = array();
0158         $generator = false;
0159         foreach ($args as $i => $value) {
0160             if (is_object($value)) {
0161                 if ($value instanceof HTMLPurifier_Token) {
0162                     // factor this out some time
0163                     if (!$generator) {
0164                         $generator = $this->context->get('Generator');
0165                     }
0166                     if (isset($value->name)) {
0167                         $subst['$'.$i.'.Name'] = $value->name;
0168                     }
0169                     if (isset($value->data)) {
0170                         $subst['$'.$i.'.Data'] = $value->data;
0171                     }
0172                     $subst['$'.$i.'.Compact'] =
0173                     $subst['$'.$i.'.Serialized'] = $generator->generateFromToken($value);
0174                     // a more complex algorithm for compact representation
0175                     // could be introduced for all types of tokens. This
0176                     // may need to be factored out into a dedicated class
0177                     if (!empty($value->attr)) {
0178                         $stripped_token = clone $value;
0179                         $stripped_token->attr = array();
0180                         $subst['$'.$i.'.Compact'] = $generator->generateFromToken($stripped_token);
0181                     }
0182                     $subst['$'.$i.'.Line'] = $value->line ? $value->line : 'unknown';
0183                 }
0184                 continue;
0185             } elseif (is_array($value)) {
0186                 $keys = array_keys($value);
0187                 if (array_keys($keys) === $keys) {
0188                     // list
0189                     $subst['$'.$i] = $this->listify($value);
0190                 } else {
0191                     // associative array
0192                     // no $i implementation yet, sorry
0193                     $subst['$'.$i.'.Keys'] = $this->listify($keys);
0194                     $subst['$'.$i.'.Values'] = $this->listify(array_values($value));
0195                 }
0196                 continue;
0197             }
0198             $subst['$' . $i] = $value;
0199         }
0200         return strtr($raw, $subst);
0201     }
0202 }
0203 
0204 // vim: et sw=4 sts=4