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.
Files changed (70) hide show
  1. lino/__init__.py +8 -3
  2. lino/api/dd.py +11 -35
  3. lino/api/doctest.py +49 -17
  4. lino/api/selenium.py +1 -1
  5. lino/core/actions.py +25 -23
  6. lino/core/actors.py +52 -23
  7. lino/core/choicelists.py +10 -8
  8. lino/core/dbtables.py +1 -1
  9. lino/core/elems.py +47 -31
  10. lino/core/fields.py +19 -9
  11. lino/core/kernel.py +26 -20
  12. lino/core/model.py +27 -16
  13. lino/core/renderer.py +2 -2
  14. lino/core/requests.py +103 -56
  15. lino/core/site.py +5 -5
  16. lino/core/store.py +5 -2
  17. lino/core/utils.py +12 -7
  18. lino/help_texts.py +7 -8
  19. lino/mixins/duplicable.py +6 -4
  20. lino/mixins/sequenced.py +17 -6
  21. lino/modlib/__init__.py +0 -2
  22. lino/modlib/changes/models.py +21 -10
  23. lino/modlib/checkdata/models.py +59 -24
  24. lino/modlib/comments/fixtures/demo2.py +12 -3
  25. lino/modlib/comments/models.py +7 -7
  26. lino/modlib/comments/ui.py +8 -5
  27. lino/modlib/export_excel/models.py +7 -5
  28. lino/modlib/extjs/__init__.py +2 -2
  29. lino/modlib/extjs/views.py +66 -22
  30. lino/modlib/help/config/makehelp/conf.tpl.py +1 -1
  31. lino/modlib/jinja/mixins.py +73 -0
  32. lino/modlib/jinja/models.py +6 -0
  33. lino/modlib/linod/__init__.py +1 -0
  34. lino/modlib/linod/choicelists.py +21 -0
  35. lino/modlib/linod/consumers.py +13 -4
  36. lino/modlib/linod/fixtures/__init__.py +0 -0
  37. lino/modlib/linod/fixtures/linod.py +32 -0
  38. lino/modlib/linod/management/commands/linod.py +6 -2
  39. lino/modlib/linod/mixins.py +18 -14
  40. lino/modlib/linod/models.py +4 -2
  41. lino/modlib/memo/mixins.py +2 -1
  42. lino/modlib/memo/parser.py +1 -1
  43. lino/modlib/notify/models.py +19 -11
  44. lino/modlib/printing/actions.py +47 -42
  45. lino/modlib/printing/choicelists.py +17 -15
  46. lino/modlib/printing/mixins.py +22 -20
  47. lino/modlib/publisher/models.py +5 -5
  48. lino/modlib/summaries/models.py +3 -2
  49. lino/modlib/system/models.py +28 -29
  50. lino/modlib/uploads/__init__.py +14 -11
  51. lino/modlib/uploads/actions.py +2 -8
  52. lino/modlib/uploads/choicelists.py +10 -10
  53. lino/modlib/uploads/fixtures/std.py +17 -0
  54. lino/modlib/uploads/mixins.py +20 -8
  55. lino/modlib/uploads/models.py +62 -38
  56. lino/modlib/uploads/ui.py +15 -9
  57. lino/utils/__init__.py +0 -1
  58. lino/utils/jscompressor.py +4 -4
  59. lino/utils/media.py +45 -23
  60. lino/utils/report.py +5 -4
  61. lino/utils/restify.py +2 -2
  62. lino/utils/soup.py +26 -8
  63. lino/utils/xml.py +19 -5
  64. {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/METADATA +1 -1
  65. {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/RECORD +68 -65
  66. lino/mixins/uploadable.py +0 -3
  67. lino/utils/requests.py +0 -55
  68. {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/WHEEL +0 -0
  69. {lino-25.2.2.dist-info → lino-25.3.0.dist-info}/licenses/AUTHORS.rst +0 -0
  70. {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 reording.
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 = "\u25B2" # ▲ Black up-pointing triangle
84
- # label = "" #
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
- label = "↓"
129
+ button_text = "↓"
124
130
  # label = "\u25bc" # triangular arrow down
125
131
  # label = "\u2193"
126
- label = "\u25BC" # ▼ Black down-pointing triangle
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(message=_("Renumbered {} of {} siblings.").format(n, qs.count()))
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
@@ -67,7 +67,5 @@ Enterprise Resources
67
67
  languages
68
68
  office
69
69
  smtpd
70
- uploads
71
-
72
70
 
73
71
  """
@@ -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("object_type", "object_id", verbose_name=_("Object"))
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("master_type", "master_id", verbose_name=_("Master"))
78
+ master = GenericForeignKey(
79
+ "master_type", "master_id", verbose_name=_("Master"))
78
80
 
79
- diff = dd.RichTextField(_("Changes"), format="plain", blank=True, editable=False)
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=(pv.date, pv.date + datetime.timedelta(1)))
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
- @dd.schedule_daily()
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" % (k, dd.obj2str(old), dd.obj2str(new)))
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, sender, dd.obj2str(sender, True))
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, sender, dd.obj2str(sender, True))
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(child)
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(child)
270
+ ChangeTypes.remove_child, request, master, sender, dd.full_model_name(
271
+ child)
261
272
  )
262
273
 
263
274
 
@@ -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 UpdateMessage(dd.Action):
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
- qs = Message.objects.filter(
51
- **gfk2lookup(Message.owner, owner, checker=chk)
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(dd.Action):
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
- if fix is None:
78
- fix = self.fix_them
79
- Message = rt.models.checkdata.Message
80
- gfk = Message.owner
81
- checkers = get_checkers_for(self.model)
82
- for obj in ar.selected_rows:
83
- assert isinstance(obj, self.model)
84
- qs = Message.objects.filter(**gfk2lookup(gfk, obj))
85
- qs.delete()
86
- for chk in checkers:
87
- chk.update_problems(ar, obj, False, fix)
88
- ar.set_response(refresh=True)
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(m)
309
- qs = Message.objects.filter(owner_type=ct, checker__in=checkers)
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
- @dd.background_task(every_unit="daily", every=1)
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:
@@ -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(user=u, comment=self).first()
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("comments.Comment", related_name="reactions_to_this")
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 *
@@ -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, observed_event=CommentEvents.created)
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
- kw["reply_to"] = constants.CHOICES_BLANK_FILTER_VALUE
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(**mkw).values_list("owner_id", flat=True)
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(hours=int(time[0]), minutes=int(time[1]))
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(value.__class__, value))
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(os.path.dirname(mf.name))
136
+ settings.SITE.makedirs_if_missing(mf.path.parent)
135
137
 
136
138
  # Render
137
- self.render(ar, mf.name)
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.get_url(ar.request))
143
+ ar.success(open_url=mf.url)
142
144
 
143
145
  def render(self, ar, file):
144
146
  workbook = ar2workbook(ar)
@@ -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: