‪TYPO3CMS  ‪main
Locales.php
Go to the documentation of this file.
1 <?php
2 
3 declare(strict_types=1);
4 
5 /*
6  * This file is part of the TYPO3 CMS project.
7  *
8  * It is free software; you can redistribute it and/or modify it under
9  * the terms of the GNU General Public License, either version 2
10  * of the License, or any later version.
11  *
12  * For the full copyright and license information, please read the
13  * LICENSE.txt file that was distributed with this source code.
14  *
15  * The TYPO3 project - inspiring people to share!
16  */
17 
19 
25 
33 {
39  protected array ‪$languages = [
40  'default' => 'English',
41  'af' => 'Afrikaans',
42  'ar' => 'Arabic',
43  'bs' => 'Bosnian',
44  'bg' => 'Bulgarian',
45  'ca' => 'Catalan',
46  'ch' => 'Chinese (Simple)',
47  'cs' => 'Czech',
48  'cy' => 'Welsh',
49  'da' => 'Danish',
50  'de' => 'German',
51  'el' => 'Greek',
52  'eo' => 'Esperanto',
53  'es' => 'Spanish',
54  'et' => 'Estonian',
55  'eu' => 'Basque',
56  'fa' => 'Persian',
57  'fi' => 'Finnish',
58  'fo' => 'Faroese',
59  'fr' => 'French',
60  'fr_CA' => 'French (Canada)',
61  'gl' => 'Galician',
62  'he' => 'Hebrew',
63  'hi' => 'Hindi',
64  'hr' => 'Croatian',
65  'hu' => 'Hungarian',
66  'is' => 'Icelandic',
67  'it' => 'Italian',
68  'ja' => 'Japanese',
69  'ka' => 'Georgian',
70  'kl' => 'Greenlandic',
71  'km' => 'Khmer',
72  'ko' => 'Korean',
73  'lb' => 'Luxembourgish',
74  'lt' => 'Lithuanian',
75  'lv' => 'Latvian',
76  'mi' => 'Maori',
77  'mk' => 'Macedonian',
78  'ms' => 'Malay',
79  'nl' => 'Dutch',
80  'no' => 'Norwegian',
81  'pl' => 'Polish',
82  'pt' => 'Portuguese',
83  'pt_BR' => 'Brazilian Portuguese',
84  'ro' => 'Romanian',
85  'ru' => 'Russian',
86  'rw' => 'Kinyarwanda',
87  'sk' => 'Slovak',
88  'sl' => 'Slovenian',
89  'sn' => 'Shona (Bantu)',
90  'sq' => 'Albanian',
91  'sr' => 'Serbian',
92  'sv' => 'Swedish',
93  'th' => 'Thai',
94  'tr' => 'Turkish',
95  'uk' => 'Ukrainian',
96  'vi' => 'Vietnamese',
97  'zh' => 'Chinese (Traditional)',
98  'zh_CN' => 'Chinese (Simplified)',
99  'zh_HK' => 'Chinese (Simplified Hong Kong)',
100  'zh_Hans_CN' => 'Chinese (Simplified Han)',
101  ];
102 
117  protected array ‪$localeDependencies = [
118  'lb' => ['de'],
119  ];
120 
121  public function ‪__construct()
122  {
123  // Allow user-defined locales
124  foreach (‪$GLOBALS['TYPO3_CONF_VARS']['SYS']['localization']['locales']['user'] ?? [] as $locale => $name) {
125  if (!is_string($locale) || $locale === '') {
126  continue;
127  }
128  if (!isset($this->‪languages[$locale])) {
129  $this->‪languages[$locale] = $name;
130  }
131  }
132  // Merge user-provided locale dependencies
133  if (is_array(‪$GLOBALS['TYPO3_CONF_VARS']['SYS']['localization']['locales']['dependencies'] ?? null)) {
134  $this->localeDependencies = array_replace_recursive(
135  $this->localeDependencies,
136  ‪$GLOBALS['TYPO3_CONF_VARS']['SYS']['localization']['locales']['dependencies']
137  );
138  }
139  }
140 
141  public function ‪createLocale(string $localeKey, array $alternativeDependencies = null): ‪Locale
142  {
143  if (strpos($localeKey, '.')) {
144  [$sanitizedLocaleKey] = explode('.', $localeKey);
145  }
146  // Find the requested language in this list based on the $languageKey
147  // Language is found. Configure it:
148  if ($localeKey === 'en' || $this->‪isValidLanguageKey($sanitizedLocaleKey ?? $localeKey)) {
149  return new ‪Locale($localeKey, $alternativeDependencies ?? $this->getLocaleDependencies($sanitizedLocaleKey ?? $localeKey));
150  }
151  return new ‪Locale();
152  }
153 
158  public function getLocales(): array
159  {
160  return array_keys($this->‪languages);
161  }
162 
163  public function ‪isValidLanguageKey(string $locale): bool
164  {
165  // "en" implicitly equals "default", so this is OK
166  if ($locale === 'en' || $locale === 'default') {
167  return true;
168  }
169  if (!isset($this->‪languages[$locale])) {
170  // the given locale is not found in the current locales, let us see if
171  // the base language (iso-639-1) is in the list of supported locales.
172  if (str_contains($locale, '_')) {
173  [$baseIsoCodeLanguageKey] = explode('_', $locale);
174  return $this->‪isValidLanguageKey($baseIsoCodeLanguageKey);
175  }
176  if (str_contains($locale, '-')) {
177  [$baseIsoCodeLanguageKey] = explode('-', $locale);
178  return $this->‪isValidLanguageKey($baseIsoCodeLanguageKey);
179  }
180  return false;
181  }
182  return true;
183  }
184 
189  public function getLanguages(): array
190  {
192  }
193 
198  public function getActiveLanguages(): array
199  {
200  return array_merge(
201  ['default'],
202  array_filter(array_values(‪$GLOBALS['TYPO3_CONF_VARS']['EXTCONF']['lang']['availableLanguages'] ?? []))
203  );
204  }
205 
206  public function isLanguageKeyAvailable(string $languageKey): bool
207  {
208  return in_array($languageKey, $this->getActiveLanguages()) || is_dir(‪Environment::getLabelsPath() . '/' . $languageKey);
209  }
210 
216  public function getLocaleDependencies(string $locale): array
217  {
218  $dependencies = [];
219  if (isset($this->localeDependencies[$locale])) {
220  $dependencies = $this->localeDependencies[$locale];
221  // Search for dependencies recursively
222  ‪$localeDependencies = $dependencies;
223  foreach (‪$localeDependencies as $dependency) {
224  if (isset($this->localeDependencies[$dependency])) {
225  $dependencies = array_merge($dependencies, $this->getLocaleDependencies($dependency));
226  }
227  }
228  }
229  // Use automatic dependency resolving.
230  // "de_AT" automatically has a dependency on "de".
231  // but only do this if the actual "de_AT" does not have a custom dependency already defined in
232  // $this->localeDependencies
233  if ($dependencies === [] && str_contains($locale, '_')) {
234  [$languageIsoCode] = explode('_', $locale);
235  // "en" = "default" is always implicitly the default fallback dependency
236  if ($languageIsoCode !== 'en') {
237  $dependencies[] = $languageIsoCode;
238  $dependencies = array_merge($dependencies, $this->getLocaleDependencies($languageIsoCode));
239  }
240  } elseif ($dependencies === [] && str_contains($locale, '-')) {
241  [$languageIsoCode] = explode('-', $locale);
242  // "en" = "default" is always implicitly the default fallback dependency
243  if ($languageIsoCode !== 'en') {
244  $dependencies[] = $languageIsoCode;
245  $dependencies = array_merge($dependencies, $this->getLocaleDependencies($languageIsoCode));
246  }
247  }
248  return array_unique($dependencies);
249  }
250 
258  public function getPreferredClientLanguage(string $languageCodesList): string
259  {
260  $allLanguageCodesFromLocales = ['en' => 'default'];
261  foreach ($this->‪languages as $locale => $localeTitle) {
262  $locale = str_replace('_', '-', $locale);
263  $allLanguageCodesFromLocales[$locale] = $locale;
264  }
265  ‪$selectedLanguage = 'default';
266  ‪$preferredLanguages = GeneralUtility::trimExplode(',', $languageCodesList);
267  // Order the preferred languages after they key
269  foreach (‪$preferredLanguages as $preferredLanguage) {
270  $quality = 1.0;
271  if (str_contains($preferredLanguage, ';q=')) {
272  [$preferredLanguage, $quality] = explode(';q=', $preferredLanguage);
273  }
274  ‪$sortedPreferredLanguages[$preferredLanguage] = $quality;
275  }
276  // Loop through the languages, with the highest priority first
277  arsort(‪$sortedPreferredLanguages, SORT_NUMERIC);
278  foreach (‪$sortedPreferredLanguages as $preferredLanguage => $quality) {
279  if (isset($allLanguageCodesFromLocales[$preferredLanguage])) {
280  ‪$selectedLanguage = $allLanguageCodesFromLocales[$preferredLanguage];
281  break;
282  }
283  // Strip the country code from the end
284  [$preferredLanguage] = explode('-', $preferredLanguage);
285  if (isset($allLanguageCodesFromLocales[$preferredLanguage])) {
286  ‪$selectedLanguage = $allLanguageCodesFromLocales[$preferredLanguage];
287  break;
288  }
289  }
290  if (!‪$selectedLanguage || ‪$selectedLanguage === 'en') {
291  ‪$selectedLanguage = 'default';
292  }
293  return str_replace('-', '_', ‪$selectedLanguage);
294  }
295 
302  public static function ‪setSystemLocaleFromSiteLanguage(‪SiteLanguage $siteLanguage): bool
303  {
304  $locale = $siteLanguage->‪getLocale()->posixFormatted();
305  if ($locale === '') {
306  return false;
307  }
308  return ‪self::setLocale($locale, $locale);
309  }
310 
320  protected static function ‪setLocale(string $locale, string $localeStringForTrigger): bool
321  {
322  $incomingLocale = $locale;
323  $availableLocales = GeneralUtility::trimExplode(',', $locale, true);
324  // If LC_NUMERIC is set e.g. to 'de_DE' PHP parses float values locale-aware resulting in strings with comma
325  // as decimal point which causes problems with value conversions - so we set all locale types except LC_NUMERIC
326  // @see https://bugs.php.net/bug.php?id=53711
327  $locale = setlocale(LC_COLLATE, ...$availableLocales);
328  if ($locale) {
329  // As str_* methods are locale aware and turkish has no upper case I
330  // Class autoloading and other checks depending on case changing break with turkish locale LC_CTYPE
331  // @see http://bugs.php.net/bug.php?id=35050
332  if (!str_starts_with($locale, 'tr')) {
333  setlocale(LC_CTYPE, ...$availableLocales);
334  }
335  setlocale(LC_MONETARY, ...$availableLocales);
336  setlocale(LC_TIME, ...$availableLocales);
337  } else {
338  // Retry again without the "utf-8" POSIX platform suffix if this is given.
339  if (str_contains($incomingLocale, '.')) {
340  [$localeWithoutModifier] = explode('.', $incomingLocale);
341  return ‪self::setLocale($localeWithoutModifier, $incomingLocale);
342  }
343  if ($localeStringForTrigger === $locale) {
344  GeneralUtility::makeInstance(LogManager::class)
345  ->getLogger(__CLASS__)
346  ->error('Locale "' . htmlspecialchars($localeStringForTrigger) . '" not found.');
347  } else {
348  GeneralUtility::makeInstance(LogManager::class)
349  ->getLogger(__CLASS__)
350  ->error('Locale "' . htmlspecialchars($localeStringForTrigger) . '" and "' . htmlspecialchars($incomingLocale) . '" not found.');
351  }
352  return false;
353  }
354  return true;
355  }
356 }
‪TYPO3\CMS\Core\Localization\Locales\$sortedPreferredLanguages
‪$sortedPreferredLanguages
Definition: Locales.php:268
‪TYPO3\CMS\Core\Localization\Locales\setSystemLocaleFromSiteLanguage
‪static bool setSystemLocaleFromSiteLanguage(SiteLanguage $siteLanguage)
Definition: Locales.php:302
‪TYPO3\CMS\Core\Localization\Locales\$preferredLanguages
‪$preferredLanguages
Definition: Locales.php:266
‪TYPO3\CMS\Core\Localization\Locales\setLocale
‪static setLocale(string $locale, string $localeStringForTrigger)
Definition: Locales.php:320
‪TYPO3\CMS\Core\Site\Entity\SiteLanguage\getLocale
‪getLocale()
Definition: SiteLanguage.php:174
‪TYPO3\CMS\Core\Core\Environment\getLabelsPath
‪static getLabelsPath()
Definition: Environment.php:233
‪TYPO3\CMS\Core\Localization\Locales\$localeDependencies
‪array $localeDependencies
Definition: Locales.php:117
‪TYPO3\CMS\Core\Localization\Locales
Definition: Locales.php:33
‪TYPO3\CMS\Core\Localization\Locales\__construct
‪__construct()
Definition: Locales.php:121
‪TYPO3\CMS\Core\Localization
Definition: CacheWarmer.php:18
‪TYPO3\CMS\Core\Site\Entity\SiteLanguage
Definition: SiteLanguage.php:27
‪TYPO3\CMS\Core\Localization\Locales\$selectedLanguage
‪foreach($this->languages as $locale=> $localeTitle) $selectedLanguage
Definition: Locales.php:265
‪TYPO3\CMS\Core\Localization\Locales\createLocale
‪createLocale(string $localeKey, array $alternativeDependencies=null)
Definition: Locales.php:141
‪TYPO3\CMS\Core\Localization\Locales\languages
‪array< non-empty-string, function getLanguages():array { return $this-> languages
Definition: Locales.php:191
‪TYPO3\CMS\Core\SingletonInterface
Definition: SingletonInterface.php:23
‪$GLOBALS
‪$GLOBALS['TYPO3_CONF_VARS']['EXTCONF']['adminpanel']['modules']
Definition: ext_localconf.php:25
‪TYPO3\CMS\Core\Log\LogManager
Definition: LogManager.php:33
‪TYPO3\CMS\Core\Core\Environment
Definition: Environment.php:41
‪TYPO3\CMS\Core\Localization\Locale
Definition: Locale.php:30
‪TYPO3\CMS\Core\Localization\Locales\isValidLanguageKey
‪array< int, getLocales():array { return array_keys( $this->languages);} public function isValidLanguageKey(string $locale):bool { if( $locale==='en'||$locale==='default') { return true;} if(!isset( $this->languages[ $locale])) { if(str_contains( $locale, '_')) {[ $baseIsoCodeLanguageKey]=explode( '_', $locale);return $this-> isValidLanguageKey($baseIsoCodeLanguageKey)
‪TYPO3\CMS\Core\Utility\GeneralUtility
Definition: GeneralUtility.php:51
‪TYPO3\CMS\Core\Localization\Locales\$languages
‪array $languages
Definition: Locales.php:39