mirror of
https://github.com/SectorLabs/django-localized-fields.git
synced 2025-10-29 18:18:57 +03:00
Add a LocalizedIntegerField
This commit is contained in:
@@ -4,6 +4,7 @@ from .uniqueslug_field import LocalizedUniqueSlugField
|
||||
from .char_field import LocalizedCharField
|
||||
from .text_field import LocalizedTextField
|
||||
from .file_field import LocalizedFileField
|
||||
from .integer_field import LocalizedIntegerField
|
||||
|
||||
|
||||
__all__ = [
|
||||
@@ -12,7 +13,8 @@ __all__ = [
|
||||
'LocalizedUniqueSlugField',
|
||||
'LocalizedCharField',
|
||||
'LocalizedTextField',
|
||||
'LocalizedFileField'
|
||||
'LocalizedFileField',
|
||||
'LocalizedIntegerField'
|
||||
]
|
||||
|
||||
try:
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import json
|
||||
|
||||
from typing import Union, List
|
||||
from typing import Union, List, Optional
|
||||
|
||||
from django.conf import settings
|
||||
from django.db.utils import IntegrityError
|
||||
@@ -53,7 +53,7 @@ class LocalizedField(HStoreField):
|
||||
setattr(model, self.name, self.descriptor_class(self))
|
||||
|
||||
@classmethod
|
||||
def from_db_value(cls, value, *_):
|
||||
def from_db_value(cls, value, *_) -> Optional[LocalizedValue]:
|
||||
"""Turns the specified database value into its Python
|
||||
equivalent.
|
||||
|
||||
|
||||
68
localized_fields/fields/integer_field.py
Normal file
68
localized_fields/fields/integer_field.py
Normal file
@@ -0,0 +1,68 @@
|
||||
from typing import Optional, Union, Dict
|
||||
|
||||
from django.conf import settings
|
||||
from django.db.utils import IntegrityError
|
||||
|
||||
from .field import LocalizedField
|
||||
from ..value import LocalizedValue, LocalizedIntegerValue
|
||||
|
||||
|
||||
class LocalizedIntegerField(LocalizedField):
|
||||
"""Stores integers as a localized value."""
|
||||
|
||||
attr_class = LocalizedIntegerValue
|
||||
|
||||
@classmethod
|
||||
def from_db_value(cls, value, *_) -> Optional[LocalizedIntegerValue]:
|
||||
db_value = super().from_db_value(value)
|
||||
if db_value is None:
|
||||
return db_value
|
||||
|
||||
# if we were used in an expression somehow then it might be
|
||||
# that we're returning an individual value or an array, so
|
||||
# we should not convert that into an :see:LocalizedIntegerValue
|
||||
if not isinstance(db_value, LocalizedValue):
|
||||
return db_value
|
||||
|
||||
return cls._convert_localized_value(db_value)
|
||||
|
||||
def to_python(self, value: Union[Dict[str, int], int, None]) -> LocalizedIntegerValue:
|
||||
"""Converts the value from a database value into a Python value."""
|
||||
|
||||
db_value = super().to_python(value)
|
||||
return self._convert_localized_value(db_value)
|
||||
|
||||
def get_prep_value(self, value: LocalizedValue) -> dict:
|
||||
"""Gets the value in a format to store into the database."""
|
||||
|
||||
prepped_value = super().get_prep_value(value)
|
||||
if prepped_value is None:
|
||||
return None
|
||||
|
||||
# make sure all values are proper integers
|
||||
for lang_code, _ in settings.LANGUAGES:
|
||||
try:
|
||||
if prepped_value[lang_code] is not None:
|
||||
int(prepped_value[lang_code])
|
||||
except (TypeError, ValueError):
|
||||
raise IntegrityError('non-integer value in column "%s.%s" violates '
|
||||
'integer constraint' % (self.name, lang_code))
|
||||
|
||||
return prepped_value
|
||||
|
||||
@staticmethod
|
||||
def _convert_localized_value(value: LocalizedValue) -> LocalizedIntegerValue:
|
||||
"""Converts from :see:LocalizedValue to :see:LocalizedIntegerValue."""
|
||||
|
||||
integer_values = {}
|
||||
for lang_code, _ in settings.LANGUAGES:
|
||||
local_value = value.get(lang_code, None)
|
||||
if local_value is None or local_value.strip() == '':
|
||||
local_value = None
|
||||
|
||||
try:
|
||||
integer_values[lang_code] = int(local_value)
|
||||
except (ValueError, TypeError):
|
||||
integer_values[lang_code] = None
|
||||
|
||||
return LocalizedIntegerValue(integer_values)
|
||||
@@ -38,7 +38,8 @@ class LocalizedValue(dict):
|
||||
"""
|
||||
|
||||
language = language or settings.LANGUAGE_CODE
|
||||
return super().get(language, default)
|
||||
value = super().get(language, default)
|
||||
return value if value is not None else default
|
||||
|
||||
def set(self, language: str, value: str):
|
||||
"""Sets the value in the specified language.
|
||||
@@ -192,6 +193,7 @@ class LocalizedFileValue(LocalizedValue):
|
||||
|
||||
def __str__(self) -> str:
|
||||
"""Returns string representation of value"""
|
||||
|
||||
return str(super().__str__())
|
||||
|
||||
@deprecation.deprecated(deprecated_in='4.6', removed_in='5.0',
|
||||
@@ -199,4 +201,30 @@ class LocalizedFileValue(LocalizedValue):
|
||||
details='Use the translate() function instead.')
|
||||
def localized(self):
|
||||
"""Returns value for current language"""
|
||||
|
||||
return self.get(translation.get_language())
|
||||
|
||||
|
||||
class LocalizedIntegerValue(LocalizedValue):
|
||||
"""All values are integers."""
|
||||
|
||||
default_value = None
|
||||
|
||||
def translate(self):
|
||||
"""Gets the value in the current language, or
|
||||
in the configured fallbck language."""
|
||||
|
||||
value = super().translate()
|
||||
if value is None or (isinstance(value, str) and value.strip() == ''):
|
||||
return None
|
||||
|
||||
return int(value)
|
||||
|
||||
def __int__(self):
|
||||
"""Gets the value in the current language as an integer."""
|
||||
|
||||
value = self.translate()
|
||||
if value is None:
|
||||
return self.default_value
|
||||
|
||||
return int(value)
|
||||
|
||||
Reference in New Issue
Block a user