-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathprompter-source.lisp
845 lines (755 loc) · 33.3 KB
/
prompter-source.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
;;;; SPDX-FileCopyrightText: Atlas Engineer LLC
;;;; SPDX-License-Identifier: BSD-3-Clause
(in-package :prompter)
;; TODO: Use methods instead of slots? Probably no, because we must be able to
;; handle anonymous sources / prompters.
;; TODO: Memoize `suggestion' computation?
;; TODO: User classes? Probably useful mostly for `source' since they may be
;; defined globally. Conversely, `prompter' is mostly used locally.
;; TODO: Performance: plists are faster, especially when it comes to modifying
;; existing attributes.
(deftype function-symbol ()
`(and symbol (satisfies fboundp)))
(defmacro with-protect ((format-string &rest args) &body body) ; TODO: Inspired by Nyxt. Move to serapeum?
"Run body while capturing all conditions and echoing them as a warning.
The warning is reported to the user as per
FORMAT-STRING and ARGS.
As a special case, the first `:condition' keyword in ARGS is replaced with the
raised condition."
(alex:with-gensyms (c)
`(handler-case (progn ,@body)
(error (,c)
(declare (ignorable ,c))
,(let* ((condition-index (position :condition args))
(new-args (if condition-index
(append (subseq args 0 condition-index)
`(,c)
(subseq args (1+ condition-index)))
args)))
`(warn ,format-string ,@new-args))))))
(defmacro run-thread (name &body body) ; TODO: Copied from Nyxt. Move to serapeum?
"Run body in a new protected new thread.
This is a \"safe\" wrapper around `bt:make-thread'."
`(bt:make-thread
(lambda ()
(if *debug-on-error*
(progn
,@body)
(with-protect ("Error on separate prompter thread: ~a" :condition)
,@body)))
:name ,name))
(define-class source ()
((name
(error "Source must have a name")
:documentation "Name which can be used to differentiate sources from one
another.")
(constructor
nil
:type (or list function)
:documentation "Function or list to set `initial-suggestions'.
If a function, it's called asynchronously with the source as argument.
The returned value is assigned to `initial-suggestions'.
If a list, it's assigned synchronously to `initial-suggestions'. The list is
guaranteed to never be modified.")
(destructor
nil
:type (or null function)
:documentation "Function called with the source as parameter to clean it up.
It's called when `destroy' is called over `prompter'.")
(initial-suggestions
'()
:reader initial-suggestions
:documentation "Suggestions used on initialization, before any user input is
processed.
On initialization this list is transformed to a list of `suggestion's with
`suggestion-maker'.
This list is never modified after initialization.")
(initial-suggestions-lock
(bt:make-lock)
:type bt:lock
:export nil
:initarg nil
:documentation "Protect `initial-suggestions' access.")
(suggestions
'()
:reader suggestions
:export t
:documentation "The current list of suggestions.
It's updated asynchronously every time the prompter input is changed.
The slot is readable even when the computation hasn't finished.
See `ready-notifier' to know when the list is final.
See `update-notifier' to know when it has been updated, to avoid polling the
list.")
(marks
'()
:type list
:reader t
:writer nil
:export t
:documentation "The list of `suggestion' values which have been marked by
the user.
Marking is only allowed when `enable-marks-p' is non-nil. When suggestions
are marked, subsequent `actions-on-return' run over all marked suggestions.
We store the values instead of the `suggestion' because `suggestion' objects are
reinstantiated between each input processing.")
(actions-on-marks
#'identity
:type (or function function-symbol (cons (or function function-symbol) *))
:documentation "The first function of this list is called automatically when
the marks change.
It does not interrupt or return the prompter.
For convenience, it may be initialized with a single function, in which case it
will be automatically turned into a list.")
(active-attributes-keys
'()
:export t
:accessor nil
:documentation "Keys of the `suggestion' attributes to process when
filtering. An empty list means all attributes are taken into account.")
(suggestion-maker
#'make-suggestion
:type (or function function-symbol)
:documentation "Function that wraps an arbitrary object into a source
`suggestion'.
This is useful to set the `suggestion' slots such as `attributes' and
`match-data' depending on the source and the input.
Called on
- arbitrary object
- (optional) source
- (optional) current input.")
(filter
#'fuzzy-match
:type (or null function function-symbol)
:documentation "Takes a `suggestion', the `source' and the `input' and
return a new `suggestion', or nil if the `suggestion' is discarded.")
(filter-preprocessor
#'delete-inexact-matches
:type (or null function function-symbol)
:documentation "Function called when input is modified, before `filter'ing the
`suggestion's.
It is passed the following arguments:
- a copy of `initial-suggestions';
- the source;
- the input.")
(filter-postprocessor
nil
:type (or null function function-symbol)
:documentation "Function called when input is modified, after `filter'ing the
`suggestion's.
It is passed the following arguments:
- the filtered suggestions;
- the source;
- the input.")
(current-input-downcase-p
nil
:type boolean
:export nil
:documentation "Whether input is downcased.
This is useful for filters to avoid recomputing it every time.")
(last-input-downcase-p
nil
:type boolean
:export nil
:documentation "Whether previous input was downcased. This is useful to
know if there is a case difference since last time and to know if we have to
recompute the match-data for instance.")
(sort-predicate
#'score>
:type (or null function)
:documentation "A predicate used to sort the `suggestion's once filtered.
The predicate works the same as the `sort' predicate.")
(actions-on-return
#'identity
:type (or function function-symbol (cons (or function function-symbol) *))
:accessor nil
:export nil
:documentation "List of funcallables that can be run on `suggestion's of
this source. This is the low-level implementation, see the `actions-on-return'
function for the public interface.
For convenience, it may be initialized with a single function or symbol, in
which case it will be automatically turned into a list.")
(update-notifier
(make-channel)
:type calispel:channel
:documentation "A channel which is written to when `filter' commits a change
to `suggestion's. A notification is only sent if at least `notification-delay'
has passed. This is useful so that clients don't have to poll `suggestion's for
changes.")
(notification-delay
0.1
:type alex:non-negative-real
:documentation "Time in seconds after which to notify `update-notifier' if
`suggestions' was modified.")
(ready-p
nil
:type boolean
:export t
:initarg nil
:documentation "Whether the source is done computing its suggestions. See
also `next-ready-p' and `all-ready-p' to wait until ready.")
(ready-channel
nil
:type (or null calispel:channel)
:export nil
:initarg nil
:documentation "Notify listener that source is ready.
The source object is sent to the channel.
If update calculation is aborted, nil is sent instead.")
(update-thread
nil
:type (or null bt:thread)
:export nil
:initarg nil
:documentation "Thread where the `filter-preprocessor', `filter' and
`filter-postprocessor' are run. We store it in a slot so that we can terminate
it.")
(attribute-thread
nil
:type (or null bt:thread)
:export nil
:initarg nil
:documentation "Thread where the attributes get asynchronously computer.
See `attribute-channel'.")
(attribute-channel
(make-channel)
:type (or null calispel:channel)
:export nil
:initarg nil
:documentation "Channel used to communicate attributes to `attribute-thread'
to compute asynchronously.")
(enable-marks-p
nil
:type boolean
:documentation "Whether multiple `suggestion's can be marked.")
(resumer
nil
:type (or null function)
:documentation "Function meant to be called with the source as argument when
the prompter is resumed.
See `resume-sources'.")
(actions-on-current-suggestion
#'identity
:type (or function function-symbol (cons (or function function-symbol) *))
:documentation "The first function of this list is called automatically on
the current-suggestion when it's changed.
It does not interrupt or return the prompter.
For convenience, it may be initialized with a single function, in which case it
will be automatically turned into a list.")
(actions-on-current-suggestion-enabled-p
nil
:type boolean
:documentation "Whether the first of `actions-on-current-suggestion' is
automatically executed."))
(:export-class-name-p t)
(:export-accessor-names-p t)
(:predicate-name-transformer 'nclasses:always-dashed-predicate-name-transformer)
(:documentation "A prompter source instance is meant to be used by a
`prompter' object. See its `sources' slot. A source is a consistent collection
of suggestions, filters and actions.
When a `prompter' `input' is set, the `update' function is called over all
sources. This function pipelines `initial-suggestions' through
`filter-preprocessor', `filter', and finally `filter-postprocessor'. If any of
these functions is nil, it's equivalent to passing the suggestions unchanged.
`filter-preprocessor' and `filter-postprocessor' are passed the whole list of
suggestions; they only set the `suggestion's once they are done. Conversely,
`filter' is passed one `suggestion' at a time and it updates `suggestion's on each
call."))
(defun default-object-attributes (object)
`(("Default" ,(princ-to-string object))))
(defmethod (setf marks) (value (source prompter:source))
(setf (slot-value source 'marks) value)
(sera:and-let* ((action (alex:ensure-function (first (actions-on-marks source))))
(_ (not (eq #'identity action))))
(run-thread "Prompter marks action thread"
(funcall action (marks source)))))
(defmethod default-action-on-current-suggestion ((source prompter:source))
"Return the default action run on the newly selected suggestion.
See `actions-on-current-suggestion'."
(first (actions-on-current-suggestion source)))
(export-always 'object-attributes)
(defgeneric object-attributes (object source)
(:method ((object t) (source prompter:source))
(declare (ignorable source))
(default-object-attributes object))
(:method :around ((object t) (source prompter:source))
(declare (ignorable source))
(loop for elem in (call-next-method)
for key = (first elem)
for tail = (rest elem)
for value = (first tail)
for rest = (rest tail)
;; NOTE: Duplicate keys are bad, because searching the alist by key
;; will always return the first occurrence, and never the second.
when (member key keys :test #'string-equal)
do (warn "Duplicate attribute names found in ~a: ~a.
Attribute names should be unique for prompter to correctly filter those."
source key)
collect key into keys
collect `(,(princ-to-string key)
,(if (functionp value) value (princ-to-string value))
,@rest)))
(:method ((object hash-table) (source prompter:source))
(declare (ignorable source))
(let ((result))
(maphash (lambda (key value)
(push (list (princ-to-string key)
(princ-to-string value))
result))
object)
(sort result #'string< :key #'first)))
(:method ((object structure-object) (source prompter:source))
(declare (ignorable source))
(or
(mapcar (lambda (slot)
(list (string-capitalize (string slot))
(princ-to-string (slot-value object slot))))
(mopu:slot-names object))
(call-next-method)))
(:method ((object list) (source prompter:source))
(declare (ignorable source))
(cond
((plist-p object)
(let ((result '()))
(alex:doplist (key value object result)
(push (list (string-capitalize key) (princ-to-string value))
result))
(nreverse result)))
((undotted-alist-p object)
(mapcar (lambda (pair)
(list
(princ-to-string (first pair))
(princ-to-string (second pair))))
object))
((alist-p object)
(mapcar (lambda (pair)
(list
(princ-to-string (first pair))
(princ-to-string (rest pair))))
object))
(t (call-next-method))))
(:documentation "Return an alist of non-dotted lists (ATTRIBUTE-KEY
ATTRIBUTE-VALUE ...) for OBJECT. Attributes are meant to describe the OBJECT
in the context of the SOURCE.
The attributes after the first two are for the application specific purposes,
like format strings or code for element display, the calling code can freely use
those to store arbitrary data.
The returned attribute-keys and attribute-values are guaranteed to be strings,
unless attribute-value is a function. In that case, the attributes are computed
asynchronously (see `active-attributes').
For structure and class instances, the alist is made of the exported slots: the
keys are the sentence-cased slot names and the values the slot values passed to
`princ-to-string'.
It's used in `make-suggestion' which can be used as a `suggestion-maker' for `source's.
It's useful to separate concerns and compose between different object attributes
and different sources (for instance, the same `object-attributes' method can be
inherited or used across different sources)."))
(define-class suggestion ()
((value
nil ; TODO: Rename `data' as with the GHT? Maybe confusing since we have `match-data'.
:type t)
(attributes
'()
:documentation "A non-dotted alist of attributes to structure the filtering.
Both the key and the value are strings or functions.
If a function, it is run asynchronously and must return a string.")
(match-data
nil
:type t
:documentation "Arbitrary data that can be used by the `filter' function and
its preprocessors. It's the responsibility of the filter to ensure the
match-data is ready for its own use.")
(score
0.0
:documentation "A score the can be set by the `filter' function and used by
the `sort-predicate'."))
(:export-class-name-p t)
(:export-accessor-names-p t)
(:predicate-name-transformer 'nclasses:always-dashed-predicate-name-transformer)
(:documentation "Suggestions are processed and listed in `source'.
It wraps arbitrary object stored in the `value' slot.
The other slots are optional.
Suggestions are made with the `suggestion-maker' slot from `source'."))
(defun pair-p (object)
(and (listp object)
(or (not (listp (rest object)))
(null (rest (rest object))))))
(defun alist-p (object)
"Return non-nil if OBJECT is an alist, dotted or undotted."
(and (listp object)
(every #'pair-p object)))
(defun undotted-alist-p (object &optional value-type)
"If VALUE-TYPE is non-nil, check if all values are of the specified type."
(and (listp object)
(every #'listp object)
(every #'listp (mapcar #'rest object))
(or (not value-type)
(every (lambda (e) (typep (first e) value-type))
(mapcar #'rest object)))))
(defun plist-p (object)
"Return non-nil if OBJECT is a plist."
(and (listp object)
(alex:proper-list-p object)
(evenp (length object))
(loop :for x :in object :by #'cddr
:always (keywordp x))))
(defun object-attributes-p (object)
(undotted-alist-p object '(or string function)))
(defmethod attribute-key ((attribute t))
(first attribute))
(defmethod attribute-value ((attribute t))
(second attribute))
(defmethod attributes-keys ((attributes t))
(mapcar #'attribute-key attributes))
(defmethod attributes-values ((attributes t))
(mapcar #'attribute-value attributes))
(defun ensure-string (object)
"Return \"\" if OBJECT is not a string."
(if (stringp object)
object
""))
(defun format-attributes (attributes)
"Performance bottleneck: This function is called as many times as they are
suggestions."
(sera:string-join (mapcar #'ensure-string (attributes-values attributes)) " "))
(defmethod initialize-instance :after ((suggestion suggestion) &key)
"Check validity."
(unless (object-attributes-p (attributes suggestion))
(warn "Attributes of ~s should be a non-dotted alist instead of ~s" (value suggestion) (attributes suggestion))
(setf (attributes suggestion) (default-object-attributes (value suggestion)))))
(defun ensure-match-data-string (suggestion source)
"Return SUGGESTION's `match-data' as a string.
If unset, set it to the return value of `format-attributes'."
(flet ((maybe-downcase (s)
(if (current-input-downcase-p source)
(string-downcase s)
s)))
(setf (match-data suggestion)
(if (and (match-data suggestion)
(typep (match-data suggestion) 'string))
(if (not (eq (last-input-downcase-p source)
(current-input-downcase-p source)))
(maybe-downcase (match-data suggestion))
(match-data suggestion))
(let ((result (format-attributes (attributes suggestion))))
(maybe-downcase result)))))
(match-data suggestion))
(export-always 'make-suggestion)
(defgeneric make-suggestion (value &optional source input)
(:method ((value t) &optional source input)
(declare (ignore input))
(make-instance 'suggestion
:value value
:attributes (object-attributes value source)))
(:documentation "Return a `suggestion' wrapping around VALUE.
Attributes are set with `object-attributes'."))
(defgeneric default-action-on-return (source)
(:method ((source source))
(first (slot-value source 'actions-on-return)))
(:documentation "Return the default action run when returning from the prompt.
See `actions-on-return'."))
(define-class yes-no-source (source)
((name "Confirm")
(yes "yes")
(no "no")
(constructor (list t nil)))
(:export-class-name-p t)
(:predicate-name-transformer 'nclasses:always-dashed-predicate-name-transformer)
(:documentation "Prompt source for yes-no questions."))
(defmethod object-attributes ((object symbol) (source yes-no-source))
`(("Answer" ,(if object (yes source) (no source)))))
(defun make-input-suggestion (suggestions source input)
(declare (ignore suggestions))
(list (funcall (suggestion-maker source) input
source input)))
(define-class raw-source (source)
((name "Input")
(filter-preprocessor 'make-input-suggestion)
(filter nil)
(enable-marks-p nil))
(:export-class-name-p t)
(:predicate-name-transformer 'nclasses:always-dashed-predicate-name-transformer)
(:documentation "Prompt source for raw user input.
Its only `suggestion' is the user input, thus it has no constructor.
If you are looking for a source that just returns its plain suggestions, use `source'."))
(defmethod initialize-instance :after ((raw-source raw-source) &key)
"Report error when RAW-SOURCE is provided bad initial arguments."
(when (constructor raw-source)
(error "Raw source should have no constructor: ~a" (constructor raw-source))))
(defun make-word-suggestions (suggestions source input)
(declare (ignore suggestions))
(mapcar (lambda (word)
(funcall (suggestion-maker source) word
source input))
(sera:words input)))
(define-class word-source (source)
((name "Input words")
(filter-preprocessor 'make-word-suggestions)
(filter nil)
(enable-marks-p t))
(:export-class-name-p t)
(:predicate-name-transformer 'nclasses:always-dashed-predicate-name-transformer)
(:documentation "Prompt source for user input words."))
(export-always 'ensure-suggestions-list)
(defgeneric ensure-suggestions-list (source elements)
(:method ((source source) elements)
(lparallel:pmapcar
(lambda (suggestion-value)
(if (suggestion-p suggestion-value)
suggestion-value
(funcall (suggestion-maker source)
suggestion-value
source)))
(uiop:ensure-list elements)))
(:documentation "Return ELEMENTS as a list of suggestions for use in SOURCE."))
(defmethod initialize-instance :after ((source source) &key)
"See the `constructor' documentation of `source'."
(let ((wait-channel (make-channel)))
(run-thread "Prompter source init thread"
(bt:acquire-lock (initial-suggestions-lock source))
;; `initial-suggestions' initialization must be done before first input can be processed.
(etypecase (constructor source)
(list
(setf (slot-value source 'initial-suggestions)
(constructor source)))
(function
;; Run constructor asynchronously.
(calispel:! wait-channel t)
(setf (slot-value source 'initial-suggestions)
(funcall (constructor source) source))))
(setf (slot-value source 'initial-suggestions)
(ensure-suggestions-list source (initial-suggestions source)))
;; TODO: Setting `suggestions' is not needed?
(setf (slot-value source 'suggestions) (initial-suggestions source))
(bt:release-lock (initial-suggestions-lock source))
(when (listp (constructor source))
;; Initial suggestions are set synchronously in this case.
(calispel:! wait-channel t)))
;; Wait until above thread has acquired the `initial-suggestions-lock'.
(calispel:? wait-channel))
(setf (actions-on-current-suggestion source)
(uiop:ensure-list (or (actions-on-current-suggestion source)
#'identity)))
(setf (actions-on-marks source)
(uiop:ensure-list (or (actions-on-marks source)
#'identity)))
(setf (slot-value source 'actions-on-return)
(uiop:ensure-list (or (slot-value source 'actions-on-return)
#'identity)))
source)
(export-always 'attributes-keys-non-default)
(defgeneric attributes-keys-non-default (source)
(:method ((source source))
(rest (attributes-keys source)))
(:documentation "Return SOURCE attributes except the default one."))
(export-always 'attributes-keys-default)
(defgeneric attributes-keys-default (source)
(:method ((source source))
(first (attributes-keys source)))
(:documentation "Return SOURCE default attribute as a non-dotted pair."))
(export-always 'attributes-default)
(defgeneric attributes-default (suggestion)
(:method ((suggestion suggestion))
(second (first (attributes suggestion))))
(:documentation "Return SUGGESTION default attribute value."))
(export-always 'attributes-non-default)
(defgeneric attributes-non-default (suggestion)
(:method ((suggestion suggestion))
(rest (attributes suggestion)))
(:documentation "Return SUGGESTION non-default attributes."))
(defmethod attributes-keys ((source source))
(attributes-keys
(alex:if-let ((sugg (first (suggestions source)))) ; TODO: Instead, ensure that SUGGESTIONS always has an element?
(attributes sugg)
(default-object-attributes ""))))
(defgeneric active-attributes-keys (source)
(:method ((source source))
(or (slot-value source 'active-attributes-keys)
(attributes-keys source)))
(:documentation "Return active attributes keys.
If the `active-attributes' slot is NIL, return all attributes keys."))
(defmethod (setf active-attributes-keys) (value (source source))
"Set active attributes to the intersection of VALUE and SOURCE attributes."
(flet ((remove-from-seq (seq &rest items)
(reduce (lambda (seq item) (remove item seq :test #'string=))
(set-difference seq items :test #'string=)
:initial-value seq)))
(setf (slot-value source 'active-attributes-keys)
(cons (attributes-keys-default source)
(apply #'remove-from-seq (attributes-keys-non-default source) value)))))
(export-always 'active-attributes)
(defgeneric active-attributes (suggestion &key source &allow-other-keys)
(:method ((suggestion suggestion)
&key (source (error "Source required"))
&allow-other-keys)
(let ((inactive-keys (set-difference (attributes-keys (attributes suggestion))
(active-attributes-keys source)
:test #'string=))
(attribute-thread nil))
(prog1
(mapcar
(lambda (attribute)
(if (functionp (attribute-value attribute))
(progn
(unless attribute-thread (setf attribute-thread (make-attribute-thread source)))
(calispel:! (attribute-channel source) (list suggestion attribute))
(list (attribute-key attribute) ""))
attribute))
(remove-if
(lambda (attr)
(find (attribute-key attr) inactive-keys :test #'string=))
(attributes suggestion)))
(when attribute-thread
;; Terminate thread:
(calispel:! (attribute-channel source) (list nil nil))))))
(:documentation "Return the active attributes of SUGGESTION.
Active attributes are queried from SOURCE."))
(defun make-attribute-thread (source)
"Return a thread that is bound to SOURCE and used to compute its `suggestion' attributes asynchronously.
Asynchronous attributes have a string-returning function as a value."
;; TODO: Notify when done updating, maybe using `update-notifier'?
(run-thread "Prompter attribute thread"
(sera:nlet lp ((sugg-attr-pair (calispel:? (attribute-channel source))))
(destructuring-bind (sugg attr) sugg-attr-pair
;; Recheck type here to protect against race conditions.
(when (functionp (attribute-value attr))
(setf (alex:assoc-value (attributes sugg) (attribute-key attr))
(list
(handler-case (funcall (attribute-value attr) (value sugg))
(error (c)
(format nil "keyword error: ~a" c)))))
(lp (calispel:? (attribute-channel source))))))))
(export-always 'marked-p)
(defun marked-p (source value)
"Return non-nil if VALUE is marked in SOURCE.
Comparison is done with `equalp'."
(find value (prompter:marks source) :test #'equalp))
(defun maybe-funcall (fn &rest args)
"Funcall FN over args.
If FN is nil, return ARGS as multiple values."
(if fn
(apply fn args)
(apply #'values args)))
(defun insert-item-at (item pred sequence) ; TODO: Arg order? Name?
"Insert ITEM in SEQUENCE after the last item FOO for which (PRED ITEM FOO) is
non-nil."
(if sequence
(let ((item-pos
(or (position-if (lambda (e) (funcall pred item e)) sequence)
(length sequence))))
(nconc (subseq sequence 0 item-pos)
(list item)
(subseq sequence item-pos)))
(list item)))
(defun make-channel (&optional size)
"Return a channel of capacity SIZE.
If SIZE is NIL, capacity is infinite."
(cond
((null size)
(make-instance 'calispel:channel
:buffer (make-instance 'jpl-queues:unbounded-fifo-queue)))
((= 0 size)
(make-instance 'calispel:channel))
((< 0 size)
(make-instance 'calispel:channel
:buffer (make-instance 'jpl-queues:bounded-fifo-queue :capacity size)))))
(defun copy-object (object &rest slot-overrides)
(let ((class-sym (class-name (class-of object))))
(apply #'make-instance class-sym
(append
slot-overrides
(alexandria:mappend
(lambda (slot)
(list (intern (symbol-name slot) "KEYWORD")
(slot-value object slot)))
(mopu:slot-names class-sym))))))
(defgeneric destroy (source)
(:method ((source source))
;; Ignore errors in case thread is already terminated.
;; REVIEW: Is there a cleaner way to do this?
(ignore-errors (bt:destroy-thread (update-thread source)))
(ignore-errors (bt:destroy-thread (attribute-thread source))))
(:documentation "Clean up the source.
SOURCE should not be used once this has been run."))
(defun update (source input new-ready-channel) ; TODO: Store `input' in the source?
"Update SOURCE to narrow down the list of `suggestion's according to INPUT.
If a previous `suggestion' computation was not finished, it is forcefully
terminated.
- First the `filter-preprocessor' is run over a copy of `initial-suggestions'.
- The resulting suggestions are passed one by one to `filter'.
When filter returns non-nil, the result is added to `suggestions' and
`update-notifier' is notified, if `notification-delay' has been exceeded or if
the last `suggestion' has been processed.
- Last the `filter-postprocessor' is run over the SOURCE and the INPUT.
Its return value is assigned to the list of suggestions.
- Finally, `ready-notifier' is fired up.
The reason we filter in 3 stages is to allow both for asynchronous and
synchronous filtering. The benefit of asynchronous filtering is that it sends
feedback to the user while the list of suggestions is being computed."
(when (and (update-thread source)
;; This is prone to a race condition, but worst case we destroy an
;; already terminated thread.
(bt:thread-alive-p (update-thread source)))
;; Note that we may be writing multiple times to this channel, but that's
;; OK, only the first value is read, so worst case the caller sees that the
;; source is terminated even though it just finished updating.
;; TODO: Destroying threads breaks ECL. Use conditions instead to terminate
;; threads properly.
(calispel:! (ready-channel source) nil)
(destroy source))
(setf (ready-channel source) new-ready-channel)
(setf (update-thread source)
(run-thread "Prompter update thread"
(flet ((wait-for-initial-suggestions ()
(bt:acquire-lock (initial-suggestions-lock source))
(bt:release-lock (initial-suggestions-lock source)))
(preprocess (initial-suggestions-copy)
(if (filter-preprocessor source)
(ensure-suggestions-list
source
(funcall (filter-preprocessor source)
initial-suggestions-copy source input))
initial-suggestions-copy))
(process! (preprocessed-suggestions)
(let ((last-notification-time (get-internal-real-time)))
(setf (slot-value source 'suggestions) '())
(if (or (str:empty? input)
(not (filter source)))
(setf (slot-value source 'suggestions) preprocessed-suggestions)
(dolist (suggestion preprocessed-suggestions)
(sera:and-let* ((processed-suggestion
(funcall (filter source) suggestion source input)))
(setf (slot-value source 'suggestions)
(insert-item-at suggestion (sort-predicate source)
(suggestions source)))
(let* ((now (get-internal-real-time))
(duration (/ (- now last-notification-time)
internal-time-units-per-second)))
(when (or (> duration (notification-delay source))
(= (length (slot-value source 'suggestions))
(length preprocessed-suggestions)))
(calispel:! (update-notifier source) t)
(setf last-notification-time now))))))))
(postprocess! ()
(when (filter-postprocessor source)
(setf (slot-value source 'suggestions)
(ensure-suggestions-list
source
(maybe-funcall (filter-postprocessor source)
(slot-value source 'suggestions)
source
input))))))
(unwind-protect
(progn
(setf (ready-p source) nil)
(wait-for-initial-suggestions)
(setf (last-input-downcase-p source) (current-input-downcase-p source))
(setf (current-input-downcase-p source) (str:downcasep input))
(process!
(preprocess
;; We copy the list of initial-suggestions so that the
;; preprocessor cannot modify them.
(mapcar #'copy-object (initial-suggestions source))))
(postprocess!))
(setf (ready-p source) t)
;; Signal this source is done:
(calispel:! new-ready-channel source))))))