Skip to content

Commit

Permalink
Add vertico-crm-mode in separate package
Browse files Browse the repository at this point in the history
  • Loading branch information
minad committed Jul 6, 2021
1 parent a091a5a commit 28462df
Show file tree
Hide file tree
Showing 2 changed files with 203 additions and 13 deletions.
28 changes: 15 additions & 13 deletions README.org
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@ a UI which behaves /correctly/ under all circumstances. By reusing the built-in
facilities system, Vertico achieves /full compatibility/ with built-in Emacs
completion commands and completion tables. Vertico only provides the completion
UI. Additional enhancements can be installed separately via complementary
packages. The code base is small and maintainable (~600 lines of code without
whitespace and comments).
packages. The code base is small and maintainable (~vertico.el~ is only about 600
lines of code without whitespace and comments).

* Features

Expand All @@ -32,6 +32,7 @@ whitespace and comments).
- Deferred completion style highlighting for performance
- Support for annotations (~annotation-function~ and ~affixation-function~)
- Support for grouping and group cycling commands (~group-function~)
- Optional enhanced =completing-read-multiple= UI (=vertico-crm-mode=)

[[https://github.com/minad/vertico/blob/main/screenshot.svg?raw=true]]

Expand All @@ -48,8 +49,12 @@ Here is an example configuration:
;; Enable vertico
(use-package vertico
:init
;; Enable the Vertico UI
(vertico-mode)

;; Optionally enable an enhanced `completing-read-multiple` UI
(vertico-crm-mode)

;; Grow and shrink the Vertico minibuffer
;; (setq vertico-resize t)

Expand All @@ -74,11 +79,6 @@ Here is an example configuration:
;; A few more useful configurations...
(use-package emacs
:init
;; Add prompt indicator to `completing-read-multiple'.
(defun crm-indicator (args)
(cons (concat "[CRM] " (car args)) (cdr args)))
(advice-add #'completing-read-multiple :filter-args #'crm-indicator)

;; Do not allow the cursor in the minibuffer prompt
(setq minibuffer-prompt-properties
'(read-only t cursor-intangible t face minibuffer-prompt))
Expand Down Expand Up @@ -193,12 +193,14 @@ to their liking - completion plays an integral part in how the users interacts
with Emacs. There are at least two other interactive completion UIs, which
follow a similar philosophy:

- [[https://github.com/raxod502/selectrum][Selectrum]]: Selectrum has a similar UI as Vertico. Vertico offers more commands
for grouping support. Selectrum supports additional Avy-style quick keys and a
horizontal display. On the other hand, Selectrum is significantly more complex
and not fully compatible with every Emacs completion command ([[https://github.com/raxod502/selectrum/issues/481][Issue #481]]),
since it uses its own filtering infrastructure, which deviates from the
standard Emacs completion facilities.
- [[https://github.com/raxod502/selectrum][Selectrum]]: Selectrum has a similar UI as Vertico. Vertico additionally has
the ability to cycle over candidates, offers a more enhanced
=completing-read-multiple= UI and additional commands for grouping support. On
the other hand, Selectrum supports Avy-style quick keys and a horizontal
display. Furthermore Selectrum is significantly more complex and not fully
compatible with every Emacs completion command ([[https://github.com/raxod502/selectrum/issues/481][Issue #481]]), since it uses its
own filtering infrastructure, which deviates from the standard Emacs
completion facilities.
- [[https://github.com/oantolin/icomplete-vertical][Icomplete-vertical]]: This package enhances the Emacs builtin Icomplete with a
vertical display. In contrast to Vertico, the candidates are rotated such that
the current candidate always appears at the top. From my perspective,
Expand Down
188 changes: 188 additions & 0 deletions vertico-crm.el
Original file line number Diff line number Diff line change
@@ -0,0 +1,188 @@
;;; vertico-crm.el --- Enhanced `completing-read-multiple' support for Vertico -*- lexical-binding: t -*-

;; Copyright (C) 2021 Free Software Foundation, Inc.

;; This file is part of GNU Emacs.

;; This program is free software: you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.

;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.

;; You should have received a copy of the GNU General Public License
;; along with this program. If not, see <http://www.gnu.org/licenses/>.

;;; Commentary:

;; This package provides `vertico-crm-mode', which sets up an enhanced
;; `completing-read-multiple' UI for Vertico.

;;; Code:

(require 'vertico)
(require 'crm)

(defvar-local vertico-crm--table nil)
(defvar-local vertico-crm--selected nil)
(defvar-local vertico-crm--count-ov nil)

(defcustom vertico-crm-count-format " (%s selected): "
"Format string used for the selection count."
:type '(choice (const nil) string)
:group 'vertico)

(defface vertico-crm-selected
'((t :inherit secondary-selection))
"Face used to highlight selected items."
:group 'vertico)

(defvar vertico-crm-map
(let ((map (make-composed-keymap nil vertico-map)))
(define-key map "\t" #'vertico-crm-select)
(define-key map [backtab] #'vertico-crm-select-erase)
map)
"Minibuffer keymap derived from `vertico-map'.")

(defun vertico-crm--update-count ()
"Update the count overlay."
(when vertico-crm--count-ov
(overlay-put vertico-crm--count-ov 'display
(and vertico-crm--selected
(format vertico-crm-count-format
(length vertico-crm--selected))))))

(defun vertico-crm--format (cand)
"Format selected candidate CAND."
;; Restore original candidate in order to preserve formatting
(setq cand (substring (or (car (all-completions cand vertico-crm--table nil)) cand)))
(add-face-text-property 0 (length cand) 'vertico-crm-selected 'append cand)
(put-text-property 0 (length cand) 'vertico-crm--selected t cand)
cand)

(defun vertico-crm--collection (str pred action)
"Programmable completion table for `vertico-crm--completing-read-multiple'.
See `completing-read' for the arguments STR, PRED and ACTION."
(pcase action
('metadata
(let* ((md (and (functionp vertico-crm--table)
(cdr (funcall vertico-crm--table str pred action))))
(group-fun (alist-get 'group-function md))
(title (substitute-command-keys "Select multiple [\\[vertico-crm-select]]")))
`(metadata
(group-function
. ,(lambda (cand transform)
(if (get-text-property 0 'vertico-crm--selected cand)
(if transform cand "Selected")
(or (and group-fun (funcall group-fun cand transform))
(if transform cand title)))))
,@md)))
('t
(nconc
(all-completions str vertico-crm--selected nil)
(cl-delete-if (lambda (x) (member x vertico-crm--selected))
(all-completions str vertico-crm--table pred))))
(_ (complete-with-action action vertico-crm--table str pred))))

(defun vertico-crm--completing-read-multiple (prompt table &optional
pred require-match initial-input
hist def inherit-input-method)
"Enhanced replacement for `completing-read-multiple'.
See `completing-read-multiple' for the arguments."
;; TODO maybe it is better to ignore initial-input or to pass it to completing-read?
;; It depends on if initial-input is used to preselect candidates or if initial-input
;; is used as a filter string. It is hard or impossible to determine this.
(let ((selected))
(minibuffer-with-setup-hook
(lambda ()
(add-hook 'minibuffer-exit-hook (lambda () (setq selected vertico-crm--selected)) nil 'local)
(when-let (pos (and vertico-crm-count-format
(string-match-p "\\(?: (default[^)]+)\\)?: \\'"
(minibuffer-prompt))))
(setq vertico-crm--count-ov (make-overlay (+ (point-min) pos)
(minibuffer-prompt-end))))
(setq vertico-crm--table table
vertico-crm--selected
(and initial-input
(mapcar #'vertico-crm--format
(split-string initial-input crm-separator 'omit-nulls))))
(vertico-crm--update-count)
(use-local-map vertico-crm-map))
(let* ((hist-sym (pcase hist
('nil 'minibuffer-history)
('t nil)
(`(,sym . ,_) sym) ;; ignore history position
(_ hist)))
(hist-val (symbol-value hist-sym))
(result
(completing-read prompt
#'vertico-crm--collection
pred
require-match
nil ;; initial-input
hist
"" ;; default
inherit-input-method)))
(setq selected (mapcar #'substring-no-properties selected))
(unless (or (equal result "") selected)
(setq selected (list result)))
(set hist-sym (append selected hist-val))
(when (consp def)
(setq def (car def)))
(if (and def (not (equal "" def)) (not selected))
(split-string def crm-separator 'omit-nulls)
selected)))))

(defun vertico-crm-select ()
"Select/deselect current candidate."
(interactive)
(let ((cand (vertico--candidate)))
(when (and (not (equal cand "")) (vertico--match-p cand))
(when (>= vertico--index 0)
(when (> vertico--total 1)
(vertico--goto (if (= (1+ vertico--index) vertico--total)
-1
(1+ vertico--index))))
(setq vertico--input t))
(if (member cand vertico-crm--selected)
;; Multi selections are not possible.
;; This is probably no problem, since this is rarely desired.
(setq vertico-crm--selected (delete cand vertico-crm--selected))
(setq vertico--lock-groups t
vertico--all-groups '("Selected")
vertico-crm--selected
(nconc vertico-crm--selected (list (vertico-crm--format cand)))))
(vertico-crm--update-count))))

(defun vertico-crm-select-erase ()
"Select/deselect current candidate and erase input."
(interactive)
(vertico-crm-select)
(delete-minibuffer-contents)
(setq vertico--lock-candidate nil))

;;;###autoload
(define-minor-mode vertico-crm-mode
"Enhanced `completing-read-multiple' support for Vertico."
:global t
(if vertico-crm-mode
(add-hook 'vertico-mode-hook #'vertico-crm--setup)
(remove-hook 'vertico-mode-hook #'vertico-crm--setup))
(vertico-crm--setup))

(defun vertico-crm--setup ()
"Setup enhanced `completing-read-multiple'."
(if (and vertico-crm-mode vertico-mode)
(progn
(advice-remove #'completing-read-multiple #'vertico--advice)
(advice-add #'completing-read-multiple :override #'vertico-crm--completing-read-multiple))
(advice-remove #'completing-read-multiple #'vertico-crm--completing-read-multiple)
(when vertico-mode
(advice-add #'completing-read-multiple :around #'vertico--advice))))

(provide 'vertico-crm)
;;; vertico-crm.el ends here

0 comments on commit 28462df

Please sign in to comment.