2

我正在尝试将 i18n 包含在 keystone.js 应用程序中。我正在这样做,就像在这个例子https://gist.github.com/JedWatson/9191081中一样,它可以工作,但我的问题是要查看当前的语言环境。我正在使用中间件通过 url 参数设置语言环境:

// middleware.js
exports.setLocale = function (req, res, next) {
    if (req.params.lang) {
        req.setLocale(req.params.lang);
    }
    else
        res.redirect('/ru/');
    next();
};

// index.js
keystone.pre('render', middleware.setLocale);

和路由

app.get('/:lang/', routes.views.index);
app.get('/:lang/blog/:category?', routes.views.blog);
app.get('/:lang/blog/post/:post', routes.views.post);
...

默认语言环境是“ru”。但在我看来

view.on('render', function (next) {
    console.log('on render', req.getLocale());
    next();
});

console.log('before render', req.getLocale());
view.render('blog');

/en/blog在它输出的路线上

------------------------------------------------
KeystoneJS Started:
qalqan is ready on default port 3000
------------------------------------------------

before render ru
on render en
GET /en/blog 304 373ms

因此,据我了解,在将变量发送到视图后会更改语言环境。在渲染之前有什么方法可以设置它吗?我知道,我可以通过 req.params 从lang每个视图中的 url 参数中获取它,但我想通过中间件为所有视图执行此操作。

4

1 回答 1

1

更新:由于 i18n 版本 0.7.0,您的代码应该可以工作。

这似乎有点违反直觉,但我认为您不应该触摸路由以避免重复 lang 参数。

不过,您确实走在正确的道路上,这就是我所做的:

// routes/middleware.js

exports.detectLang = function(req, res, next) {
    var match = req.url.match(/^\/(de|en)([\/\?].*)?$/i);

    if (match) {
        req.setLocale(match[1]);
        // Make locale available in template
        // (necessary until i18n 0.6.x)
        res.locals.locale = req.getLocale();
        // reset the URL for routing
        req.url = match[2] || '/';
    } (else) {
        // Here you can redirect to default locale if you want
    }

    next();
}

现在只需在定义所有其他 res.locals 之前使用中间件:

// routes/index.js

keystone.pre('routes', i18n.init);
keystone.pre('routes', middleware.detectLang);

现在,您可以在所有 URL 前面加上'/' + res.locals.locale,用户将在您的应用程序周围的任何地方保留他们的语言环境。

于 2016-02-11T12:16:28.430 回答