django-fast-treenode 3.0.7__py3-none-any.whl → 3.0.8__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: django-fast-treenode
3
- Version: 3.0.7
3
+ Version: 3.0.8
4
4
  Summary: Treenode Framework for supporting tree (hierarchical) data structure in Django projects
5
5
  Home-page: https://django-fast-treenode.readthedocs.io/
6
6
  Author: Timur Kady
@@ -1,4 +1,4 @@
1
- django_fast_treenode-3.0.7.dist-info/licenses/LICENSE,sha256=SSYqS84FCnAW7tAxmjBKU8qAa8Jv4VGPuSSGeHwWtJE,1095
1
+ django_fast_treenode-3.0.8.dist-info/licenses/LICENSE,sha256=SSYqS84FCnAW7tAxmjBKU8qAa8Jv4VGPuSSGeHwWtJE,1095
2
2
  treenode/__init__.py,sha256=3z1hWpHyy4wg6uz7HCmRi9FaXYeN5CfANVpa77UIoPw,53
3
3
  treenode/apps.py,sha256=QlwjNDM9rkUoWB8Vm8-OkS6lNx0-aTByuGZlu9wrQMs,1832
4
4
  treenode/cache.py,sha256=2jUiiecfFxwB7QFukpU4u0FnDzGH6hNRfo6KAYvs6vM,8447
@@ -7,18 +7,18 @@ treenode/settings.py,sha256=oSkcKXNVd28HXrlWZIH2VYinCMq-UdCDlX4KD0Qc_Xk,631
7
7
  treenode/signals.py,sha256=ERrlKjGqhYaPYVKKRk1JBBlPFOmJKpJ6bXsJavcTlo0,518
8
8
  treenode/tests.py,sha256=2uDafv3Ns6f7Vy1ekUtgYxCZEi1KRyesZDTAFhYcX-E,63
9
9
  treenode/urls.py,sha256=krHvVigc_dxC0z5hEd2rgeH6th8jW7qJY3Qbia-419Y,240
10
- treenode/version.py,sha256=-I7mgQs7Zf31bXpDTd5OymDS52Y_ye4ZPw9yQjKTS9E,220
10
+ treenode/version.py,sha256=5TH3IchoS9D8YEFnTIlRrjoALxMdemh1PCN0Bb9Jnss,220
11
11
  treenode/widgets.py,sha256=VItPvN9XgaSRI_MZjKEmtaHDJcn2bDIQIppwKjXmYQM,4017
12
12
  treenode/admin/__init__.py,sha256=XNEYHdF5lKb0vpdlVxdR2fxj5oUgzyx1YyCwsv0gxHw,100
13
- treenode/admin/admin.py,sha256=2-74ESGEZ7s6wfPzep6OQlz1uD_80WFExemtwbQiD3M,8289
13
+ treenode/admin/admin.py,sha256=uORuVSx-p0MYXLRVl54cBa7IAL9zicyeYTX3Cv5pBDc,7462
14
14
  treenode/admin/changelist.py,sha256=KUYS9MaR8Ck_1xmMqupobxWKarrJEqmHuEG32CL01Bo,1662
15
15
  treenode/admin/exporter.py,sha256=QE74V6W3tvwA5kCvBt1MmVlLOaWh-o8EU63cgmiwD5Q,5724
16
16
  treenode/admin/importer.py,sha256=hK3D-1DZcoowGblRluGzng3n5Bf__hMsbNaIGXRpRdg,6263
17
17
  treenode/admin/mixin.py,sha256=jOkKKt0CpknGokluW8T1YB7VhkCkaAAMYBuptbYDXAw,10648
18
18
  treenode/managers/__init__.py,sha256=c7F9Ku9489Hv6lTpUY2nbyBlWFCXBWAkNBm4xTKcjL8,186
19
19
  treenode/managers/managers.py,sha256=8OaFxtajyR1d7-UHyiUbifMBEF9cjfHTIEYPkYUWmt0,7166
20
- treenode/managers/queries.py,sha256=pmZ07QZQbTTJN0ur6-pQGua2vGXVCVisylL1LWNXMmo,10655
21
- treenode/managers/tasks.py,sha256=b8deUAbCpD1Yov-PpjKNAx49rL4fejkGf64ih5JziF0,7138
20
+ treenode/managers/queries.py,sha256=Kepax8SDn7G5tOlPRWBCp5Oyp49O5iMITCMBoNCm_Ak,10655
21
+ treenode/managers/tasks.py,sha256=nVlGDxNnFlZS1-oU8UH_yTXJERcGrAZ9kCmKwKO5WuY,7138
22
22
  treenode/models/__init__.py,sha256=iR4ksCKoayvkIWWgGk6OUGHZC3D0mzAtgdBcS2vQPBw,188
23
23
  treenode/models/decorators.py,sha256=N2dcnWqSCiEXDcYCf0zVijrbGUC8kYlqOLi_GKFmECU,1457
24
24
  treenode/models/factory.py,sha256=sPUSrvo1za-r6ny3B8ptwevyjO8-iUpPNrT0eSD2kvI,1786
@@ -38,12 +38,12 @@ treenode/models/mixins/update.py,sha256=oCZkMnfT23n2n3_mnokDrtTLS_jO_lJRwG3ENkH_
38
38
  treenode/static/.gitkeep,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
39
39
  treenode/static/css/.gitkeep,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
40
40
  treenode/static/css/tree_widget.css,sha256=YVIHuZT_E5ck5N7wvYvuwgV_m-4Ta_uGFiNUoTw-zbk,4861
41
- treenode/static/css/treenode_admin.css,sha256=_AVEO52atudOVBspgH4ecubbgGUeKhcC13WSxo7QQAw,3021
41
+ treenode/static/css/treenode_admin.css,sha256=w7yyyf4aFsJKlr_YJPVq6ap5ZvA5Co7ZCYq6ipuV958,3291
42
42
  treenode/static/css/treenode_tabs.css,sha256=pbulaUEhNMQFNWXFQrRN1zMyKjoSqLAnjNygpouPUfE,879
43
43
  treenode/static/js/.gitkeep,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
44
44
  treenode/static/js/lz-string.min.js,sha256=TAnTJQd2AlLqT9M2TU7GFjnoj9SIfwLeZnpEtLkP624,4718
45
45
  treenode/static/js/tree_widget.js,sha256=2LU8IIJD5TXbWVGjCEgrj5aYnUYtwBZECzj25XLtxNc,9633
46
- treenode/static/js/treenode_admin.js,sha256=fkxAj50QxwpPVdQxDNxQZ5vR0goUt53h8FzOv_fjxBQ,15945
46
+ treenode/static/js/treenode_admin.js,sha256=IU3Zj2Q-WBxftl0zsjeRPrt1xOzzdEKlzGEMThi3hrk,9706
47
47
  treenode/static/vendors/jquery-ui/AUTHORS.txt,sha256=geZ7JfVsTJq8Wgjrcm1p56Zqjxx7PL9FkxhKiN8dQg8,14848
48
48
  treenode/static/vendors/jquery-ui/LICENSE.txt,sha256=9kkCXPZcjc2FiLCTeIO6elK9ttNRKUZKNqFgLrACUPE,1818
49
49
  treenode/static/vendors/jquery-ui/index.html,sha256=RSv8cA662qJdc02OX-UmpaBJoZrngoQqnNWVZNn9GQQ,24229
@@ -67,12 +67,12 @@ treenode/templates/.gitkeep,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
67
67
  treenode/templates/treenode/.gitkeep,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
68
68
  treenode/templates/treenode/admin/.gitkeep,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
69
69
  treenode/templates/treenode/admin/treenode_ajax_rows.html,sha256=zFyPaTbSyxRjOqQ85SMv__qTIYDjEna6chYODBypDZA,224
70
- treenode/templates/treenode/admin/treenode_changelist.html,sha256=ALU8iZLCOIiiGIf5RTdv3Bb694pG_YyQ-wcbz1CkBKc,654
70
+ treenode/templates/treenode/admin/treenode_changelist.html,sha256=JiUwX33w5WwegX7gM2ctw-bx9XOyXwKt9QB_6g7Dfz8,2283
71
71
  treenode/templates/treenode/admin/treenode_import_export.html,sha256=K6L_JnhGpWgQdduXSfAJ0UuTBfSN4ylFw9w_e5tPDDE,2926
72
- treenode/templates/treenode/admin/treenode_rows.html,sha256=S1XtZXWMUdfgjDoKG7OJwZg81a_5IKPp7tyrrHhJbBk,1695
72
+ treenode/templates/treenode/admin/treenode_rows.html,sha256=de90wKpNdVB2C5CGnIkStBB0dWCDkmWr_pluV_NgBUs,1857
73
73
  treenode/templates/treenode/widgets/tree_widget.html,sha256=GKcCU-B2FkkJ2BSOuXOw9e_PdYTtADcvyITEXqOlZ9Y,723
74
74
  treenode/templatetags/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
75
- treenode/templatetags/treenode_admin.py,sha256=0Xz5tS_Etf2FU69Ww7GfQ6u2_jh97jymD1Gc4XCpHbc,1204
75
+ treenode/templatetags/treenode_admin.py,sha256=5f5oqAS4zC_f0kkJRsm5MqXHjsKJXn8GslbZcbMivxg,2736
76
76
  treenode/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
77
77
  treenode/utils/db/__init__.py,sha256=RwicAcJSI1nhIPWLdT7j9TFsgOc9834VDn9lVn54GlY,247
78
78
  treenode/utils/db/compiler.py,sha256=PgD9ybS5H8OUHw1gkFBQHhnrf5HiCx8QXUMRhydwh7o,3824
@@ -81,13 +81,13 @@ treenode/utils/db/service.py,sha256=PF85Yhz2xUWFFCzpLYotmiNTZXXEH61rhswslSxEUds,
81
81
  treenode/utils/db/sqlcompat.py,sha256=K71ggkKIvpdTtHQ6Y4qcbo6cj2eYiEfy6DlVBr8Po1E,4460
82
82
  treenode/utils/db/sqlquery.py,sha256=KXcfKbbaBF-D134H_2DiPQtjedR79SJNXPJc0msZYEc,1938
83
83
  treenode/views/__init__.py,sha256=ppxbBx51TUaKstJFpAd_DTmbKjbZGmVMLNYSpgUKnd0,111
84
- treenode/views/autoapi.py,sha256=o75e8IFsogbhZN_rbx3BKVnoruD96nWelnC5UzOqUDw,3628
85
- treenode/views/autocomplete.py,sha256=Z7cBnC4Ihdyxm8zlbnG6CkZdVkM3TOTWRpw5mdhaIVA,1469
86
- treenode/views/children.py,sha256=bygXaEBExxG3zIPL34_PYHLFFIqlQU2naqPIlyQ6e-s,1152
87
- treenode/views/common.py,sha256=mrmr40R91XVbMWcz5GZT-OjpnQ87F7XQZxu1W6rqpqI,617
84
+ treenode/views/autoapi.py,sha256=IAQfhWey4z-QBuAn-QDHhpJSFCF702sHtqYOA6vK4BY,3537
85
+ treenode/views/autocomplete.py,sha256=ERGJT4jfCE2GUv94Ja8SgJyG1FOAyqIeB0PKH1fyU7g,1417
86
+ treenode/views/children.py,sha256=seO0SNKriRY2FJOO1oZgX42iKolXMf3EF76GrfdJZLQ,1111
87
+ treenode/views/common.py,sha256=kUN3IgMZCdNdJ2haxB9MTGcn2rctyFUAHNagzcu9wXk,594
88
88
  treenode/views/crud.py,sha256=RI5rdyD4hZTszjZFThByxi_lkAeJlqbDCXFkD8iyzKE,7424
89
89
  treenode/views/search.py,sha256=c_GyooT3jyoNa96bBxfoWruRN1wIw-ZGYvwGKkGojTs,1501
90
- django_fast_treenode-3.0.7.dist-info/METADATA,sha256=7scCSnoGv6UFkQHQV1yLuZBtXr3PSU1Lii3qQi43r_o,10249
91
- django_fast_treenode-3.0.7.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
92
- django_fast_treenode-3.0.7.dist-info/top_level.txt,sha256=fmgxHbXyx1O2MPi_9kjx8aL9L-8TmV0gre4Go8XgqFk,9
93
- django_fast_treenode-3.0.7.dist-info/RECORD,,
90
+ django_fast_treenode-3.0.8.dist-info/METADATA,sha256=-DOMkpuNAIruYNiRcCc7g6zGVpbWjter7IEDb9k10mg,10249
91
+ django_fast_treenode-3.0.8.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
92
+ django_fast_treenode-3.0.8.dist-info/top_level.txt,sha256=fmgxHbXyx1O2MPi_9kjx8aL9L-8TmV0gre4Go8XgqFk,9
93
+ django_fast_treenode-3.0.8.dist-info/RECORD,,
treenode/admin/admin.py CHANGED
@@ -2,7 +2,26 @@
2
2
  """
3
3
  TreeNode Admin Model Class
4
4
 
5
- Version: 3.0.7
5
+ Modified admin panel for django-fast-treenode. Solves the following problems:
6
+ - Set list_per_page = 10000 to display all elements at once.
7
+ - Hidden standard pagination via CSS
8
+ - Disabled counting the total number of elements to speed up loading
9
+ - Accordion works regardless of the display mode
10
+ Two modes are supported:
11
+ - Indented - with indents and icons
12
+ - Breadcrumbs - with breadcrumbs
13
+ All modes have links to editing objects.
14
+
15
+ - Expand buttons for nodes with children
16
+
17
+ Additional features:
18
+ - Control panel with "Expand All" / "Collapse All" buttons
19
+ - Saving the state of the tree between page transitions
20
+ - Smooth animations when expanding/collapsing
21
+ - Counting the total number of nodes in the tree
22
+ - Recursive hiding of grandchildren when collapsing the parent
23
+
24
+ Version: 3.1.0
6
25
  Author: Timur Kady
7
26
  Email: timurkady@yandex.com
8
27
  """
@@ -10,13 +29,11 @@ Email: timurkady@yandex.com
10
29
 
11
30
  from django.contrib import admin
12
31
  from django.db import models
13
- from django.http import HttpResponseRedirect
14
32
  from django.urls import reverse
15
33
  from django.utils.html import escape
16
34
  from django.utils.safestring import mark_safe
17
35
  from django.utils.translation import gettext_lazy as _
18
36
 
19
- from .changelist import TreeNodeChangeList
20
37
  from .mixin import AdminMixin
21
38
  from ..forms import TreeNodeForm
22
39
  from ..widgets import TreeWidget
@@ -34,14 +51,12 @@ class TreeNodeModelAdmin(AdminMixin, admin.ModelAdmin):
34
51
  # Режимы отображения
35
52
  TREENODE_DISPLAY_MODE_ACCORDION = 'accordion'
36
53
  TREENODE_DISPLAY_MODE_BREADCRUMBS = 'breadcrumbs'
37
- TREENODE_DISPLAY_MODE_INDENTATION = 'indentation'
38
54
  treenode_display_mode = TREENODE_DISPLAY_MODE_ACCORDION
39
55
 
40
56
  form = TreeNodeForm
41
57
  importer_class = None
42
58
  exporter_class = None
43
59
  ordering = []
44
- list_per_page = 1000
45
60
 
46
61
  formfield_overrides = {
47
62
  models.ForeignKey: {'widget': TreeWidget()},
@@ -89,96 +104,69 @@ class TreeNodeModelAdmin(AdminMixin, admin.ModelAdmin):
89
104
 
90
105
  toggle.short_description = _("Expand")
91
106
 
92
- def _get_treenode_field_display(self, request, obj):
93
- """Return HTML for the tree node in list view."""
94
- level = obj.get_depth()
95
- edit_url = reverse(
96
- f"admin:{obj._meta.app_label}_{obj._meta.model_name}_change",
97
- args=[obj.pk]
98
- )
107
+ def get_changelist(self, request, **kwargs):
108
+ """Get changelist."""
109
+ ChangeList = super().get_changelist(request, **kwargs)
99
110
 
100
- if self.treenode_display_mode == self.TREENODE_DISPLAY_MODE_ACCORDION:
101
- icon = "📄 " if obj.is_leaf() else "📁 "
102
- content = (
103
- f'<span style="padding-left: {level * 1.5}em;">'
104
- f'{icon}<a href="{edit_url}">{str(obj)}</a></span>'
105
- )
106
- elif self.treenode_display_mode == self.TREENODE_DISPLAY_MODE_BREADCRUMBS:
107
- breadcrumbs = obj.get_ancestors(include_self=True)
108
- model_label = obj._meta.app_label
109
- model_name = obj._meta.model_name
110
- href = f"admin:{model_label}_{model_name}_change"
111
-
112
- display_attr = getattr(obj, 'display_field', 'id')
113
-
114
- records = [
115
- (
116
- getattr(item, display_attr, item.pk),
117
- reverse(href, args=[item.pk]),
118
- )
119
- for item in breadcrumbs
120
- ]
121
-
122
- content = " / ".join([
123
- f'<a href="{url}">{escape(label)}</a>'
124
- for label, url in records
125
- ])
126
-
127
- elif self.treenode_display_mode == self.TREENODE_DISPLAY_MODE_INDENTATION: # noqa
128
- indent = "&mdash;" * level
129
- content = f'{indent}<a href="{edit_url}">{str(obj)}</a>'
130
- else:
131
- content = f'<a href="{edit_url}">{str(obj)}</a>'
132
-
133
- html = (
134
- f'<div class="treenode-wrapper" '
135
- f'data-treenode-pk="{obj.pk}" '
136
- f'data-treenode-depth="{level}" '
137
- f'data-treenode-parent="{obj.parent_id or ""}">'
138
- f'<span class="treenode-content">{content}</span>'
139
- f'</div>'
140
- )
141
- return mark_safe(html)
111
+ class NoPaginationChangeList(ChangeList):
112
+ """Suppress pagination."""
142
113
 
143
- def get_list_display(self, request):
144
- """Generate list_display dynamically with tree-aware columns."""
145
- # Define callable that replaces display field with tree field
146
- def treenode_field(obj):
147
- return self._get_treenode_field_display(request, obj)
148
- treenode_field.short_description = self.model._meta.verbose_name
114
+ def get_results(self, request):
115
+ """Get result."""
116
+ super().get_results(request)
117
+ self.paginator.show_all = True
118
+ self.result_count = len(self.result_list)
119
+ self.full_result_count = len(self.result_list)
120
+ self.can_show_all = False
121
+ self.multi_page = False
122
+ self.actions = self.model_admin.get_actions(request)
149
123
 
150
- display_field = getattr(self.model, 'display_field', '__str__')
124
+ return NoPaginationChangeList
151
125
 
152
- user_list_display = list(super().get_list_display(request))
153
- # If the list is empty or only contains __str__, replace it entirely
154
- if not user_list_display or user_list_display == ['__str__']:
155
- user_list_display = [treenode_field]
156
- else:
157
- try:
158
- pos = user_list_display.index(display_field)
159
- user_list_display.pop(pos)
160
- user_list_display.insert(pos, treenode_field)
161
- except ValueError:
162
- user_list_display.insert(0, treenode_field)
126
+ def get_changelist_instance(self, request):
127
+ """
128
+ Get changelist instance.
163
129
 
164
- return (self.drag, self.toggle) + tuple(user_list_display)
130
+ Make sure our custom ChangeList is used without pagination.
131
+ """
132
+ ChangeList = self.get_changelist(request)
165
133
 
166
- def get_list_display_links(self, request, list_display):
167
- """Get display list links."""
168
- return ('treenode_field',)
134
+ return ChangeList(
135
+ request,
136
+ self.model,
137
+ self.get_list_display(request),
138
+ self.get_list_display_links(
139
+ request,
140
+ self.get_list_display(request)
141
+ ),
142
+ self.get_list_filter(request),
143
+ self.date_hierarchy,
144
+ self.search_fields,
145
+ self.list_select_related,
146
+ self.list_per_page,
147
+ self.list_max_show_all,
148
+ self.list_editable,
149
+ self,
150
+ sortable_by=self.get_sortable_by(request),
151
+ search_help_text=self.get_search_help_text(request),
152
+ )
169
153
 
170
154
  def get_queryset(self, request):
171
- """By default: only root nodes, unless searching or editing."""
155
+ """Get queryset."""
172
156
  qs = super().get_queryset(request)
157
+ return qs.select_related('parent')\
158
+ .prefetch_related('children')\
159
+ .order_by('_path')
173
160
 
174
- if request.GET.get("q"):
175
- return qs
161
+ def get_list_display(self, request):
162
+ """Get list_display."""
163
+ def treenode_field(obj):
164
+ return self._get_treenode_field_display(request, obj)
176
165
 
177
- resolved = request.resolver_match
178
- if resolved and resolved.url_name.endswith("_change"):
179
- return qs
166
+ description = str(self.model._meta.verbose_name)
167
+ treenode_field.short_description = description
180
168
 
181
- return qs.filter(parent__isnull=True)
169
+ return (self.drag, self.toggle, treenode_field)
182
170
 
183
171
  def get_form(self, request, obj=None, **kwargs):
184
172
  """Get Form method."""
@@ -189,56 +177,46 @@ class TreeNodeModelAdmin(AdminMixin, admin.ModelAdmin):
189
177
 
190
178
  def get_search_fields(self, request):
191
179
  """Get search fields."""
192
- return [getattr(self.model, 'treenode_display_field', 'id')]
193
-
194
- def get_changelist(self, request, **kwargs):
195
- """Get ChangeList Class."""
196
- return TreeNodeChangeList
197
-
198
- def get_ordering(self, request):
199
- """Get ordering."""
200
- return None
201
-
202
- def changelist_view(self, request, extra_context=None):
203
- """Changelist View."""
204
- extra_context = extra_context or {}
205
- # TODO
206
- extra_context['import_export_enabled'] = self.import_export
207
- extra_context['num_sorted_fields'] = len(self.get_ordering(request) or []) # noqa: D501
208
-
209
- response = super().changelist_view(request, extra_context=extra_context)
210
-
211
- # If response is a redirect, then there is no point in updating
212
- # ChangeList
213
- if isinstance(response, HttpResponseRedirect):
214
- return response
180
+ return [getattr(self.model, 'display_field', 'id') or 'id']
215
181
 
216
- ChangeListClass = self.get_changelist(request)
217
- cl = ChangeListClass(
218
- request,
219
- self.model,
220
- self.list_display,
221
- self.list_display_links,
222
- self.list_filter,
223
- self.date_hierarchy,
224
- self.search_fields,
225
- self.list_select_related,
226
- self.list_per_page,
227
- self.list_max_show_all,
228
- self.list_editable,
229
- self,
230
- self.get_sortable_by(request),
231
- self.get_search_help_text(request),
182
+ def _get_treenode_field_display(self, request, obj):
183
+ """
184
+ Generate HTML to display tree nodes.
185
+
186
+ Depending on the selected display mode (accordion or breadcrumbs),
187
+ do the following:
188
+ - For accordion mode: add indents and icons.
189
+ - For breadcrumbs mode: display breadcrumb path.
190
+ """
191
+ level = obj.get_depth()
192
+ display_field = getattr(obj, "display_field", None)
193
+ edit_url = reverse(
194
+ f"admin:{obj._meta.app_label}_{obj._meta.model_name}_change",
195
+ args=[obj.pk]
232
196
  )
197
+ icon = ""
198
+ text = ""
199
+ padding = ""
200
+ closing = ""
201
+
202
+ if self.treenode_display_mode == self.TREENODE_DISPLAY_MODE_ACCORDION:
203
+ icon = "📄 " if obj.is_leaf() else "📁 "
204
+ text = getattr(obj, display_field, str(obj))
205
+ padding = f'<span style="padding-left: {level * 1.5}em;">'
206
+ closing = "</span>"
207
+ elif self.treenode_display_mode == self.TREENODE_DISPLAY_MODE_BREADCRUMBS: # noqa
208
+ if display_field:
209
+ breadcrumbs = obj.get_breadcrumbs(attr=display_field)
210
+ else:
211
+ breadcrumbs = [str(item) for item in obj.get_ancestors()]
233
212
 
234
- cl.get_results(request)
235
- cl.result_list = self.render_changelist_rows(cl.result_list, request)
213
+ text = "/" + "/".join([escape(label) for label in breadcrumbs])
236
214
 
237
- return response
215
+ content = f'{padding}{icon}<a href="{edit_url}">{escape(text)}</a>{closing}' # noqa
216
+ return mark_safe(content)
238
217
 
239
218
  def get_list_per_page(self, request):
240
219
  """Get list per page."""
241
220
  return 999999
242
221
 
243
-
244
222
  # The End
@@ -5,7 +5,7 @@ Low-level SQL Query Manager.
5
5
  Encapsulates all logic to retrieve related primary keys based on relationships
6
6
  (e.g., ancestors, children, descendants, siblings, family, root) using raw SQL.
7
7
 
8
- Version: 3.0.7
8
+ Version: 3.0.0
9
9
  Author: Timur Kady
10
10
  Email: timurkady@yandex.com
11
11
  """
@@ -2,7 +2,7 @@
2
2
  """
3
3
  TreeNode TaskQuery manager
4
4
 
5
- Version: 3.0.4
5
+ Version: 3.0.1
6
6
  Author: Timur Kady
7
7
  Email: timurkady@yandex.com
8
8
  """
@@ -66,6 +66,27 @@ Email: timurkady@yandex.com
66
66
  opacity: 1.0;
67
67
  }
68
68
 
69
+ .treenode-toolbar{
70
+ display: flex;
71
+ }
72
+
73
+ .treenode-toolbar {
74
+ margin: 15px 0px;
75
+ }
76
+
77
+ .treenode-button {
78
+ padding: 5px !important;
79
+ margin-left: 15px !important;
80
+ }
81
+
82
+ tr.treenode-hidden {
83
+ display: none;
84
+ }
85
+
86
+ td.action-checkbox{
87
+ text-align: center;
88
+ }
89
+
69
90
  .dark-theme .treenode-toggle {
70
91
  color: #ccc;
71
92
  background-color: #444;