django-unfold 0.26.0__py3-none-any.whl → 0.28.0__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- {django_unfold-0.26.0.dist-info → django_unfold-0.28.0.dist-info}/METADATA +54 -5
- {django_unfold-0.26.0.dist-info → django_unfold-0.28.0.dist-info}/RECORD +33 -26
- unfold/admin.py +6 -157
- unfold/contrib/forms/templates/unfold/forms/array.html +31 -0
- unfold/contrib/forms/widgets.py +58 -3
- unfold/contrib/import_export/forms.py +21 -7
- unfold/contrib/import_export/templates/admin/import_export/change_list_export.html +5 -0
- unfold/contrib/import_export/templates/admin/import_export/export.html +1 -1
- unfold/dataclasses.py +2 -0
- unfold/decorators.py +3 -0
- unfold/fields.py +208 -0
- unfold/static/unfold/css/styles.css +1 -1
- unfold/templates/admin/change_form.html +0 -2
- unfold/templates/admin/edit_inline/tabular.html +4 -6
- unfold/templates/admin/includes/fieldset.html +2 -32
- unfold/templates/admin/login.html +4 -0
- unfold/templates/admin/submit_line.html +1 -1
- unfold/templates/unfold/helpers/attrs.html +1 -0
- unfold/templates/unfold/helpers/display_header.html +1 -1
- unfold/templates/unfold/helpers/field_readonly.html +1 -3
- unfold/templates/unfold/helpers/field_readonly_value.html +1 -0
- unfold/templates/unfold/helpers/fieldset_row.html +53 -0
- unfold/templates/unfold/helpers/search_results.html +10 -10
- unfold/templates/unfold/layouts/base.html +11 -0
- unfold/templates/unfold/layouts/base_simple.html +7 -1
- unfold/templates/unfold/widgets/clearable_file_input.html +1 -1
- unfold/templates/unfold/widgets/clearable_file_input_small.html +1 -1
- unfold/templates/unfold/widgets/foreign_key_raw_id.html +7 -13
- unfold/utils.py +21 -1
- unfold/views.py +18 -6
- unfold/widgets.py +12 -1
- {django_unfold-0.26.0.dist-info → django_unfold-0.28.0.dist-info}/LICENSE.md +0 -0
- {django_unfold-0.26.0.dist-info → django_unfold-0.28.0.dist-info}/WHEEL +0 -0
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: django-unfold
|
3
|
-
Version: 0.
|
3
|
+
Version: 0.28.0
|
4
4
|
Summary: Modern Django admin theme for seamless interface development
|
5
5
|
Home-page: https://unfoldadmin.com
|
6
6
|
License: MIT
|
@@ -35,13 +35,13 @@ Description-Content-Type: text/markdown
|
|
35
35
|
|
36
36
|
Unfold is theme for Django admin incorporating most common practises for building full-fledged admin areas. It is designed to work at the top of default administration provided by Django.
|
37
37
|
|
38
|
-
- **Unfold:** demo site is available at [unfoldadmin.com](https://unfoldadmin.com)
|
38
|
+
- **Unfold:** demo site is available at [unfoldadmin.com](https://unfoldadmin.com?utm_medium=github&utm_source=unfold)
|
39
39
|
- **Formula:** repository with demo implementation at [github.com/unfoldadmin/formula](https://github.com/unfoldadmin/formula)
|
40
40
|
- **Turbo:** Django & Next.js boilerplate implementing Unfold at [github.com/unfoldadmin/turbo](https://github.com/unfoldadmin/turbo)
|
41
41
|
|
42
42
|
## Are you using Unfold and need a help?<!-- omit from toc -->
|
43
43
|
|
44
|
-
Did you decide to start using Unfold but you don't have time to make the switch from native Django admin? [Get in touch with us](https://unfoldadmin.com
|
44
|
+
Did you decide to start using Unfold but you don't have time to make the switch from native Django admin? [Get in touch with us](https://unfoldadmin.com/?utm_medium=github&utm_source=unfold) and let's supercharge development by using our know-how.
|
45
45
|
|
46
46
|
## Features <!-- omit from toc -->
|
47
47
|
|
@@ -52,15 +52,17 @@ Did you decide to start using Unfold but you don't have time to make the switch
|
|
52
52
|
- **Dependencies:** completely based only on `django.contrib.admin`
|
53
53
|
- **Actions:** multiple ways how to define actions within different parts of admin
|
54
54
|
- **WYSIWYG:** built-in support for WYSIWYG (Trix)
|
55
|
+
- **Array widget:** built-in widget for `django.contrib.postgres.fields.ArrayField`
|
55
56
|
- **Filters:** custom dropdown, numeric, datetime, and text fields
|
56
57
|
- **Dashboard:** custom components for rapid dashboard development
|
57
58
|
- **Model tabs:** define custom tab navigations for models
|
58
59
|
- **Fieldset tabs:** merge several fielsets into tabs in change form
|
59
60
|
- **Colors:** possibility to override default color scheme
|
61
|
+
- **Changeform modes:** display fields in changeform in compressed mode
|
60
62
|
- **Third party packages:** default support for multiple popular applications
|
61
63
|
- **Environment label**: distinguish between environments by displaying a label
|
62
64
|
- **Nonrelated inlines**: displays nonrelated model as inline in changeform
|
63
|
-
- **Parallel admin**: support for default admin in parallel with Unfold. [Admin migration guide](https://unfoldadmin.com/blog/migrating-django-admin-unfold
|
65
|
+
- **Parallel admin**: support for default admin in parallel with Unfold. [Admin migration guide](https://unfoldadmin.com/blog/migrating-django-admin-unfold/?utm_medium=github&utm_source=unfold)
|
64
66
|
- **VS Code**: project configuration and development container is included
|
65
67
|
|
66
68
|
## Table of contents <!-- omit from toc -->
|
@@ -79,6 +81,7 @@ Did you decide to start using Unfold but you don't have time to make the switch
|
|
79
81
|
- [Dropdown filters](#dropdown-filters)
|
80
82
|
- [Numeric filters](#numeric-filters)
|
81
83
|
- [Date/time filters](#datetime-filters)
|
84
|
+
- [Custom admin pages](#custom-admin-pages)
|
82
85
|
- [Nonrelated inlines](#nonrelated-inlines)
|
83
86
|
- [Display decorator](#display-decorator)
|
84
87
|
- [Change form tabs](#change-form-tabs)
|
@@ -321,13 +324,17 @@ def permission_callback(request):
|
|
321
324
|
|
322
325
|
from django import models
|
323
326
|
from django.contrib import admin
|
327
|
+
from django.contrib.postgres.fields import ArrayField
|
324
328
|
from django.db import models
|
325
329
|
from unfold.admin import ModelAdmin
|
326
|
-
from unfold.contrib.forms.widgets import WysiwygWidget
|
330
|
+
from unfold.contrib.forms.widgets import ArrayWidget, WysiwygWidget
|
327
331
|
|
328
332
|
|
329
333
|
@admin.register(MyModel)
|
330
334
|
class CustomAdminClass(ModelAdmin):
|
335
|
+
# Display fields in changeform in compressed mode
|
336
|
+
compressed_fields = True # Default: False
|
337
|
+
|
331
338
|
# Preprocess content of readonly fields before render
|
332
339
|
readonly_preprocess_fields = {
|
333
340
|
"model_field_name": "html.unescape",
|
@@ -346,6 +353,9 @@ class CustomAdminClass(ModelAdmin):
|
|
346
353
|
formfield_overrides = {
|
347
354
|
models.TextField: {
|
348
355
|
"widget": WysiwygWidget,
|
356
|
+
},
|
357
|
+
ArrayField: {
|
358
|
+
"widget": ArrayWidget,
|
349
359
|
}
|
350
360
|
}
|
351
361
|
```
|
@@ -636,6 +646,45 @@ class YourModelAdmin(ModelAdmin):
|
|
636
646
|
)
|
637
647
|
```
|
638
648
|
|
649
|
+
## Custom admin pages
|
650
|
+
|
651
|
+
By default, Unfold provides a basic view mixin which helps with creation of basic views which are part of Unfold UI. The implementation requires creation of class based view inheriting from `unfold.views.UnfoldModelAdminViewMixin`. It is important to add `title` and `permissions_required` properties.
|
652
|
+
|
653
|
+
```python
|
654
|
+
# admin.py
|
655
|
+
|
656
|
+
from django.views.generic import TemplateView
|
657
|
+
from unfold.admin import ModelAdmin
|
658
|
+
from unfold.views import UnfoldModelAdminViewMixin
|
659
|
+
|
660
|
+
|
661
|
+
class MyClassBasedView(UnfoldModelAdminViewMixin, TemplateView):
|
662
|
+
title = "Custom Title" # required: custom page header title
|
663
|
+
permissions_required = () # required: tuple of permissions
|
664
|
+
template_name = "some/template/path.html"
|
665
|
+
|
666
|
+
|
667
|
+
class CustomAdmin(ModelAdmin):
|
668
|
+
def get_urls(self):
|
669
|
+
return super().get_urls() + [
|
670
|
+
path(
|
671
|
+
"custom-url-path",
|
672
|
+
MyClassBasedView.as_view(model_admin=self), # IMPORTANT: model_admin is required
|
673
|
+
name="custom_name"
|
674
|
+
),
|
675
|
+
]
|
676
|
+
```
|
677
|
+
|
678
|
+
The template is straightforward, extend from `unfold/layouts/base.html` and the UI will display all Unfold components like header or sidebar with all menu items. Then all content needs to be located in `content` block.
|
679
|
+
|
680
|
+
```django-html
|
681
|
+
{% extends "unfold/layouts/base.html" %}
|
682
|
+
|
683
|
+
{% block content %}
|
684
|
+
Content here
|
685
|
+
{% endblock %}
|
686
|
+
```
|
687
|
+
|
639
688
|
## Nonrelated inlines
|
640
689
|
|
641
690
|
To display inlines which are not related (no foreign key pointing at the main model) to the model instance in changeform, you can use nonrelated inlines which are included in `unfold.contrib.inlines` module. Make sure this module is included in `INSTALLED_APPS` in settings.py.
|
@@ -1,5 +1,5 @@
|
|
1
1
|
unfold/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
2
|
-
unfold/admin.py,sha256=
|
2
|
+
unfold/admin.py,sha256=Q8my6Ui10yDW_KqTwfPK8p51u9SAj2n5f8DFF8lxYuk,18991
|
3
3
|
unfold/apps.py,sha256=SlBXPYrUd2uXn67qFbRvbXSUk3XFWrF4-5WELgDCvho,381
|
4
4
|
unfold/checks.py,sha256=Smgji9w19hnYjJElJ_FJnnyTEAE-E-OUB6otHu7lasY,1670
|
5
5
|
unfold/contrib/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
@@ -23,9 +23,10 @@ unfold/contrib/forms/apps.py,sha256=Di0TMzVuRpVxLG-8Bjdq5ALCSf5r7u2xVhD0jU6H5Sc,
|
|
23
23
|
unfold/contrib/forms/static/unfold/forms/css/trix.css,sha256=TH9WdnaZrmwI8hAEydwjobdrBzSw_KYdRTSQDuD-8hE,20027
|
24
24
|
unfold/contrib/forms/static/unfold/forms/js/trix.config.js,sha256=spkNBlJVk_pqido_rM6yywQxkJ3Kqb7DMLiBgpKksdA,858
|
25
25
|
unfold/contrib/forms/static/unfold/forms/js/trix.js,sha256=Pao0XiVeDiRawfTkGDg_np6CxB-oXPrUDI9akWc87oc,174157
|
26
|
+
unfold/contrib/forms/templates/unfold/forms/array.html,sha256=11silyHbsJA0U_ksS8MvfFOJKC_qKTAwXxoMIB78APk,1507
|
26
27
|
unfold/contrib/forms/templates/unfold/forms/helpers/toolbar.html,sha256=yS8Zy-UrzvZ5RUYwdprQzREffnYq0NlIbXBfZM2UB04,9700
|
27
28
|
unfold/contrib/forms/templates/unfold/forms/wysiwyg.html,sha256=4ZefV6XrjJlUczcuSw8BhvMJUFSZPSXo1IkgkBivh5g,351
|
28
|
-
unfold/contrib/forms/widgets.py,sha256=
|
29
|
+
unfold/contrib/forms/widgets.py,sha256=yWug9YQ6FWI7hqvNZqhCwJhlfC_gMh9GELVGTE2Tw9k,2813
|
29
30
|
unfold/contrib/guardian/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
30
31
|
unfold/contrib/guardian/apps.py,sha256=ObJqwh4vHxkD4XfduP5IQAiYiWZxsXUOUqF1_R1GsRI,136
|
31
32
|
unfold/contrib/guardian/templates/admin/guardian/model/change_form.html,sha256=FSJc4MYYWyzZAy8Ay0b7Ov-cUo-oELHOM5fQehM54Lg,403
|
@@ -37,13 +38,14 @@ unfold/contrib/guardian/templates/unfold/guardian/group_form.html,sha256=P8WMC5E
|
|
37
38
|
unfold/contrib/guardian/templates/unfold/guardian/user_form.html,sha256=ci7FRrhTEKbFKKxsJ-07_dWXBYz4mqXPoqu5HfqYLaM,4132
|
38
39
|
unfold/contrib/import_export/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
39
40
|
unfold/contrib/import_export/apps.py,sha256=SdJu6Qh90VqGWY19FSDhhpUqhTbaIYsJKny3zX5baHI,149
|
40
|
-
unfold/contrib/import_export/forms.py,sha256=
|
41
|
+
unfold/contrib/import_export/forms.py,sha256=O8BBKJIdqLg3-uUFQyDV9-L6EoJ7EVSX3LJBP1zegGw,2037
|
41
42
|
unfold/contrib/import_export/templates/admin/import_export/base.html,sha256=loL2qcV-f8aAzkHss_I4IkwfgemVW2CjOu_aiBxdwX0,357
|
42
43
|
unfold/contrib/import_export/templates/admin/import_export/change_form.html,sha256=_YorgltKO0CRSaLBAp67XnGMUBWEP_XewsUff0hlrE4,446
|
44
|
+
unfold/contrib/import_export/templates/admin/import_export/change_list_export.html,sha256=vZb2XVoJrDSjM1msiZqkQPMLQGjuqRHuLNdt11hmbQs,164
|
43
45
|
unfold/contrib/import_export/templates/admin/import_export/change_list_export_item.html,sha256=pTDeqPKOlCPKH2dxMIfPnWuc2wVDzB7AzL73WbxSnRY,257
|
44
46
|
unfold/contrib/import_export/templates/admin/import_export/change_list_import_export.html,sha256=JdKd6P2Ot9Ou4yg4CywTauuE1UiTz_mRvDwlx3vj3LI,229
|
45
47
|
unfold/contrib/import_export/templates/admin/import_export/change_list_import_item.html,sha256=XUuRxnsx9YQbKvW-E_JGl_ha7kpTSGSoRefOTTizuX0,233
|
46
|
-
unfold/contrib/import_export/templates/admin/import_export/export.html,sha256=
|
48
|
+
unfold/contrib/import_export/templates/admin/import_export/export.html,sha256=NZ2bZ0yvskp_twIjX9LSROMmicXAU8dDXemxdII9_K0,3447
|
47
49
|
unfold/contrib/import_export/templates/admin/import_export/import.html,sha256=P54_f3s96PV87Bo-FCZfmsn9DkRXLOB36r7HYF6y7GM,2075
|
48
50
|
unfold/contrib/import_export/templates/admin/import_export/import_confirm.html,sha256=M-acK4XSLHuPFD_NJashGYvPPeJrJsC-3LMvHs3lRis,867
|
49
51
|
unfold/contrib/import_export/templates/admin/import_export/import_errors.html,sha256=0DmJvZs31u-E2Y53yySci86cTnG9aUnOzvfYrOo0lYA,1422
|
@@ -62,14 +64,15 @@ unfold/contrib/simple_history/templates/simple_history/_object_history_list.html
|
|
62
64
|
unfold/contrib/simple_history/templates/simple_history/object_history.html,sha256=AZ6uQRr7wKxV_rys5hGTVGYtVS-Fp5eHIqiXYW8FB1c,847
|
63
65
|
unfold/contrib/simple_history/templates/simple_history/object_history_form.html,sha256=MOL3Tw3Nk3Rnq1koRV7yeCev4CP06_4xqAIOQk1M7FU,2290
|
64
66
|
unfold/contrib/simple_history/templates/simple_history/submit_line.html,sha256=ns9CEkU4HwKHhhj8qj_9UXvzp0viGtD1tp93GV2WRCs,1703
|
65
|
-
unfold/dataclasses.py,sha256=
|
66
|
-
unfold/decorators.py,sha256=
|
67
|
+
unfold/dataclasses.py,sha256=XssBT3nfeFO-oekKDWrX6abIyrIW1P8CPzzCv1TRYFM,266
|
68
|
+
unfold/decorators.py,sha256=6E4vPVwK0IQDAiDPg9pgyypRqciX_gR0jwITDcrSc8U,3367
|
67
69
|
unfold/exceptions.py,sha256=gcCj1ox61E137bk_0Cqy4YC3SttdPgB-fiJUqpmyHSE,43
|
70
|
+
unfold/fields.py,sha256=Zegz_og7aHpdvUdFlUcIYVPB9TjPGjybhR2QNuq1b7A,7422
|
68
71
|
unfold/forms.py,sha256=GXEm3CFwglyuEbGdVyEMJTB45Gs-_RvGGlXJEkPy2kw,3688
|
69
72
|
unfold/settings.py,sha256=--TdTSWdOA8TQGW4-vjJkjy_zEyd_kZwBr3BIuQ8hzI,1208
|
70
73
|
unfold/sites.py,sha256=Gy_i43j2nizW2g8-mas5icvtk-beKism_CznATW6Ia8,12586
|
71
74
|
unfold/static/unfold/css/simplebar.css,sha256=5LLaEM11pKi6JFCOLt4XKuZxTpT9rpdq_tNlaQytFlU,4647
|
72
|
-
unfold/static/unfold/css/styles.css,sha256=
|
75
|
+
unfold/static/unfold/css/styles.css,sha256=CU9SCfH3nREI3C6yP9DFAvvr_23lSg9H6rAwM0_w958,91898
|
73
76
|
unfold/static/unfold/fonts/inter/Inter-Bold.woff2,sha256=O88EyjAeRPE_QEyKBKpK5wf2epUOEu8wwjj5bnhCZqE,46552
|
74
77
|
unfold/static/unfold/fonts/inter/Inter-Medium.woff2,sha256=O88EyjAeRPE_QEyKBKpK5wf2epUOEu8wwjj5bnhCZqE,46552
|
75
78
|
unfold/static/unfold/fonts/inter/Inter-Regular.woff2,sha256=O88EyjAeRPE_QEyKBKpK5wf2epUOEu8wwjj5bnhCZqE,46552
|
@@ -91,7 +94,7 @@ unfold/templates/admin/auth/user/add_form.html,sha256=iLig-vd2YExXsj0xGBwYhZ4kGU
|
|
91
94
|
unfold/templates/admin/auth/user/change_password.html,sha256=-Wa9ml3yss-kDz0YQxCiwoxs91KQD8eetCt5l6xekWM,2892
|
92
95
|
unfold/templates/admin/base.html,sha256=MGqtCcydXZPnp6dasaWktyd9D6rdUYX01rFGAv7Zkm4,2226
|
93
96
|
unfold/templates/admin/base_site.html,sha256=3ckWrcAdd7Pw1hk6Zwyknab_Qb-rteV9-mXhMnfo6VI,361
|
94
|
-
unfold/templates/admin/change_form.html,sha256=
|
97
|
+
unfold/templates/admin/change_form.html,sha256=s7iHvcH1coJQu7QaDk6vx_tvgxMG8SQWEy4J9bBllzY,5361
|
95
98
|
unfold/templates/admin/change_form_object_tools.html,sha256=eyeH-i2HgEM0Yi-OJA2D1VnKJyC19A_my1IDGxxoP8Y,593
|
96
99
|
unfold/templates/admin/change_list.html,sha256=18GDZswc1c0xtw2BcKti9SX95Ar9e1BX_HSY0K79g_8,5102
|
97
100
|
unfold/templates/admin/change_list_object_tools.html,sha256=cmMiT2nT20Ph5yfpj9aHPr76Z-JP4aSXp0o-Rnad28s,147
|
@@ -100,17 +103,17 @@ unfold/templates/admin/date_hierarchy.html,sha256=BfUPbsLpHZVa40BHBahz1H9RSVuz36
|
|
100
103
|
unfold/templates/admin/delete_confirmation.html,sha256=hpa2E14oZEXBBs6W1qdNQuF650TIO2Rhr52Q6UfwVeQ,5166
|
101
104
|
unfold/templates/admin/delete_selected_confirmation.html,sha256=Foka2yvwAMEZre-Kh1KNadRzrCotdKM2U4e6AJQYZu8,4941
|
102
105
|
unfold/templates/admin/edit_inline/stacked.html,sha256=HG-Dj42gcKNofHRVjg0ltIER2oJGYUd9GN_B7lDv7rQ,4580
|
103
|
-
unfold/templates/admin/edit_inline/tabular.html,sha256=
|
106
|
+
unfold/templates/admin/edit_inline/tabular.html,sha256=aMy6kk3ZDBtgZFf1pO30vHWmONvCYor9yoiR9Day4xI,13042
|
104
107
|
unfold/templates/admin/filter.html,sha256=dkrFkei-EAlldIU8DrgvSChzWQuUOu6-LS_qlZxdfFw,1708
|
105
|
-
unfold/templates/admin/includes/fieldset.html,sha256=
|
108
|
+
unfold/templates/admin/includes/fieldset.html,sha256=lMVwBifFWKvLvHqZ6yjP6Xf6BJFzi-EOf5JHIxEHmRI,888
|
106
109
|
unfold/templates/admin/includes/object_delete_summary.html,sha256=Nv69SCzyJHFX14iJFfodxKM0IIpQegKZH0fvKB15QJI,468
|
107
110
|
unfold/templates/admin/index.html,sha256=pkGdKWdD3zzOvkRdELvdb15sleSpfl4eHPA14PAh7z0,684
|
108
|
-
unfold/templates/admin/login.html,sha256=
|
111
|
+
unfold/templates/admin/login.html,sha256=ePmGmS9MrqvR6_6hnRSYdW57XKmMIuVGTL9I6jCw9BM,3792
|
109
112
|
unfold/templates/admin/nav_sidebar.html,sha256=63lUhsHfrjZowplnOEq4BkGD9jlX0bVQw5VrAMJqf5M,1178
|
110
113
|
unfold/templates/admin/object_history.html,sha256=PsbhXFd_3SCB9YkSJeHESp2VqjNlHtUW26mRtaZ-MD0,5069
|
111
114
|
unfold/templates/admin/pagination.html,sha256=KWTPV7_hVSZ1374a-pqHXhnOueNQKu1UnSUYirrWtCk,1173
|
112
115
|
unfold/templates/admin/search_form.html,sha256=8fJlPYHQDCm4Je05wwdNJuJQR6ChLgghWmo-yHSybMs,1099
|
113
|
-
unfold/templates/admin/submit_line.html,sha256=
|
116
|
+
unfold/templates/admin/submit_line.html,sha256=X7xg5vgWfHFWJOuXOLshma_LWWI0H9GHDmMYD1JqfI4,4353
|
114
117
|
unfold/templates/auth/widgets/read_only_password_hash.html,sha256=Li9efo-3cFC5zj9im0SPfc62R4ZNVPQhs24H1U7xmD8,785
|
115
118
|
unfold/templates/registration/logged_out.html,sha256=E7RHtB6AGQwgUIiV7dwJ1DbdfNvEhzJARONnB6jRLhQ,1028
|
116
119
|
unfold/templates/registration/password_change_done.html,sha256=i1ZzfTwZHWNWoN9_xHZDdcgLdTOVbTFFD1HUSuG0LkY,1062
|
@@ -131,12 +134,15 @@ unfold/templates/unfold/helpers/actions_row.html,sha256=1xd39zx38NOoKuDuxAG7PHeu
|
|
131
134
|
unfold/templates/unfold/helpers/add_link.html,sha256=mIgpKrwqBO1oJ4cwPQWSX1oUHBwHJmy5-2TxUHf-1bo,808
|
132
135
|
unfold/templates/unfold/helpers/app_list.html,sha256=lFnW8p9DcZbI9t3_ee9JX9ERHA0NRL2V88zpzuG4jq8,4720
|
133
136
|
unfold/templates/unfold/helpers/app_list_default.html,sha256=vZkw1F7oHOKReNkdHRYjhuNdA1nNdvSD4wbDmf0bnsM,4102
|
137
|
+
unfold/templates/unfold/helpers/attrs.html,sha256=Mwpj72kuwYj8hOT3J2T8qx6f1r_4xwwaS1slHA-82jI,166
|
134
138
|
unfold/templates/unfold/helpers/boolean.html,sha256=p_WOlytoXvDwta76WgcV4JSWKpBgKf4amhqmHF798F8,564
|
135
139
|
unfold/templates/unfold/helpers/breadcrumb_item.html,sha256=k_1j57UV0WtzFFlMKaewj4NLbR_DhXI6RzCHThblZLw,234
|
136
|
-
unfold/templates/unfold/helpers/display_header.html,sha256=
|
140
|
+
unfold/templates/unfold/helpers/display_header.html,sha256=HiuaIJ6Y8DSM99OFypLO_2uQadZIw7tSg1aJJpFEXkM,1161
|
137
141
|
unfold/templates/unfold/helpers/display_label.html,sha256=LS9DWzYjHkYLV27sZDwyXlg2sLJ0AlId9FbjnXpsbfg,317
|
138
142
|
unfold/templates/unfold/helpers/field.html,sha256=Ds-zUHkdyxamfUCVNhxvtM0XoJg9OCA0QcsLbLWv4oo,882
|
139
|
-
unfold/templates/unfold/helpers/field_readonly.html,sha256=
|
143
|
+
unfold/templates/unfold/helpers/field_readonly.html,sha256=O0gHEW46OWt1oUUk0lZiyR-mztWv_7IH6GpKRm2wUw4,235
|
144
|
+
unfold/templates/unfold/helpers/field_readonly_value.html,sha256=wdzHVKaI96S-S1MaV-odKXDdT_MRkfWMcXdUBhRlTDY,490
|
145
|
+
unfold/templates/unfold/helpers/fieldset_row.html,sha256=pcVBMudv1QDZp840S9a_8BuFb1DzUWBmsPUZI7lfgDo,2751
|
140
146
|
unfold/templates/unfold/helpers/fieldsets_tabs.html,sha256=V3bgW75eozaBDty-xfciGafhCWq_Ba5HfQkk92yRc9A,1445
|
141
147
|
unfold/templates/unfold/helpers/form_errors.html,sha256=EwerIJptSCWXvtAJ1IZKfEn98qlShBIGavsTThbklAs,266
|
142
148
|
unfold/templates/unfold/helpers/form_label.html,sha256=SR4U6iK9w4oels6iGY_Da-yN4BbXQVN9zCDlBGGXcw8,310
|
@@ -152,7 +158,7 @@ unfold/templates/unfold/helpers/navigation.html,sha256=Nmgk6_690AeHgKE0-YS7tQYq2
|
|
152
158
|
unfold/templates/unfold/helpers/pagination_current_item.html,sha256=4cZ2KLVcP0Y7xuGyXgexDQ07r94cgM5Gnmtv11dkRPQ,69
|
153
159
|
unfold/templates/unfold/helpers/pagination_ellipsis.html,sha256=Ar9Ntf2I_79mIVW5Hadwkn1Kx1Lj3d8eIXNXI1IIBfg,56
|
154
160
|
unfold/templates/unfold/helpers/search.html,sha256=T3JLlzEeHTEpX6qfjNQ0cQPW2rtVIOyE9quEyVHVXsA,1382
|
155
|
-
unfold/templates/unfold/helpers/search_results.html,sha256=
|
161
|
+
unfold/templates/unfold/helpers/search_results.html,sha256=N5CNMJSF91jbJS6OQP0nANlnwVeS9UmnuOJ8QLitl2c,774
|
156
162
|
unfold/templates/unfold/helpers/site_icon.html,sha256=RO-R5yRt6yOx41Z8dpDP4lzwMXFdz8Dp5j8vGUtHzh0,789
|
157
163
|
unfold/templates/unfold/helpers/site_logo.html,sha256=05tqXzHy--pluceRQ2jDUZCFX9DjPHdBqGaieUv9sXk,424
|
158
164
|
unfold/templates/unfold/helpers/submit.html,sha256=oSzq85LRLhdOlbFtFZFhYm6ucT95u6LunTeSTDClszQ,206
|
@@ -161,12 +167,13 @@ unfold/templates/unfold/helpers/tab_list.html,sha256=WMFSx0HEvylI_AOIPtFuWff1ePJ
|
|
161
167
|
unfold/templates/unfold/helpers/theme_switch.html,sha256=skkl6fYUnYLM7fAPivHxWjnOnuQSKa-7aDxh7I8dGIg,2266
|
162
168
|
unfold/templates/unfold/helpers/userlinks.html,sha256=qWjtBt9Q_tU8a874ii0Qqg8t_d-SSYBTB_3QZfNlx9g,634
|
163
169
|
unfold/templates/unfold/helpers/welcomemsg.html,sha256=noRysgSENef4_53pXaTiBCy2or6lQm1ZtmCQVODAB1c,1120
|
164
|
-
unfold/templates/unfold/layouts/
|
170
|
+
unfold/templates/unfold/layouts/base.html,sha256=bAXZDbyiyxNiE-49mqr7pHUFhC2mHZQzIDUY-js_yZ0,379
|
171
|
+
unfold/templates/unfold/layouts/base_simple.html,sha256=T37-ncKZBLNnOlsUQq90rGl3sLdRjFaxWCPG399pkbE,1179
|
165
172
|
unfold/templates/unfold/layouts/skeleton.html,sha256=iXrUiggVp36vmBIia5p32c-9Ruy0PxkFFQogDpvENbk,3419
|
166
|
-
unfold/templates/unfold/widgets/clearable_file_input.html,sha256=
|
167
|
-
unfold/templates/unfold/widgets/clearable_file_input_small.html,sha256=
|
173
|
+
unfold/templates/unfold/widgets/clearable_file_input.html,sha256=gAJsfyCnanOyeN4ypp1y7r76znvITV7FYTyWvPsmlDs,1882
|
174
|
+
unfold/templates/unfold/widgets/clearable_file_input_small.html,sha256=LPgiKuMAYPiEupFxdlwc1TozNRV1w9ekdRuv-5P4fgw,2531
|
168
175
|
unfold/templates/unfold/widgets/date.html,sha256=WXo2LG1v_gBZBSg-zocj7oujMKI0MWLYCIFfB04HMLQ,122
|
169
|
-
unfold/templates/unfold/widgets/foreign_key_raw_id.html,sha256=
|
176
|
+
unfold/templates/unfold/widgets/foreign_key_raw_id.html,sha256=BgXi4ziywtkWmZuUye5bbJ6yeEoHDJB_2lkwXX8hc6c,1026
|
170
177
|
unfold/templates/unfold/widgets/radio.html,sha256=3WcmclQNg7R_pRjEHL1dHkGjAzWlWNYnhHkAirC4nuA,646
|
171
178
|
unfold/templates/unfold/widgets/radio_option.html,sha256=IZgPx-aWKJuxrSalJ3K50RFd1vwSpb9Qk0yZwfV78_A,368
|
172
179
|
unfold/templates/unfold/widgets/range.html,sha256=28FBtSUgUcG82vpk_I27Lbs5oWZOV_oMzVhx4wj3-Ik,262
|
@@ -181,10 +188,10 @@ unfold/templatetags/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSu
|
|
181
188
|
unfold/templatetags/unfold.py,sha256=HFe0GrTD4va0lLzsZhxqjEOONmehqOWdf5vulkxgfGU,6302
|
182
189
|
unfold/templatetags/unfold_list.py,sha256=5xAjQX0_JnVwDaj-wGkGqbjOAtp-a18koWIKj5VfBz0,13867
|
183
190
|
unfold/typing.py,sha256=1P8PWM2oeaceUJtA5j071RbKEBpHYaux441u7Hd6wv4,643
|
184
|
-
unfold/utils.py,sha256=
|
185
|
-
unfold/views.py,sha256=
|
186
|
-
unfold/widgets.py,sha256=
|
187
|
-
django_unfold-0.
|
188
|
-
django_unfold-0.
|
189
|
-
django_unfold-0.
|
190
|
-
django_unfold-0.
|
191
|
+
unfold/utils.py,sha256=zZdJE4FmwRd7p5a7sJiAoZjBOJitXJduOq7BulyppWM,4803
|
192
|
+
unfold/views.py,sha256=hQCyeeMa9kcJV1IZeeYqj8PGW7J4QWME8n-5n0UGmiU,1003
|
193
|
+
unfold/widgets.py,sha256=w9BnJ6LYaFJJG7eJGJ-NKwqSHh6iUExN4P94WRvZO_c,15575
|
194
|
+
django_unfold-0.28.0.dist-info/LICENSE.md,sha256=Ltk_quRyyvV3J5v3brtOqmibeZSw2Hrb8bY1W3ya0Ik,1077
|
195
|
+
django_unfold-0.28.0.dist-info/METADATA,sha256=_i3r1OdWajR9BKmzMzYuPv91IyuKVKMvSTu6ufCwl8E,50658
|
196
|
+
django_unfold-0.28.0.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
|
197
|
+
django_unfold-0.28.0.dist-info/RECORD,,
|
unfold/admin.py
CHANGED
@@ -7,50 +7,30 @@ from django.contrib.admin import ModelAdmin as BaseModelAdmin
|
|
7
7
|
from django.contrib.admin import StackedInline as BaseStackedInline
|
8
8
|
from django.contrib.admin import TabularInline as BaseTabularInline
|
9
9
|
from django.contrib.admin import display, helpers
|
10
|
-
from django.contrib.admin.utils import lookup_field
|
11
10
|
from django.contrib.admin.widgets import RelatedFieldWidgetWrapper
|
12
|
-
from django.core.exceptions import ObjectDoesNotExist
|
13
11
|
from django.db import models
|
14
|
-
from django.db.models import
|
15
|
-
BLANK_CHOICE_DASH,
|
16
|
-
ForeignObjectRel,
|
17
|
-
JSONField,
|
18
|
-
ManyToManyRel,
|
19
|
-
Model,
|
20
|
-
OneToOneField,
|
21
|
-
)
|
12
|
+
from django.db.models import BLANK_CHOICE_DASH, Model
|
22
13
|
from django.db.models.fields import Field
|
23
14
|
from django.db.models.fields.related import ForeignKey, ManyToManyField
|
24
15
|
from django.forms import Form
|
25
16
|
from django.forms.fields import TypedChoiceField
|
26
|
-
from django.forms.models import
|
27
|
-
ModelChoiceField,
|
28
|
-
ModelMultipleChoiceField,
|
29
|
-
)
|
30
|
-
from django.forms.utils import flatatt
|
17
|
+
from django.forms.models import ModelChoiceField, ModelMultipleChoiceField
|
31
18
|
from django.forms.widgets import SelectMultiple
|
32
19
|
from django.http import HttpRequest, HttpResponse
|
33
20
|
from django.shortcuts import redirect
|
34
|
-
from django.template.defaultfilters import linebreaksbr
|
35
21
|
from django.template.response import TemplateResponse
|
36
22
|
from django.urls import URLPattern, path, reverse
|
37
|
-
from django.utils.
|
38
|
-
from django.utils.module_loading import import_string
|
39
|
-
from django.utils.safestring import SafeText, mark_safe
|
40
|
-
from django.utils.text import capfirst
|
23
|
+
from django.utils.safestring import mark_safe
|
41
24
|
from django.utils.translation import gettext_lazy as _
|
42
25
|
from django.views import View
|
43
26
|
|
44
27
|
from .checks import UnfoldModelAdminChecks
|
45
28
|
from .dataclasses import UnfoldAction
|
46
29
|
from .exceptions import UnfoldException
|
30
|
+
from .fields import UnfoldAdminField, UnfoldAdminReadonlyField
|
47
31
|
from .forms import ActionForm
|
48
|
-
from .settings import get_config
|
49
32
|
from .typing import FieldsetsType
|
50
|
-
from .utils import display_for_field
|
51
33
|
from .widgets import (
|
52
|
-
CHECKBOX_LABEL_CLASSES,
|
53
|
-
LABEL_CLASSES,
|
54
34
|
SELECT_CLASSES,
|
55
35
|
UnfoldAdminBigIntegerFieldWidget,
|
56
36
|
UnfoldAdminDecimalFieldWidget,
|
@@ -90,8 +70,6 @@ try:
|
|
90
70
|
except ImportError:
|
91
71
|
HAS_MONEY = False
|
92
72
|
|
93
|
-
checkbox = UnfoldBooleanWidget({"class": "action-select"}, lambda value: False)
|
94
|
-
|
95
73
|
FORMFIELD_OVERRIDES = {
|
96
74
|
models.DateTimeField: {
|
97
75
|
"form_class": forms.SplitDateTimeField,
|
@@ -141,140 +119,10 @@ FORMFIELD_OVERRIDES_INLINE.update(
|
|
141
119
|
}
|
142
120
|
)
|
143
121
|
|
144
|
-
|
145
|
-
class UnfoldAdminField(helpers.AdminField):
|
146
|
-
def label_tag(self) -> SafeText:
|
147
|
-
classes = []
|
148
|
-
if not self.field.field.widget.__class__.__name__.startswith(
|
149
|
-
"Unfold"
|
150
|
-
) and not self.field.field.widget.template_name.startswith("unfold"):
|
151
|
-
return super().label_tag()
|
152
|
-
|
153
|
-
# TODO load config from current AdminSite (override Fieldline.__iter__ method)
|
154
|
-
for lang, flag in get_config()["EXTENSIONS"]["modeltranslation"][
|
155
|
-
"flags"
|
156
|
-
].items():
|
157
|
-
if f"[{lang}]" in self.field.label:
|
158
|
-
self.field.label = self.field.label.replace(f"[{lang}]", flag)
|
159
|
-
break
|
160
|
-
|
161
|
-
contents = conditional_escape(self.field.label)
|
162
|
-
|
163
|
-
if self.is_checkbox:
|
164
|
-
classes.append(" ".join(CHECKBOX_LABEL_CLASSES))
|
165
|
-
else:
|
166
|
-
classes.append(" ".join(LABEL_CLASSES))
|
167
|
-
|
168
|
-
if self.field.field.required:
|
169
|
-
classes.append("required")
|
170
|
-
|
171
|
-
attrs = {"class": " ".join(classes)} if classes else {}
|
172
|
-
required = mark_safe(' <span class="text-red-600">*</span>')
|
173
|
-
|
174
|
-
return self.field.label_tag(
|
175
|
-
contents=mark_safe(contents),
|
176
|
-
attrs=attrs,
|
177
|
-
label_suffix=required if self.field.field.required else "",
|
178
|
-
)
|
179
|
-
|
122
|
+
checkbox = UnfoldBooleanWidget({"class": "action-select"}, lambda value: False)
|
180
123
|
|
181
124
|
helpers.AdminField = UnfoldAdminField
|
182
125
|
|
183
|
-
|
184
|
-
class UnfoldAdminReadonlyField(helpers.AdminReadonlyField):
|
185
|
-
def label_tag(self) -> SafeText:
|
186
|
-
if not isinstance(self.model_admin, ModelAdmin) and not isinstance(
|
187
|
-
self.model_admin, ModelAdminMixin
|
188
|
-
):
|
189
|
-
return super().label_tag()
|
190
|
-
|
191
|
-
attrs = {
|
192
|
-
"class": " ".join(LABEL_CLASSES + ["mb-2"]),
|
193
|
-
}
|
194
|
-
|
195
|
-
label = self.field["label"]
|
196
|
-
|
197
|
-
return format_html(
|
198
|
-
"<label{}>{}{}</label>",
|
199
|
-
flatatt(attrs),
|
200
|
-
capfirst(label),
|
201
|
-
self.form.label_suffix,
|
202
|
-
)
|
203
|
-
|
204
|
-
def is_json(self) -> bool:
|
205
|
-
field, obj, model_admin = (
|
206
|
-
self.field["field"],
|
207
|
-
self.form.instance,
|
208
|
-
self.model_admin,
|
209
|
-
)
|
210
|
-
|
211
|
-
try:
|
212
|
-
f, attr, value = lookup_field(field, obj, model_admin)
|
213
|
-
except (AttributeError, ValueError, ObjectDoesNotExist):
|
214
|
-
return False
|
215
|
-
|
216
|
-
return isinstance(f, JSONField)
|
217
|
-
|
218
|
-
def contents(self) -> str:
|
219
|
-
contents = self._get_contents()
|
220
|
-
contents = self._preprocess_field(contents)
|
221
|
-
return contents
|
222
|
-
|
223
|
-
def _get_contents(self) -> str:
|
224
|
-
from django.contrib.admin.templatetags.admin_list import _boolean_icon
|
225
|
-
|
226
|
-
field, obj, model_admin = (
|
227
|
-
self.field["field"],
|
228
|
-
self.form.instance,
|
229
|
-
self.model_admin,
|
230
|
-
)
|
231
|
-
try:
|
232
|
-
f, attr, value = lookup_field(field, obj, model_admin)
|
233
|
-
except (AttributeError, ValueError, ObjectDoesNotExist):
|
234
|
-
result_repr = self.empty_value_display
|
235
|
-
else:
|
236
|
-
if field in self.form.fields:
|
237
|
-
widget = self.form[field].field.widget
|
238
|
-
# This isn't elegant but suffices for contrib.auth's
|
239
|
-
# ReadOnlyPasswordHashWidget.
|
240
|
-
if getattr(widget, "read_only", False):
|
241
|
-
return widget.render(field, value)
|
242
|
-
if f is None:
|
243
|
-
if getattr(attr, "boolean", False):
|
244
|
-
result_repr = _boolean_icon(value)
|
245
|
-
else:
|
246
|
-
if hasattr(value, "__html__"):
|
247
|
-
result_repr = value
|
248
|
-
else:
|
249
|
-
result_repr = linebreaksbr(value)
|
250
|
-
else:
|
251
|
-
if isinstance(f.remote_field, ManyToManyRel) and value is not None:
|
252
|
-
result_repr = ", ".join(map(str, value.all()))
|
253
|
-
elif (
|
254
|
-
isinstance(f.remote_field, (ForeignObjectRel, OneToOneField))
|
255
|
-
and value is not None
|
256
|
-
):
|
257
|
-
result_repr = self.get_admin_url(f.remote_field, value)
|
258
|
-
else:
|
259
|
-
result_repr = display_for_field(value, f, self.empty_value_display)
|
260
|
-
return conditional_escape(result_repr)
|
261
|
-
result_repr = linebreaksbr(result_repr)
|
262
|
-
return conditional_escape(result_repr)
|
263
|
-
|
264
|
-
def _preprocess_field(self, contents: str) -> str:
|
265
|
-
if (
|
266
|
-
hasattr(self.model_admin, "readonly_preprocess_fields")
|
267
|
-
and self.field["field"] in self.model_admin.readonly_preprocess_fields
|
268
|
-
):
|
269
|
-
func = self.model_admin.readonly_preprocess_fields[self.field["field"]]
|
270
|
-
if isinstance(func, str):
|
271
|
-
contents = import_string(func)(contents)
|
272
|
-
elif callable(func):
|
273
|
-
contents = func(contents)
|
274
|
-
|
275
|
-
return contents
|
276
|
-
|
277
|
-
|
278
126
|
helpers.AdminReadonlyField = UnfoldAdminReadonlyField
|
279
127
|
|
280
128
|
|
@@ -616,6 +464,7 @@ class ModelAdmin(ModelAdminMixin, BaseModelAdmin):
|
|
616
464
|
method=method,
|
617
465
|
description=self._get_action_description(method, action),
|
618
466
|
path=self._get_action_url(method, action),
|
467
|
+
attrs=method.attrs if hasattr(method, "attrs") else None,
|
619
468
|
)
|
620
469
|
|
621
470
|
@staticmethod
|
@@ -0,0 +1,31 @@
|
|
1
|
+
{% load i18n %}
|
2
|
+
|
3
|
+
<div class="flex flex-col gap-4" x-data="{items: []}">
|
4
|
+
{% for subwidget in widget.subwidgets %}
|
5
|
+
<div class="flex flex-row">
|
6
|
+
{% with widget=subwidget %}
|
7
|
+
{% include widget.template_name %}
|
8
|
+
{% endwith %}
|
9
|
+
|
10
|
+
<a x-on:click="$el.parentElement.remove()" class="bg-white border cursor-pointer flex items-center h-9.5 justify-center ml-2 rounded shadow-sm shrink-0 text-red-600 text-sm w-9.5 dark:bg-gray-900 dark:border-gray-700 dark:text-red-500">
|
11
|
+
<span class="material-symbols-outlined text-sm">delete</span>
|
12
|
+
</a>
|
13
|
+
</div>
|
14
|
+
{% endfor %}
|
15
|
+
|
16
|
+
<template x-for="(item, index) in items" :key="item.key">
|
17
|
+
<div class="flex flex-row">
|
18
|
+
{% include template.template_name with widget=template %}
|
19
|
+
|
20
|
+
<a x-on:click="items.splice(index, 1)" class="bg-white border cursor-pointer flex items-center h-9.5 justify-center ml-2 rounded shadow-sm shrink-0 text-red-600 text-sm w-9.5 dark:bg-gray-900 dark:border-gray-700 dark:text-red-500">
|
21
|
+
<span class="material-symbols-outlined text-sm">delete</span>
|
22
|
+
</a>
|
23
|
+
</div>
|
24
|
+
</template>
|
25
|
+
|
26
|
+
<div class="flex flex-row">
|
27
|
+
<div x-on:click="items.push({ key: new Date().getTime()})" class="bg-primary-600 border border-transparent cursor-pointer font-medium inline-block px-3 py-2 rounded-md text-sm text-white w-full lg:w-auto">
|
28
|
+
{% trans "Add new item" %}
|
29
|
+
</div>
|
30
|
+
</div>
|
31
|
+
</div>
|
unfold/contrib/forms/widgets.py
CHANGED
@@ -1,7 +1,10 @@
|
|
1
|
-
from typing import Any, Dict, Optional
|
1
|
+
from typing import Any, Dict, List, Optional, Union
|
2
2
|
|
3
|
-
from django.
|
4
|
-
from
|
3
|
+
from django.core.validators import EMPTY_VALUES
|
4
|
+
from django.forms import MultiWidget, Widget
|
5
|
+
from django.http import QueryDict
|
6
|
+
from django.utils.datastructures import MultiValueDict
|
7
|
+
from unfold.widgets import PROSE_CLASSES, UnfoldAdminTextInputWidget
|
5
8
|
|
6
9
|
WYSIWYG_CLASSES = [
|
7
10
|
*PROSE_CLASSES,
|
@@ -22,6 +25,58 @@ WYSIWYG_CLASSES = [
|
|
22
25
|
]
|
23
26
|
|
24
27
|
|
28
|
+
class ArrayWidget(MultiWidget):
|
29
|
+
template_name = "unfold/forms/array.html"
|
30
|
+
widget_class = UnfoldAdminTextInputWidget
|
31
|
+
|
32
|
+
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
33
|
+
widgets = [self.widget_class]
|
34
|
+
super().__init__(widgets)
|
35
|
+
|
36
|
+
def get_context(self, name: str, value: str, attrs: Dict) -> Dict:
|
37
|
+
self._resolve_widgets(value)
|
38
|
+
context = super().get_context(name, value, attrs)
|
39
|
+
template_widget = UnfoldAdminTextInputWidget()
|
40
|
+
template_widget.name = name
|
41
|
+
|
42
|
+
context.update({"template": template_widget})
|
43
|
+
return context
|
44
|
+
|
45
|
+
def value_from_datadict(
|
46
|
+
self, data: QueryDict, files: MultiValueDict, name: str
|
47
|
+
) -> List:
|
48
|
+
values = []
|
49
|
+
|
50
|
+
for item in data.getlist(name):
|
51
|
+
if item not in EMPTY_VALUES:
|
52
|
+
values.append(item)
|
53
|
+
|
54
|
+
return values
|
55
|
+
|
56
|
+
def value_omitted_from_data(
|
57
|
+
self, data: QueryDict, files: MultiValueDict, name: str
|
58
|
+
) -> List:
|
59
|
+
return data.getlist(name) not in [[""], *EMPTY_VALUES]
|
60
|
+
|
61
|
+
def decompress(self, value: Union[str, List]) -> List:
|
62
|
+
if isinstance(value, List):
|
63
|
+
return value.split(",")
|
64
|
+
|
65
|
+
return []
|
66
|
+
|
67
|
+
def _resolve_widgets(self, value: Optional[Union[List, str]]) -> None:
|
68
|
+
if value is None:
|
69
|
+
value = []
|
70
|
+
|
71
|
+
elif isinstance(value, List):
|
72
|
+
self.widgets = [self.widget_class for item in value]
|
73
|
+
else:
|
74
|
+
self.widgets = [self.widget_class for item in value.split(",")]
|
75
|
+
|
76
|
+
self.widgets_names = ["" for i in range(len(self.widgets))]
|
77
|
+
self.widgets = [w() if isinstance(w, type) else w for w in self.widgets]
|
78
|
+
|
79
|
+
|
25
80
|
class WysiwygWidget(Widget):
|
26
81
|
template_name = "unfold/forms/wysiwyg.html"
|
27
82
|
|
@@ -15,23 +15,37 @@ class ImportForm(BaseImportForm):
|
|
15
15
|
def __init__(self, *args, **kwargs):
|
16
16
|
super().__init__(*args, **kwargs)
|
17
17
|
|
18
|
-
self.fields["resource"].widget.attrs["class"] = " ".join(
|
19
|
-
|
20
|
-
|
18
|
+
self.fields["resource"].widget.attrs["class"] = " ".join(
|
19
|
+
[self.fields["resource"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
20
|
+
)
|
21
|
+
self.fields["import_file"].widget = UnfoldAdminFileFieldWidget(
|
22
|
+
attrs=self.fields["import_file"].widget.attrs
|
23
|
+
)
|
24
|
+
self.fields["format"].widget.attrs["class"] = " ".join(
|
25
|
+
[self.fields["format"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
26
|
+
)
|
21
27
|
|
22
28
|
|
23
29
|
class ExportForm(BaseExportForm):
|
24
30
|
def __init__(self, *args, **kwargs):
|
25
31
|
super().__init__(*args, **kwargs)
|
26
|
-
self.fields["resource"].widget.attrs["class"] = " ".join(
|
27
|
-
|
32
|
+
self.fields["resource"].widget.attrs["class"] = " ".join(
|
33
|
+
[self.fields["resource"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
34
|
+
)
|
35
|
+
self.fields["format"].widget.attrs["class"] = " ".join(
|
36
|
+
[self.fields["format"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
37
|
+
)
|
28
38
|
|
29
39
|
|
30
40
|
class SelectableFieldsExportForm(BaseSelectableFieldsExportForm):
|
31
41
|
def __init__(self, formats, resources, **kwargs):
|
32
42
|
super().__init__(formats, resources, **kwargs)
|
33
|
-
self.fields["resource"].widget.attrs["class"] = " ".join(
|
34
|
-
|
43
|
+
self.fields["resource"].widget.attrs["class"] = " ".join(
|
44
|
+
[self.fields["resource"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
45
|
+
)
|
46
|
+
self.fields["format"].widget.attrs["class"] = " ".join(
|
47
|
+
[self.fields["format"].widget.attrs.get("class", ""), *SELECT_CLASSES]
|
48
|
+
)
|
35
49
|
|
36
50
|
for _key, field in self.fields.items():
|
37
51
|
if isinstance(field, BooleanField):
|