Django高级编程之自定义Field实现多语言


Posted in Python onJuly 02, 2019

自定义数据库字段

扩展默认的models.CharField和models.TextField使之成为支持多语言的字段。
可以轻松实现复用,无需配置多余选项

from django.conf import settings
from django.db import models
from django.utils.translation import get_language


class MultilingualField(models.Field):
 SUPPORTED_FIELD_TYPES = [models.CharField, models.TextField]

 def __init__(self, verbose_name=None, **kwargs):
  self.localized_field_model = None
  for model in MultilingualField.SUPPORTED_FIELD_TYPES:
   if issubclass(self.__class__, model):
    self.localized_field_model = model
  self._blank = kwargs.get("blank", False)
  self._editable = kwargs.get("editable", True)
  super().__init__(verbose_name, **kwargs)

 @staticmethod
 def localized_field_name(name, lang_code):
  lang_code_safe = lang_code.replace("-", "_")
  return f"{name}_{lang_code_safe}"

 def get_localized_field(self, lang_code, lang_name):
  _blank = (self._blank
     if lang_code == settings.LANGUAGE_CODE
     else True)
  localized_field = self.localized_field_model(
   f"{self.verbose_name} ({lang_name})",
   name=self.name,
   primary_key=self.primary_key,
   max_length=self.max_length,
   unique=self.unique,
   blank=_blank,
   null=False, # we ignore the null argument!
   db_index=self.db_index,
   default=self.default or "",
   editable=self._editable,
   serialize=self.serialize,
   choices=self.choices,
   help_text=self.help_text,
   db_column=None,
   db_tablespace=self.db_tablespace)
  return localized_field

 def contribute_to_class(self, cls, name,
       private_only=False):
  def translated_value(self):
   language = get_language()
   val = self.__dict__.get(
    MultilingualField.localized_field_name(
      name, language))
   if not val:
    val = self.__dict__.get(
     MultilingualField.localized_field_name(
       name, settings.LANGUAGE_CODE))
   return val

  # generate language-specific fields dynamically
  if not cls._meta.abstract:
   if self.localized_field_model:
    for lang_code, lang_name in settings.LANGUAGES:
     localized_field = self.get_localized_field(
      lang_code, lang_name)
     localized_field.contribute_to_class(
       cls,
       MultilingualField.localized_field_name(
         name, lang_code))

    setattr(cls, name, property(translated_value))
   else:
    super().contribute_to_class(
     cls, name, private_only)


class MultilingualCharField(models.CharField, MultilingualField):
 pass


class MultilingualTextField(models.TextField, MultilingualField):
 pass

这里定义了 MultilingualCharField 和 MultilingualTextField字段

使用方法

settings.py中配置多语言

LANGUAGE_CODE = 'zh-hans'
TIME_ZONE = 'Asia/Shanghai'

LANGUAGES = (
 ('en-us', 'US English'),
 ('zh-hans', 'Asia/Shanghai')
)

默认语言设置为中文,多语言为英语

models.py中使用字段

from django.db import models
from django.utils.translation import ugettext_lazy as _

from utils.fields import (
 MultilingualCharField,
 MultilingualTextField
)

class Item(models.Model):
 title = MultilingualCharField(_('Title'), max_length=200)
 description = MultilingualTextField(_('Description'), blank=True)
 content = MultilingualTextField(_('Content'))

 def __str__(self):
  return self.title

效果图

Django高级编程之自定义Field实现多语言

可以看到,数据库字段自动生成了相应语言的字段

当用户语言切换到其他,可以自动适配实现多语言

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python 算法 排序实现快速排序
Jun 05 Python
Python中getattr函数和hasattr函数作用详解
Jun 14 Python
Python 遍历子文件和所有子文件夹的代码实例
Dec 21 Python
Python中的__slots__示例详解
Jul 06 Python
python实现多人聊天室
Mar 31 Python
Python实现将多个空格换为一个空格.md的方法
Dec 20 Python
浅谈python新式类和旧式类区别
Apr 26 Python
python3.6+django2.0+mysql搭建网站过程详解
Jul 24 Python
Django REST Framework之频率限制的使用
Sep 29 Python
Python 执行矩阵与线性代数运算
Aug 01 Python
Python+OpenCV图像处理——打印图片属性、设置存储路径、调用摄像头
Oct 22 Python
Python 实现绘制子图及子图刻度的变换等问题
May 31 Python
python 杀死自身进程的实现方法
Jul 01 #Python
python 判断linux进程,并杀死进程的实现方法
Jul 01 #Python
PyCharm-错误-找不到指定文件python.exe的解决方法
Jul 01 #Python
解决pycharm 工具栏Tool中找不到Run manager.py Task的问题
Jul 01 #Python
简单了解python中对象的取反运算符
Jul 01 #Python
python做反被爬保护的方法
Jul 01 #Python
python全栈知识点总结
Jul 01 #Python
You might like
php自动获取目录下的模板的代码
2010/08/08 PHP
通过5个php实例细致说明传值与传引用的区别
2012/08/08 PHP
php返回字符串中所有单词的方法
2015/03/09 PHP
浅谈PHP中try{}catch{}的使用方法
2016/12/09 PHP
利用PHP访问MySql数据库的逻辑操作以及增删改查的实例讲解
2017/08/30 PHP
javascript 验证日期的函数
2010/03/18 Javascript
jQuery中prevAll()方法用法实例
2015/01/08 Javascript
jQuery扁平化风格下拉框美化插件FancySelect使用指南
2015/02/10 Javascript
js限制文本框只能输入整数或者带小数点的数字
2015/04/27 Javascript
jQuery实现的图文高亮滚动切换特效实例
2015/08/10 Javascript
javascript生成随机数方法汇总
2015/11/12 Javascript
利用CSS3在Angular中实现动画
2016/01/15 Javascript
javascript将url解析为json格式的两种方法
2017/08/18 Javascript
详解A标签中href=""的几种用法
2017/08/20 Javascript
Bootstrap框架建立树形菜单(Tree)的实例代码
2017/10/30 Javascript
js 公式编辑器 - 自定义匹配规则 - 带提示下拉框 - 动态获取光标像素坐标
2018/01/04 Javascript
AjaxUpLoad.js实现文件上传
2018/03/05 Javascript
Angular 封装并发布组件的方法示例
2018/04/19 Javascript
浅析微信小程序modal弹窗关闭默认会执行cancel问题
2019/10/14 Javascript
react实现复选框全选和反选组件效果
2020/08/25 Javascript
Python的Tornado框架异步编程入门实例
2015/04/24 Python
Python去除字符串两端空格的方法
2015/05/21 Python
Python PIL图片添加字体的例子
2019/08/22 Python
python 如何上传包到pypi
2020/12/24 Python
基于PyInstaller各参数的含义说明
2021/03/04 Python
CSS3弹性盒模型开发笔记(二)
2016/04/26 HTML / CSS
澳大利亚领先的宠物用品商店:VetSupply
2017/09/08 全球购物
Molly Bracken法国电子商店:法国女性时尚品牌
2019/07/24 全球购物
个人找工作自荐信格式
2013/09/21 职场文书
运动会的口号
2014/06/09 职场文书
中学教师师德师风演讲稿
2014/08/22 职场文书
党的群众路线教育实践活动领导班子整改方案
2014/10/25 职场文书
安全主题班会教案
2015/08/12 职场文书
写给汽车4S店的创业计划书,拿来即用!
2019/08/09 职场文书
pandas求平均数和中位数的方法实例
2021/08/04 Python
Java 多态分析
2022/04/26 Java/Android