1
1
"""Enumeration support for django model forms"""
2
2
3
+ import typing as t
3
4
from copy import copy
4
5
from decimal import DecimalException
5
6
from enum import Enum , Flag
6
7
from functools import reduce
7
8
from operator import or_
8
- from typing import Any , Iterable , List , Optional , Protocol , Sequence , Tuple , Type , Union
9
9
10
10
from django .core .exceptions import ValidationError
11
11
from django .forms .fields import (
42
42
]
43
43
44
44
45
- _SelectChoices = Iterable [Union [Tuple [Any , Any ], Tuple [str , Iterable [Tuple [Any , Any ]]]]]
45
+ _SelectChoices = t .Iterable [
46
+ t .Union [t .Tuple [t .Any , t .Any ], t .Tuple [str , t .Iterable [t .Tuple [t .Any , t .Any ]]]]
47
+ ]
46
48
47
- _Choice = Tuple [Any , Any ]
48
- _ChoiceNamedGroup = Tuple [str , Iterable [_Choice ]]
49
- _FieldChoices = Iterable [Union [_Choice , _ChoiceNamedGroup ]]
49
+ _Choice = t . Tuple [t . Any , t . Any ]
50
+ _ChoiceNamedGroup = t . Tuple [str , t . Iterable [_Choice ]]
51
+ _FieldChoices = t . Iterable [t . Union [_Choice , _ChoiceNamedGroup ]]
50
52
51
53
52
- class _ChoicesCallable (Protocol ):
54
+ class _ChoicesCallable (t . Protocol ):
53
55
def __call__ (self ) -> _FieldChoices : ... # pragma: no cover
54
56
55
57
56
- _ChoicesParameter = Union [_FieldChoices , _ChoicesCallable ]
58
+ _ChoicesParameter = t . Union [_FieldChoices , _ChoicesCallable ]
57
59
58
60
59
- class _CoerceCallable (Protocol ):
60
- def __call__ (self , value : Any , / ) -> Any : ... # pragma: no cover
61
+ class _CoerceCallable (t . Protocol ):
62
+ def __call__ (self , value : t . Any , / ) -> t . Any : ... # pragma: no cover
61
63
62
64
63
65
class _Unspecified :
@@ -81,7 +83,7 @@ def render(self, *args, **kwargs):
81
83
one of our choices, we add it as an option.
82
84
"""
83
85
84
- value : Any = getattr (kwargs .get ("value" ), "value" , kwargs .get ("value" ))
86
+ value : t . Any = getattr (kwargs .get ("value" ), "value" , kwargs .get ("value" ))
85
87
if value not in EnumChoiceField .empty_values and value not in (
86
88
choice [0 ] for choice in self .choices
87
89
):
@@ -135,9 +137,9 @@ class FlagMixin:
135
137
This mixin adapts a widget to work with :class:`~enum.IntFlag` types.
136
138
"""
137
139
138
- enum : Optional [Type [Flag ]]
140
+ enum : t . Optional [t . Type [Flag ]]
139
141
140
- def __init__ (self , enum : Optional [Type [Flag ]] = None , ** kwargs ):
142
+ def __init__ (self , enum : t . Optional [t . Type [Flag ]] = None , ** kwargs ):
141
143
self .enum = enum
142
144
super ().__init__ (** kwargs )
143
145
@@ -223,29 +225,29 @@ class ChoiceFieldMixin(
223
225
:param kwargs: Any additional parameters to pass to ChoiceField base class.
224
226
"""
225
227
226
- _enum_ : Optional [Type [Enum ]] = None
227
- _primitive_ : Optional [Type ] = None
228
+ _enum_ : t . Optional [t . Type [Enum ]] = None
229
+ _primitive_ : t . Optional [t . Type ] = None
228
230
_strict_ : bool = True
229
- empty_value : Any = ""
230
- empty_values : Sequence [Any ] = list (TypedChoiceField .empty_values )
231
+ empty_value : t . Any = ""
232
+ empty_values : t . Sequence [t . Any ] = list (TypedChoiceField .empty_values )
231
233
232
234
_empty_value_overridden_ : bool = False
233
235
_empty_values_overridden_ : bool = False
234
236
235
237
choices : _ChoicesParameter
236
238
237
- non_strict_widget : Optional [Type [ChoiceWidget ]] = NonStrictSelect
239
+ non_strict_widget : t . Optional [t . Type [ChoiceWidget ]] = NonStrictSelect
238
240
239
241
def __init__ (
240
242
self ,
241
- enum : Optional [Type [Enum ]] = _enum_ ,
242
- primitive : Optional [Type ] = _primitive_ ,
243
+ enum : t . Optional [t . Type [Enum ]] = _enum_ ,
244
+ primitive : t . Optional [t . Type ] = _primitive_ ,
243
245
* ,
244
- empty_value : Any = _Unspecified ,
246
+ empty_value : t . Any = _Unspecified ,
245
247
strict : bool = _strict_ ,
246
- empty_values : Union [List [Any ], Type [_Unspecified ]] = _Unspecified ,
248
+ empty_values : t . Union [t . List [t . Any ], t . Type [_Unspecified ]] = _Unspecified ,
247
249
choices : _ChoicesParameter = (),
248
- coerce : Optional [_CoerceCallable ] = None ,
250
+ coerce : t . Optional [_CoerceCallable ] = None ,
249
251
** kwargs ,
250
252
):
251
253
self ._strict_ = strict
@@ -328,30 +330,30 @@ def enum(self, enum):
328
330
f"specify a non-conflicting empty_value."
329
331
)
330
332
331
- def _coerce_to_value_type (self , value : Any ) -> Any :
333
+ def _coerce_to_value_type (self , value : t . Any ) -> t . Any :
332
334
"""Coerce the value to the enumerations value type"""
333
335
return self .primitive (value )
334
336
335
- def prepare_value (self , value : Any ) -> Any :
337
+ def prepare_value (self , value : t . Any ) -> t . Any :
336
338
"""Must return the raw enumeration value type"""
337
339
value = self ._coerce (value )
338
340
return super ().prepare_value (
339
341
value .value if isinstance (value , self .enum ) else value
340
342
)
341
343
342
- def to_python (self , value : Any ) -> Any :
344
+ def to_python (self , value : t . Any ) -> t . Any :
343
345
"""Return the value as its full enumeration object"""
344
346
return self ._coerce (value )
345
347
346
- def valid_value (self , value : Any ) -> bool :
348
+ def valid_value (self , value : t . Any ) -> bool :
347
349
"""Return false if this value is not valid"""
348
350
try :
349
351
self ._coerce (value )
350
352
return True
351
353
except ValidationError :
352
354
return False
353
355
354
- def default_coerce (self , value : Any ) -> Any :
356
+ def default_coerce (self , value : t . Any ) -> t . Any :
355
357
"""
356
358
Attempt conversion of value to an enumeration value and return it
357
359
if successful.
@@ -421,6 +423,10 @@ class EnumMultipleChoiceField( # type: ignore
421
423
422
424
non_strict_widget = NonStrictSelectMultiple
423
425
426
+ def has_changed (self , initial , data ):
427
+ # TODO
428
+ return super ().has_changed (initial , data )
429
+
424
430
425
431
class EnumFlagField (ChoiceFieldMixin , TypedMultipleChoiceField ): # type: ignore
426
432
"""
@@ -441,11 +447,11 @@ class EnumFlagField(ChoiceFieldMixin, TypedMultipleChoiceField): # type: ignore
441
447
442
448
def __init__ (
443
449
self ,
444
- enum : Optional [Type [Flag ]] = None ,
450
+ enum : t . Optional [t . Type [Flag ]] = None ,
445
451
* ,
446
- empty_value : Any = _Unspecified ,
452
+ empty_value : t . Any = _Unspecified ,
447
453
strict : bool = ChoiceFieldMixin ._strict_ ,
448
- empty_values : Union [List [Any ], Type [_Unspecified ]] = _Unspecified ,
454
+ empty_values : t . Union [t . List [t . Any ], t . Type [_Unspecified ]] = _Unspecified ,
449
455
choices : _ChoicesParameter = (),
450
456
** kwargs ,
451
457
):
@@ -466,11 +472,23 @@ def __init__(
466
472
** kwargs ,
467
473
)
468
474
469
- def _coerce (self , value : Any ) -> Any :
475
+ def _coerce (self , value : t . Any ) -> t . Any :
470
476
"""Combine the values into a single flag using |"""
471
477
if self .enum and isinstance (value , self .enum ):
472
478
return value
473
479
values = TypedMultipleChoiceField ._coerce (self , value ) # type: ignore[attr-defined]
474
480
if values :
475
481
return reduce (or_ , values )
476
482
return self .empty_value
483
+
484
+ def has_changed (self , initial , data ):
485
+ return super ().has_changed (
486
+ * (
487
+ [str (en .value ) for en in decompose (initial )]
488
+ if isinstance (initial , Flag )
489
+ else initial ,
490
+ [str (en .value ) for en in decompose (data )]
491
+ if isinstance (data , Flag )
492
+ else data ,
493
+ )
494
+ )
0 commit comments