Yii2语言国际化配置Twig翻译解决方案
转载
Yii2语言国际化配置Twig翻译解决方案
我自己在写项目的时候,不喜欢使用php自身的模板,主要是各种PHP标签让我烦,而且对Html的标签兼容也不够友好,所以我后面采用了twig模板,配置之类的也是很方便,写起来也很顺手,但是在Yii2语言国际化翻译这块就遇到了坑,当我们指定文件类型,除了处理php扩展的之外,也处理twig扩展的文件的时候,就不会解析twig中的内容,因为不符合PHP的标签处理逻辑,在PHP中我们使用Yii::t(),但是在twig中使用的是Yii.t()这个函数在translator的配置中,显得很乏力,而且看源码也可以发现,实际上也只处理php文件,网上找了很多针对这个问题的处理方式,似乎也没有几个使用的。现在看下我是如何解决的
第一步 显示修改i18n配置
将twig扩展加入进去,修改后如下:
return [ 'color' => null, 'interactive' => true, 'help' => null, 'sourcePath' => '@app', 'messagePath' => '@app/messages', 'languages' => ['zh-CN', 'ru-RU'], 'translator' => 'Yii::t', // 翻译器 'sort' => false, 'overwrite' => true, 'removeUnused' => false, 'markUnused' => true, 'except' => [ '.svn', '.git', '.gitignore', '.gitkeep', '.hgignore', '.hgkeep', '/messages', '/BaseYii.php', 'vendor', 'tests', 'runtime', 'migrations', ], 'only' => [ '*.php', '*.twig', // 添加模板扩展 ], 'format' => 'php', 'db' => 'db', 'sourceMessageTable' => '{{%source_message}}', 'messageTable' => '{{%message}}', 'catalog' => 'messages', 'ignoreCategories' => [], 'phpFileHeader' => '', 'phpDocBlock' => null, ];
第二步 继承重写
创建文件app/commands/TranslatorController.php,内容如下
<?php namespace app\commands; use yii\console\controllers\MessageController; use yii\helpers\Console; /** * Extracts messages to be translated from source files. * * @author durban.zhang <durban.zhang@gmail.com> */ class TranslatorController extends MessageController { public function init() { parent::init(); } /** * This command echoes what you have entered as the message. * @param string $message the message to be echoed. */ protected function extractMessages($fileName, $translator, $ignoreCategories = []) { $messages = []; $extInfo = pathinfo($fileName, PATHINFO_EXTENSION); if ('twig' == $extInfo) { $coloredFileName = Console::ansiFormat($fileName, [Console::FG_CYAN]); $this->stdout("Extracting messages from $coloredFileName...\n"); $subject = file_get_contents($fileName); $preg = '/\{\{ Yii\.t\(\'(.*?)\', \'(.*?)\'\) \}\}/'; $content = preg_replace($preg, "<?php Yii::t('$1', '$2'); ?>", $subject); $tokens = token_get_all($content); foreach ((array) $translator as $currentTranslator) { $translatorTokens = token_get_all('<?php ' . $currentTranslator); array_shift($translatorTokens); $messages = array_merge_recursive( $messages, $this->extractMessagesFromTokens( $tokens, $translatorTokens, $ignoreCategories)); } $this->stdout("\n"); } else { $messages = parent::extractMessages($fileName, $translator, $ignoreCategories); } return $messages; } protected function extractMessagesFromTokens(array $tokens, array $translatorTokens, array $ignoreCategories) { $messages = []; $translatorTokensCount = count($translatorTokens); $matchedTokensCount = 0; $buffer = []; $pendingParenthesisCount = 0;foreach ($tokens as $token) { // finding out translator call if ($matchedTokensCount < $translatorTokensCount) { if ($this->tokensEqual($token, $translatorTokens[$matchedTokensCount])) { $matchedTokensCount++; } else { $matchedTokensCount = 0; } } elseif ($matchedTokensCount === $translatorTokensCount) { // translator found // end of function call if ($this->tokensEqual(')', $token)) { $pendingParenthesisCount--; if (0 === $pendingParenthesisCount) { // end of translator call or end of something that we can't extract if (isset($buffer[0][0], $buffer[1], $buffer[2][0]) && T_CONSTANT_ENCAPSED_STRING === $buffer[0][0] && ',' === $buffer[1] && T_CONSTANT_ENCAPSED_STRING === $buffer[2][0]) { // is valid call we can extract $category = stripcslashes($buffer[0][1]); $category = mb_substr($category, 1, -1); if (!$this->isCategoryIgnored($category, $ignoreCategories)) { $message = stripcslashes($buffer[2][1]); $message = mb_substr($message, 1, -1); $messages[$category][] = $message; } $nestedTokens = array_slice($buffer, 3); if (count($nestedTokens) > $translatorTokensCount) { // search for possible nested translator calls $messages = array_merge_recursive($messages, $this->extractMessagesFromTokens($nestedTokens, $translatorTokens, $ignoreCategories)); } } else { // invalid call or dynamic call we can't extract $line = Console::ansiFormat($this->getLine($buffer), [Console::FG_CYAN]); $skipping = Console::ansiFormat('Skipping line', [Console::FG_YELLOW]); $this->stdout("$skipping $line. Make sure both category and message are static strings.\n"); } // prepare for the next match $matchedTokensCount = 0; $pendingParenthesisCount = 0; $buffer = []; } else { $buffer[] = $token; } } elseif ($this->tokensEqual('(', $token)) { // count beginning of function call, skipping translator beginning if ($pendingParenthesisCount > 0) { $buffer[] = $token; } $pendingParenthesisCount++; } elseif (isset($token[0]) && !in_array($token[0], [T_WHITESPACE, T_COMMENT])) { // ignore comments and whitespaces $buffer[] = $token; } } } return $messages; } }
从上面的代码可以看出,实现了两个方法
extractMessages和extractMessagesFromTokens
本来是不需要实现extractMessagesFromTokens这个方法的,但是父类中的private,试过了实例化类,但是需要传递需要的参数,为了避免出现问题,暂时没用实例化的方式。
最后extractMessagesFromTokens 这个只是父类的copy版本。那么重点就在extractMessages
判断如果文件是twig文件,则进行处理,处理逻辑我写的也很简单,暂时能解决问题
$preg = '/\{\{ Yii\.t\(\'(.*?)\', \'(.*?)\'\) \}\}/'; $content = preg_replace($preg, "<?php Yii::t('$1', '$2'); ?>", $subject); $tokens = token_get_all($content);
主要是将Yii.t转为了含有PHP标签的字符串,实际上父类也只是针对代码进行字符串的过滤处理,以此类推的话,如果有其他模板的话也可以采用此方法
到这里就处理完了,
但是执行的命令不是下面这个
./yii message/extract @app/config/i18n.php
而是下面这个
./yii translator/extract @app/config/i18n.php
OK,问题解决,如您有其他方案,请指教

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Java工程师必备书单
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/a724888/article/details/82026061 江湖路险,你我同行。 Java开发工程师一般负责后端开发,当然也有专门做Java Web的工程师,但是随着前后端的分离,越来越多的Java工程师需要往大后端方向发展。 今天我们就来介绍一下Java后端开发者的书单。 首先要感谢一下江南白衣大大的后端书架,让我在初学阶段读到了很多好书,直到现在都印象深刻。 我在两年的学习历程中看了很多的书,其中不乏XXX入门到精通,XXX王者归来,XXX指南什么的。 虽然这类书确实毛病很多,但是作为非科班的我来说,当时还是看的津津有味。直到后来我看到一些优秀的书籍,以及白衣哥的书架,我才逐渐认识到看一些精品书籍的好处。 所以我们今天就从这些精品书籍中挑选一些优秀书籍来推荐给大家,当然其中有一些书我自己也没有时间看完。 接下来我们按照后端技术栈各个部分的内容来推荐书籍。 网络 1 TCP/IP卷一 这本其实我刚开始没看太懂,可能是当时太水的原因,但是一般是大牛力荐的书。...
- 下一篇
python网络编程
python提供的网络服务: 标准的BSD Sockets API,可以访问底层操作系统Socket接口的全部方法(基本的Socket) 高级别的网络服务模块SocketServer提供服务器中心类,简化网络服务器的开发。 · Socket()函数: socket.socket([family[,type[,protol]]]) family: 套接字家族(AF_UNIX或者AF_INET) type: 套接字类型可以根据是面向连接的还是非连接分为SOCK_STREAM或者SOCK_DERAM protocol: 一般默认为0 Socket对象方法:s.bind() /s.listen()/s.accept() 客户端套接字: s.connect()/s.connect_ex() 公共用途的套接字函数: s.recv()/s.send()/s.sendall()/s.recvfrom()/s.close()/s.getpeername()/s.getsockname()/s.setsockopt(level,optname,value)/s.getsockopt(level,optna...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- Windows10,CentOS7,CentOS8安装Nodejs环境
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Windows10,CentOS7,CentOS8安装MongoDB4.0.16
- CentOS6,CentOS7官方镜像安装Oracle11G
- Red5直播服务器,属于Java语言的直播服务器
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- SpringBoot2整合Redis,开启缓存,提高访问速度