1
|
|
|
""" |
2
|
|
|
All fields that are attached to the models. |
3
|
|
|
|
4
|
|
|
The core design of django-parler is to attach descriptor fields |
5
|
|
|
to the shared model, which then proxies the get/set calls to the translated model. |
6
|
|
|
|
7
|
|
|
The :class:`TranslatedField` objects are automatically added to the shared model, |
8
|
|
|
but may be added explicitly as well. This also allows to set the ``any_language`` configuration option. |
9
|
|
|
It's also useful for abstract models; add a :class:`TranslatedField` to |
10
|
|
|
indicate that the derived model is expected to provide that translatable field. |
11
|
|
|
""" |
12
|
|
|
from __future__ import unicode_literals |
13
|
|
|
|
14
|
|
|
import django |
15
|
|
|
from django.forms.forms import pretty_name |
16
|
|
|
|
17
|
|
|
|
18
|
|
|
# TODO: inherit RelatedField? |
19
|
|
|
class TranslatedField(object): |
20
|
|
|
""" |
21
|
|
|
Proxy field attached to a model. |
22
|
|
|
|
23
|
|
|
The field is automatically added to the shared model. |
24
|
|
|
However, this can be assigned manually to be more explicit, or to pass the ``any_language`` value. |
25
|
|
|
The ``any_language=True`` option causes the attribute to always return a translated value, |
26
|
|
|
even when the current language and fallback are missing. |
27
|
|
|
This can be useful for "title" attributes for example. |
28
|
|
|
|
29
|
|
|
Example: |
30
|
|
|
|
31
|
|
|
.. code-block:: python |
32
|
|
|
|
33
|
|
|
from django.db import models |
34
|
|
|
from parler.models import TranslatableModel, TranslatedFieldsModel |
35
|
|
|
|
36
|
|
|
class MyModel(TranslatableModel): |
37
|
|
|
title = TranslatedField(any_language=True) # Add with any-fallback support |
38
|
|
|
slug = TranslatedField() # Optional, but explicitly mentioned |
39
|
|
|
|
40
|
|
|
|
41
|
|
|
class MyModelTranslation(TranslatedFieldsModel): |
42
|
|
|
# Manual model class: |
43
|
|
|
master = models.ForeignKey(MyModel, related_name='translations', null=True) |
44
|
|
|
title = models.CharField("Title", max_length=200) |
45
|
|
|
slug = models.SlugField("Slug") |
46
|
|
|
""" |
47
|
|
|
|
48
|
|
|
def __init__(self, any_language=False): |
49
|
|
|
self.model = None |
50
|
|
|
self.name = None |
51
|
|
|
self.any_language = any_language |
52
|
|
|
self._meta = None |
53
|
|
|
|
54
|
|
|
def contribute_to_class(self, cls, name): |
55
|
|
|
#super(TranslatedField, self).contribute_to_class(cls, name) |
56
|
|
|
self.model = cls |
57
|
|
|
self.name = name |
58
|
|
|
|
59
|
|
|
# Add the proxy attribute |
60
|
|
|
setattr(cls, self.name, TranslatedFieldDescriptor(self)) |
61
|
|
|
|
62
|
|
|
@property |
63
|
|
|
def meta(self): |
64
|
|
|
if self._meta is None: |
65
|
|
|
self._meta = self.model._parler_meta._get_extension_by_field(self.name) |
66
|
|
|
return self._meta |
67
|
|
|
|
68
|
|
|
|
69
|
|
|
class TranslatedFieldDescriptor(object): |
70
|
|
|
""" |
71
|
|
|
Descriptor for translated attributes. |
72
|
|
|
|
73
|
|
|
This attribute proxies all get/set calls to the translated model. |
74
|
|
|
""" |
75
|
|
|
|
76
|
|
|
def __init__(self, field): |
77
|
|
|
""" |
78
|
|
|
:type field: TranslatedField |
79
|
|
|
""" |
80
|
|
|
self.field = field |
81
|
|
|
|
82
|
|
|
def __get__(self, instance, instance_type=None): |
83
|
|
|
if not instance: |
84
|
|
|
# Return the class attribute when asked for by the admin. |
85
|
|
|
return self |
86
|
|
|
|
87
|
|
|
# Auto create is useless for __get__, will return empty titles everywhere. |
88
|
|
|
# Better use a fallback instead, just like gettext does. |
89
|
|
|
translation = None |
90
|
|
|
meta = self.field.meta |
91
|
|
|
try: |
92
|
|
|
translation = instance._get_translated_model(use_fallback=True, meta=meta) |
93
|
|
|
except meta.model.DoesNotExist as e: |
94
|
|
|
if self.field.any_language: |
95
|
|
|
translation = instance._get_any_translated_model(meta=meta) # returns None on error. |
96
|
|
|
|
97
|
|
|
if translation is None: |
98
|
|
|
# Improve error message |
99
|
|
|
e.args = ("{1}\nAttempted to read attribute {0}.".format(self.field.name, e.args[0]),) |
100
|
|
|
raise |
101
|
|
|
|
102
|
|
|
return getattr(translation, self.field.name) |
103
|
|
|
|
104
|
|
|
def __set__(self, instance, value): |
105
|
|
|
if instance is None: |
106
|
|
|
raise AttributeError("{0} must be accessed via instance".format(self.field.opts.object_name)) |
107
|
|
|
|
108
|
|
|
# When assigning the property, assign to the current language. |
109
|
|
|
# No fallback is used in this case. |
110
|
|
|
translation = instance._get_translated_model(use_fallback=False, auto_create=True, meta=self.field.meta) |
111
|
|
|
setattr(translation, self.field.name, value) |
112
|
|
|
|
113
|
|
|
def __delete__(self, instance): |
114
|
|
|
# No autocreate or fallback, as this is delete. |
115
|
|
|
# Rather blow it all up when the attribute doesn't exist. |
116
|
|
|
# Similar to getting a KeyError on `del dict['UNKNOWN_KEY']` |
117
|
|
|
translation = instance._get_translated_model(meta=self.field.meta) |
118
|
|
|
delattr(translation, self.field.name) |
119
|
|
|
|
120
|
|
|
def __repr__(self): |
121
|
|
|
return "<{0} for {1}.{2}>".format(self.__class__.__name__, self.field.model.__name__, self.field.name) |
122
|
|
|
|
123
|
|
|
@property |
124
|
|
|
def short_description(self): |
125
|
|
|
""" |
126
|
|
|
Ensure that the admin ``list_display`` renders the correct verbose name for translated fields. |
127
|
|
|
|
128
|
|
|
The :func:`~django.contrib.admin.utils.label_for_field` function |
129
|
|
|
uses :func:`~django.db.models.Options.get_field_by_name` to find the find and ``verbose_name``. |
130
|
|
|
However, for translated fields, this option does not exist, |
131
|
|
|
hence it falls back to reading the attribute and trying ``short_description``. |
132
|
|
|
Ideally, translated fields should also appear in this list, to be treated like regular fields. |
133
|
|
|
""" |
134
|
|
|
translations_model = self.field.meta.model |
135
|
|
|
if translations_model is None: |
136
|
|
|
# This only happens with abstract models. The code is accessing the descriptor at the base model directly, |
137
|
|
|
# not the upgraded descriptor version that contribute_translations() installed. |
138
|
|
|
# Fallback to what the admin label_for_field() would have done otherwise. |
139
|
|
|
return pretty_name(self.field.name) |
140
|
|
|
|
141
|
|
|
if django.VERSION >= (1, 8): |
142
|
|
|
field = translations_model._meta.get_field(self.field.name) |
143
|
|
|
else: |
144
|
|
|
field = translations_model._meta.get_field_by_name(self.field.name)[0] |
145
|
|
|
|
146
|
|
|
return field.verbose_name |
147
|
|
|
|
148
|
|
|
|
149
|
|
|
class LanguageCodeDescriptor(object): |
150
|
|
|
""" |
151
|
|
|
This is the property to access the ``language_code`` in the ``TranslatableModel``. |
152
|
|
|
""" |
153
|
|
|
|
154
|
|
|
def __get__(self, instance, instance_type=None): |
155
|
|
|
if not instance: |
156
|
|
|
raise AttributeError("language_code must be accessed via instance") |
157
|
|
|
|
158
|
|
|
return instance._current_language |
159
|
|
|
|
160
|
|
|
def __set__(self, instance, value): |
161
|
|
|
raise AttributeError("The 'language_code' attribute cannot be changed directly! Use the set_current_language() method instead.") |
162
|
|
|
|
163
|
|
|
def __delete__(self, instance): |
164
|
|
|
raise AttributeError("The 'language_code' attribute cannot be deleted!") |
165
|
|
|
|
166
|
|
|
|
167
|
|
|
try: |
168
|
|
|
from south.modelsinspector import add_ignored_fields |
169
|
|
|
except ImportError: |
170
|
|
|
pass |
171
|
|
|
else: |
172
|
|
|
_name_re = "^" + __name__.replace(".", "\.") |
173
|
|
|
add_ignored_fields(( |
174
|
|
|
_name_re + "\.TranslatedField", |
175
|
|
|
)) |
176
|
|
|
|