Proyectos de Subversion Moodle

Rev

| Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1 efrain 1
<?php
2
 
3
/**
4
 * Class responsible for generating HTMLPurifier_Language objects, managing
5
 * caching and fallbacks.
6
 * @note Thanks to MediaWiki for the general logic, although this version
7
 *       has been entirely rewritten
8
 * @todo Serialized cache for languages
9
 */
10
class HTMLPurifier_LanguageFactory
11
{
12
 
13
    /**
14
     * Cache of language code information used to load HTMLPurifier_Language objects.
15
     * Structure is: $factory->cache[$language_code][$key] = $value
16
     * @type array
17
     */
18
    public $cache;
19
 
20
    /**
21
     * Valid keys in the HTMLPurifier_Language object. Designates which
22
     * variables to slurp out of a message file.
23
     * @type array
24
     */
25
    public $keys = array('fallback', 'messages', 'errorNames');
26
 
27
    /**
28
     * Instance to validate language codes.
29
     * @type HTMLPurifier_AttrDef_Lang
30
     *
31
     */
32
    protected $validator;
33
 
34
    /**
35
     * Cached copy of dirname(__FILE__), directory of current file without
36
     * trailing slash.
37
     * @type string
38
     */
39
    protected $dir;
40
 
41
    /**
42
     * Keys whose contents are a hash map and can be merged.
43
     * @type array
44
     */
45
    protected $mergeable_keys_map = array('messages' => true, 'errorNames' => true);
46
 
47
    /**
48
     * Keys whose contents are a list and can be merged.
49
     * @value array lookup
50
     */
51
    protected $mergeable_keys_list = array();
52
 
53
    /**
54
     * Retrieve sole instance of the factory.
55
     * @param HTMLPurifier_LanguageFactory $prototype Optional prototype to overload sole instance with,
56
     *                   or bool true to reset to default factory.
57
     * @return HTMLPurifier_LanguageFactory
58
     */
59
    public static function instance($prototype = null)
60
    {
61
        static $instance = null;
62
        if ($prototype !== null) {
63
            $instance = $prototype;
64
        } elseif ($instance === null || $prototype == true) {
65
            $instance = new HTMLPurifier_LanguageFactory();
66
            $instance->setup();
67
        }
68
        return $instance;
69
    }
70
 
71
    /**
72
     * Sets up the singleton, much like a constructor
73
     * @note Prevents people from getting this outside of the singleton
74
     */
75
    public function setup()
76
    {
77
        $this->validator = new HTMLPurifier_AttrDef_Lang();
78
        $this->dir = HTMLPURIFIER_PREFIX . '/HTMLPurifier';
79
    }
80
 
81
    /**
82
     * Creates a language object, handles class fallbacks
83
     * @param HTMLPurifier_Config $config
84
     * @param HTMLPurifier_Context $context
85
     * @param bool|string $code Code to override configuration with. Private parameter.
86
     * @return HTMLPurifier_Language
87
     */
88
    public function create($config, $context, $code = false)
89
    {
90
        // validate language code
91
        if ($code === false) {
92
            $code = $this->validator->validate(
93
                $config->get('Core.Language'),
94
                $config,
95
                $context
96
            );
97
        } else {
98
            $code = $this->validator->validate($code, $config, $context);
99
        }
100
        if ($code === false) {
101
            $code = 'en'; // malformed code becomes English
102
        }
103
 
104
        $pcode = str_replace('-', '_', $code); // make valid PHP classname
105
        static $depth = 0; // recursion protection
106
 
107
        if ($code == 'en') {
108
            $lang = new HTMLPurifier_Language($config, $context);
109
        } else {
110
            $class = 'HTMLPurifier_Language_' . $pcode;
111
            $file  = $this->dir . '/Language/classes/' . $code . '.php';
112
            if (file_exists($file) || class_exists($class)) {
113
                $lang = new $class($config, $context);
114
            } else {
115
                // Go fallback
116
                $raw_fallback = $this->getFallbackFor($code);
117
                $fallback = $raw_fallback ? $raw_fallback : 'en';
118
                $depth++;
119
                $lang = $this->create($config, $context, $fallback);
120
                if (!$raw_fallback) {
121
                    $lang->error = true;
122
                }
123
                $depth--;
124
            }
125
        }
126
        $lang->code = $code;
127
        return $lang;
128
    }
129
 
130
    /**
131
     * Returns the fallback language for language
132
     * @note Loads the original language into cache
133
     * @param string $code language code
134
     * @return string|bool
135
     */
136
    public function getFallbackFor($code)
137
    {
138
        $this->loadLanguage($code);
139
        return $this->cache[$code]['fallback'];
140
    }
141
 
142
    /**
143
     * Loads language into the cache, handles message file and fallbacks
144
     * @param string $code language code
145
     */
146
    public function loadLanguage($code)
147
    {
148
        static $languages_seen = array(); // recursion guard
149
 
150
        // abort if we've already loaded it
151
        if (isset($this->cache[$code])) {
152
            return;
153
        }
154
 
155
        // generate filename
156
        $filename = $this->dir . '/Language/messages/' . $code . '.php';
157
 
158
        // default fallback : may be overwritten by the ensuing include
159
        $fallback = ($code != 'en') ? 'en' : false;
160
 
161
        // load primary localisation
162
        if (!file_exists($filename)) {
163
            // skip the include: will rely solely on fallback
164
            $filename = $this->dir . '/Language/messages/en.php';
165
            $cache = array();
166
        } else {
167
            include $filename;
168
            $cache = compact($this->keys);
169
        }
170
 
171
        // load fallback localisation
172
        if (!empty($fallback)) {
173
 
174
            // infinite recursion guard
175
            if (isset($languages_seen[$code])) {
176
                trigger_error(
177
                    'Circular fallback reference in language ' .
178
                    $code,
179
                    E_USER_ERROR
180
                );
181
                $fallback = 'en';
182
            }
183
            $language_seen[$code] = true;
184
 
185
            // load the fallback recursively
186
            $this->loadLanguage($fallback);
187
            $fallback_cache = $this->cache[$fallback];
188
 
189
            // merge fallback with current language
190
            foreach ($this->keys as $key) {
191
                if (isset($cache[$key]) && isset($fallback_cache[$key])) {
192
                    if (isset($this->mergeable_keys_map[$key])) {
193
                        $cache[$key] = $cache[$key] + $fallback_cache[$key];
194
                    } elseif (isset($this->mergeable_keys_list[$key])) {
195
                        $cache[$key] = array_merge($fallback_cache[$key], $cache[$key]);
196
                    }
197
                } else {
198
                    $cache[$key] = $fallback_cache[$key];
199
                }
200
            }
201
        }
202
 
203
        // save to cache for later retrieval
204
        $this->cache[$code] = $cache;
205
        return;
206
    }
207
}
208
 
209
// vim: et sw=4 sts=4