lino 25.3.3__py3-none-any.whl → 25.4.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 (38) hide show
  1. lino/__init__.py +7 -1
  2. lino/api/dd.py +1 -1
  3. lino/config/admin_main_base.html +4 -3
  4. lino/help_texts.py +12 -14
  5. lino/locale/bn/LC_MESSAGES/django.po +1112 -907
  6. lino/locale/de/LC_MESSAGES/django.mo +0 -0
  7. lino/locale/de/LC_MESSAGES/django.po +1066 -918
  8. lino/locale/django.pot +1041 -900
  9. lino/locale/es/LC_MESSAGES/django.po +1103 -902
  10. lino/locale/et/LC_MESSAGES/django.po +1103 -915
  11. lino/locale/fr/LC_MESSAGES/django.po +1105 -916
  12. lino/locale/nl/LC_MESSAGES/django.po +1102 -917
  13. lino/locale/pt_BR/LC_MESSAGES/django.po +1099 -900
  14. lino/locale/zh_Hant/LC_MESSAGES/django.po +1099 -900
  15. lino/management/commands/buildcache.py +48 -3
  16. lino/management/commands/prep.py +1 -1
  17. lino/mixins/duplicable.py +1 -1
  18. lino/modlib/__init__.py +0 -1
  19. lino/modlib/checkdata/models.py +33 -15
  20. lino/modlib/comments/fixtures/demo2.py +1 -9
  21. lino/modlib/extjs/views.py +2 -3
  22. lino/modlib/linod/mixins.py +94 -41
  23. lino/modlib/linod/models.py +1 -1
  24. lino/modlib/memo/__init__.py +4 -0
  25. lino/modlib/memo/models.py +47 -1
  26. lino/modlib/users/ui.py +4 -2
  27. lino/modlib/weasyprint/__init__.py +12 -20
  28. lino/modlib/weasyprint/config/weasyprint/base.weasy.html +13 -2
  29. lino/sphinxcontrib/logo/__init__.py +2 -1
  30. lino/utils/__init__.py +1 -0
  31. lino/utils/dbhash.py +5 -3
  32. lino/utils/format_date.py +9 -4
  33. {lino-25.3.3.dist-info → lino-25.4.0.dist-info}/METADATA +1 -1
  34. {lino-25.3.3.dist-info → lino-25.4.0.dist-info}/RECORD +37 -38
  35. lino/management/commands/buildsite.py +0 -67
  36. {lino-25.3.3.dist-info → lino-25.4.0.dist-info}/WHEEL +0 -0
  37. {lino-25.3.3.dist-info → lino-25.4.0.dist-info}/licenses/AUTHORS.rst +0 -0
  38. {lino-25.3.3.dist-info → lino-25.4.0.dist-info}/licenses/COPYING +0 -0
@@ -1,13 +1,58 @@
1
1
  # -*- coding: UTF-8 -*-
2
- # Copyright 2023 Rumma & Ko Ltd.
2
+ # Copyright 2009-2023 Rumma & Ko Ltd.
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
- from django.core.management.base import BaseCommand
5
+ from click import confirm
6
+ from django.core.management.base import BaseCommand, CommandError
7
+ from django.core.management import call_command
6
8
  from django.conf import settings
9
+ from lino import logger
7
10
 
8
11
 
9
12
  class Command(BaseCommand):
13
+ """Build the site cache files and run collectstatic for this Lino site."""
14
+
15
+ def add_arguments(self, parser):
16
+ super().add_arguments(parser)
17
+ parser.add_argument(
18
+ "-b", "--batch", "--noinput",
19
+ action="store_false",
20
+ dest="interactive",
21
+ default=True,
22
+ help="Do not prompt for input of any kind.",
23
+ ),
24
+
10
25
  def handle(self, *args, **options):
26
+ interactive = options.get("interactive")
11
27
  verbosity = options.get("verbosity")
12
- # print("20231005", verbosity)
28
+ project_dir = settings.SITE.project_dir
29
+
30
+ options = dict(interactive=False, verbosity=verbosity)
31
+
32
+ if interactive:
33
+ msg = "Build everything for ({})".format(project_dir)
34
+ msg += ".\nAre you sure?"
35
+ if not confirm(msg, default=True):
36
+ raise CommandError("User abort.")
37
+
38
+ # the following log message was useful on Travis 20150104
39
+ if verbosity > 0:
40
+ logger.info("`buildcache` started on %s.", project_dir)
41
+
42
+ # pth = project_dir / "settings.py"
43
+ # if pth.exists():
44
+ # pth.touch()
45
+
46
+ call_command("collectstatic", **options)
47
+
13
48
  settings.SITE.build_site_cache(force=True, verbosity=verbosity)
49
+
50
+ # if settings.SITE.is_installed("help"):
51
+ # call_command("makehelp", verbosity=verbosity)
52
+
53
+ # for p in settings.SITE.installed_plugins:
54
+ # p.on_buildsite(settings.SITE, verbosity=verbosity)
55
+
56
+ # settings.SITE.clear_site_config()
57
+
58
+ logger.info("`buildcache` finished on %s.", project_dir)
@@ -15,7 +15,7 @@ class Command(BaseCommand):
15
15
  super().add_arguments(parser)
16
16
  (
17
17
  parser.add_argument(
18
- "--noinput",
18
+ "-b", "--batch", "--noinput",
19
19
  action="store_false",
20
20
  dest="interactive",
21
21
  default=True,
lino/mixins/duplicable.py CHANGED
@@ -29,7 +29,7 @@ class Duplicate(actions.Action):
29
29
  sort_index = 11
30
30
  show_in_workflow = False
31
31
  # readonly = False # like ShowInsert. See docs/blog/2012/0726
32
- callable_from = "t"
32
+ callable_from = "td"
33
33
 
34
34
  # required_roles = set([Expert])
35
35
 
lino/modlib/__init__.py CHANGED
@@ -55,7 +55,6 @@ Communication
55
55
 
56
56
  comments
57
57
  notify
58
- linod
59
58
 
60
59
  Enterprise Resources
61
60
  ====================
@@ -1,5 +1,5 @@
1
1
  # -*- coding: UTF-8 -*-
2
- # Copyright 2015-2023 Rumma & Ko Ltd
2
+ # Copyright 2015-2025 Rumma & Ko Ltd
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
5
  from collections import OrderedDict
@@ -9,18 +9,17 @@ from django.conf import settings
9
9
  from django.utils import translation
10
10
  from django.template.defaultfilters import pluralize
11
11
 
12
+ from lino import logger
13
+ from lino.core import constants
12
14
  from lino.core.gfks import gfk2lookup
13
15
  from lino.modlib.gfks.mixins import Controllable
14
16
  from lino.modlib.users.mixins import UserAuthored
15
17
  from lino.modlib.linod.choicelists import background_task
16
18
  from lino.core.roles import SiteStaff
17
-
18
19
  from lino.api import dd, rt, _
19
20
 
20
21
  from .choicelists import Checker, Checkers
21
-
22
22
  from .roles import CheckdataUser
23
- from lino.core import constants
24
23
 
25
24
 
26
25
  class CheckerAction(dd.Action):
@@ -32,6 +31,8 @@ class CheckerAction(dd.Action):
32
31
  Message = rt.models.checkdata.Message
33
32
  gfk = Message.owner
34
33
  for obj in objects:
34
+ if checkers is None:
35
+ checkers = get_checkers_for(obj.__class__)
35
36
  qs = Message.objects.filter(**gfk2lookup(gfk, obj))
36
37
  qs.delete()
37
38
  for chk in checkers:
@@ -84,12 +85,12 @@ class UpdateMessagesByController(CheckerAction):
84
85
  combo_group = "checkdata"
85
86
  required_roles = dd.login_required()
86
87
 
87
- def __init__(self, model):
88
- self.model = model
89
- super().__init__()
88
+ # def __init__(self, model=None, **kwargs):
89
+ # self.model = model
90
+ # super().__init__(**kwargs)
90
91
 
91
92
  def run_from_ui(self, ar, fix=None):
92
- self.run_it(ar, fix, get_checkers_for(self.model), ar.selected_rows)
93
+ self.run_it(ar, fix, None, ar.selected_rows)
93
94
  # if fix is None:
94
95
  # fix = self.fix_them
95
96
  # Message = rt.models.checkdata.Message
@@ -109,6 +110,16 @@ class FixMessagesByController(UpdateMessagesByController):
109
110
  fix_them = True
110
111
 
111
112
 
113
+ class QuickFixMessagesByController(UpdateMessagesByController):
114
+ # label = _("Fix data problems")
115
+ fix_them = True
116
+ combo_group = None
117
+ # icon_name = "lightning"
118
+ icon_name = None
119
+ button_text = ' ⚡ ' # 26A1
120
+ # button_text = "✓" # u"\u2713"
121
+
122
+
112
123
  class FixAllProblems(CheckerAction):
113
124
  select_rows = False
114
125
  show_in_plain = True
@@ -119,7 +130,7 @@ class FixAllProblems(CheckerAction):
119
130
 
120
131
  def run_from_ui(self, ar, fix=None):
121
132
  mi = ar.master_instance
122
- print(f"20250307 {mi}")
133
+ # print(f"20250307 {mi}")
123
134
  self.run_it(ar, fix, get_checkers_for(mi.__class__), [mi])
124
135
  ar.set_response(refresh=True)
125
136
 
@@ -264,11 +275,14 @@ def set_checkdata_actions(sender, **kw):
264
275
  if m is None:
265
276
  continue
266
277
  assert m is not Message
267
- m.define_action(check_data=UpdateMessagesByController(m))
268
- m.define_action(fix_problems=FixMessagesByController(m))
269
- if True:
270
- # don't add it automatically because appdev might prefer
271
- # to show it in a detail_layout:
278
+ # if hasattr(m, 'check_data'):
279
+ if (label := getattr(m, 'quickfix_checkdata_label', None)):
280
+ # print(f"20250324 Customized quickfix_checkdata_label {label} for {m}")
281
+ m.define_action(quick_fix=QuickFixMessagesByController(label=label))
282
+ else:
283
+ # print(f"20250324 Default checkdata buttons for {m}")
284
+ m.define_action(check_data=UpdateMessagesByController())
285
+ m.define_action(fix_problems=FixMessagesByController())
272
286
  m.define_action(
273
287
  show_problems=dd.ShowSlaveTable(
274
288
  MessagesByOwner,
@@ -280,7 +294,11 @@ def set_checkdata_actions(sender, **kw):
280
294
 
281
295
 
282
296
  def get_checkers_for(model):
283
- return get_checkable_models()[model]
297
+ checkers = []
298
+ for m, lst in get_checkable_models().items():
299
+ if m is not None and issubclass(model, m):
300
+ checkers += lst
301
+ return checkers
284
302
 
285
303
 
286
304
  def check_instance(obj, **kwargs):
@@ -4,6 +4,7 @@
4
4
  Adds some demo comments.
5
5
 
6
6
  """
7
+
7
8
  import datetime
8
9
  from lino.utils import i2t
9
10
  from lino.utils import Cycler
@@ -32,12 +33,6 @@ plain1 = "Some plain text."
32
33
  plain2 = "Two paragraphs of plain text.\n\nThe second paragraph."
33
34
  # plain2 += " With an 👁 (U+1F441)." #5855 (Jane fails to store certain unicode characters)
34
35
 
35
- imageDataURL = """"""
36
- body_with_img = f"""\
37
- <p>Here is an image:</p>
38
- <p><img src="{imageDataURL}" class="bar"></p>\
39
- """
40
-
41
36
  BODIES = Cycler(
42
37
  [styled, table, lorem, short_lorem, breaking, cond_comment, plain1, plain2]
43
38
  )
@@ -45,9 +40,6 @@ BODIES = Cycler(
45
40
  BODIES.items.insert(0, "")
46
41
  BODIES.items.insert(0, "")
47
42
 
48
- if dd.is_installed('blogs'):
49
- BODIES.items.append(body_with_img)
50
-
51
43
 
52
44
  def objects():
53
45
  Comment = rt.models.comments.Comment
@@ -321,8 +321,7 @@ NOT_FOUND = "%s has no row with primary key %r"
321
321
 
322
322
  class ApiElement(View):
323
323
  """
324
- The view that responds to
325
- ``r'api/(?P<app_label>\w+)/(?P<actor>\w+)/(?P<pk>[^/]+)$'``.
324
+ The view that responds to ``api/app_label/actor/pk``.
326
325
  """
327
326
 
328
327
  @method_decorator(ensure_csrf_cookie)
@@ -354,7 +353,7 @@ class ApiElement(View):
354
353
  ar = ba.create_request(
355
354
  request=request, renderer=settings.SITE.kernel.default_renderer)
356
355
  if pk and pk != "-99999" and pk != "-99998":
357
- if issubclass(rpt.model, models.Model):
356
+ if rpt.model is not None and issubclass(rpt.model, models.Model):
358
357
  try:
359
358
  ar.set_selected_pks(pk)
360
359
  except rpt.model.DoesNotExist:
@@ -38,6 +38,11 @@ class RunNow(dd.Action):
38
38
  # icon_name = 'bell'
39
39
  # icon_name = 'lightning'
40
40
 
41
+ # def get_action_permission(self, ar, obj, state):
42
+ # if obj.requested_at or obj.is_running():
43
+ # return False
44
+ # return super().get_action_permission(ar, obj, state)
45
+
41
46
  def run_from_ui(self, ar, **kwargs):
42
47
  # print("20231102 RunNow", ar.selected_rows)
43
48
  for obj in ar.selected_rows:
@@ -45,10 +50,11 @@ class RunNow(dd.Action):
45
50
  if True: # dd.plugins.linod.use_channels:
46
51
  obj.last_start_time = None
47
52
  obj.last_end_time = None
53
+ obj.requested_at = timezone.now()
48
54
  obj.message = "{} requested to run this task at {}.".format(
49
55
  ar.get_user(), dd.ftl(timezone.now())
50
56
  )
51
- obj.disabled = False
57
+ # obj.disabled = False
52
58
  obj.full_clean()
53
59
  obj.save()
54
60
  else:
@@ -57,6 +63,30 @@ class RunNow(dd.Action):
57
63
  ar.set_response(refresh=True)
58
64
 
59
65
 
66
+ class CancelRun(dd.Action):
67
+ label = _("Cancel request")
68
+ help_text = _("Cancel the request to run this task asap.")
69
+ select_rows = True
70
+ button_text = "🗙" # ⛒
71
+ # icon_name = 'bell'
72
+ # icon_name = 'lightning'
73
+
74
+ # def get_action_permission(self, ar, obj, state):
75
+ # if obj.requested_at is None:
76
+ # return False
77
+ # return super().get_action_permission(ar, obj, state)
78
+
79
+ def run_from_ui(self, ar, **kwargs):
80
+ for obj in ar.selected_rows:
81
+ assert issubclass(obj.__class__, Runnable)
82
+ obj.requested_at = None
83
+ obj.message = "{} cancelled the request to run this task.".format(
84
+ ar.get_user())
85
+ obj.full_clean()
86
+ obj.save()
87
+ ar.set_response(refresh=True)
88
+
89
+
60
90
  class Runnable(Sequenced, RecurrenceSet):
61
91
  class Meta:
62
92
  abstract = True
@@ -66,6 +96,7 @@ class Runnable(Sequenced, RecurrenceSet):
66
96
  last_start_time = dd.DateTimeField(
67
97
  _("Started at"), null=True, editable=False)
68
98
  last_end_time = dd.DateTimeField(_("Ended at"), null=True, editable=False)
99
+ requested_at = dd.DateTimeField(_("Requested at"), null=True, editable=False)
69
100
  message = dd.RichTextField(
70
101
  _("Logged messages"), format="plain", editable=False)
71
102
 
@@ -74,12 +105,23 @@ class Runnable(Sequenced, RecurrenceSet):
74
105
  name = models.CharField(_("Name"), max_length=200, blank=True)
75
106
 
76
107
  run_now = RunNow()
108
+ cancel_run = CancelRun()
77
109
 
78
110
  def __str__(self):
111
+ # r = "{} ({} #{})".format(
112
+ # self.name, self._meta.verbose_name, self.seqno)
79
113
  r = "{} #{} ({})".format(
80
114
  self._meta.verbose_name, self.seqno, self.name)
81
115
  return r
82
116
 
117
+ def disabled_fields(self, ar):
118
+ rv = super().disabled_fields(ar)
119
+ if self.requested_at is None:
120
+ rv.add('cancel_run')
121
+ else:
122
+ rv.add('run_now')
123
+ return rv
124
+
83
125
  def full_clean(self, *args, **kwargs):
84
126
  super().full_clean(*args, **kwargs)
85
127
  # 20250213 The following caused 'Invalid procedure invoicing.Task for
@@ -114,6 +156,7 @@ class Runnable(Sequenced, RecurrenceSet):
114
156
  # ar.info("Start %s with logging level %s", astr(self), self.log_level)
115
157
  # forget about any previous run:
116
158
  self.last_start_time = timezone.now()
159
+ self.requested_at = None
117
160
  self.last_end_time = None
118
161
  self.message = ""
119
162
  # print("20231102 full_clean")
@@ -150,17 +193,20 @@ class Runnable(Sequenced, RecurrenceSet):
150
193
  @dd.displayfield("Status")
151
194
  def status(self, ar=None):
152
195
  if self.is_running():
153
- return _("Running since {}").format(dd.ftl(self.last_start_time))
196
+ return _("Running since {}").format(dd.ftf(self.last_start_time))
197
+ if self.requested_at is not None:
198
+ return _("Requested to run asap (since {})").format(
199
+ dd.ftf(self.requested_at))
154
200
  if self.disabled:
155
201
  return _("Disabled")
156
202
  if self.last_start_time is None or self.last_end_time is None:
157
- if self.every_unit in (Recurrences.never, None):
203
+ if self.every_unit in {Recurrences.never, None}:
158
204
  return _("Not scheduled")
159
205
  return _("Scheduled to run asap")
160
206
  next_time = self.get_next_suggested_date(self.last_end_time)
161
207
  if next_time is None:
162
208
  return _("Not scheduled")
163
- return _("Scheduled to run at {}").format(dd.ftl(next_time))
209
+ return _("Scheduled to run at {}").format(dd.ftf(next_time))
164
210
 
165
211
 
166
212
  async def start_task_runner(ar=None, max_count=None):
@@ -177,53 +223,60 @@ async def start_task_runner(ar=None, max_count=None):
177
223
  next_time = now + \
178
224
  timedelta(seconds=dd.plugins.linod.background_sleep_time)
179
225
 
226
+ tasks = []
180
227
  for cls in Procedures.task_classes():
181
228
  # asyncio.ensure_future(m.start_task_runner(ar.spawn_request()))
182
229
  # print("20240424b")
183
- tasks = cls.objects.filter(disabled=False).order_by("seqno")
230
+ async for obj in cls.objects.filter(
231
+ requested_at__isnull=False).order_by("requested_at"):
232
+ tasks.append(obj)
233
+ for cls in Procedures.task_classes():
234
+ async for obj in cls.objects.filter(
235
+ requested_at__isnull=True, disabled=False).order_by("seqno"):
236
+ tasks.append(obj)
184
237
  # print("20240424c")
185
238
  # async for self in tasks:
186
- async for self in tasks:
187
- # print("20240424d")
188
- # raise Warning("20231230")
189
- if self.last_end_time is None and self.last_start_time is not None:
190
- run_duration = now - self.last_start_time
191
- if run_duration > timedelta(hours=2):
192
- msg = "Kill {} because it has been running more than 2 hours".format(
193
- astr(self)
194
- )
195
- await ar.adebug(msg)
196
- self.last_end_time = now
197
- self.message = msg
198
- await sync_to_async(self.full_clean)()
199
- # self.full_clean()
200
- await self.asave()
201
- # self.disabled = True
202
- else:
203
- await ar.adebug("Skip running task %s", astr(self))
204
- continue
205
-
206
- if self.last_end_time is not None:
207
- nst = await sync_to_async(self.get_next_suggested_date)(
208
- self.last_end_time, ar.logger
239
+ for self in tasks:
240
+ # print("20240424d")
241
+ # raise Warning("20231230")
242
+ if self.last_end_time is None and self.last_start_time is not None:
243
+ run_duration = now - self.last_start_time
244
+ if run_duration > timedelta(hours=2):
245
+ msg = "Killed {} because running more than 2 hours".format(
246
+ astr(self)
209
247
  )
210
- if nst is None:
211
- await ar.adebug("No time suggested to start %s", astr(self))
212
- continue
213
- if nst > now:
214
- await ar.adebug("Too early to start %s", astr(self))
215
- next_time = min(next_time, nst)
216
- continue
217
-
218
- # await ar.adebug("Start %s", self)
219
- # print("20231021 1 gonna start", self)
220
- await self.start_task(ar)
221
- assert self.last_end_time is not None
248
+ await ar.adebug(msg)
249
+ self.last_end_time = now
250
+ self.message = msg
251
+ await sync_to_async(self.full_clean)()
252
+ # self.full_clean()
253
+ await self.asave()
254
+ # self.disabled = True
255
+ else:
256
+ await ar.adebug("Skip running task %s", astr(self))
257
+ continue
258
+
259
+ if self.requested_at is None and self.last_end_time is not None:
222
260
  nst = await sync_to_async(self.get_next_suggested_date)(
223
261
  self.last_end_time, ar.logger
224
262
  )
225
- if nst is not None:
263
+ if nst is None:
264
+ await ar.adebug("No time suggested to start %s", astr(self))
265
+ continue
266
+ if nst > now:
267
+ await ar.adebug("Too early to start %s", astr(self))
226
268
  next_time = min(next_time, nst)
269
+ continue
270
+
271
+ # await ar.adebug("Start %s", self)
272
+ # print("20231021 1 gonna start", self)
273
+ await self.start_task(ar)
274
+ assert self.last_end_time is not None
275
+ nst = await sync_to_async(self.get_next_suggested_date)(
276
+ self.last_end_time, ar.logger
277
+ )
278
+ if nst is not None:
279
+ next_time = min(next_time, nst)
227
280
 
228
281
  count += 1
229
282
  if max_count is not None and count >= max_count:
@@ -38,7 +38,7 @@ class SystemTasks(dd.Table):
38
38
  name
39
39
  every every_unit
40
40
  log_level disabled status
41
- last_start_time last_end_time
41
+ requested_at last_start_time last_end_time
42
42
  message
43
43
  """
44
44
  insert_layout = """
@@ -158,3 +158,7 @@ class Plugin(ad.Plugin):
158
158
  mg = site.plugins.office
159
159
  m = m.add_menu(mg.app_label, mg.verbose_name)
160
160
  m.add_action("memo.Mentions")
161
+ # m.add_action("about.About.insert_reference")
162
+
163
+ # def get_quicklinks(self):
164
+ # yield "about.About.insert_reference"
@@ -14,8 +14,9 @@ from lino.core.roles import SiteStaff
14
14
  from lino.core.gfks import gfk2lookup
15
15
  from lino.modlib.gfks.mixins import Controllable
16
16
  from lino.modlib.gfks.fields import GenericForeignKey, GenericForeignKeyIdField
17
+ from lino.modlib.about.models import About
17
18
  from .parser import split_name_rest
18
- # from .mixins import *
19
+ from .mixins import MemoReferrable
19
20
 
20
21
  # Translators: will also be concatenated with '(type)' and '(object)'
21
22
  target_label = _("Target")
@@ -71,8 +72,10 @@ class Mention(Controllable):
71
72
  obj = super()
72
73
  return obj.as_summary_item(ar, text, **kwargs)
73
74
 
75
+
74
76
  dd.update_field(Mention, 'owner', verbose_name=_("Referrer"))
75
77
 
78
+
76
79
  class Mentions(dd.Table):
77
80
  required_roles = dd.login_required(SiteStaff)
78
81
  editable = False
@@ -89,8 +92,51 @@ class Mentions(dd.Table):
89
92
  # column_names = "target *"
90
93
  # default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
91
94
 
95
+
92
96
  class MentionsByTarget(Mentions):
93
97
  label = _("Mentioned by")
94
98
  master_key = "target"
95
99
  column_names = "owner *"
96
100
  default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
101
+
102
+
103
+ CONTENT_TYPE_FIELD = dd.ForeignKey(ContentType, editable=True)
104
+
105
+
106
+ class InsertReference(dd.Action):
107
+
108
+ label = _("Insert reference")
109
+ select_rows = False
110
+ parameters = dict(
111
+ content_type=CONTENT_TYPE_FIELD,
112
+ primary_key=GenericForeignKeyIdField(CONTENT_TYPE_FIELD, editable=True)
113
+ )
114
+ params_layout = """
115
+ content_type
116
+ primary_key
117
+ """
118
+
119
+ @dd.chooser()
120
+ def content_type_choices(self):
121
+ for obj in rt.models.gfks.ContentType.objects.all():
122
+ if issubclass(obj.model_class(), MemoReferrable):
123
+ yield obj
124
+
125
+ @dd.chooser(instance_values=True)
126
+ def primary_key_choices(cls, content_type):
127
+ if content_type is None:
128
+ # print("You must select a content type")
129
+ # return []
130
+ return [("", _("You must select a content type"))]
131
+ m = content_type.model_class()
132
+ return m.objects.all()
133
+
134
+ def run_from_ui(self, ar, **kwargs):
135
+ pv = ar.action_param_values
136
+ ct = pv.content_type.model_class()
137
+ obj = ct.objects.get(pk=pv.primary_key)
138
+ txt = obj.obj2memo()
139
+ ar.success(message=txt)
140
+
141
+
142
+ About.insert_reference = InsertReference()