Language.php 4.4 KB

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