Language.php 4.2 KB

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