13
13
TypedChoiceField ,
14
14
TypedMultipleChoiceField ,
15
15
)
16
- from django .forms .widgets import ChoiceWidget , Select , SelectMultiple
16
+ from django .forms .widgets import (
17
+ CheckboxSelectMultiple ,
18
+ ChoiceWidget ,
19
+ RadioSelect ,
20
+ Select ,
21
+ SelectMultiple ,
22
+ )
17
23
18
24
from django_enum .utils import choices as get_choices
19
25
from django_enum .utils import (
20
26
decompose ,
21
27
determine_primitive ,
28
+ get_set_bits ,
22
29
get_set_values ,
23
30
with_typehint ,
24
31
)
27
34
"NonStrictSelect" ,
28
35
"NonStrictSelectMultiple" ,
29
36
"FlagSelectMultiple" ,
30
- "FlagNonStrictSelectMultiple " ,
37
+ "NonStrictRadioSelect " ,
31
38
"ChoiceFieldMixin" ,
32
39
"EnumChoiceField" ,
33
40
"EnumMultipleChoiceField" ,
@@ -60,7 +67,7 @@ class _Unspecified:
60
67
"""
61
68
62
69
63
- class NonStrictMixin ( with_typehint ( Select )): # type: ignore
70
+ class NonStrictMixin :
64
71
"""
65
72
Mixin to add non-strict behavior to a widget, this makes sure the set value
66
73
appears as a choice if it is not one of the enumeration choices.
@@ -79,19 +86,53 @@ def render(self, *args, **kwargs):
79
86
choice [0 ] for choice in self .choices
80
87
):
81
88
self .choices = list (self .choices ) + [(value , str (value ))]
82
- return super ().render (* args , ** kwargs )
89
+ return super ().render (* args , ** kwargs ) # type: ignore[misc]
90
+
91
+
92
+ class NonStrictFlagMixin :
93
+ """
94
+ Mixin to add non-strict behavior to a multiple choice flag widget, this makes sure
95
+ that set flags outside of the enumerated flags will show up as choices. They will
96
+ be displayed as the index of the set bit.
97
+ """
98
+
99
+ choices : _SelectChoices
100
+
101
+ def render (self , * args , ** kwargs ):
102
+ """
103
+ Before rendering if we're a non-strict flag field and bits are set that are
104
+ not part of our flag enumeration we add them as (integer value, bit index)
105
+ to our (value, label) choice list.
106
+ """
107
+
108
+ raw_choices = zip (
109
+ get_set_values (kwargs .get ("value" )), get_set_bits (kwargs .get ("value" ))
110
+ )
111
+ self .choices = list (self .choices )
112
+ choice_values = set (choice [0 ] for choice in self .choices )
113
+ for value , label in raw_choices :
114
+ if value not in choice_values :
115
+ self .choices .append ((value , label ))
116
+ return super ().render (* args , ** kwargs ) # type: ignore[misc]
83
117
84
118
85
119
class NonStrictSelect (NonStrictMixin , Select ):
86
120
"""
87
- A Select widget for non-strict EnumChoiceFields that includes any existing
88
- non-conforming value as a choice option .
121
+ This widget renders a select box that includes an option for each value on the
122
+ enumeration .
89
123
"""
90
124
91
125
92
- class FlagSelectMultiple ( SelectMultiple ):
126
+ class NonStrictRadioSelect ( NonStrictMixin , RadioSelect ):
93
127
"""
94
- A SelectMultiple widget for EnumFlagFields.
128
+ This widget renders a radio select field that includes an option for each value on
129
+ the enumeration and for any non-value that is set.
130
+ """
131
+
132
+
133
+ class FlagMixin :
134
+ """
135
+ This mixin adapts a widget to work with :class:`~enum.IntFlag` types.
95
136
"""
96
137
97
138
enum : Optional [Type [Flag ]]
@@ -119,17 +160,45 @@ def format_value(self, value):
119
160
return value
120
161
121
162
163
+ class FlagSelectMultiple (FlagMixin , SelectMultiple ):
164
+ """
165
+ This widget will render :class:`~enum.IntFlag` types as a multi select field with
166
+ an option for each flag value.
167
+ """
168
+
169
+
170
+ class FlagCheckbox (FlagMixin , CheckboxSelectMultiple ):
171
+ """
172
+ This widget will render :class:`~enum.IntFlag` types as checkboxes with a checkbox
173
+ for each flag value.
174
+ """
175
+
176
+
122
177
class NonStrictSelectMultiple (NonStrictMixin , SelectMultiple ):
123
178
"""
124
- A SelectMultiple widget for non-strict EnumFlagFields that includes any
125
- existing non-conforming value as a choice option .
179
+ This widget will render a multi select box that includes an option for each
180
+ value on the enumeration and for any non-value that is passed in .
126
181
"""
127
182
128
183
129
- class FlagNonStrictSelectMultiple (NonStrictMixin , FlagSelectMultiple ):
184
+ class NonStrictFlagSelectMultiple (NonStrictFlagMixin , FlagSelectMultiple ):
185
+ """
186
+ This widget will render a multi select box that includes an option for each flag
187
+ on the enumeration and also for each bit lot listed in the enumeration that is set
188
+ on the value.
189
+
190
+ Options for extra bits only appear if they are set. You should pass choices to the
191
+ form field if you want additional options to always appear.
130
192
"""
131
- A SelectMultiple widget for non-strict EnumFlagFields that includes any
132
- existing non-conforming value as a choice option.
193
+
194
+
195
+ class NonStrictFlagCheckbox (NonStrictFlagMixin , FlagCheckbox ):
196
+ """
197
+ This widget will render a checkbox for each flag on the enumeration and also
198
+ for each bit not listed in the enumeration that is set on the value.
199
+
200
+ Checkboxes for extra bits only appear if they are set. You should pass choices to
201
+ the form field if you want additional checkboxes to always appear.
133
202
"""
134
203
135
204
@@ -165,7 +234,7 @@ class ChoiceFieldMixin(
165
234
166
235
choices : _ChoicesParameter
167
236
168
- non_strict_widget : Type [ChoiceWidget ] = NonStrictSelect
237
+ non_strict_widget : Optional [ Type [ChoiceWidget ] ] = NonStrictSelect
169
238
170
239
def __init__ (
171
240
self ,
@@ -181,7 +250,7 @@ def __init__(
181
250
):
182
251
self ._strict_ = strict
183
252
self ._primitive_ = primitive
184
- if not self .strict :
253
+ if not self .strict and self . non_strict_widget :
185
254
kwargs .setdefault ("widget" , self .non_strict_widget )
186
255
187
256
if empty_values is _Unspecified :
@@ -294,7 +363,7 @@ def default_coerce(self, value: Any) -> Any:
294
363
295
364
:param value: The value to convert
296
365
:raises ValidationError: if a valid return value cannot be determined.
297
- :return : An enumeration value or the canonical empty value if value is
366
+ :returns : An enumeration value or the canonical empty value if value is
298
367
one of our empty_values, or the value itself if this is a
299
368
non-strict field and the value is of a matching primitive type
300
369
"""
@@ -368,7 +437,7 @@ class EnumFlagField(ChoiceFieldMixin, TypedMultipleChoiceField): # type: ignore
368
437
"""
369
438
370
439
widget = FlagSelectMultiple
371
- non_strict_widget = FlagNonStrictSelectMultiple
440
+ non_strict_widget = NonStrictFlagSelectMultiple
372
441
373
442
def __init__ (
374
443
self ,
@@ -380,10 +449,12 @@ def __init__(
380
449
choices : _ChoicesParameter = (),
381
450
** kwargs ,
382
451
):
383
- kwargs .setdefault (
384
- "widget" ,
385
- self .widget (enum = enum ) if strict else self .non_strict_widget (enum = enum ), # type: ignore[call-arg]
452
+ widget = kwargs .get (
453
+ "widget" , self .widget if self .strict else self .non_strict_widget
386
454
)
455
+ if isinstance (widget , type ) and issubclass (widget , FlagMixin ):
456
+ widget = widget (enum = enum )
457
+ kwargs ["widget" ] = widget
387
458
super ().__init__ (
388
459
enum = enum ,
389
460
empty_value = (
0 commit comments