Django实现国际化的步骤

Django国际化的详细文档: http://djangobook.py3k.cn/2.0/chapter19/ (中文) 英文好的可以看: https://docs.djangoproject.com/en/1.4/topics/i18n/

在这里写下Django实现国际化的基本步骤,详细内容见如上链接;

(只讲步骤,不谈原理;因为我讲不清,原理在上面链接中讲的很清楚;)

静态翻译html

一.)使用Django模板引擎的解析实现国际化,说白了就是 {% trans %}标签国际化

1.)将 {% load i18n %} 放在模板最前面;在html(默认的格式), txt, xml等Django模板支持的文件中将需要国际化的字段用 {% trans '****'%}表示;

2.)在Django项目根目录新建一个locale目录,然后在项目根目录下执行django-admin.py makemessages -l zh_CN (最后一个字段是语言代码;如果需要翻译的模板不是html,则需要通过参数-e 指明, 如: django-admin.py makemessages -l de -e txt),为每个要支持的语言执行一次; 该命令将生成对应语言的po文件,如locale/zh_CN/LC_MESSAGES/django.po。

3.)在每个po文件中,为所有需要翻译的字段填写对应语言的翻译内容; 每个msgstr对应上面的msgid;翻译完后将po文件保存为'UTF-8 无BOM'格式;

4.)执行django-admin.py compilemessages将所有的po文件转成便于编译模块处理的*.mo文件;

5.)在settings里面修改LANGUAGE_CODE查看效果;

静态翻译外部js

二.)外部javascript的国际化

1.)在js文件中将需要国际化的字段用 gettext('***')代替;

2.)在项目根目录下创建一个三级目录, 如: transjs/jsi18n/locale;在transjs和jsi18n下都放入__init__.py文件,保证改目录能够被导入;将要被国际化的js文件拷贝到jsi18n下;

3.)将transjs/jsi18n安装到INSTALL_APPS下;

4.)在html中,在导入需要翻译的js前,添加<script type='text/javascript' src='/jsi18n'></script>;

5.)在全局的urls.py中添加如下代码

js_info_dict = {

'packages': ('transjs.jsi18n',),

}

urlpatterns += patterns('',

(r'^jsi18n/$', 'django.views.i18n.javascript_catalog', js_info_dict),

)

6.)在trans/jsi18n下执行django-admin.py makemessages -d djangojs -l de,生成po文件,在po文件中填入相应语言的翻译内容;

7.)在trans/jsi18n下执行django-admin.py compilemessages编译po文件,生成mo文件;

8.)在settings里面修改LANGUAGE_CODE查看效果;

以上是静态翻译的基本步骤,是根据settings.py中的LANGUAGE_CODE翻译的,下面看看如何动态翻译

三.)动态翻译(提供语言列表让用户选择,根据用户选择的语言动态切换),以下步骤是在完成了上面一,二的基础上进行的;

1.)在settings.py中设置所支持的语言列表; 如支持中文和英文

LANGUAGES = (

('zh', '简体中文'),

('en', 'English'),

)

2.)添加中间件LocaleMiddleware, 在MIDDLEWARE_CLASSES设置中增加 'django.middleware.locale.LocaleMiddleware';必须放在 SessionMiddleware和CacheMiddleware的后面;

3.)重定向视图,Django自带了一个 django.views.i18n.set_language 视图,作用是设置用户语言偏好并重定向返回到指定的页面;

在URLconf中加入下面这行代码来激活这个视图:

(r'^i18n/', include('django.conf.urls.i18n')),

4.)提交切换语言的请求,下面是一个请求的表单例子;在next的value中指定切换语言后显示的页面;

<form action='/i18n/setlang/' method='post'>

<input name='next' type='hidden' value='/next/page/' />

<select name='language'>

{% for lang in LANGUAGES %}

<option value='{{ lang.0 }}'>{{ lang.1 }}</option>

{% endfor %}

</select>

<input type='submit' value='Go' />

</form>

四.)翻译的内容中含有参数的情况

1.)在html中使用模板参数,此时必须使用{% blocktrans %}代替{% trans %},例如:

{% blocktrans %}This string will have {{ value }} inside.{% endblocktrans %}

其中{{ value }}是一个django模板变量,由django在将模板转换成具体的html时传入;

2.)如果要使用模板过滤器来翻译一个模板表达式,需要在翻译的这段文本中将表达式绑定到一个本地变量中:

{% blocktrans with value|filter as myvar %} This will have {{ myvar }} inside. {% endblocktrans %}

myvar就是用来绑定表达式本地变量;filter是django所支持的所有过滤器,下面是个使用length的例子;

例如:有一个模板变量{{ funcname }},在翻译串内需要用到该变量的字符长度:

{% blocktrans with funcname|length as len %}

There is {{ len }} letter in {{ funcname }}.

{% endblocktrans %}

len :本地变量,用来绑定(或者说保存){{funcname}}的长度,然后在作为参数出现在翻译字符串内;

{{ funcname }}: 就是1.)里面所讲的模板变量;

这里又引出了另一个问题,就是如果len大于1怎么办? Good question,its answer is using {% plural %} tag. 简单的说就是把单复数形式的句子都写一遍,用{% plural %}区分开,先写单数形式,再写复数形式,实际翻译时,会自动根据len的值来选择句子,具体如下:

{% blocktrans with funcname|length as len %}

There is only one letter in {{funcname}}.

{% plural %}

There are {{ len }} letters in {{ funcname }}.

{% endblocktrans %}

3.)js中使用参数;这个也许有更精妙的方法,但是目前我只知道下面的方法;

在需要翻译的字符串里放入一个占位符,然后在得到翻译的字串后,将占位符用实际的值取代;

如; gettext('Nice to meet you, %s');

这个地方用 %s 作为占位符;在得到翻译的字串后,将 %s 替换成实际的人名即可;

<完>


原理分析:

django.middleware.locale.LocaleMiddleware

LocaleMiddleware 按照如下算法确定用户的语言:

  • 1. 首先,在当前用户的 session 的中查找django_language(LANGUAGE_SESSION_KEY = '_language')键;

  • 2. 如未找到,它会找寻一个cookie,key为LANGUAGE_COOKIE_NAME = 'django_language'

  • 3. 还找不到的话,它会在 HTTP 请求头部里查找Accept-Language, 该头部是你的浏览器发送的,并且按优先顺序告诉服务器你的语言偏好。 Django会尝试头部中的每一个语种直到它发现一个可用的翻译。

  • 4. 以上都失败了的话, 就使用全局的 LANGUAGE_CODE 设定值。


  • 只有在 LANGUAGES 设置中列出的语言才能被选用。 若希望将语言限制为所提供语言中的某些(因为应用程序并不提供所有语言的表示),则将 LANGUAGES 设置为所希望提供语言的列表,例如: 例如:


  • LANGUAGES = (
      ('de', _('German')),
      ('en', _('English')),
    )

一旦LocaleMiddleware决定用户的偏好,它会让这个偏好作为request.LANGUAGE_CODE对每一个HttpRequest有效。请随意在你的视图代码中读一读这个值。 以下是一个简单的例子:

def hello_world(request):
    if request.LANGUAGE_CODE == 'de-at':
        return HttpResponse("You prefer to read Austrian German.")
    else:
        return HttpResponse("You prefer to read another language.")

注意,对于静态翻译(无中间件)而言,此语言在settings.LANGUAGE_CODE中,而对于动态翻译(中间件),它在request.LANGUAGE_CODE中。


你可能感兴趣的:(Django实现国际化的步骤)