lino 25.2.2__py3-none-any.whl → 25.3.0__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.
- lino/__init__.py +8 -3
- lino/api/dd.py +11 -35
- lino/api/doctest.py +49 -17
- lino/api/selenium.py +1 -1
- lino/core/actions.py +25 -23
- lino/core/actors.py +52 -23
- lino/core/choicelists.py +10 -8
- lino/core/dbtables.py +1 -1
- lino/core/elems.py +47 -31
- lino/core/fields.py +19 -9
- lino/core/kernel.py +26 -20
- lino/core/model.py +27 -16
- lino/core/renderer.py +2 -2
- lino/core/requests.py +103 -56
- lino/core/site.py +5 -5
- lino/core/store.py +5 -2
- lino/core/utils.py +12 -7
- lino/help_texts.py +7 -8
- lino/mixins/duplicable.py +6 -4
- lino/mixins/sequenced.py +17 -6
- lino/modlib/__init__.py +0 -2
- lino/modlib/changes/models.py +21 -10
- lino/modlib/checkdata/models.py +59 -24
- lino/modlib/comments/fixtures/demo2.py +12 -3
- lino/modlib/comments/models.py +7 -7
- lino/modlib/comments/ui.py +8 -5
- lino/modlib/export_excel/models.py +7 -5
- lino/modlib/extjs/__init__.py +2 -2
- lino/modlib/extjs/views.py +66 -22
- lino/modlib/help/config/makehelp/conf.tpl.py +1 -1
- lino/modlib/jinja/mixins.py +73 -0
- lino/modlib/jinja/models.py +6 -0
- lino/modlib/linod/__init__.py +1 -0
- lino/modlib/linod/choicelists.py +21 -0
- lino/modlib/linod/consumers.py +13 -4
- lino/modlib/linod/fixtures/__init__.py +0 -0
- lino/modlib/linod/fixtures/linod.py +32 -0
- lino/modlib/linod/management/commands/linod.py +6 -2
- lino/modlib/linod/mixins.py +18 -14
- lino/modlib/linod/models.py +4 -2
- lino/modlib/memo/mixins.py +2 -1
- lino/modlib/memo/parser.py +1 -1
- lino/modlib/notify/models.py +19 -11
- lino/modlib/printing/actions.py +47 -42
- lino/modlib/printing/choicelists.py +17 -15
- lino/modlib/printing/mixins.py +22 -20
- lino/modlib/publisher/models.py +5 -5
- lino/modlib/summaries/models.py +3 -2
- lino/modlib/system/models.py +28 -29
- lino/modlib/uploads/__init__.py +14 -11
- lino/modlib/uploads/actions.py +2 -8
- lino/modlib/uploads/choicelists.py +10 -10
- lino/modlib/uploads/fixtures/std.py +17 -0
- lino/modlib/uploads/mixins.py +20 -8
- lino/modlib/uploads/models.py +62 -38
- lino/modlib/uploads/ui.py +15 -9
- lino/utils/__init__.py +0 -1
- lino/utils/jscompressor.py +4 -4
- lino/utils/media.py +45 -23
- lino/utils/report.py +5 -4
- lino/utils/restify.py +2 -2
- lino/utils/soup.py +26 -8
- lino/utils/xml.py +19 -5
- {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/METADATA +1 -1
- {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/RECORD +68 -65
- lino/mixins/uploadable.py +0 -3
- lino/utils/requests.py +0 -55
- {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/WHEEL +0 -0
- {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/licenses/AUTHORS.rst +0 -0
- {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/licenses/COPYING +0 -0
lino/mixins/sequenced.py
CHANGED
@@ -33,7 +33,7 @@ class MoveByN(actions.Action):
|
|
33
33
|
This action is available on any :class:`Sequenced` object as
|
34
34
|
:attr:`Sequenced.move_by_n`.
|
35
35
|
|
36
|
-
It is currently only used by React to allow for drag and drop
|
36
|
+
It is currently only used by React to allow for drag and drop reordering.
|
37
37
|
|
38
38
|
"""
|
39
39
|
|
@@ -80,9 +80,11 @@ class MoveUp(actions.Action):
|
|
80
80
|
# label = _("Up")
|
81
81
|
# label = "\u2191" thin arrow up
|
82
82
|
# label = "\u25b2" # triangular arrow up
|
83
|
-
label = "
|
84
|
-
#
|
83
|
+
label = _("Move up")
|
84
|
+
# button_text = "\u25B2" # ▲ Black up-pointing triangle
|
85
|
+
button_text = "↑"
|
85
86
|
custom_handler = True
|
87
|
+
callable_from = "t"
|
86
88
|
# icon_name = 'arrow_up'
|
87
89
|
# ~ icon_file = 'arrow_up.png'
|
88
90
|
readonly = False
|
@@ -92,9 +94,12 @@ class MoveUp(actions.Action):
|
|
92
94
|
return False
|
93
95
|
if not super().get_action_permission(ar, obj, state):
|
94
96
|
return False
|
97
|
+
# if ar.order_by is None or "seqno" not in ar.order_by:
|
98
|
+
# return False
|
95
99
|
if ar.get_total_count() == 0:
|
96
100
|
return False
|
97
101
|
if ar.data_iterator[0] == obj:
|
102
|
+
# print(f"20250305 first of {ar.data_iterator}")
|
98
103
|
return False
|
99
104
|
# print("20161128", obj.seqno, ar.data_iterator.count())
|
100
105
|
return True
|
@@ -119,14 +124,16 @@ class MoveDown(actions.Action):
|
|
119
124
|
|
120
125
|
"""
|
121
126
|
|
127
|
+
label = _("Move down")
|
122
128
|
# label = _("Down")
|
123
|
-
|
129
|
+
button_text = "↓"
|
124
130
|
# label = "\u25bc" # triangular arrow down
|
125
131
|
# label = "\u2193"
|
126
|
-
|
132
|
+
# button_text = "\u25BC" # ▼ Black down-pointing triangle
|
127
133
|
# icon_name = 'arrow_down'
|
128
134
|
custom_handler = True
|
129
135
|
# ~ icon_file = 'arrow_down.png'
|
136
|
+
callable_from = "t"
|
130
137
|
readonly = False
|
131
138
|
|
132
139
|
def get_action_permission(self, ar, obj, state):
|
@@ -134,10 +141,13 @@ class MoveDown(actions.Action):
|
|
134
141
|
return False
|
135
142
|
if not super().get_action_permission(ar, obj, state):
|
136
143
|
return False
|
144
|
+
# if ar.order_by is None or "seqno" not in ar.order_by:
|
145
|
+
# return False
|
137
146
|
n = ar.get_total_count()
|
138
147
|
if n == 0:
|
139
148
|
return False
|
140
149
|
if ar.data_iterator[n - 1] == obj:
|
150
|
+
# print(f"20250305 last of {ar.data_iterator}")
|
141
151
|
return False
|
142
152
|
# ~ if obj.__class__.__name__=='Entry' and obj.seqno == 25:
|
143
153
|
# ~ print 20130706, ar.data_iterator.count(), ar.data_iterator
|
@@ -315,7 +325,8 @@ class Sequenced(Duplicable):
|
|
315
325
|
|
316
326
|
seq_no += 1
|
317
327
|
|
318
|
-
ar.success(
|
328
|
+
ar.success(
|
329
|
+
message=_("Renumbered {} of {} siblings.").format(n, qs.count()))
|
319
330
|
ar.set_response(refresh_all=True)
|
320
331
|
|
321
332
|
@fields.displayfield(_("Move"))
|
lino/modlib/__init__.py
CHANGED
lino/modlib/changes/models.py
CHANGED
@@ -64,7 +64,8 @@ class Change(UserAuthored):
|
|
64
64
|
related_name="changes_by_object",
|
65
65
|
)
|
66
66
|
object_id = GenericForeignKeyIdField(object_type, blank=True, null=True)
|
67
|
-
object = GenericForeignKey(
|
67
|
+
object = GenericForeignKey(
|
68
|
+
"object_type", "object_id", verbose_name=_("Object"))
|
68
69
|
|
69
70
|
master_type = dd.ForeignKey(
|
70
71
|
"contenttypes.ContentType",
|
@@ -74,9 +75,11 @@ class Change(UserAuthored):
|
|
74
75
|
related_name="changes_by_master",
|
75
76
|
)
|
76
77
|
master_id = GenericForeignKeyIdField(master_type, blank=True, null=True)
|
77
|
-
master = GenericForeignKey(
|
78
|
+
master = GenericForeignKey(
|
79
|
+
"master_type", "master_id", verbose_name=_("Master"))
|
78
80
|
|
79
|
-
diff = dd.RichTextField(_("Changes"), format="plain",
|
81
|
+
diff = dd.RichTextField(_("Changes"), format="plain",
|
82
|
+
blank=True, editable=False)
|
80
83
|
changed_fields = dd.CharField(_("Fields"), max_length=250, blank=True)
|
81
84
|
|
82
85
|
def __str__(self):
|
@@ -129,7 +132,8 @@ class Changes(dd.Table):
|
|
129
132
|
if pv.change_type:
|
130
133
|
qs = qs.filter(type=pv.change_type)
|
131
134
|
if pv.date:
|
132
|
-
qs = qs.filter(time__range=(
|
135
|
+
qs = qs.filter(time__range=(
|
136
|
+
pv.date, pv.date + datetime.timedelta(1)))
|
133
137
|
# if settings.SITE.user_model and ar.param_values.user:
|
134
138
|
# qs = qs.filter(user=ar.param_values.user)
|
135
139
|
if pv.object_type:
|
@@ -167,7 +171,9 @@ def log_change(type, request, master, obj, msg="", changed_fields=""):
|
|
167
171
|
remove_after = dd.plugins.changes.remove_after
|
168
172
|
if remove_after:
|
169
173
|
|
170
|
-
|
174
|
+
from lino.modlib.linod.choicelists import schedule_daily
|
175
|
+
|
176
|
+
@schedule_daily()
|
171
177
|
def delete_older_changes(ar):
|
172
178
|
days = datetime.timedelta(days=remove_after)
|
173
179
|
# django.core.exceptions.FieldError: Cannot resolve keyword 'time_lt' into field. Choices are: changed_fields, diff, id, list_item, master, master_id, master_type, master_type_id, name_column, navigation_panel, object, object_id, object_type, object_type_id, overview, time, type, user, user_id, workflow_buttons
|
@@ -206,7 +212,8 @@ def on_update(sender=None, watcher=None, request=None, **kw):
|
|
206
212
|
changes = []
|
207
213
|
for k, old, new in watcher.get_updates(cs.ignored_fields):
|
208
214
|
changed_fields += k + " "
|
209
|
-
changes.append("%s : %s --> %s" %
|
215
|
+
changes.append("%s : %s --> %s" %
|
216
|
+
(k, dd.obj2str(old), dd.obj2str(new)))
|
210
217
|
if len(changes) == 0:
|
211
218
|
msg = "(no changes)"
|
212
219
|
elif len(changes) == 1:
|
@@ -229,7 +236,8 @@ def on_delete(sender=None, request=None, **kw):
|
|
229
236
|
master = get_master(sender)
|
230
237
|
if master is None:
|
231
238
|
return
|
232
|
-
log_change(ChangeTypes.delete, request, master,
|
239
|
+
log_change(ChangeTypes.delete, request, master,
|
240
|
+
sender, dd.obj2str(sender, True))
|
233
241
|
|
234
242
|
|
235
243
|
@receiver(on_ui_created)
|
@@ -238,7 +246,8 @@ def on_ui_created(sender=None, request=None, **kw):
|
|
238
246
|
master = get_master(sender)
|
239
247
|
if master is None:
|
240
248
|
return
|
241
|
-
log_change(ChangeTypes.create, request, master,
|
249
|
+
log_change(ChangeTypes.create, request, master,
|
250
|
+
sender, dd.obj2str(sender, True))
|
242
251
|
|
243
252
|
|
244
253
|
@receiver(pre_add_child)
|
@@ -247,7 +256,8 @@ def on_add_child(sender=None, request=None, child=None, **kw):
|
|
247
256
|
if master is None:
|
248
257
|
return
|
249
258
|
log_change(
|
250
|
-
ChangeTypes.add_child, request, master, sender, dd.full_model_name(
|
259
|
+
ChangeTypes.add_child, request, master, sender, dd.full_model_name(
|
260
|
+
child)
|
251
261
|
)
|
252
262
|
|
253
263
|
|
@@ -257,7 +267,8 @@ def on_remove_child(sender=None, request=None, child=None, **kw):
|
|
257
267
|
if master is None:
|
258
268
|
return
|
259
269
|
log_change(
|
260
|
-
ChangeTypes.remove_child, request, master, sender, dd.full_model_name(
|
270
|
+
ChangeTypes.remove_child, request, master, sender, dd.full_model_name(
|
271
|
+
child)
|
261
272
|
)
|
262
273
|
|
263
274
|
|
lino/modlib/checkdata/models.py
CHANGED
@@ -12,6 +12,7 @@ from django.template.defaultfilters import pluralize
|
|
12
12
|
from lino.core.gfks import gfk2lookup
|
13
13
|
from lino.modlib.gfks.mixins import Controllable
|
14
14
|
from lino.modlib.users.mixins import UserAuthored
|
15
|
+
from lino.modlib.linod.choicelists import background_task
|
15
16
|
from lino.core.roles import SiteStaff
|
16
17
|
|
17
18
|
from lino.api import dd, rt, _
|
@@ -22,12 +23,27 @@ from .roles import CheckdataUser
|
|
22
23
|
from lino.core import constants
|
23
24
|
|
24
25
|
|
25
|
-
class
|
26
|
+
class CheckerAction(dd.Action):
|
27
|
+
fix_them = False
|
28
|
+
|
29
|
+
def run_it(self, ar, fix, checkers, objects):
|
30
|
+
if fix is None:
|
31
|
+
fix = self.fix_them
|
32
|
+
Message = rt.models.checkdata.Message
|
33
|
+
gfk = Message.owner
|
34
|
+
for obj in objects:
|
35
|
+
qs = Message.objects.filter(**gfk2lookup(gfk, obj))
|
36
|
+
qs.delete()
|
37
|
+
for chk in checkers:
|
38
|
+
chk.update_problems(ar, obj, False, fix)
|
39
|
+
ar.set_response(refresh=True)
|
40
|
+
|
41
|
+
|
42
|
+
class UpdateMessage(CheckerAction):
|
26
43
|
icon_name = "bell"
|
27
44
|
ui5_icon_name = "sap-icon://bell"
|
28
45
|
label = _("Check data")
|
29
46
|
combo_group = "checkdata"
|
30
|
-
fix_them = False
|
31
47
|
sort_index = 90
|
32
48
|
# custom_handler = True
|
33
49
|
# select_rows = False
|
@@ -47,11 +63,11 @@ class UpdateMessage(dd.Action):
|
|
47
63
|
# has been deleted.
|
48
64
|
obj.delete()
|
49
65
|
else:
|
50
|
-
|
51
|
-
|
52
|
-
)
|
53
|
-
qs.delete()
|
54
|
-
chk.update_problems(ar, owner, False, fix)
|
66
|
+
self.run_it(ar, fix, [chk], [owner])
|
67
|
+
# qs = Message.objects.filter(
|
68
|
+
# **gfk2lookup(Message.owner, owner, checker=chk))
|
69
|
+
# qs.delete()
|
70
|
+
# chk.update_problems(ar, owner, False, fix)
|
55
71
|
ar.set_response(refresh_all=True)
|
56
72
|
|
57
73
|
|
@@ -61,12 +77,11 @@ class FixProblem(UpdateMessage):
|
|
61
77
|
sort_index = 91
|
62
78
|
|
63
79
|
|
64
|
-
class UpdateMessagesByController(
|
80
|
+
class UpdateMessagesByController(CheckerAction):
|
65
81
|
icon_name = "bell"
|
66
82
|
ui5_icon_name = "sap-icon://bell"
|
67
83
|
label = _("Check data")
|
68
84
|
combo_group = "checkdata"
|
69
|
-
fix_them = False
|
70
85
|
required_roles = dd.login_required()
|
71
86
|
|
72
87
|
def __init__(self, model):
|
@@ -74,18 +89,19 @@ class UpdateMessagesByController(dd.Action):
|
|
74
89
|
super().__init__()
|
75
90
|
|
76
91
|
def run_from_ui(self, ar, fix=None):
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
92
|
+
self.run_it(ar, fix, get_checkers_for(self.model), ar.selected_rows)
|
93
|
+
# if fix is None:
|
94
|
+
# fix = self.fix_them
|
95
|
+
# Message = rt.models.checkdata.Message
|
96
|
+
# gfk = Message.owner
|
97
|
+
# checkers = get_checkers_for(self.model)
|
98
|
+
# for obj in ar.selected_rows:
|
99
|
+
# assert isinstance(obj, self.model)
|
100
|
+
# qs = Message.objects.filter(**gfk2lookup(gfk, obj))
|
101
|
+
# qs.delete()
|
102
|
+
# for chk in checkers:
|
103
|
+
# chk.update_problems(ar, obj, False, fix)
|
104
|
+
# ar.set_response(refresh=True)
|
89
105
|
|
90
106
|
|
91
107
|
class FixMessagesByController(UpdateMessagesByController):
|
@@ -93,6 +109,21 @@ class FixMessagesByController(UpdateMessagesByController):
|
|
93
109
|
fix_them = True
|
94
110
|
|
95
111
|
|
112
|
+
class FixAllProblems(CheckerAction):
|
113
|
+
select_rows = False
|
114
|
+
show_in_plain = True
|
115
|
+
# http_method = "POST"
|
116
|
+
label = _("Fix all data problems")
|
117
|
+
button_text = "✓" # u"\u2713"
|
118
|
+
fix_them = True
|
119
|
+
|
120
|
+
def run_from_ui(self, ar, fix=None):
|
121
|
+
mi = ar.master_instance
|
122
|
+
print(f"20250307 {mi}")
|
123
|
+
self.run_it(ar, fix, get_checkers_for(mi.__class__), [mi])
|
124
|
+
ar.set_response(refresh=True)
|
125
|
+
|
126
|
+
|
96
127
|
class Message(Controllable, UserAuthored):
|
97
128
|
class Meta(object):
|
98
129
|
app_label = "checkdata"
|
@@ -182,6 +213,8 @@ class MessagesByOwner(Messages):
|
|
182
213
|
column_names = "message checker user #fixable *"
|
183
214
|
default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
|
184
215
|
|
216
|
+
fix_all_problems = FixAllProblems()
|
217
|
+
|
185
218
|
|
186
219
|
# This was the first use case of a slave table with something else than a model
|
187
220
|
# instance as its master
|
@@ -305,8 +338,10 @@ def check_data(ar, args=[], fix=True, prune=False):
|
|
305
338
|
assert not m._meta.abstract
|
306
339
|
if settings.SITE.is_hidden_plugin(m._meta.app_label):
|
307
340
|
continue
|
308
|
-
ct = rt.models.contenttypes.ContentType.objects.get_for_model(
|
309
|
-
|
341
|
+
ct = rt.models.contenttypes.ContentType.objects.get_for_model(
|
342
|
+
m)
|
343
|
+
qs = Message.objects.filter(
|
344
|
+
owner_type=ct, checker__in=checkers)
|
310
345
|
qs.delete()
|
311
346
|
name = str(m._meta.verbose_name_plural)
|
312
347
|
qs = m.objects.all()
|
@@ -346,7 +381,7 @@ def check_data(ar, args=[], fix=True, prune=False):
|
|
346
381
|
ar.logger.info(msg, done, what, found, fixed)
|
347
382
|
|
348
383
|
|
349
|
-
@
|
384
|
+
@background_task(every_unit="daily", every=1)
|
350
385
|
def checkdata(ar):
|
351
386
|
"""Run all data checkers."""
|
352
387
|
check_data(ar, fix=False)
|
@@ -32,9 +32,21 @@ plain1 = "Some plain text."
|
|
32
32
|
plain2 = "Two paragraphs of plain text.\n\nThe second paragraph."
|
33
33
|
# plain2 += " With an 👁 (U+1F441)." #5855 (Jane fails to store certain unicode characters)
|
34
34
|
|
35
|
+
imageDataURL = """"""
|
36
|
+
body_with_img = f"""\
|
37
|
+
<p>Here is an image:</p>
|
38
|
+
<p><img src="{imageDataURL}" class="bar"></p>\
|
39
|
+
"""
|
40
|
+
|
35
41
|
BODIES = Cycler(
|
36
42
|
[styled, table, lorem, short_lorem, breaking, cond_comment, plain1, plain2]
|
37
43
|
)
|
44
|
+
# Add two empty bodies that will be filled later:
|
45
|
+
BODIES.items.insert(0, "")
|
46
|
+
BODIES.items.insert(0, "")
|
47
|
+
|
48
|
+
if dd.is_installed('blogs'):
|
49
|
+
BODIES.items.append(body_with_img)
|
38
50
|
|
39
51
|
|
40
52
|
def objects():
|
@@ -44,9 +56,6 @@ def objects():
|
|
44
56
|
# use_linod = settings.SITE.use_linod
|
45
57
|
# settings.SITE.use_linod = False
|
46
58
|
|
47
|
-
# Add two empty bodies that will be filled later:
|
48
|
-
BODIES.items.insert(0, "")
|
49
|
-
BODIES.items.insert(0, "")
|
50
59
|
MENTIONED = Cycler()
|
51
60
|
for model in rt.models_by_base(Commentable):
|
52
61
|
if model.memo_command is not None:
|
lino/modlib/comments/models.py
CHANGED
@@ -3,6 +3,8 @@
|
|
3
3
|
# License: GNU Affero General Public License v3 (see file COPYING for details)
|
4
4
|
|
5
5
|
# from html import escape
|
6
|
+
from .ui import *
|
7
|
+
from lino.modlib.checkdata.choicelists import Checker
|
6
8
|
from django.contrib.humanize.templatetags.humanize import naturaltime
|
7
9
|
from django.db import models
|
8
10
|
from django.db.models import Q
|
@@ -10,7 +12,7 @@ from django.core import validators
|
|
10
12
|
from django.utils.html import mark_safe, format_html, SafeString
|
11
13
|
from django.contrib.contenttypes.models import ContentType
|
12
14
|
from django.conf import settings
|
13
|
-
from lino.utils.html import E, tostring, fromstring
|
15
|
+
# from lino.utils.html import E, tostring, fromstring
|
14
16
|
|
15
17
|
from lino.api import dd, rt, _
|
16
18
|
|
@@ -144,7 +146,8 @@ class Comment(
|
|
144
146
|
u = ar.get_user()
|
145
147
|
if u.is_anonymous:
|
146
148
|
return
|
147
|
-
mr = rt.models.comments.Reaction.objects.filter(
|
149
|
+
mr = rt.models.comments.Reaction.objects.filter(
|
150
|
+
user=u, comment=self).first()
|
148
151
|
if mr:
|
149
152
|
return mr.emotion
|
150
153
|
|
@@ -399,7 +402,8 @@ class Reaction(CreatedModified, UserAuthored, DateRangeObservable):
|
|
399
402
|
|
400
403
|
allow_cascaded_delete = "user comment"
|
401
404
|
|
402
|
-
comment = dd.ForeignKey(
|
405
|
+
comment = dd.ForeignKey(
|
406
|
+
"comments.Comment", related_name="reactions_to_this")
|
403
407
|
emotion = Emotions.field(default="ok")
|
404
408
|
|
405
409
|
def as_summary_item(self, ar, text=None, **kwargs):
|
@@ -418,8 +422,6 @@ class Reaction(CreatedModified, UserAuthored, DateRangeObservable):
|
|
418
422
|
#
|
419
423
|
# mp.register_django_model('comment', Comment)
|
420
424
|
|
421
|
-
from lino.modlib.checkdata.choicelists import Checker
|
422
|
-
|
423
425
|
|
424
426
|
class CommentChecker(Checker):
|
425
427
|
# temporary checker to fix #4084 (Comment.owner is empty when replying to a comment)
|
@@ -436,5 +438,3 @@ class CommentChecker(Checker):
|
|
436
438
|
|
437
439
|
|
438
440
|
CommentChecker.activate()
|
439
|
-
|
440
|
-
from .ui import *
|
lino/modlib/comments/ui.py
CHANGED
@@ -2,6 +2,7 @@
|
|
2
2
|
# Copyright 2013-2023 Rumma & Ko Ltd
|
3
3
|
# License: GNU Affero General Public License v3 (see file COPYING for details)
|
4
4
|
|
5
|
+
from lino.modlib.publisher.choicelists import PageFillers
|
5
6
|
from django.utils.translation import ngettext
|
6
7
|
from django.contrib.humanize.templatetags.humanize import naturaltime
|
7
8
|
from django.contrib.contenttypes.models import ContentType
|
@@ -125,7 +126,8 @@ class Comments(dd.Table):
|
|
125
126
|
|
126
127
|
@classmethod
|
127
128
|
def comments_created(cls, user, sd, ed):
|
128
|
-
pv = dict(user=user, start_date=sd, end_date=ed,
|
129
|
+
pv = dict(user=user, start_date=sd, end_date=ed,
|
130
|
+
observed_event=CommentEvents.created)
|
129
131
|
# pv = dict(start_date=sd, end_date=ed, observed_event=CommentEvents.created)
|
130
132
|
return cls.request(user=user, param_values=pv)
|
131
133
|
|
@@ -206,6 +208,7 @@ class CommentsByType(CommentsByX):
|
|
206
208
|
master_key = "comment_type"
|
207
209
|
column_names = "body created user *"
|
208
210
|
|
211
|
+
|
209
212
|
# TODO: rename CommentsByRFC to CommentsByOwner
|
210
213
|
class CommentsByRFC(CommentsByX):
|
211
214
|
master_key = "owner"
|
@@ -232,7 +235,8 @@ class CommentsByRFC(CommentsByX):
|
|
232
235
|
@classmethod
|
233
236
|
def param_defaults(cls, ar, **kw):
|
234
237
|
kw = super().param_defaults(ar, **kw)
|
235
|
-
|
238
|
+
if ar.display_mode == constants.DISPLAY_MODE_STORY:
|
239
|
+
kw["reply_to"] = constants.CHOICES_BLANK_FILTER_VALUE
|
236
240
|
return kw
|
237
241
|
|
238
242
|
# @classmethod
|
@@ -275,7 +279,8 @@ class CommentsByMentioned(CommentsByX):
|
|
275
279
|
assert not cls.model._meta.abstract
|
276
280
|
ct = ContentType.objects.get_for_model(cls.model)
|
277
281
|
mkw = gfk2lookup(Mention.target, mi, owner_type=ct)
|
278
|
-
mentions = Mention.objects.filter(
|
282
|
+
mentions = Mention.objects.filter(
|
283
|
+
**mkw).values_list("owner_id", flat=True)
|
279
284
|
# mentions = [o.comment_id for o in Mention.objects.filter(**mkw)]
|
280
285
|
# print(mkw, mentions)
|
281
286
|
# return super(CommentsByMentioned, cls).get_filter_kw(ar, **kw)
|
@@ -312,6 +317,4 @@ class ReactionsByComment(Reactions):
|
|
312
317
|
default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
|
313
318
|
|
314
319
|
|
315
|
-
from lino.modlib.publisher.choicelists import PageFillers
|
316
|
-
|
317
320
|
PageFillers.add_item(RecentComments)
|
@@ -86,7 +86,8 @@ def ar2workbook(ar, column_names=None):
|
|
86
86
|
time = time.strip("-")
|
87
87
|
negative = True
|
88
88
|
time = time.split(":")
|
89
|
-
value = datetime.timedelta(
|
89
|
+
value = datetime.timedelta(
|
90
|
+
hours=int(time[0]), minutes=int(time[1]))
|
90
91
|
if negative: # Make negative.
|
91
92
|
value = value - value - value
|
92
93
|
elif iselement(value) or isinstance(value, SafeString):
|
@@ -111,7 +112,8 @@ def ar2workbook(ar, column_names=None):
|
|
111
112
|
cell.style = style
|
112
113
|
cell.value = value
|
113
114
|
except ValueError as e:
|
114
|
-
raise Exception("20190222 {} {}".format(
|
115
|
+
raise Exception("20190222 {} {}".format(
|
116
|
+
value.__class__, value))
|
115
117
|
|
116
118
|
return workbook
|
117
119
|
|
@@ -131,14 +133,14 @@ class ExportExcelAction(actions.Action):
|
|
131
133
|
def run_from_ui(self, ar, **kw):
|
132
134
|
# Prepare tmp file
|
133
135
|
mf = TmpMediaFile(ar, "xlsx")
|
134
|
-
settings.SITE.makedirs_if_missing(
|
136
|
+
settings.SITE.makedirs_if_missing(mf.path.parent)
|
135
137
|
|
136
138
|
# Render
|
137
|
-
self.render(ar, mf.
|
139
|
+
self.render(ar, mf.path)
|
138
140
|
|
139
141
|
# Tell client that the action was successful and that it
|
140
142
|
# should open a new browser window on the generated file.
|
141
|
-
ar.success(open_url=mf.
|
143
|
+
ar.success(open_url=mf.url)
|
142
144
|
|
143
145
|
def render(self, ar, file):
|
144
146
|
workbook = ar2workbook(ar)
|
lino/modlib/extjs/__init__.py
CHANGED
@@ -250,12 +250,12 @@ class Plugin(Plugin):
|
|
250
250
|
),
|
251
251
|
url(
|
252
252
|
rx + r"choices/(?P<app_label>\w+)/(?P<rptname>\w+)/"
|
253
|
-
"(?P<fldname>\w+)$",
|
253
|
+
r"(?P<fldname>\w+)$",
|
254
254
|
views.Choices.as_view(),
|
255
255
|
),
|
256
256
|
url(
|
257
257
|
rx + r"apchoices/(?P<app_label>\w+)/(?P<actor>\w+)/"
|
258
|
-
"(?P<an>\w+)/(?P<field>\w+)$",
|
258
|
+
r"(?P<an>\w+)/(?P<field>\w+)$",
|
259
259
|
views.ActionParamChoices.as_view(),
|
260
260
|
),
|
261
261
|
# the thread_id can be a negative number:
|