我想集成一个简单的HTML表单,允许用户更改Symfony2 Web应用程序的语言(即从页面en/faq转到fr/faq)。如何以适当的方式做到这一点?Symfony2语言选择器
我已经发现了与Symfony的做的一个很好的方式,但不与Symfony2的:http://symfony.com/blog/play-with-the-user-language
我想集成一个简单的HTML表单,允许用户更改Symfony2 Web应用程序的语言(即从页面en/faq转到fr/faq)。如何以适当的方式做到这一点?Symfony2语言选择器
我已经发现了与Symfony的做的一个很好的方式,但不与Symfony2的:http://symfony.com/blog/play-with-the-user-language
在你的路由定义使用_locale
参数自动设置用户区域。
见http://symfony.com/doc/current/book/translation.html#the-locale-and-the-url
你需要调用$this->get('session')->setLocale($locale)
控制器内(由Symfony的2.1 '请求' 代替 '会话')。
我创建了一个表格,而我通过语言的数组:
<?php
class LanguageType extends AbstractType
{
public function buildForm(FormBuilder $builder, array $options)
{
$langs = $options['languages'];
$langs = array_combine($langs, $langs);
foreach ($langs as &$lang) {
$lang = \Locale::getDisplayName($lang);
}
$builder->add('language', 'choice', array(
'choices' => $langs,
'required' => false,
));
}
public function getDefaultOptions(array $options)
{
return array(
'languages' => array('fr_FR', 'en_GB'),
'csrf_protection' => false,
);
}
public function getName()
{
return 'my_language';
}
}
我提交此表到一个单独的动作在控制器中,我设定的区域,并返回一个重定向最后一页:
<?php
class LanguageController extends Controller
{
public function switchLanguageAction()
{
$form = $this->get('form.factory')->create(
new LanguageType(),
array('language' => $this->get('session')->getLocale()),
array('languages' => $this->container->getParameter('roger.admin.languages', null))
);
$request = $this->get('request');
$form->bindRequest($request);
$referer = $request->headers->get('referer');
if ($form->isValid()) {
$locale = $form->get('language')->getData();
$router = $this->get('router');
// Create URL path to pass it to matcher
$urlParts = parse_url($referer);
$basePath = $request->getBaseUrl();
$path = str_replace($basePath, '', $urlParts['path']);
// Match route and get it's arguments
$route = $router->match($path);
$routeAttrs = array_replace($route, array('_locale' => $locale));
$routeName = $routeAttrs['_route'];
unset($routeAttrs['_route']);
// Set Locale
$this->get('session')->setLocale($locale);
return new RedirectResponse($router->generate($routeName, $routeAttrs));
}
return new RedirectResponse($referer);
}
}
适用于任何有效的语言环境(你将它们作为“语言”选项,同时创造你的形式),提供的PHP扩展国际启用。如果不是,则需要用手动创建的区域名称列表替换\Locale::getDisplayName($lang)
。
请注意:引用者可能会被某些极客重写,然后'$ router-> match()'会抛出一个异常,因为没有匹配。一个后备可能会很好:-) –
我还没有做到这一点的形式,但只是在屏幕顶部的小旗子图像。每个标志都是指向当前页面的链接,但URL中的双字母语言代码会被相应标志的语言替换。我的布局模板具有下面的代码:
的replaceLanguageInUrl功能在我的枝条延伸类中定义:
public function getFunctions()
{
return array(
'replaceLanguageInUrl' => new \Twig_Function_Method($this, 'replaceLanguageInUrl'),
);
}
public function replaceLanguageInUrl($currentLanguage, $newLanguage, $url)
{
//EDIT BEGIN
if (strpos($url,$currentLanguage) == false) {
$url = getBaseUrl($url).'/'.$currentLanguage;
}
//EDIT END
return str_replace('/' . $currentLanguage . '/', '/' . $newLanguage . '/', $url);
}
当点击一个标志,同样的页面加载,但在新的语言。这也会在会话中自动设置新的语言。
我在本地做过,但比wdev的解决方案更简单一些,我用一些图片(标志)作为按钮。单击标志时,将设置新的区域设置,并使用新语言刷新页面(通过重定向)。您需要使用Symfony's translation system。下面是代码:
控制器:
public function englishAction(Request $request)
{
$this->get('session')->setLocale('en_US');
return $this->redirect($request->headers->get('referer'));
}
public function chineseAction(Request $request)
{
$this->get('session')->setLocale('zh_CN');
return $this->redirect($request->headers->get('referer'));
}
public function frenchAction(Request $request)
{
$this->get('session')->setLocale('fr_FR');
return $this->redirect($request->headers->get('referer'));
}
模板:
<ul class="nav pull-right">
<li>
<a href="{{ path('english') }}"><img src="{{ asset('bundles/fkmywebsite/images/flag-en.png') }}" alt="English Language" height="30" width="18" /></a>
</li>
<li>
<a href="{{ path('chinese') }}"><img src="{{ asset('bundles/fkmywebsite/images/flag-cn.jpg') }}" alt="Chinese Language" height="30" width="18" /></a>
</li>
<li>
<a href="{{ path('french') }}"><img src="{{ asset('bundles/fkmywebsite/images/flag-fr.png') }}" alt="French Language" height="30" width="18" /></a>
</li>
</ul>
编辑:此解决方案可与Symfony2.0,为Symfony2.1检查this question
最简单的方法我发现是直接在树枝模板中做的。至少,它适用于2.2:
<ul class="lang-menu">
<li><a href="{{ path(app.request.get('_route'), app.request.get('_route_params')|merge({'_locale': 'ca'})) }}">Català</a></li>
<li><a href="{{ path(app.request.get('_route'), app.request.get('_route_params')|merge({'_locale': 'en'})) }}">English</a></li>
</ul>
是的,这是我为每个页面所做的,但是我也想让用户通过在列表中选择他想要的语言。 – cvsoftware