Lang.php 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkPHP [ WE CAN DO IT JUST THINK ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2006~2019 http://thinkphp.cn All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
  8. // +----------------------------------------------------------------------
  9. // | Author: liu21st <liu21st@gmail.com>
  10. // +----------------------------------------------------------------------
  11. declare (strict_types = 1);
  12. namespace think;
  13. /**
  14. * 多语言管理类
  15. * @package think
  16. */
  17. class Lang
  18. {
  19. /**
  20. * 配置参数
  21. * @var array
  22. */
  23. protected $config = [
  24. // 默认语言
  25. 'default_lang' => 'zh-cn',
  26. // 允许的语言列表
  27. 'allow_lang_list' => [],
  28. // 是否使用Cookie记录
  29. 'use_cookie' => true,
  30. // 扩展语言包
  31. 'extend_list' => [],
  32. // 多语言cookie变量
  33. 'cookie_var' => 'think_lang',
  34. // 多语言自动侦测变量名
  35. 'detect_var' => 'lang',
  36. // Accept-Language转义为对应语言包名称
  37. 'accept_language' => [
  38. 'zh-hans-cn' => 'zh-cn',
  39. ],
  40. // 是否支持语言分组
  41. 'allow_group' => false,
  42. ];
  43. /**
  44. * 多语言信息
  45. * @var array
  46. */
  47. private $lang = [];
  48. /**
  49. * 当前语言
  50. * @var string
  51. */
  52. private $range = 'zh-cn';
  53. /**
  54. * 构造方法
  55. * @access public
  56. * @param array $config
  57. */
  58. public function __construct(array $config = [])
  59. {
  60. $this->config = array_merge($this->config, array_change_key_case($config));
  61. $this->range = $this->config['default_lang'];
  62. }
  63. public static function __make(Config $config)
  64. {
  65. return new static($config->get('lang'));
  66. }
  67. /**
  68. * 设置当前语言
  69. * @access public
  70. * @param string $lang 语言
  71. * @return void
  72. */
  73. public function setLangSet(string $lang): void
  74. {
  75. $this->range = $lang;
  76. }
  77. /**
  78. * 获取当前语言
  79. * @access public
  80. * @return string
  81. */
  82. public function getLangSet(): string
  83. {
  84. return $this->range;
  85. }
  86. /**
  87. * 获取默认语言
  88. * @access public
  89. * @return string
  90. */
  91. public function defaultLangSet()
  92. {
  93. return $this->config['default_lang'];
  94. }
  95. /**
  96. * 加载语言定义(不区分大小写)
  97. * @access public
  98. * @param string|array $file 语言文件
  99. * @param string $range 语言作用域
  100. * @return array
  101. */
  102. public function load($file, $range = ''): array
  103. {
  104. $range = $range ?: $this->range;
  105. if (!isset($this->lang[$range])) {
  106. $this->lang[$range] = [];
  107. }
  108. $lang = [];
  109. foreach ((array) $file as $_file) {
  110. if (is_file($_file)) {
  111. $result = $this->parse($_file);
  112. $lang = array_change_key_case($result) + $lang;
  113. }
  114. }
  115. if (!empty($lang)) {
  116. $this->lang[$range] = $lang + $this->lang[$range];
  117. }
  118. return $this->lang[$range];
  119. }
  120. /**
  121. * 解析语言文件
  122. * @access protected
  123. * @param string $file 语言文件名
  124. * @return array
  125. */
  126. protected function parse(string $file): array
  127. {
  128. $type = pathinfo($file, PATHINFO_EXTENSION);
  129. switch ($type) {
  130. case 'php':
  131. $result = include $file;
  132. break;
  133. case 'yml':
  134. case 'yaml':
  135. if (function_exists('yaml_parse_file')) {
  136. $result = yaml_parse_file($file);
  137. }
  138. break;
  139. }
  140. return isset($result) && is_array($result) ? $result : [];
  141. }
  142. /**
  143. * 判断是否存在语言定义(不区分大小写)
  144. * @access public
  145. * @param string|null $name 语言变量
  146. * @param string $range 语言作用域
  147. * @return bool
  148. */
  149. public function has(string $name, string $range = ''): bool
  150. {
  151. $range = $range ?: $this->range;
  152. if ($this->config['allow_group'] && strpos($name, '.')) {
  153. list($name1, $name2) = explode('.', $name, 2);
  154. return isset($this->lang[$range][strtolower($name1)][$name2]);
  155. }
  156. return isset($this->lang[$range][strtolower($name)]);
  157. }
  158. /**
  159. * 获取语言定义(不区分大小写)
  160. * @access public
  161. * @param string|null $name 语言变量
  162. * @param array $vars 变量替换
  163. * @param string $range 语言作用域
  164. * @return mixed
  165. */
  166. public function get(string $name = null, array $vars = [], string $range = '')
  167. {
  168. $range = $range ?: $this->range;
  169. // 空参数返回所有定义
  170. if (is_null($name)) {
  171. return $this->lang[$range] ?? [];
  172. }
  173. if ($this->config['allow_group'] && strpos($name, '.')) {
  174. list($name1, $name2) = explode('.', $name, 2);
  175. $value = $this->lang[$range][strtolower($name1)][$name2] ?? $name;
  176. } else {
  177. $value = $this->lang[$range][strtolower($name)] ?? $name;
  178. }
  179. // 变量解析
  180. if (!empty($vars) && is_array($vars)) {
  181. /**
  182. * Notes:
  183. * 为了检测的方便,数字索引的判断仅仅是参数数组的第一个元素的key为数字0
  184. * 数字索引采用的是系统的 sprintf 函数替换,用法请参考 sprintf 函数
  185. */
  186. if (key($vars) === 0) {
  187. // 数字索引解析
  188. array_unshift($vars, $value);
  189. $value = call_user_func_array('sprintf', $vars);
  190. } else {
  191. // 关联索引解析
  192. $replace = array_keys($vars);
  193. foreach ($replace as &$v) {
  194. $v = "{:{$v}}";
  195. }
  196. $value = str_replace($replace, $vars, $value);
  197. }
  198. }
  199. return $value;
  200. }
  201. /**
  202. * 自动侦测设置获取语言选择
  203. * @access public
  204. * @param Request $request
  205. * @return string
  206. */
  207. public function detect(Request $request): string
  208. {
  209. // 自动侦测设置获取语言选择
  210. $langSet = '';
  211. if ($request->get($this->config['detect_var'])) {
  212. // url中设置了语言变量
  213. $langSet = strtolower($request->get($this->config['detect_var']));
  214. } elseif ($request->cookie($this->config['cookie_var'])) {
  215. // Cookie中设置了语言变量
  216. $langSet = strtolower($request->cookie($this->config['cookie_var']));
  217. } elseif ($request->server('HTTP_ACCEPT_LANGUAGE')) {
  218. // 自动侦测浏览器语言
  219. $match = preg_match('/^([a-z\d\-]+)/i', $request->server('HTTP_ACCEPT_LANGUAGE'), $matches);
  220. if ($match) {
  221. $langSet = strtolower($matches[1]);
  222. if (isset($this->config['accept_language'][$langSet])) {
  223. $langSet = $this->config['accept_language'][$langSet];
  224. }
  225. }
  226. }
  227. if (empty($this->config['allow_lang_list']) || in_array($langSet, $this->config['allow_lang_list'])) {
  228. // 合法的语言
  229. $this->range = $langSet;
  230. }
  231. return $this->range;
  232. }
  233. /**
  234. * 保存当前语言到Cookie
  235. * @access public
  236. * @param Cookie $cookie Cookie对象
  237. * @return void
  238. */
  239. public function saveToCookie(Cookie $cookie)
  240. {
  241. if ($this->config['use_cookie']) {
  242. $cookie->set($this->config['cookie_var'], $this->range);
  243. }
  244. }
  245. }