django-unfold 0.26.0__tar.gz → 0.27.0__tar.gz

Sign up to get free protection for your applications and to get access to all the features.
Files changed (198) hide show
  1. {django_unfold-0.26.0 → django_unfold-0.27.0}/PKG-INFO +11 -2
  2. {django_unfold-0.26.0 → django_unfold-0.27.0}/README.md +10 -1
  3. {django_unfold-0.26.0 → django_unfold-0.27.0}/pyproject.toml +1 -1
  4. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/admin.py +5 -157
  5. django_unfold-0.27.0/src/unfold/contrib/forms/templates/unfold/forms/array.html +31 -0
  6. django_unfold-0.27.0/src/unfold/contrib/forms/widgets.py +97 -0
  7. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/decorators.py +3 -0
  8. django_unfold-0.27.0/src/unfold/fields.py +200 -0
  9. django_unfold-0.27.0/src/unfold/static/unfold/css/styles.css +1 -0
  10. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/change_form.html +0 -2
  11. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/edit_inline/tabular.html +4 -6
  12. django_unfold-0.27.0/src/unfold/templates/admin/includes/fieldset.html +21 -0
  13. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/display_header.html +1 -1
  14. django_unfold-0.27.0/src/unfold/templates/unfold/helpers/field_readonly.html +7 -0
  15. django_unfold-0.27.0/src/unfold/templates/unfold/helpers/field_readonly_value.html +1 -0
  16. django_unfold-0.27.0/src/unfold/templates/unfold/helpers/fieldset_row.html +53 -0
  17. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/clearable_file_input.html +1 -1
  18. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/foreign_key_raw_id.html +7 -13
  19. django_unfold-0.26.0/src/unfold/contrib/forms/widgets.py +0 -42
  20. django_unfold-0.26.0/src/unfold/static/unfold/css/styles.css +0 -1
  21. django_unfold-0.26.0/src/unfold/templates/admin/includes/fieldset.html +0 -51
  22. django_unfold-0.26.0/src/unfold/templates/unfold/helpers/field_readonly.html +0 -9
  23. {django_unfold-0.26.0 → django_unfold-0.27.0}/LICENSE.md +0 -0
  24. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/__init__.py +0 -0
  25. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/apps.py +0 -0
  26. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/checks.py +0 -0
  27. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/__init__.py +0 -0
  28. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/__init__.py +0 -0
  29. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/admin.py +0 -0
  30. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/apps.py +0 -0
  31. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/forms.py +0 -0
  32. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/static/unfold/filters/css/nouislider.min.css +0 -0
  33. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/static/unfold/filters/js/DateTimeShortcuts.js +0 -0
  34. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/static/unfold/filters/js/admin-numeric-filter.js +0 -0
  35. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/static/unfold/filters/js/nouislider.min.js +0 -0
  36. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/static/unfold/filters/js/wNumb.min.js +0 -0
  37. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_date_range.html +0 -0
  38. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_datetime_range.html +0 -0
  39. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_field.html +0 -0
  40. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_numeric_range.html +0 -0
  41. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_numeric_single.html +0 -0
  42. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/filters/templates/unfold/filters/filters_numeric_slider.html +0 -0
  43. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/__init__.py +0 -0
  44. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/apps.py +0 -0
  45. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/static/unfold/forms/css/trix.css +0 -0
  46. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/static/unfold/forms/js/trix.config.js +0 -0
  47. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/static/unfold/forms/js/trix.js +0 -0
  48. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/templates/unfold/forms/helpers/toolbar.html +0 -0
  49. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/forms/templates/unfold/forms/wysiwyg.html +0 -0
  50. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/__init__.py +0 -0
  51. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/apps.py +0 -0
  52. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/admin/guardian/model/change_form.html +0 -0
  53. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/admin/guardian/model/field.html +0 -0
  54. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/admin/guardian/model/obj_perms_manage.html +0 -0
  55. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/admin/guardian/model/obj_perms_manage_group.html +0 -0
  56. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/admin/guardian/model/obj_perms_manage_user.html +0 -0
  57. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/unfold/guardian/group_form.html +0 -0
  58. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/guardian/templates/unfold/guardian/user_form.html +0 -0
  59. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/__init__.py +0 -0
  60. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/apps.py +0 -0
  61. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/forms.py +0 -0
  62. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/base.html +0 -0
  63. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/change_form.html +0 -0
  64. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/change_list_export_item.html +0 -0
  65. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/change_list_import_export.html +0 -0
  66. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/change_list_import_item.html +0 -0
  67. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/export.html +0 -0
  68. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import.html +0 -0
  69. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import_confirm.html +0 -0
  70. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import_errors.html +0 -0
  71. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import_form.html +0 -0
  72. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import_preview.html +0 -0
  73. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/import_validation.html +0 -0
  74. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/import_export/templates/admin/import_export/resource_fields_list.html +0 -0
  75. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/inlines/__init__.py +0 -0
  76. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/inlines/admin.py +0 -0
  77. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/inlines/apps.py +0 -0
  78. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/inlines/checks.py +0 -0
  79. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/inlines/forms.py +0 -0
  80. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/__init__.py +0 -0
  81. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/apps.py +0 -0
  82. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/templates/simple_history/_object_history_list.html +0 -0
  83. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/templates/simple_history/object_history.html +0 -0
  84. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/templates/simple_history/object_history_form.html +0 -0
  85. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/contrib/simple_history/templates/simple_history/submit_line.html +0 -0
  86. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/dataclasses.py +0 -0
  87. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/exceptions.py +0 -0
  88. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/forms.py +0 -0
  89. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/settings.py +0 -0
  90. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/sites.py +0 -0
  91. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/css/simplebar.css +0 -0
  92. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/inter/Inter-Bold.woff2 +0 -0
  93. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/inter/Inter-Medium.woff2 +0 -0
  94. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/inter/Inter-Regular.woff2 +0 -0
  95. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/inter/Inter-SemiBold.woff2 +0 -0
  96. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/inter/styles.css +0 -0
  97. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/material-symbols/Material-Symbols-Outlined.woff2 +0 -0
  98. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/fonts/material-symbols/styles.css +0 -0
  99. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/alpine.js +0 -0
  100. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/alpine.persist.js +0 -0
  101. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/app.js +0 -0
  102. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/chart.js +0 -0
  103. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/htmx.js +0 -0
  104. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/static/unfold/js/simplebar.js +0 -0
  105. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/styles.css +0 -0
  106. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/actions.html +0 -0
  107. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/app_index.html +0 -0
  108. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/app_list.html +0 -0
  109. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/auth/user/add_form.html +0 -0
  110. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/auth/user/change_password.html +0 -0
  111. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/base.html +0 -0
  112. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/base_site.html +0 -0
  113. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/change_form_object_tools.html +0 -0
  114. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/change_list.html +0 -0
  115. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/change_list_object_tools.html +0 -0
  116. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/change_list_results.html +0 -0
  117. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/date_hierarchy.html +0 -0
  118. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/delete_confirmation.html +0 -0
  119. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/delete_selected_confirmation.html +0 -0
  120. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/edit_inline/stacked.html +0 -0
  121. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/filter.html +0 -0
  122. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/includes/object_delete_summary.html +0 -0
  123. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/index.html +0 -0
  124. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/login.html +0 -0
  125. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/nav_sidebar.html +0 -0
  126. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/object_history.html +0 -0
  127. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/pagination.html +0 -0
  128. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/search_form.html +0 -0
  129. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/admin/submit_line.html +0 -0
  130. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/auth/widgets/read_only_password_hash.html +0 -0
  131. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/registration/logged_out.html +0 -0
  132. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/registration/password_change_done.html +0 -0
  133. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/registration/password_change_form.html +0 -0
  134. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/change_list_filter.html +0 -0
  135. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/card.html +0 -0
  136. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/chart/bar.html +0 -0
  137. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/chart/line.html +0 -0
  138. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/container.html +0 -0
  139. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/flex.html +0 -0
  140. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/navigation.html +0 -0
  141. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/progress.html +0 -0
  142. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/separator.html +0 -0
  143. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/text.html +0 -0
  144. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/components/title.html +0 -0
  145. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/account_links.html +0 -0
  146. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/actions_row.html +0 -0
  147. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/add_link.html +0 -0
  148. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/app_list.html +0 -0
  149. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/app_list_default.html +0 -0
  150. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/boolean.html +0 -0
  151. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/breadcrumb_item.html +0 -0
  152. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/display_label.html +0 -0
  153. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/field.html +0 -0
  154. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/fieldsets_tabs.html +0 -0
  155. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/form_errors.html +0 -0
  156. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/form_label.html +0 -0
  157. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/header.html +0 -0
  158. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/help_text.html +0 -0
  159. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/history.html +0 -0
  160. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/label.html +0 -0
  161. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/messages/error.html +0 -0
  162. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/messages/errornote.html +0 -0
  163. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/messages/info.html +0 -0
  164. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/messages.html +0 -0
  165. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/navigation.html +0 -0
  166. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/pagination_current_item.html +0 -0
  167. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/pagination_ellipsis.html +0 -0
  168. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/search.html +0 -0
  169. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/search_results.html +0 -0
  170. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/site_icon.html +0 -0
  171. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/site_logo.html +0 -0
  172. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/submit.html +0 -0
  173. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/tab_action.html +0 -0
  174. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/tab_list.html +0 -0
  175. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/theme_switch.html +0 -0
  176. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/userlinks.html +0 -0
  177. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/helpers/welcomemsg.html +0 -0
  178. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/layouts/base_simple.html +0 -0
  179. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/layouts/skeleton.html +0 -0
  180. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/clearable_file_input_small.html +0 -0
  181. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/date.html +0 -0
  182. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/radio.html +0 -0
  183. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/radio_option.html +0 -0
  184. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/range.html +0 -0
  185. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/related_widget_wrapper.html +0 -0
  186. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/split_datetime.html +0 -0
  187. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/split_datetime_vertical.html +0 -0
  188. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/split_money.html +0 -0
  189. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/textarea.html +0 -0
  190. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/textarea_expandable.html +0 -0
  191. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templates/unfold/widgets/time.html +0 -0
  192. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templatetags/__init__.py +0 -0
  193. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templatetags/unfold.py +0 -0
  194. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/templatetags/unfold_list.py +0 -0
  195. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/typing.py +0 -0
  196. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/utils.py +0 -0
  197. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/views.py +0 -0
  198. {django_unfold-0.26.0 → django_unfold-0.27.0}/src/unfold/widgets.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: django-unfold
3
- Version: 0.26.0
3
+ Version: 0.27.0
4
4
  Summary: Modern Django admin theme for seamless interface development
5
5
  Home-page: https://unfoldadmin.com
6
6
  License: MIT
@@ -52,11 +52,13 @@ 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
@@ -321,13 +323,17 @@ def permission_callback(request):
321
323
 
322
324
  from django import models
323
325
  from django.contrib import admin
326
+ from django.contrib.postgres.fields import ArrayField
324
327
  from django.db import models
325
328
  from unfold.admin import ModelAdmin
326
- from unfold.contrib.forms.widgets import WysiwygWidget
329
+ from unfold.contrib.forms.widgets import ArrayWidget, WysiwygWidget
327
330
 
328
331
 
329
332
  @admin.register(MyModel)
330
333
  class CustomAdminClass(ModelAdmin):
334
+ # Display fields in changeform in compressed mode
335
+ compressed_fields = True # Default: False
336
+
331
337
  # Preprocess content of readonly fields before render
332
338
  readonly_preprocess_fields = {
333
339
  "model_field_name": "html.unescape",
@@ -346,6 +352,9 @@ class CustomAdminClass(ModelAdmin):
346
352
  formfield_overrides = {
347
353
  models.TextField: {
348
354
  "widget": WysiwygWidget,
355
+ },
356
+ ArrayField: {
357
+ "widget": ArrayWidget,
349
358
  }
350
359
  }
351
360
  ```
@@ -28,11 +28,13 @@ Did you decide to start using Unfold but you don't have time to make the switch
28
28
  - **Dependencies:** completely based only on `django.contrib.admin`
29
29
  - **Actions:** multiple ways how to define actions within different parts of admin
30
30
  - **WYSIWYG:** built-in support for WYSIWYG (Trix)
31
+ - **Array widget:** built-in widget for `django.contrib.postgres.fields.ArrayField`
31
32
  - **Filters:** custom dropdown, numeric, datetime, and text fields
32
33
  - **Dashboard:** custom components for rapid dashboard development
33
34
  - **Model tabs:** define custom tab navigations for models
34
35
  - **Fieldset tabs:** merge several fielsets into tabs in change form
35
36
  - **Colors:** possibility to override default color scheme
37
+ - **Changeform modes:** display fields in changeform in compressed mode
36
38
  - **Third party packages:** default support for multiple popular applications
37
39
  - **Environment label**: distinguish between environments by displaying a label
38
40
  - **Nonrelated inlines**: displays nonrelated model as inline in changeform
@@ -297,13 +299,17 @@ def permission_callback(request):
297
299
 
298
300
  from django import models
299
301
  from django.contrib import admin
302
+ from django.contrib.postgres.fields import ArrayField
300
303
  from django.db import models
301
304
  from unfold.admin import ModelAdmin
302
- from unfold.contrib.forms.widgets import WysiwygWidget
305
+ from unfold.contrib.forms.widgets import ArrayWidget, WysiwygWidget
303
306
 
304
307
 
305
308
  @admin.register(MyModel)
306
309
  class CustomAdminClass(ModelAdmin):
310
+ # Display fields in changeform in compressed mode
311
+ compressed_fields = True # Default: False
312
+
307
313
  # Preprocess content of readonly fields before render
308
314
  readonly_preprocess_fields = {
309
315
  "model_field_name": "html.unescape",
@@ -322,6 +328,9 @@ class CustomAdminClass(ModelAdmin):
322
328
  formfield_overrides = {
323
329
  models.TextField: {
324
330
  "widget": WysiwygWidget,
331
+ },
332
+ ArrayField: {
333
+ "widget": ArrayWidget,
325
334
  }
326
335
  }
327
336
  ```
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "django-unfold"
3
- version = "0.26.0"
3
+ version = "0.27.0"
4
4
  description = "Modern Django admin theme for seamless interface development"
5
5
  license = "MIT"
6
6
  readme = "README.md"
@@ -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.html import conditional_escape, format_html
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
 
@@ -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>
@@ -0,0 +1,97 @@
1
+ from typing import Any, Dict, List, Optional, Union
2
+
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
8
+
9
+ WYSIWYG_CLASSES = [
10
+ *PROSE_CLASSES,
11
+ "border",
12
+ "border-gray-200",
13
+ "border-t-0",
14
+ "group-[.errors]:border-red-600",
15
+ "max-w-none",
16
+ "p-4",
17
+ "rounded-b-md",
18
+ "rounded-t-none",
19
+ "text-gray-500",
20
+ "w-full",
21
+ "focus:outline-none",
22
+ "dark:border-gray-700",
23
+ "dark:text-gray-400",
24
+ "dark:group-[.errors]:border-red-500",
25
+ ]
26
+
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
+
80
+ class WysiwygWidget(Widget):
81
+ template_name = "unfold/forms/wysiwyg.html"
82
+
83
+ class Media:
84
+ css = {"all": ("unfold/forms/css/trix.css",)}
85
+ js = (
86
+ "unfold/forms/js/trix.js",
87
+ "unfold/forms/js/trix.config.js",
88
+ )
89
+
90
+ def __init__(self, attrs: Optional[Dict[str, Any]] = None) -> None:
91
+ super().__init__(attrs)
92
+
93
+ self.attrs.update(
94
+ {
95
+ "class": " ".join(WYSIWYG_CLASSES),
96
+ }
97
+ )
@@ -58,6 +58,7 @@ def display(
58
58
  function: Optional[Callable[[Model], Any]] = None,
59
59
  *,
60
60
  boolean: Optional[bool] = None,
61
+ image: Optional[bool] = None,
61
62
  ordering: Optional[Union[str, Combinable, BaseExpression]] = None,
62
63
  description: Optional[str] = None,
63
64
  empty_value: Optional[str] = None,
@@ -72,6 +73,8 @@ def display(
72
73
  )
73
74
  if boolean is not None:
74
75
  func.boolean = boolean
76
+ if image is not None:
77
+ func.image = image
75
78
  if ordering is not None:
76
79
  func.admin_order_field = ordering
77
80
  if description is not None:
@@ -0,0 +1,200 @@
1
+ from django.contrib.admin import helpers
2
+ from django.contrib.admin.utils import lookup_field, quote
3
+ from django.core.exceptions import ObjectDoesNotExist
4
+ from django.db import models
5
+ from django.db.models import (
6
+ ForeignObjectRel,
7
+ ImageField,
8
+ JSONField,
9
+ ManyToManyRel,
10
+ OneToOneField,
11
+ )
12
+ from django.forms.utils import flatatt
13
+ from django.template.defaultfilters import linebreaksbr
14
+ from django.urls import NoReverseMatch, reverse
15
+ from django.utils.html import conditional_escape, format_html
16
+ from django.utils.module_loading import import_string
17
+ from django.utils.safestring import SafeText, mark_safe
18
+ from django.utils.text import capfirst
19
+
20
+ from .settings import get_config
21
+ from .utils import display_for_field
22
+ from .widgets import CHECKBOX_LABEL_CLASSES, LABEL_CLASSES
23
+
24
+
25
+ class UnfoldAdminReadonlyField(helpers.AdminReadonlyField):
26
+ def label_tag(self) -> SafeText:
27
+ from .admin import ModelAdmin, ModelAdminMixin
28
+
29
+ if not isinstance(self.model_admin, ModelAdmin) and not isinstance(
30
+ self.model_admin, ModelAdminMixin
31
+ ):
32
+ return super().label_tag()
33
+
34
+ attrs = {
35
+ "class": " ".join(LABEL_CLASSES + ["mb-2"]),
36
+ }
37
+
38
+ label = self.field["label"]
39
+
40
+ return format_html(
41
+ "<label{}>{}{}</label>",
42
+ flatatt(attrs),
43
+ capfirst(label),
44
+ self.form.label_suffix,
45
+ )
46
+
47
+ def is_json(self) -> bool:
48
+ field, obj, model_admin = (
49
+ self.field["field"],
50
+ self.form.instance,
51
+ self.model_admin,
52
+ )
53
+
54
+ try:
55
+ f, attr, value = lookup_field(field, obj, model_admin)
56
+ except (AttributeError, ValueError, ObjectDoesNotExist):
57
+ return False
58
+
59
+ return isinstance(f, JSONField)
60
+
61
+ def is_image(self) -> bool:
62
+ field, obj, model_admin = (
63
+ self.field["field"],
64
+ self.form.instance,
65
+ self.model_admin,
66
+ )
67
+
68
+ try:
69
+ f, attr, value = lookup_field(field, obj, model_admin)
70
+ except (AttributeError, ValueError, ObjectDoesNotExist):
71
+ return False
72
+
73
+ if hasattr(attr, "image"):
74
+ return attr.image
75
+ elif (
76
+ isinstance(attr, property)
77
+ and hasattr(attr, "fget")
78
+ and hasattr(attr.fget, "image")
79
+ ):
80
+ return attr.fget.image
81
+
82
+ return isinstance(f, ImageField)
83
+
84
+ def contents(self) -> str:
85
+ contents = self._get_contents()
86
+ contents = self._preprocess_field(contents)
87
+ return contents
88
+
89
+ def get_admin_url(self, remote_field, remote_obj):
90
+ url_name = f"admin:{remote_field.model._meta.app_label}_{remote_field.model._meta.model_name}_change"
91
+ try:
92
+ url = reverse(
93
+ url_name,
94
+ args=[quote(remote_obj.pk)],
95
+ current_app=self.model_admin.admin_site.name,
96
+ )
97
+ return format_html(
98
+ '<a href="{}" class="text-primary-600 underline whitespace-nowrap">{}</a>',
99
+ url,
100
+ remote_obj,
101
+ )
102
+ except NoReverseMatch:
103
+ return str(remote_obj)
104
+
105
+ def _get_contents(self) -> str:
106
+ from django.contrib.admin.templatetags.admin_list import _boolean_icon
107
+
108
+ field, obj, model_admin = (
109
+ self.field["field"],
110
+ self.form.instance,
111
+ self.model_admin,
112
+ )
113
+ try:
114
+ f, attr, value = lookup_field(field, obj, model_admin)
115
+ except (AttributeError, ValueError, ObjectDoesNotExist):
116
+ result_repr = self.empty_value_display
117
+ else:
118
+ if field in self.form.fields:
119
+ widget = self.form[field].field.widget
120
+ # This isn't elegant but suffices for contrib.auth's
121
+ # ReadOnlyPasswordHashWidget.
122
+ if getattr(widget, "read_only", False):
123
+ return widget.render(field, value)
124
+
125
+ if f is None:
126
+ if getattr(attr, "boolean", False):
127
+ result_repr = _boolean_icon(value)
128
+ else:
129
+ if hasattr(value, "__html__"):
130
+ result_repr = value
131
+ else:
132
+ result_repr = linebreaksbr(value)
133
+ else:
134
+ if isinstance(f.remote_field, ManyToManyRel) and value is not None:
135
+ result_repr = ", ".join(map(str, value.all()))
136
+ elif (
137
+ isinstance(f.remote_field, (ForeignObjectRel, OneToOneField))
138
+ and value is not None
139
+ ):
140
+ result_repr = self.get_admin_url(f.remote_field, value)
141
+ elif isinstance(f, models.URLField):
142
+ return format_html(
143
+ '<a href="{}" class="text-primary-600 underline whitespace-nowrap">{}</a>',
144
+ value,
145
+ value,
146
+ )
147
+ else:
148
+ result_repr = display_for_field(value, f, self.empty_value_display)
149
+ return conditional_escape(result_repr)
150
+ result_repr = linebreaksbr(result_repr)
151
+ return conditional_escape(result_repr)
152
+
153
+ def _preprocess_field(self, contents: str) -> str:
154
+ if (
155
+ hasattr(self.model_admin, "readonly_preprocess_fields")
156
+ and self.field["field"] in self.model_admin.readonly_preprocess_fields
157
+ ):
158
+ func = self.model_admin.readonly_preprocess_fields[self.field["field"]]
159
+ if isinstance(func, str):
160
+ contents = import_string(func)(contents)
161
+ elif callable(func):
162
+ contents = func(contents)
163
+
164
+ return contents
165
+
166
+
167
+ class UnfoldAdminField(helpers.AdminField):
168
+ def label_tag(self) -> SafeText:
169
+ classes = []
170
+ if not self.field.field.widget.__class__.__name__.startswith(
171
+ "Unfold"
172
+ ) and not self.field.field.widget.template_name.startswith("unfold"):
173
+ return super().label_tag()
174
+
175
+ # TODO load config from current AdminSite (override Fieldline.__iter__ method)
176
+ for lang, flag in get_config()["EXTENSIONS"]["modeltranslation"][
177
+ "flags"
178
+ ].items():
179
+ if f"[{lang}]" in self.field.label:
180
+ self.field.label = self.field.label.replace(f"[{lang}]", flag)
181
+ break
182
+
183
+ contents = conditional_escape(self.field.label)
184
+
185
+ if self.is_checkbox:
186
+ classes.append(" ".join(CHECKBOX_LABEL_CLASSES))
187
+ else:
188
+ classes.append(" ".join(LABEL_CLASSES))
189
+
190
+ if self.field.field.required:
191
+ classes.append("required")
192
+
193
+ attrs = {"class": " ".join(classes)} if classes else {}
194
+ required = mark_safe(' <span class="text-red-600">*</span>')
195
+
196
+ return self.field.label_tag(
197
+ contents=mark_safe(contents),
198
+ attrs=attrs,
199
+ label_suffix=required if self.field.field.required else "",
200
+ )