尝试从Django模板更改语言的问题

时间:2013-08-22 23:33:35

标签: python django internationalization django-i18n

我需要包含两个按钮或链接,以允许用户更改英语和西班牙语之间的语言。我已阅读the docs并试了一下:

<form action="/i18n/setlang/" method="post">{% csrf_token %}
    <input name="language" type="hidden" value="es" />
    <input type="submit" value="ES" />
</form>

但是,每次单击按钮时,页面都会重新加载,但语言根本不会更改。我错过了什么吗?

注意:我没有设置next,因为我只想用所需语言重新加载当前页面。

如果我使用文档提供的默认表单,结果是相同的:页面重新加载但语言没有改变:

<form action="{% url 'set_language' %}" method="post">
    {% csrf_token %}
    <input name="next" type="hidden" value="{{ redirect_to }}" />
    <select name="language">
        {% get_language_info_list for LANGUAGES as languages %}
        {% for language in languages %}
            <option value="{{ language.code }}"{% if language.code == LANGUAGE_CODE %} selected="selected"{% endif %}>
                {{ language.name_local }} ({{ language.code }})
            </option>
        {% endfor %}
    </select>
    <input type="submit" value="Go" />
</form>

更新

经过进一步测试后,我发现在i18n_patterns中同时使用patternsurls.py时出现问题。目前我的文件看起来像:

urlpatterns = i18n_patterns('',
    url(r'^contents/', include('contents.urls')),
    url(r'^events/', include('events.urls')),
    # ...
)
urlpatterns += patterns('',
    url(r'^i18n/', include('django.conf.urls.i18n')),
)

这似乎不起作用。但是,如果我删除i18n_patterns并将其更改为patterns,那么它似乎有效:

urlpatterns = patterns('',
    url(r'^contents/', include('contents.urls')),
    url(r'^events/', include('events.urls')),
    # ...
)
urlpatterns += patterns('',
    url(r'^i18n/', include('django.conf.urls.i18n')),
)

文档说你不必把它包含在i18n_patterns里面,所以我认为这应该有用,但事实并非如此!如果您在django.conf.urls.i18n之前或之后添加i18n_patterns并不一定无关紧要。

8 个答案:

答案 0 :(得分:19)

经过更多测试并感谢@AronYsidoro链接的related question,我终于找到了问题,这是一个非常简单的解决方案,实际上解决了这个问题。

首先,让我解释一下这个问题:当您使用i18_patterns中的urls.py预先添加语言代码时,如果您调用网址set_language来更改语言而未指定{{ 1}},默认为当前版本,但使用前置的旧语言代码!所以,语言又恢复原状!而且,如果您明确指定next,则必须确保在开头不包含语言代码。

如果您使用next{{ request.path }}{{ request.get_full_path }}指定为当前页面,则无法恢复语言代码。

那么,在使用next时,我们如何删除此不受欢迎的语言代码以重新加载当前页面并更改语言?很容易,我们只需切片3个第一个字符(斜线和两个字符语言代码)!

这里有两个例子。第一个是select的形式(选择语言),另一个是按钮形式(每种语言)。

我真的希望这有助于其他人。您只需复制并粘贴代码即可。但是,如果使用“按钮形式”,则只需将语言设置为所需的语言即可!

从列表中更改语言:

i18n_patterns

将语言更改为按钮:

<form action="{% url 'set_language' %}" method="post">
  {% csrf_token %}
  <input name="next" type="hidden" value="{{ request.get_full_path|slice:'3:' }}" />
  <select name="language">
    {% get_language_info_list for LANGUAGES as languages %}
      {% for language in languages %}
        <option value="{{ language.code }}"{% if language.code == LANGUAGE_CODE %} selected="selected"{% endif %}>
          {{ language.name_local }} ({{ language.code }})
        </option>
      {% endfor %}
  </select>
  <input type="submit" value="Change" />
</form>

答案 1 :(得分:10)

可能选项的总结:

使用select

更改用户的会话语言

关于Django docs的例子,有一个非常详尽的描述。

使用按钮

更改用户的会话语言

没有必要像@Caumons建议的那样为每个按钮重复一个表单,而是只需在表单中包含尽可能多的按钮作为语言。

<form action="{% url 'set_language' %}" method="post">
    {% csrf_token %}
    <input name="next" type="hidden" value="{{ request.get_full_path|slice:'3:' }}" />
    <ul class="nav navbar-nav navbar-right language menu">
        {% get_current_language as LANGUAGE_CODE %}
        {% get_available_languages as LANGUAGES %}
        {% get_language_info_list for LANGUAGES as languages %}
        {% for language in languages %}
            <li>
                <button type="submit"
                        name="language"
                        value="{{ language.code }}"
                        class="{% if language.code == LANGUAGE_CODE %}selected{% endif %}">
                    {{ language.name_local }}
                </button>
            </li>
        {% endfor %}
    </ul>
</form>

您当然可以设置按钮的样式,使其看起来像链接或其他任何内容。

更改链接显示的语言

如果不需要更改默认用户会话语言,则可以使用简单链接更改内容:

<ul class="nav navbar-nav navbar-right language menu">
    {% get_current_language as LANGUAGE_CODE %}
    {% get_available_languages as LANGUAGES %}
    {% get_language_info_list for LANGUAGES as languages %}
    {% for language in languages %}
        <li>
            <a href="/{{ language.code }}{{ request.get_full_path|slice:'3:' }}"
               class="{% if language.code == LANGUAGE_CODE %}selected{% endif %}"
               lang="{{ language.code }}">
                {{ language.name_local }}
            </a>
        </li>
    {% endfor %}
</ul>

SEO

如果使用表单来更改会话语言,我不完全确定内容是否友好,就像Django建议的那样。因此,链接<a>标记可能会隐藏在<button>元素下方。

答案 2 :(得分:2)

此链接:

Django: i18n - change language

可能会有更改语言的内容。

答案 3 :(得分:2)

如果在您当前的系统中,您只有2种语言,那么只需使用如下:

{% ifequal LANGUAGE_CODE "en" %}
       <a href="/es{{ request.get_full_path }}">Spanish</a>
{% else %}
       <a href="/en{{ request.get_full_path }}">English</a>
{% endifequal %}

不需要表格,网址和提交等。它对我有用。

答案 4 :(得分:2)

如果只需要两种语言,例如。英文和法文,您已经在settings.py中定义了它,并且已经设置了默认语言,并且已经在主应用程序中正确配置了urls.py。然后,只需在模板(或部分,顶栏等)中使用它即可。btn-kinito "btn-header只是样式类,您可以使用css或JS进行操作。

内部的循环或迭代只是循环浏览LANGUAGES[]中定义的settings.py列表,然后创建一个按钮。带有字符“ |”加上一个&nbsp;空间,使它看起来很可爱,因为我们只有两个郎。

{% url 'set_language' %}是Django的名为set_language的重定向视图,它重定向到URL。这就是为什么在这种情况下,需要在主应用程序的urls.py中放入path('i18n/', include('django.conf.urls.i18n')),的原因。因此,在为列表中的每种语言创建按钮之后,您将能够重定向到该URL。

  <div class="btn-header">
    <form action="{% url 'set_language' %}" method="post">
      {% csrf_token %}
      <input name="next" type="hidden" value="{{ redirect_to }}" />
        {% get_current_language as LANGUAGE_CODE %}
        {% get_available_languages as LANGUAGES %}
        {% get_language_info_list for LANGUAGES as languages %}
        {% for language in languages %}
          <button type="submit" name="language" value="{{ language.code }}"
            class="btn-kinito">
            {{ language.code }}
          </button>|&nbsp;
        {% endfor %}
    </form>
  </div>

对于urls.py,我认为它可能看起来像这样:

from django.contrib import admin
from django.urls import path, include
from django.conf import settings
from django.conf.urls.static import static
from django.conf.urls.i18n import i18n_patterns


# I don't want my admin translated
urlpatterns = [
    path('admin/', admin.site.urls),
]

urlpatterns += i18n_patterns (
    path('i18n/', include('django.conf.urls.i18n')),
    path('', include('pages.urls')),
    path('cats', include('cats.urls')),
    path('dogs', include('dogs.urls')),
    prefix_default_language=False,
 ) + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

prefix_default_language=False,是可选的,它会从url中删除默认的语言前缀,如果只有两种或三种语言,这是有意义的。尽管过去我确实遇到了问题,但prefix_default_language=False,却无法正常工作。

如何解决prefix_default_language=False,无法正常工作或无法从网址中删除默认语言前缀的问题

在我的settings.py中,我进行了更改: LANGUAGE_CODE = 'en-us'LANGUAGE_CODE = 'en'

(似乎已经解决了)

答案 5 :(得分:1)

{% load i18n %}

{% get_current_language as LANGUAGE_CODE %}
{% get_available_languages as LANGUAGES %}
{% get_language_info_list for LANGUAGES as languages %}

 <div class="btn-header">
    <form action="{% url 'set_language' %}" method="post">
      {% csrf_token %}
      <input name="next" type="hidden" value="{{ redirect_to }}" />
        {% for language in languages %}
            {% if language.code != LANGUAGE_CODE %}
                <button type="submit" name="language" value="{{ language.code }}">{{ language.name }}</button>
            {% endif %}
        {% endfor %}
    </form>
  </div>

如果您有两种语言。它只会向您显示关闭状态。

https://www.loom.com/share/9319c0e9204f417a8eec897965ce3a96

答案 6 :(得分:0)

除了添加此处建议的表单:

<form action="{% url 'set_language' %}" method="post">
  {% csrf_token %}
  {{ request.get_full_path_info|slice:'3:'}}
  <input name="next" type="hidden" value="{{ languageless_url }}" />
  <ul class="nav navbar-nav navbar-right language menu">
    {% get_current_language as LANGUAGE_CODE %}
    {% get_available_languages as LANGUAGES %}
    {% get_language_info_list for LANGUAGES as languages %}
    {% for language in languages %}
    <li>
      <button type="submit" name="language" value="{{ language.code }}"
        class="{% if language.code == LANGUAGE_CODE %}selected{% endif %}">
        {{ language.code }}
      </button>
    </li>
    {% endfor %}
  </ul>
</form>

我建议添加一个上下文处理器(app.context_processors.py):

def language_processor(request):
"""
This is needed for language picker to work
"""
return {
    'languageless_url':
    '/' + '/'.join(request.get_full_path().split('/')[2:])
}

这允许将逻辑放在模板之外。 另外,不要忘记在模板设置中添加处理器:

            'context_processors': [
            'app.context_processors.language_processor',

答案 7 :(得分:0)

Django 3.02

<div class="uk-flex">

    {% get_current_language as LANGUAGE_CODE %}
    {% get_available_languages as LANGUAGES %}
    {% get_language_info_list for LANGUAGES as languages %}
    <div class="languages">
        <p>{% trans "Language" %}:</p>
        <ul class="languages">
        {% for language in languages %}
            <li>
                <a href="/{{ language.code }}/{{ request.get_full_path |slice:'4:'}}" {% if language.code == LANGUAGE_CODE %} class="selected"{% endif %}>
                {{ language.name_local }}
                </a>
            </li>
        {% endfor %}
        </ul>
    </div>
</div>