2 Commits

Author SHA1 Message Date
  Dirk Alders 4d77be2721 Instructions moved to piki website 2 months ago
  Dirk Alders 18dd102fe2 Tree view added 2 months ago
9 changed files with 96 additions and 84 deletions
  1. 1
    61
      README.md
  2. 5
    0
      data/system-pages/tree/meta.json
  3. 2
    0
      data/system-pages/tree/page
  4. 3
    3
      pages/__init__.py
  5. 17
    8
      pages/context.py
  6. 1
    1
      pages/help.py
  7. 62
    6
      pages/page.py
  8. 4
    4
      pages/views.py
  9. 1
    1
      themes

+ 1
- 61
README.md View File

@@ -1,62 +1,2 @@
1 1
 # piki
2
-
3
-Piki is a minimal wiki.
4
-
5
-
6
-## Installation
7
-### Get the repository
8
-####Go to the subfolder, where you want to create your new Piki-Application (here ~/tmp)
9
-    cd ~/tmp
10
-#### Clone the repository
11
-    git clone https://git.mount-mockery.de/application/piki.git
12
-#### Change to your repository and initialise it completely
13
-    cd piki
14
-    git submodule init
15
-    git submodule update
16
-
17
-
18
-### Create your virtual environment
19
-#### Create python3 environment
20
-    python3 -m venv venv
21
-#### Activate the environment
22
-    source venv/bin/activate
23
-#### Install PaTT Requirements
24
-    pip install -r requirements.txt
25
-
26
-## Configuration and Initialisation of Piki
27
-### Create your config File
28
-#### Copy the config example
29
-    cp config_example/config.py .
30
-    chmod 700 config.py
31
-
32
-#### Set a secret key
33
-Edit config.py and add a SECRET_KEY. Generate the secret e.g by executing the following command:
34
-
35
-    python manage.py
36
-
37
-At the End of the error message you'll see a random secret:
38
-
39
-KeyError: "You need to create a config.py file including at least a SECRET_KEY definition (e.g.: --> **'HERE IS THE RANDOM SECRET ;-)'** <--)."
40
-
41
-
42
-
43
-### Create your initial database and first user for Patt
44
-    python manage.py migrate
45
-    python manage.py createsuperuser
46
-
47
-### Finalise Configuration
48
-Now there are two ways to finalise your configuration. The first way is for a test or development system. The other is for a production System.
49
-
50
-1. **Test or development System:** Edit config.py and set the Variable DEBUG to True.
51
-
52
-2. **Production System:** Edit config.py and set the Variable ALLOWED_HOSTS. Execute "python manage.py collectstatic" to create a folder including all static files. Then add PaTT to your server configuration. See also [Django Documnetation](https://docs.djangoproject.com/en/3.1/howto/deployment/) for further information.
53
-
54
-## Start the Test or development System
55
-### Go to the folder, where your PaTT-Application is locates (here ~/tmp/piki)
56
-    cd ~/tmp/piki
57
-
58
-###Activate your Virtual Environment
59
-    source activate
60
-
61
-###Start the Server
62
-    python manage.py runserver
2
+Piki is a minimal wiki. You find further information under https://piki.mount-mockery.de

+ 5
- 0
data/system-pages/tree/meta.json View File

@@ -0,0 +1,5 @@
1
+{
2
+    "modified_time": 1729022206,
3
+    "modified_user": "system-page",
4
+    "creation_time": 1729022206
5
+}

+ 2
- 0
data/system-pages/tree/page View File

@@ -0,0 +1,2 @@
1
+= Tree
2
+<<allpagestree>>

+ 3
- 3
pages/__init__.py View File

@@ -10,15 +10,15 @@ def params(**kwargs):
10 10
     return params
11 11
 
12 12
 
13
-def url_page(request, rel_path, **kwargs):
13
+def url_page(rel_path, **kwargs):
14 14
     return reverse('page-page', kwargs={'rel_path': rel_path}) + params(**kwargs)
15 15
 
16 16
 
17
-def url_helpview(request, page):
17
+def url_helpview(page):
18 18
     return reverse('page-helpview', kwargs={'page': page})
19 19
 
20 20
 
21
-def url_edit(request, rel_path, **kwargs):
21
+def url_edit(rel_path, **kwargs):
22 22
     return reverse('page-edit', kwargs={'rel_path': rel_path}) + params(**kwargs)
23 23
 
24 24
 

+ 17
- 8
pages/context.py View File

@@ -20,6 +20,7 @@ BACK_UID = 'back'
20 20
 EDIT_UID = 'edit'
21 21
 HELP_UID = 'help'
22 22
 INDEX_UID = 'index'
23
+TREE_UID = 'tree'
23 24
 NAVIGATION_ENTRY_UID = 'navigation-%s'
24 25
 
25 26
 
@@ -65,7 +66,7 @@ def menubar(context, request, caller_name, **kwargs):
65 66
     if not cms_mode_active(request):
66 67
         menubar_users(bar, request)
67 68
         add_help_menu(request, bar, "current_help_page" in kwargs)
68
-    add_index_menu(request, bar, kwargs.get("rel_path", ''))
69
+    add_nav_links(request, bar, kwargs.get("rel_path", ''))
69 70
     finalise_bar(request, bar)
70 71
 
71 72
 
@@ -100,7 +101,7 @@ def navigation_entry_parameters(request, path):
100 101
         NAVIGATION_ENTRY_UID % os.path.basename(path),          # uid
101 102
         '/' + os.path.basename(path),                           # name
102 103
         None,                                                   # icon
103
-        pages.url_page(request, path),                          # url
104
+        pages.url_page(path),                                   # url
104 105
         False,                                                  # left
105 106
         False                                                   # active
106 107
     )
@@ -111,21 +112,29 @@ def add_help_menu(request, bar, active):
111 112
         HELP_UID,                                   # uid
112 113
         _('Help'),                                  # name
113 114
         color_icon_url(request, 'help.png'),        # icon
114
-        pages.url_helpview(request, 'main'),        # url
115
+        pages.url_helpview('main'),                 # url
115 116
         True,                                       # left
116 117
         active                                      # active
117 118
     )
118 119
 
119 120
 
120
-def add_index_menu(request, bar, rel_path):
121
+def add_nav_links(request, bar, rel_path):
121 122
     bar.append_entry(
122 123
         INDEX_UID,                                  # uid
123 124
         _('Index'),                                 # name
124 125
         color_icon_url(request, 'edit.png'),        # icon
125
-        pages.url_page(request, 'index'),           # url
126
+        pages.url_page('index'),                    # url
126 127
         True,                                       # left
127 128
         request.path == "/page/index"               # active
128 129
     )
130
+    bar.append_entry(
131
+        TREE_UID,                                   # uid
132
+        _('Tree'),                                  # name
133
+        color_icon_url(request, 'tree.png'),        # icon
134
+        pages.url_page('tree'),                     # url
135
+        True,                                       # left
136
+        request.path == "/page/tree"                # active
137
+    )
129 138
 
130 139
 
131 140
 def add_edit_menu(request, bar, rel_path):
@@ -133,7 +142,7 @@ def add_edit_menu(request, bar, rel_path):
133 142
         EDIT_UID,                                   # uid
134 143
         _('Edit'),                                  # name
135 144
         color_icon_url(request, 'edit2.png'),       # icon
136
-        pages.url_edit(request, rel_path),          # url
145
+        pages.url_edit(rel_path),                   # url
137 146
         True,                                       # left
138 147
         False                                       # active
139 148
     )
@@ -156,7 +165,7 @@ def add_meta_menu(request, bar, rel_path):
156 165
             EDIT_UID,                                       # uid
157 166
             _('Page'),                                      # name
158 167
             color_icon_url(request, 'display.png'),         # icon
159
-            pages.url_page(request, rel_path),              # url
168
+            pages.url_page(rel_path),                       # url
160 169
             True,                                           # left
161 170
             False                                           # active
162 171
         )
@@ -165,7 +174,7 @@ def add_meta_menu(request, bar, rel_path):
165 174
             EDIT_UID,                                       # uid
166 175
             _('Meta'),                                      # name
167 176
             color_icon_url(request, 'info.png'),            # icon
168
-            pages.url_page(request, rel_path, meta=None),   # url
177
+            pages.url_page(rel_path, meta=None),            # url
169 178
             True,                                           # left
170 179
             False                                           # active
171 180
         )

+ 1
- 1
pages/help.py View File

@@ -131,7 +131,7 @@ def actionbar(context, request, current_help_page=None, **kwargs):
131 131
             HELP_UID + '-%s' % name.lower(),                        # uid
132 132
             _(name),                                                # name
133 133
             color_icon_url(request, num + '.png'),                  # icon
134
-            pages.url_helpview(request, name.lower()),              # url
134
+            pages.url_helpview(name.lower()),                       # url
135 135
             True,                                                   # left
136 136
             name.lower() == current_help_page,                      # active
137 137
         )

+ 62
- 6
pages/page.py View File

@@ -22,6 +22,7 @@ def full_path_all_pages(expression="*"):
22 22
     system_pages = fstools.dirlist(settings.SYSTEM_PAGES_ROOT, expression=expression, rekursive=False)
23 23
     system_pages = [os.path.join(settings.PAGES_ROOT, os.path.basename(path)) for path in system_pages]
24 24
     pages = fstools.dirlist(settings.PAGES_ROOT, expression=expression, rekursive=False)
25
+    # TODO: strip path, if page or meta.json is missing
25 26
     return list(set(system_pages + pages))
26 27
 
27 28
 
@@ -197,15 +198,15 @@ class page_django(page_data):
197 198
             name = _("Current")
198 199
             meta += f"| {name} \
199 200
                       | {timestamp_to_datetime(self._request, mtime)} \
200
-                      | [[{url_page(self._request, self.rel_path)} | Page]] \
201
-                      | [[{url_page(self._request, self.rel_path, meta=None)} | Meta]]\n"
201
+                      | [[{url_page(self.rel_path)} | Page]] \
202
+                      | [[{url_page(self.rel_path, meta=None)} | Meta]]\n"
202 203
             # History
203 204
             for num in reversed(hnl):
204 205
                 p = page_wrapped(self._request, self._path, history_version=num)
205 206
                 meta += f"| {num} \
206 207
                           | {timestamp_to_datetime(self._request, p.modified_time)} \
207
-                          | [[{url_page(self._request, p.rel_path, history=num)} | Page]] \
208
-                          | [[{url_page(self._request, p.rel_path, meta=None, history=num)} | Meta]] (with page changes)\n"
208
+                          | [[{url_page(p.rel_path, history=num)} | Page]] \
209
+                          | [[{url_page(p.rel_path, meta=None, history=num)} | Meta]] (with page changes)\n"
209 210
         # Diff
210 211
         html_diff = ""
211 212
         if self._history_version:
@@ -222,6 +223,8 @@ class page_django(page_data):
222 223
         macros = {
223 224
             "subpages": self.macro_subpages,
224 225
             "allpages": self.macro_allpages,
226
+            "subpagetree": self.macro_subpagetree,
227
+            "allpagestree": self.macro_allpagestree,
225 228
         }
226 229
         return mycreole.render(request, txt, self.attachment_path, macros=macros)
227 230
 
@@ -231,6 +234,7 @@ class page_django(page_data):
231 234
 
232 235
     def macro_subpages(self, *args, **kwargs):
233 236
         allpages = kwargs.pop("allpages", False)
237
+        tree = kwargs.pop("tree", False)
234 238
         #
235 239
 
236 240
         def parse_depth(s: str):
@@ -263,7 +267,19 @@ class page_django(page_data):
263 267
             self._request,
264 268
             [page_django(self._request, path) for path in full_path_all_pages(expression)]
265 269
         )
266
-        return pl.html_list(depth=depth, filter_str=filter_str, parent_rel_path=parent_rel_path)
270
+        if tree:
271
+            return page_tree(pl).html()
272
+        else:
273
+            return pl.html_list(depth=depth, filter_str=filter_str, parent_rel_path=parent_rel_path)
274
+
275
+    def macro_allpagestree(self, *args, **kwargs):
276
+        kwargs["allpages"] = True
277
+        kwargs["tree"] = True
278
+        return self.macro_subpages(*args, **kwargs)
279
+
280
+    def macro_subpagetree(self, * args, **kwargs):
281
+        kwargs["tree"] = True
282
+        return self.macro_subpages(*args, **kwargs)
267 283
 
268 284
 
269 285
 class page_list(list):
@@ -290,13 +306,50 @@ class page_list(list):
290 306
                     if last_char != first_char:
291 307
                         last_char = first_char
292 308
                         rv += f"=== {first_char}\n"
293
-                    rv += f"* [[{url_page(self._request, page.rel_path)} | {name} ]]\n"
309
+                    rv += f"* [[{url_page(page.rel_path)} | {name} ]]\n"
294 310
         return rv
295 311
 
296 312
     def html_list(self, depth=9999, filter_str='', parent_rel_path=''):
297 313
         return mycreole.render_simple(self.creole_list(depth, filter_str, parent_rel_path))
298 314
 
299 315
 
316
+class page_tree(dict):
317
+    T_PATTERN = "├── "
318
+    L_PATTERN = "└── "
319
+    I_PATTERN = "│   "
320
+    D_PATTERN = "   &nbsp;&nbsp;&nbsp;"
321
+
322
+    def __init__(self, pl: page_list):
323
+        super().__init__()
324
+        for page in pl:
325
+            store_item = self
326
+            for entry in page.rel_path.split("/"):
327
+                if not entry in store_item:
328
+                    store_item[entry] = {}
329
+                store_item = store_item[entry]
330
+
331
+    def html(self, rel_path=None, fill=""):
332
+        base = self
333
+        try:
334
+            for key in rel_path.split("/"):
335
+                base = base[key]
336
+        except AttributeError:
337
+            rel_path = ''
338
+        #
339
+        rv = ""
340
+        #
341
+        l = len(base)
342
+        for entry in sorted(list(base.keys())):
343
+            l -= 1
344
+            page_path = os.path.join(rel_path, entry)
345
+            page = page_wrapped(None, page_path)
346
+            if page.is_available():
347
+                entry = f'<a href="{url_page(page_path)}">{entry}</a>'
348
+            rv += fill + (self.L_PATTERN if l == 0 else self.T_PATTERN) + entry + "<br>\n"
349
+            rv += self.html(page_path, fill=fill+(self.D_PATTERN if l == 0 else self.I_PATTERN))
350
+        return rv
351
+
352
+
300 353
 class page_wrapped(object):
301 354
     """
302 355
     This class holds different page and meta instances and decides which will be used in which case.
@@ -395,6 +448,9 @@ class page_wrapped(object):
395 448
         rv = page.attachment_path
396 449
         return rv
397 450
 
451
+    def is_available(self):
452
+        return self._page.is_available() or self._system_page.is_available()
453
+
398 454
     @property
399 455
     def raw_page_src(self):
400 456
         page = self.__page_choose__()

+ 4
- 4
pages/views.py View File

@@ -23,7 +23,7 @@ logger = logging.getLogger(settings.ROOT_LOGGER_NAME).getChild(__name__)
23 23
 
24 24
 
25 25
 def root(request):
26
-    return HttpResponseRedirect(url_page(request, config.STARTPAGE))
26
+    return HttpResponseRedirect(url_page(config.STARTPAGE))
27 27
 
28 28
 
29 29
 def page(request, rel_path):
@@ -92,7 +92,7 @@ def edit(request, rel_path):
92 92
                     messages.edit_success(request)
93 93
                 else:
94 94
                     messages.edit_no_change(request)
95
-                return HttpResponseRedirect(url_page(request, rel_path))
95
+                return HttpResponseRedirect(url_page(rel_path))
96 96
             elif preview is not None:
97 97
                 form = EditForm(page_data=page_txt, page_tags=tags)
98 98
                 #
@@ -107,10 +107,10 @@ def edit(request, rel_path):
107 107
                 )
108 108
                 return render(request, 'pages/page_form.html', context=context)
109 109
             else:
110
-                return HttpResponseRedirect(url_page(request, rel_path))
110
+                return HttpResponseRedirect(url_page(rel_path))
111 111
     else:
112 112
         messages.permission_denied_msg_page(request, rel_path)
113
-        return HttpResponseRedirect(url_page(request, rel_path))
113
+        return HttpResponseRedirect(url_page(rel_path))
114 114
 
115 115
 
116 116
 def search(request):

+ 1
- 1
themes

@@ -1 +1 @@
1
-Subproject commit c63d8731fe85374ef74abf24b0d205bce24f66b1
1
+Subproject commit c13b45a7f736c0e8658fbd24ff826ad9b5d336e6

Loading…
Cancel
Save