django-fast-treenode 3.0.7__tar.gz → 3.0.8__tar.gz

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.
Files changed (125) hide show
  1. {django_fast_treenode-3.0.7/django_fast_treenode.egg-info → django_fast_treenode-3.0.8}/PKG-INFO +1 -1
  2. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8/django_fast_treenode.egg-info}/PKG-INFO +1 -1
  3. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/admin.md +0 -3
  4. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/pyproject.toml +1 -1
  5. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/setup.py +1 -1
  6. django_fast_treenode-3.0.8/treenode/admin/admin.py +222 -0
  7. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/managers/queries.py +1 -1
  8. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/managers/tasks.py +1 -1
  9. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/css/treenode_admin.css +21 -0
  10. django_fast_treenode-3.0.8/treenode/static/js/treenode_admin.js +322 -0
  11. django_fast_treenode-3.0.8/treenode/templates/treenode/admin/treenode_changelist.html +71 -0
  12. django_fast_treenode-3.0.8/treenode/templates/treenode/admin/treenode_rows.html +54 -0
  13. django_fast_treenode-3.0.8/treenode/templatetags/treenode_admin.py +89 -0
  14. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/version.py +2 -2
  15. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/autoapi.py +91 -91
  16. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/autocomplete.py +52 -52
  17. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/children.py +41 -41
  18. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/common.py +23 -23
  19. django_fast_treenode-3.0.7/treenode/admin/admin.py +0 -244
  20. django_fast_treenode-3.0.7/treenode/static/js/treenode_admin.js +0 -531
  21. django_fast_treenode-3.0.7/treenode/templates/treenode/admin/treenode_changelist.html +0 -25
  22. django_fast_treenode-3.0.7/treenode/templates/treenode/admin/treenode_rows.html +0 -57
  23. django_fast_treenode-3.0.7/treenode/templatetags/treenode_admin.py +0 -46
  24. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/LICENSE +0 -0
  25. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/MANIFEST.in +0 -0
  26. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/README.md +0 -0
  27. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/django_fast_treenode.egg-info/SOURCES.txt +0 -0
  28. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/django_fast_treenode.egg-info/dependency_links.txt +0 -0
  29. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/django_fast_treenode.egg-info/requires.txt +0 -0
  30. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/django_fast_treenode.egg-info/top_level.txt +0 -0
  31. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/.gitignore +0 -0
  32. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/.nojekyll +0 -0
  33. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/about.md +0 -0
  34. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/api.md +0 -0
  35. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/apifirst.md +0 -0
  36. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/cache.md +0 -0
  37. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/customization.md +0 -0
  38. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/dnd.md +0 -0
  39. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/import_export.md +0 -0
  40. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/index.md +0 -0
  41. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/insert-after.jpg +0 -0
  42. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/insert-as-child.jpg +0 -0
  43. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/installation.md +0 -0
  44. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/migration.md +0 -0
  45. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/models.md +0 -0
  46. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/requirements.txt +0 -0
  47. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/roadmap.md +0 -0
  48. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/docs/using.md +0 -0
  49. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/setup.cfg +0 -0
  50. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/tests/test_suite.py +0 -0
  51. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/__init__.py +0 -0
  52. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/admin/__init__.py +0 -0
  53. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/admin/changelist.py +0 -0
  54. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/admin/exporter.py +0 -0
  55. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/admin/importer.py +0 -0
  56. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/admin/mixin.py +0 -0
  57. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/apps.py +0 -0
  58. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/cache.py +0 -0
  59. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/forms.py +0 -0
  60. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/managers/__init__.py +0 -0
  61. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/managers/managers.py +0 -0
  62. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/__init__.py +0 -0
  63. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/decorators.py +0 -0
  64. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/factory.py +0 -0
  65. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/__init__.py +0 -0
  66. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/ancestors.py +0 -0
  67. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/children.py +0 -0
  68. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/descendants.py +0 -0
  69. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/family.py +0 -0
  70. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/logical.py +0 -0
  71. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/node.py +0 -0
  72. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/properties.py +0 -0
  73. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/roots.py +0 -0
  74. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/siblings.py +0 -0
  75. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/tree.py +0 -0
  76. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/mixins/update.py +0 -0
  77. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/models/models.py +0 -0
  78. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/settings.py +0 -0
  79. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/signals.py +0 -0
  80. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/.gitkeep +0 -0
  81. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/css/.gitkeep +0 -0
  82. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/css/tree_widget.css +0 -0
  83. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/css/treenode_tabs.css +0 -0
  84. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/js/.gitkeep +0 -0
  85. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/js/lz-string.min.js +0 -0
  86. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/js/tree_widget.js +0 -0
  87. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/AUTHORS.txt +0 -0
  88. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/LICENSE.txt +0 -0
  89. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/external/jquery/jquery.js +0 -0
  90. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_444444_256x240.png +0 -0
  91. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_555555_256x240.png +0 -0
  92. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_777620_256x240.png +0 -0
  93. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_777777_256x240.png +0 -0
  94. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_cc0000_256x240.png +0 -0
  95. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/images/ui-icons_ffffff_256x240.png +0 -0
  96. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/index.html +0 -0
  97. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.css +0 -0
  98. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.js +0 -0
  99. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.min.css +0 -0
  100. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.min.js +0 -0
  101. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.structure.css +0 -0
  102. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.structure.min.css +0 -0
  103. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.theme.css +0 -0
  104. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/jquery-ui.theme.min.css +0 -0
  105. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/static/vendors/jquery-ui/package.json +0 -0
  106. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/.gitkeep +0 -0
  107. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/treenode/.gitkeep +0 -0
  108. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/treenode/admin/.gitkeep +0 -0
  109. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/treenode/admin/treenode_ajax_rows.html +0 -0
  110. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/treenode/admin/treenode_import_export.html +0 -0
  111. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templates/treenode/widgets/tree_widget.html +0 -0
  112. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/templatetags/__init__.py +0 -0
  113. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/tests.py +0 -0
  114. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/urls.py +0 -0
  115. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/__init__.py +0 -0
  116. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/__init__.py +0 -0
  117. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/compiler.py +0 -0
  118. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/db_vendor.py +0 -0
  119. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/service.py +0 -0
  120. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/sqlcompat.py +0 -0
  121. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/utils/db/sqlquery.py +0 -0
  122. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/__init__.py +0 -0
  123. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/crud.py +0 -0
  124. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/views/search.py +0 -0
  125. {django_fast_treenode-3.0.7 → django_fast_treenode-3.0.8}/treenode/widgets.py +0 -0
@@ -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,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
@@ -16,7 +16,6 @@ class CategoryAdmin(TreeNodeModelAdmin):
16
16
  # Set the display mode: 'accordion', 'breadcrumbs', or 'indentation'
17
17
  treenode_display_mode = TreeNodeModelAdmin.TREENODE_DISPLAY_MODE_ACCORDION
18
18
  # treenode_display_mode = TreeNodeModelAdmin.TREENODE_DISPLAY_MODE_BREADCRUMBS
19
- # treenode_display_mode = TreeNodeModelAdmin.TREENODE_DISPLAY_MODE_INDENTATION
20
19
 
21
20
  list_display = ("name",)
22
21
  search_fields = ("name",)
@@ -30,8 +29,6 @@ You can choose from three display modes:
30
29
  Expands/collapses nodes dynamically.
31
30
  - **`TREENODE_DISPLAY_MODE_BREADCRUMBS`**
32
31
  Displays the tree as a sequence of **breadcrumbs**, making it easy to navigate.
33
- - **`TREENODE_DISPLAY_MODE_INDENTATION`**
34
- Uses a **long dash** (`———`) to indicate nesting levels, providing a simple visual structure.
35
32
 
36
33
  The accordion mode is **always active**, and the setting only affects how nodes are displayed.
37
34
 
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "django-fast-treenode"
7
- version = "3.0.7"
7
+ version = "3.0.8"
8
8
  description = "Treenode Framework for supporting tree (hierarchical) data structure in Django projects"
9
9
  readme = "README.md"
10
10
  authors = [{ name = "Timur Kady", email = "timurkady@yandex.com" }]
@@ -2,7 +2,7 @@ from setuptools import setup, find_packages
2
2
 
3
3
  setup(
4
4
  name='django-fast-treenode',
5
- version='3.0.7',
5
+ version='3.0.8',
6
6
  description='Treenode Framework for supporting tree (hierarchical) data structure in Django projects',
7
7
  long_description=open('README.md', encoding='utf-8').read(),
8
8
  long_description_content_type='text/markdown',
@@ -0,0 +1,222 @@
1
+ # -*- coding: utf-8 -*-
2
+ """
3
+ TreeNode Admin Model Class
4
+
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
25
+ Author: Timur Kady
26
+ Email: timurkady@yandex.com
27
+ """
28
+
29
+
30
+ from django.contrib import admin
31
+ from django.db import models
32
+ from django.urls import reverse
33
+ from django.utils.html import escape
34
+ from django.utils.safestring import mark_safe
35
+ from django.utils.translation import gettext_lazy as _
36
+
37
+ from .mixin import AdminMixin
38
+ from ..forms import TreeNodeForm
39
+ from ..widgets import TreeWidget
40
+ from .importer import TreeNodeImporter
41
+ from .exporter import TreeNodeExporter
42
+
43
+ import logging
44
+
45
+ logger = logging.getLogger(__name__)
46
+
47
+
48
+ class TreeNodeModelAdmin(AdminMixin, admin.ModelAdmin):
49
+ """Admin for TreeNodeModel."""
50
+
51
+ # Режимы отображения
52
+ TREENODE_DISPLAY_MODE_ACCORDION = 'accordion'
53
+ TREENODE_DISPLAY_MODE_BREADCRUMBS = 'breadcrumbs'
54
+ treenode_display_mode = TREENODE_DISPLAY_MODE_ACCORDION
55
+
56
+ form = TreeNodeForm
57
+ importer_class = None
58
+ exporter_class = None
59
+ ordering = []
60
+
61
+ formfield_overrides = {
62
+ models.ForeignKey: {'widget': TreeWidget()},
63
+ }
64
+
65
+ change_list_template = "treenode/admin/treenode_changelist.html"
66
+ import_export = True
67
+
68
+ class Media:
69
+ """Meta Class."""
70
+
71
+ css = {"all": (
72
+ "css/treenode_admin.css",
73
+ "vendors/jquery-ui/jquery-ui.css",
74
+ )}
75
+ js = (
76
+ "vendors/jquery-ui/jquery-ui.js",
77
+ # "js/lz-string.min.js",
78
+ "js/treenode_admin.js",
79
+ )
80
+
81
+ def __init__(self, model, admin_site):
82
+ """Init method."""
83
+ super().__init__(model, admin_site)
84
+
85
+ if not self.list_display:
86
+ self.list_display = [field.name for field in model._meta.fields]
87
+
88
+ self.TreeNodeImporter = self.importer_class or TreeNodeImporter
89
+ self.TreeNodeExporter = self.exporter_class or TreeNodeExporter
90
+
91
+ def drag(self, obj):
92
+ """Drag and drop сolumn."""
93
+ return mark_safe('<span class="treenode-drag-handle">&#9776;</span>')
94
+
95
+ drag.short_description = _("Move")
96
+
97
+ def toggle(self, obj):
98
+ """Toggle column."""
99
+ if obj.get_children_count() > 0:
100
+ return mark_safe(
101
+ f'<button class="treenode-toggle" data-node-id="{obj.pk}">►</button>' # noqa
102
+ )
103
+ return mark_safe('<div class="treenode-space">&nbsp;</div>')
104
+
105
+ toggle.short_description = _("Expand")
106
+
107
+ def get_changelist(self, request, **kwargs):
108
+ """Get changelist."""
109
+ ChangeList = super().get_changelist(request, **kwargs)
110
+
111
+ class NoPaginationChangeList(ChangeList):
112
+ """Suppress pagination."""
113
+
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)
123
+
124
+ return NoPaginationChangeList
125
+
126
+ def get_changelist_instance(self, request):
127
+ """
128
+ Get changelist instance.
129
+
130
+ Make sure our custom ChangeList is used without pagination.
131
+ """
132
+ ChangeList = self.get_changelist(request)
133
+
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
+ )
153
+
154
+ def get_queryset(self, request):
155
+ """Get queryset."""
156
+ qs = super().get_queryset(request)
157
+ return qs.select_related('parent')\
158
+ .prefetch_related('children')\
159
+ .order_by('_path')
160
+
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)
165
+
166
+ description = str(self.model._meta.verbose_name)
167
+ treenode_field.short_description = description
168
+
169
+ return (self.drag, self.toggle, treenode_field)
170
+
171
+ def get_form(self, request, obj=None, **kwargs):
172
+ """Get Form method."""
173
+ form = super().get_form(request, obj, **kwargs)
174
+ if "parent" in form.base_fields:
175
+ form.base_fields["parent"].widget = TreeWidget()
176
+ return form
177
+
178
+ def get_search_fields(self, request):
179
+ """Get search fields."""
180
+ return [getattr(self.model, 'display_field', 'id') or 'id']
181
+
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]
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()]
212
+
213
+ text = "/" + "/".join([escape(label) for label in breadcrumbs])
214
+
215
+ content = f'{padding}{icon}<a href="{edit_url}">{escape(text)}</a>{closing}' # noqa
216
+ return mark_safe(content)
217
+
218
+ def get_list_per_page(self, request):
219
+ """Get list per page."""
220
+ return 999999
221
+
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;
@@ -0,0 +1,322 @@
1
+ /**
2
+ * treenode_admin.js
3
+ *
4
+ * Cleaned version 3.1.0 — TreeNode Admin extension for Django Admin.
5
+ * - No AJAX loading of children
6
+ * - No persistence in localStorage
7
+ * - Local-only expand/collapse logic
8
+ * - Compatible with pre-rendered full tree
9
+ *
10
+ * Version: 3.1.0
11
+ * Author: Timur Kady
12
+ * Email: timurkady@yandex.com
13
+ */
14
+
15
+ (function($) {
16
+
17
+ // ------------------------------- //
18
+ // Helpers //
19
+ // ------------------------------- //
20
+
21
+ function getCookie(name) {
22
+ let cookieValue = null;
23
+ if (document.cookie && document.cookie !== '') {
24
+ const cookies = document.cookie.split(';');
25
+ for (let i = 0; i < cookies.length; i++) {
26
+ const cookie = cookies[i].trim();
27
+ if (cookie.startsWith(name + '=')) {
28
+ cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
29
+ break;
30
+ }
31
+ }
32
+ }
33
+ return cookieValue;
34
+ }
35
+
36
+ // ------------------------------- //
37
+ // AJAX Setup //
38
+ // ------------------------------- //
39
+
40
+ const csrftoken = getCookie('csrftoken');
41
+
42
+ $.ajaxSetup({
43
+ beforeSend: function(xhr, settings) {
44
+ if (!/^https?:.*/.test(settings.url)) {
45
+ xhr.setRequestHeader("X-CSRFToken", csrftoken);
46
+ }
47
+ $('body').css('cursor', 'wait');
48
+ },
49
+ complete: function() {
50
+ $('body').css('cursor', 'default');
51
+ },
52
+ cache: false
53
+ });
54
+
55
+ // ------------------------------- //
56
+ // Visual feedback //
57
+ // ------------------------------- //
58
+
59
+ const TreeFx = {
60
+ flashInsert(nodeId) {
61
+ const $row = $(`tr[data-node-id="${nodeId}"]`);
62
+ if (!$row.length) return;
63
+
64
+ $row.addClass("flash-insert");
65
+ setTimeout(() => $row.removeClass("flash-insert"), 1000);
66
+ },
67
+
68
+ markDragging($item, enable) {
69
+ $item.toggleClass('dragging', enable);
70
+ }
71
+ };
72
+
73
+ // ------------------------------- //
74
+ // Tree logic //
75
+ // ------------------------------- //
76
+
77
+ var ChangeList = {
78
+ $tableBody: null,
79
+ isShiftPressed: false,
80
+ activeTargetRow: null,
81
+ isMoving: false,
82
+
83
+ init: function() {
84
+ this.$tableBody = $('table#result_list tbody');
85
+ this.bindEvents();
86
+ this.enableDragAndDrop();
87
+ },
88
+
89
+ // Save expanded nodes to localStorage
90
+ saveTree: function() {
91
+ if (!this.$tableBody) return;
92
+
93
+ // Сохраняем ТОЛЬКО те node_id, которые раскрыты
94
+ this.expandedNodes = this.$tableBody.find(".treenode-toggle").map(function() {
95
+ const $btn = $(this);
96
+ return $btn.data("expanded") ? $btn.data("node-id") : null;
97
+ }).get().filter(Boolean);
98
+
99
+ if (this.expandedNodes.length === 0) {
100
+ localStorage.removeItem("treenode_expanded");
101
+ } else {
102
+ localStorage.setItem("treenode_expanded", JSON.stringify(this.expandedNodes));
103
+ }
104
+
105
+ const count = $("#result_list tbody tr").length;
106
+ if (count > 0) {
107
+ $("p.paginator").first().text(`${count} ${ChangeList.label}`);
108
+ localStorage.setItem("label", ChangeList.label);
109
+ }
110
+ },
111
+
112
+
113
+ // Restore expanded nodes from localStorage
114
+ restoreTreeState: function() {
115
+ const expanded = JSON.parse(localStorage.getItem("treenode_expanded") || "[]");
116
+ for (const nodeId of expanded) {
117
+ ChangeList.expandNode(nodeId);
118
+ }
119
+ },
120
+
121
+ expandNode: function(nodeId) {
122
+ const $row = this.$tableBody.find(`tr[data-node-id="${nodeId}"]`);
123
+ const $btn = $row.find(".treenode-toggle");
124
+ $btn.text("▼").data("expanded", true);
125
+ this.showChildren(nodeId);
126
+ this.saveTree();
127
+ },
128
+
129
+ collapseNode: function(nodeId) {
130
+ const $row = this.$tableBody.find(`tr[data-node-id="${nodeId}"]`);
131
+ const $btn = $row.find(".treenode-toggle");
132
+ $btn.text("►").data("expanded", false); // <-- ВАЖНО!
133
+ this.hideChildrenRecursive(nodeId);
134
+ this.saveTree();
135
+ },
136
+
137
+ toggleNode: function($btn) {
138
+ const nodeId = $btn.data("node-id");
139
+ const isExpanded = $btn.data("expanded");
140
+ if (isExpanded) {
141
+ this.collapseNode(nodeId);
142
+ } else {
143
+ this.expandNode(nodeId);
144
+ }
145
+ },
146
+
147
+ showChildren: function(parentId) {
148
+ const $children = this.$tableBody.find(`tr[data-parent-id="${parentId}"]`);
149
+ $children.removeClass("treenode-hidden");
150
+
151
+ $children.each((_, child) => {
152
+ const $child = $(child);
153
+ const childId = $child.data("node-id");
154
+ const $toggle = $child.find(".treenode-toggle");
155
+ if ($toggle.data("expanded")) {
156
+ this.showChildren(childId);
157
+ }
158
+ });
159
+ },
160
+
161
+ hideChildrenRecursive: function(parentId) {
162
+ const $children = this.$tableBody.find(`tr[data-parent-id="${parentId}"]`);
163
+ $children.addClass("treenode-hidden");
164
+
165
+ $children.each((_, child) => {
166
+ const childId = $(child).data("node-id");
167
+ this.hideChildrenRecursive(childId);
168
+ });
169
+ },
170
+
171
+ expandAll: function() {
172
+ const self = this;
173
+ this.$tableBody.find("button.treenode-toggle").each(function() {
174
+ self.expandNode($(this).data("node-id"));
175
+ });
176
+ },
177
+
178
+ collapseAll: function() {
179
+ const self = this;
180
+ this.$tableBody.find("button.treenode-toggle").each(function() {
181
+ self.collapseNode($(this).data("node-id"));
182
+ });
183
+ },
184
+
185
+ bindEvents: function() {
186
+ const self = this;
187
+
188
+ $(document).on("click", "button.treenode-toggle", function(e) {
189
+ e.preventDefault();
190
+ self.toggleNode($(this));
191
+ });
192
+
193
+ $(document).on("click", ".treenode-expand-all", function() {
194
+ self.expandAll();
195
+ });
196
+
197
+ $(document).on("click", ".treenode-collapse-all", function() {
198
+ self.collapseAll();
199
+ });
200
+
201
+ $(document).on("keydown", function(e) {
202
+ if (e.key === "Shift") {
203
+ self.isShiftPressed = true;
204
+ self.updateDndHighlight();
205
+ }
206
+ }).on("keyup", function(e) {
207
+ if (e.key === "Shift") {
208
+ self.isShiftPressed = false;
209
+ self.updateDndHighlight();
210
+ }
211
+ });
212
+
213
+ $('#result_list').on('change', '#action-toggle', function() {
214
+ $('input.action-select').prop('checked', this.checked);
215
+ });
216
+ },
217
+
218
+ enableDragAndDrop: function() {
219
+ const self = this;
220
+
221
+ this.$tableBody.sortable({
222
+ items: "tr",
223
+ handle: ".treenode-drag-handle",
224
+ placeholder: "treenode-placeholder",
225
+ helper: function(e, tr) {
226
+ const $originals = tr.children();
227
+ const $helper = tr.clone();
228
+ $helper.find('[id]').removeAttr('id');
229
+ $helper.children().each(function(index) {
230
+ $(this).width($originals.eq(index).width());
231
+ });
232
+ return $helper;
233
+ },
234
+ start: function(e, ui) {
235
+ TreeFx.markDragging(ui.item, true);
236
+ },
237
+ over: function(e, ui) {
238
+ self.updateDndHighlight();
239
+ },
240
+ stop: function(e, ui) {
241
+ TreeFx.markDragging(ui.item, false);
242
+ const $item = ui.item;
243
+ const nodeId = $item.data("node-id");
244
+ const prevId = $item.prev().data("node-id") || null;
245
+ const mode = self.isShiftPressed ? 'child' : 'after';
246
+
247
+ if (self.activeTargetRow) {
248
+ self.activeTargetRow.removeClass("target-as-child");
249
+ self.activeTargetRow = null;
250
+ }
251
+
252
+ self.applyMove(nodeId, prevId, mode);
253
+ TreeFx.flashInsert(nodeId);
254
+ },
255
+ });
256
+ },
257
+
258
+ updateDndHighlight: function() {
259
+ const $placeholder = this.$tableBody.find("tr.treenode-placeholder");
260
+ const $target = $placeholder.prev();
261
+
262
+ if (!$target.length || !$target.data("node-id")) {
263
+ this.$tableBody.find("tr.target-as-child").removeClass("target-as-child");
264
+ this.activeTargetRow = null;
265
+ return;
266
+ }
267
+
268
+ if (this.isShiftPressed) {
269
+ if (!this.activeTargetRow || !this.activeTargetRow.is($target)) {
270
+ this.$tableBody.find("tr.target-as-child").removeClass("target-as-child");
271
+ $target.addClass("target-as-child");
272
+ this.activeTargetRow = $target;
273
+ }
274
+ } else {
275
+ if (this.activeTargetRow) {
276
+ this.activeTargetRow.removeClass("target-as-child");
277
+ this.activeTargetRow = null;
278
+ }
279
+ }
280
+ },
281
+
282
+ applyMove: function(nodeId, targetId, mode) {
283
+ if (this.isMoving) return;
284
+
285
+ this.isMoving = true;
286
+ this.activeTargetRow = null;
287
+
288
+ const params = {
289
+ node_id: nodeId,
290
+ target_id: targetId,
291
+ mode: mode
292
+ };
293
+
294
+ $.ajax({
295
+ url: 'move/',
296
+ method: 'POST',
297
+ data: params,
298
+ dataType: 'json',
299
+ success: function(data) {
300
+ const msg = data.message || "Node moved successfully.";
301
+ $("<li class='success'>" + msg + "</li>").appendTo(".messagelist");
302
+ },
303
+ error: function(xhr, status, error) {
304
+ const fallback = "Error moving node.";
305
+ $("<li class='error'>" + (xhr.responseText || fallback) + "</li>").appendTo(".messagelist");
306
+ },
307
+ complete: function() {
308
+ ChangeList.isMoving = false;
309
+ location.reload();
310
+ }
311
+ });
312
+ }
313
+ };
314
+
315
+ $(document).ready(function () {
316
+ if ($("table#result_list").length) {
317
+ ChangeList.init();
318
+ ChangeList.restoreTreeState();
319
+ }
320
+ });
321
+
322
+ })(django.jQuery || window.jQuery);