|
| 1 | +from django import forms |
| 2 | +from django.core.exceptions import ValidationError |
| 3 | +from django.forms import formset_factory, model_to_dict |
| 4 | +from django.forms.models import modelform_factory |
| 5 | +from django.utils.html import format_html, format_html_join |
| 6 | + |
| 7 | + |
| 8 | +def models_to_dicts(models): |
| 9 | + """ |
| 10 | + Convert initial data (which is a list of model instances or None) to a |
| 11 | + list of dictionary data suitable for a formset. |
| 12 | + """ |
| 13 | + return [model_to_dict(model) for model in models or []] |
| 14 | + |
| 15 | + |
| 16 | +class EmbeddedModelArrayField(forms.Field): |
| 17 | + def __init__(self, model, prefix, max_length=None, *args, **kwargs): |
| 18 | + self.model = model |
| 19 | + self.prefix = prefix |
| 20 | + self.formset = formset_factory( |
| 21 | + form=modelform_factory(model, fields="__all__"), |
| 22 | + can_delete=True, |
| 23 | + max_num=max_length, |
| 24 | + extra=3, |
| 25 | + validate_max=True, |
| 26 | + ) |
| 27 | + kwargs["widget"] = EmbeddedModelArrayWidget() |
| 28 | + super().__init__(*args, **kwargs) |
| 29 | + |
| 30 | + def clean(self, value): |
| 31 | + if not value: |
| 32 | + return [] |
| 33 | + formset = self.formset(value, prefix=self.prefix) |
| 34 | + if not formset.is_valid(): |
| 35 | + raise ValidationError(formset.errors + formset.non_form_errors()) |
| 36 | + cleaned_data = [] |
| 37 | + for data in formset.cleaned_data: |
| 38 | + # The fallback to True skips empty forms. |
| 39 | + if data.get("DELETE", True): |
| 40 | + continue |
| 41 | + data.pop("DELETE") # The "delete" checkbox isn't part of model data. |
| 42 | + cleaned_data.append(self.model(**data)) |
| 43 | + return cleaned_data |
| 44 | + |
| 45 | + def has_changed(self, initial, data): |
| 46 | + formset = self.formset(data, initial=models_to_dicts(initial), prefix=self.prefix) |
| 47 | + return formset.has_changed() |
| 48 | + |
| 49 | + def get_bound_field(self, form, field_name): |
| 50 | + return EmbeddedModelArrayBoundField(form, self, field_name) |
| 51 | + |
| 52 | + |
| 53 | +class EmbeddedModelArrayBoundField(forms.BoundField): |
| 54 | + def __init__(self, form, field, name): |
| 55 | + super().__init__(form, field, name) |
| 56 | + self.formset = field.formset( |
| 57 | + self.data if form.is_bound else None, |
| 58 | + initial=models_to_dicts(self.initial), |
| 59 | + prefix=self.html_name, |
| 60 | + ) |
| 61 | + |
| 62 | + def __str__(self): |
| 63 | + body = format_html_join( |
| 64 | + "\n", "<tbody>{}</tbody>", ((form.as_table(),) for form in self.formset) |
| 65 | + ) |
| 66 | + return format_html("<table>\n{}\n</table>\n{}", body, self.formset.management_form) |
| 67 | + |
| 68 | + |
| 69 | +class EmbeddedModelArrayWidget(forms.Widget): |
| 70 | + """ |
| 71 | + This widget extracts the data for EmbeddedModelArrayFormField's formset. |
| 72 | + It is never rendered. |
| 73 | + """ |
| 74 | + |
| 75 | + def value_from_datadict(self, data, files, name): |
| 76 | + return {key: data[key] for key in data if key.startswith(f"{name}-")} |
0 commit comments