Language.php 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. <?php
  2. namespace Tools\Utility;
  3. /**
  4. * Parses Browser detected preferred language.
  5. */
  6. class Language {
  7. /**
  8. * Parse languages from a browser language list.
  9. *
  10. * Options
  11. * - forceLowerCase: defaults to true
  12. *
  13. * @param string|null $languageList List of locales/language codes.
  14. * @param array|bool|null $options Flags to forceLowerCase or removeDuplicates locales/language codes
  15. * deprecated: Set to true/false to toggle lowercase
  16. *
  17. * @return array
  18. */
  19. public static function parseLanguageList($languageList = null, $options = []) {
  20. $defaultOptions = [
  21. 'forceLowerCase' => true,
  22. ];
  23. if (!is_array($options)) {
  24. $options = ['forceLowerCase' => $options];
  25. }
  26. $options += $defaultOptions;
  27. if ($languageList === null) {
  28. if (!env('HTTP_ACCEPT_LANGUAGE')) {
  29. return [];
  30. }
  31. /** @var string $languageList */
  32. $languageList = env('HTTP_ACCEPT_LANGUAGE');
  33. }
  34. $languages = [];
  35. $languagesRanks = [];
  36. $languageRanges = explode(',', trim($languageList));
  37. foreach ($languageRanges as $languageRange) {
  38. $pattern = '/(\*|[a-zA-Z0-9]{1,8}(?:-[a-zA-Z0-9]{1,8})*)(?:\s*;\s*q\s*=\s*(0(?:\.\d{0,3})|1(?:\.0{0,3})))?/';
  39. if (preg_match($pattern, trim($languageRange), $match)) {
  40. if (!isset($match[2])) {
  41. $rank = '1.0';
  42. } else {
  43. $rank = (string)(float)($match[2]);
  44. }
  45. if (!isset($languages[$rank])) {
  46. if ($rank === '1') {
  47. $rank = '1.0';
  48. }
  49. $languages[$rank] = [];
  50. }
  51. $language = $match[1];
  52. if ($options['forceLowerCase']) {
  53. $language = strtolower($language);
  54. } else {
  55. /** @var string $language */
  56. $language = substr_replace($language, strtolower(substr($language, 0, 2)), 0, 2);
  57. if (strlen($language) === 5) {
  58. /** @var string $language */
  59. $language = substr_replace($language, strtoupper(substr($language, 3, 2)), 3, 2);
  60. }
  61. }
  62. if (array_key_exists($language, $languagesRanks) === false) {
  63. $languages[$rank][] = $language;
  64. $languagesRanks[$language] = $rank;
  65. } elseif ($rank > $languagesRanks[$language]) {
  66. foreach ($languages as $existRank => $existLangs) {
  67. $key = array_search($existLangs, $languages, true);
  68. if ($key !== false) {
  69. unset($languages[$existRank][$key]);
  70. if (empty($languages[$existRank])) {
  71. unset($languages[$existRank]);
  72. }
  73. }
  74. }
  75. $languages[$rank][] = $language;
  76. $languagesRanks[$language] = $rank;
  77. }
  78. }
  79. }
  80. krsort($languages);
  81. return $languages;
  82. }
  83. /**
  84. * Compares two parsed arrays of language tags and find the matches
  85. *
  86. * @param array<string> $accepted
  87. * @param array $available
  88. * @return string|null
  89. */
  90. public static function findFirstMatch(array $accepted, array $available = []) {
  91. $matches = static::findMatches($accepted, $available);
  92. if (!$matches) {
  93. return null;
  94. }
  95. $match = array_shift($matches);
  96. if (!$match) {
  97. return null;
  98. }
  99. return array_shift($match);
  100. }
  101. /**
  102. * Compares two parsed arrays of language tags and find the matches
  103. *
  104. * @param array<string> $accepted
  105. * @param array $available
  106. * @return array
  107. */
  108. public static function findMatches(array $accepted, array $available = []) {
  109. $matches = [];
  110. if (!$available) {
  111. $available = static::parseLanguageList();
  112. }
  113. foreach ($accepted as $acceptedValue) {
  114. foreach ($available as $availableQuality => $availableValues) {
  115. $availableQuality = (float)$availableQuality;
  116. if ($availableQuality === 0.0) {
  117. continue;
  118. }
  119. foreach ($availableValues as $availableValue) {
  120. $matchingGrade = static::_matchLanguage($acceptedValue, $availableValue);
  121. if ($matchingGrade > 0) {
  122. $q = (string)($availableQuality * $matchingGrade);
  123. if ($q === '1') {
  124. $q = '1.0';
  125. }
  126. if (!isset($matches[$q])) {
  127. $matches[$q] = [];
  128. }
  129. if (!in_array($availableValue, $matches[$q])) {
  130. $matches[$q][] = $availableValue;
  131. }
  132. }
  133. }
  134. }
  135. }
  136. krsort($matches);
  137. return $matches;
  138. }
  139. /**
  140. * Compare two language tags and distinguish the degree of matching
  141. *
  142. * @param string $a
  143. * @param string $b
  144. * @return float
  145. */
  146. protected static function _matchLanguage($a, $b) {
  147. $a = explode('-', strtolower($a));
  148. $b = explode('-', strtolower($b));
  149. for ($i = 0, $n = min(count($a), count($b)); $i < $n; $i++) {
  150. if ($a[$i] !== $b[$i]) {
  151. break;
  152. }
  153. }
  154. return $i === 0 ? 0 : (float)$i / count($a);
  155. }
  156. }