Skip to content

Commit abfa0f4

Browse files
committed
fix #107 fix #106
1 parent 8bb0bd2 commit abfa0f4

File tree

8 files changed

+348
-31
lines changed

8 files changed

+348
-31
lines changed

doc/source/changelog.rst

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ v2.2.0 (2025-03-23)
88
===================
99

1010
* Fixed `Enum types that resolve to primitives of str or int but that do not inherit from those types can result in validation errors. <https://github.com/bckohan/django-enum/issues/109>`_
11+
* Implemented `Support checkboxes for FlagEnumField <https://github.com/bckohan/django-enum/issues/107>`_
12+
* Implemented `Support radio buttons for EnumChoiceField <https://github.com/bckohan/django-enum/issues/106>`_
1113
* Implemented `If default is not provided for flag fields it should be Flag(0). <https://github.com/bckohan/django-enum/issues/105>`_
1214
* Fixed `EnumFlagFields set empty values to Flag(0) when model field has null=True, default=None <https://github.com/bckohan/django-enum/issues/104>`_
1315
* Fixed `Large enum fields that inherit from binaryfield have editable=False by default <https://github.com/bckohan/django-enum/issues/103>`_

src/django_enum/fields.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1241,16 +1241,16 @@ def formfield(self, form_class=None, choices_form_class=None, **kwargs):
12411241
from django_enum.forms import (
12421242
ChoiceFieldMixin,
12431243
EnumFlagField,
1244-
FlagNonStrictSelectMultiple,
12451244
FlagSelectMultiple,
1245+
NonStrictFlagSelectMultiple,
12461246
)
12471247

12481248
kwargs["empty_value"] = None if self.default is None else self.enum(0)
12491249
kwargs.setdefault(
12501250
"widget",
12511251
FlagSelectMultiple(enum=self.enum)
12521252
if self.strict
1253-
else FlagNonStrictSelectMultiple(enum=self.enum),
1253+
else NonStrictFlagSelectMultiple(enum=self.enum),
12541254
)
12551255

12561256
form_field = Field.formfield(

src/django_enum/forms.py

+91-20
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,19 @@
1313
TypedChoiceField,
1414
TypedMultipleChoiceField,
1515
)
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+
)
1723

1824
from django_enum.utils import choices as get_choices
1925
from django_enum.utils import (
2026
decompose,
2127
determine_primitive,
28+
get_set_bits,
2229
get_set_values,
2330
with_typehint,
2431
)
@@ -27,7 +34,7 @@
2734
"NonStrictSelect",
2835
"NonStrictSelectMultiple",
2936
"FlagSelectMultiple",
30-
"FlagNonStrictSelectMultiple",
37+
"NonStrictRadioSelect",
3138
"ChoiceFieldMixin",
3239
"EnumChoiceField",
3340
"EnumMultipleChoiceField",
@@ -60,7 +67,7 @@ class _Unspecified:
6067
"""
6168

6269

63-
class NonStrictMixin(with_typehint(Select)): # type: ignore
70+
class NonStrictMixin:
6471
"""
6572
Mixin to add non-strict behavior to a widget, this makes sure the set value
6673
appears as a choice if it is not one of the enumeration choices.
@@ -79,19 +86,53 @@ def render(self, *args, **kwargs):
7986
choice[0] for choice in self.choices
8087
):
8188
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]
83117

84118

85119
class NonStrictSelect(NonStrictMixin, Select):
86120
"""
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.
89123
"""
90124

91125

92-
class FlagSelectMultiple(SelectMultiple):
126+
class NonStrictRadioSelect(NonStrictMixin, RadioSelect):
93127
"""
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.
95136
"""
96137

97138
enum: Optional[Type[Flag]]
@@ -119,17 +160,45 @@ def format_value(self, value):
119160
return value
120161

121162

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+
122177
class NonStrictSelectMultiple(NonStrictMixin, SelectMultiple):
123178
"""
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.
126181
"""
127182

128183

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.
130192
"""
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.
133202
"""
134203

135204

@@ -165,7 +234,7 @@ class ChoiceFieldMixin(
165234

166235
choices: _ChoicesParameter
167236

168-
non_strict_widget: Type[ChoiceWidget] = NonStrictSelect
237+
non_strict_widget: Optional[Type[ChoiceWidget]] = NonStrictSelect
169238

170239
def __init__(
171240
self,
@@ -181,7 +250,7 @@ def __init__(
181250
):
182251
self._strict_ = strict
183252
self._primitive_ = primitive
184-
if not self.strict:
253+
if not self.strict and self.non_strict_widget:
185254
kwargs.setdefault("widget", self.non_strict_widget)
186255

187256
if empty_values is _Unspecified:
@@ -294,7 +363,7 @@ def default_coerce(self, value: Any) -> Any:
294363
295364
:param value: The value to convert
296365
: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
298367
one of our empty_values, or the value itself if this is a
299368
non-strict field and the value is of a matching primitive type
300369
"""
@@ -368,7 +437,7 @@ class EnumFlagField(ChoiceFieldMixin, TypedMultipleChoiceField): # type: ignore
368437
"""
369438

370439
widget = FlagSelectMultiple
371-
non_strict_widget = FlagNonStrictSelectMultiple
440+
non_strict_widget = NonStrictFlagSelectMultiple
372441

373442
def __init__(
374443
self,
@@ -380,10 +449,12 @@ def __init__(
380449
choices: _ChoicesParameter = (),
381450
**kwargs,
382451
):
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
386454
)
455+
if isinstance(widget, type) and issubclass(widget, FlagMixin):
456+
widget = widget(enum=enum)
457+
kwargs["widget"] = widget
387458
super().__init__(
388459
enum=enum,
389460
empty_value=(

src/django_enum/utils.py

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
"with_typehint",
1818
"SupportedPrimitive",
1919
"decimal_params",
20+
"get_set_values",
2021
"get_set_bits",
2122
]
2223

tests/djenum/admin.py

+50-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,22 @@
11
from django.contrib import admin
22

3-
from django.forms import ModelForm
3+
from django.forms import ModelForm, RadioSelect
4+
from django_enum.forms import (
5+
NonStrictRadioSelect,
6+
FlagCheckbox,
7+
NonStrictFlagCheckbox,
8+
)
9+
from django_enum.utils import decompose
10+
11+
from tests.djenum.enums import TextEnum, GNSSConstellation
412
from tests.djenum.models import (
513
AdminDisplayBug35,
614
EnumTester,
715
NullBlankFormTester,
816
NullableBlankFormTester,
917
Bug53Tester,
1018
NullableStrFormTester,
19+
AltWidgetTester,
1120
)
1221

1322
admin.site.register(EnumTester)
@@ -23,3 +32,43 @@ class AdminDisplayBug35Admin(admin.ModelAdmin):
2332

2433

2534
admin.site.register(AdminDisplayBug35, AdminDisplayBug35Admin)
35+
36+
37+
class AltWidgetAdminForm(ModelForm):
38+
class Meta:
39+
model = AltWidgetTester
40+
fields = "__all__"
41+
widgets = {
42+
"text": RadioSelect,
43+
"text_null": RadioSelect,
44+
"text_non_strict": NonStrictRadioSelect,
45+
"constellation": FlagCheckbox,
46+
"constellation_null": FlagCheckbox,
47+
"constellation_non_strict": NonStrictFlagCheckbox,
48+
}
49+
50+
51+
class AltWidgetAdmin(admin.ModelAdmin):
52+
form = AltWidgetAdminForm
53+
list_display = (
54+
"text",
55+
"text_null",
56+
"text_non_strict",
57+
"constellations",
58+
"constellations_null",
59+
"constellations_non_strict",
60+
)
61+
62+
def constellations(self, obj):
63+
return ", ".join([str(c.name) for c in decompose(obj.constellation)])
64+
65+
def constellations_null(self, obj):
66+
if obj.constellation_null is None:
67+
return "None"
68+
return ", ".join([str(c.name) for c in decompose(obj.constellation_null)])
69+
70+
def constellations_non_strict(self, obj):
71+
return ", ".join([str(c.name) for c in decompose(obj.constellation_non_strict)])
72+
73+
74+
admin.site.register(AltWidgetTester, AltWidgetAdmin)

tests/djenum/enums.py

+8
Original file line numberDiff line numberDiff line change
@@ -390,3 +390,11 @@ class StrTestEnum(str, Enum):
390390

391391
def __str__(self):
392392
return self.value
393+
394+
395+
class GNSSConstellation(IntFlag):
396+
GPS = 1 << 0
397+
GLONASS = 1 << 1
398+
GALILEO = 1 << 2
399+
BEIDOU = 1 << 3
400+
QZSS = 1 << 4

tests/djenum/models.py

+14
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@
4040
TextEnum,
4141
TimeEnum,
4242
NullableConstants,
43+
GNSSConstellation,
4344
)
4445

4546

@@ -381,3 +382,16 @@ class Bug53Tester(models.Model):
381382
)
382383
int_blank_null_false = EnumField(ExternEnum, null=False, blank=True)
383384
int_blank_null_true = EnumField(ExternEnum, null=True, blank=True)
385+
386+
387+
class AltWidgetTester(models.Model):
388+
text = EnumField(TextEnum, default=TextEnum.VALUE1)
389+
text_null = EnumField(TextEnum, default=None, blank=True, null=True)
390+
text_non_strict = EnumField(
391+
TextEnum, default=TextEnum.VALUE1, strict=False, max_length=10
392+
)
393+
constellation = EnumField(GNSSConstellation)
394+
constellation_null = EnumField(
395+
GNSSConstellation, null=True, blank=True, default=None
396+
)
397+
constellation_non_strict = EnumField(GNSSConstellation, strict=False)

0 commit comments

Comments
 (0)