Skip to content
csxtan edited this page Feb 6, 2024 · 10 revisions

Consult is documented in the [[https://github.com/minad/consult/blob/main/README.org][README]], see in particular the [[https://github.com/minad/consult/blob/main/README.org#configuration][configuration section]]. On this wiki page auxiliary configuration and small utility commands are documented. Feel free to contribute your own useful configuration snippets, candidate multi sources or Consult-related commands!

The snippets on this page REQUIRE [[https://www.gnu.org/software/emacs/manual/html_node/elisp/Lexical-Binding.html][lexical binding]]. Lexical binding is usually specified as a [[https://www.gnu.org/software/emacs/manual/html_node/emacs/File-Variables.html#File-Variables][file-local variable]]. To activate lexical binding, add ;; -- lexical-binding: t -- to the top of your source file. If you use a literate org file, adding this block to the very top of it should do the trick:

#+begin_example ,#+begin_src emacs-lisp :comments no :tangle yes ;; -- lexical-binding: t -- ,#+end_src #+end_example

  • Configuration ** Manual preview for non-Consult commands using Embark

#+begin_src emacs-lisp (define-key minibuffer-local-map (kbd "M-.") #'my-embark-preview) (defun my-embark-preview () "Previews candidate in vertico buffer, unless it's a consult command" (interactive) (unless (bound-and-true-p consult--preview-function) (save-selected-window (let ((embark-quit-after-action nil)) (embark-dwim)))))

#+end_src

** Toggle preview during active completion session

It is possible to enable/disable preview during an active =completing-read= session, by writing a small command. See [[https://github.com/minad/consult/issues/233][#233]].

#+begin_src emacs-lisp (defvar-local consult-toggle-preview-orig nil)

(defun consult-toggle-preview () "Command to enable/disable preview." (interactive) (if consult-toggle-preview-orig (setq consult--preview-function consult-toggle-preview-orig consult-toggle-preview-orig nil) (setq consult-toggle-preview-orig consult--preview-function consult--preview-function #'ignore)))

(define-key vertico-map (kbd "M-P") #'consult-toggle-preview) #+end_src

** Configure S-up/S-down preview keys

It is possible to configure multiple preview keys, for example S-up/S-down, such that one can scroll over the list of candidates while doing preview. For Vertico the following configuration can be used.

#+begin_src emacs-lisp (define-key vertico-map [S-up] #'vertico-previous) (define-key vertico-map [S-down] #'vertico-next) (consult-customize consult-recent-file :preview-key '([S-up] [S-down])) #+end_src

** Add command-local keybinding

For example if =consult-line= is bound to =C-s=, you may want to load the latest search term when pressing =C-s C-s=. This can be achieved by binding =C-s= in the consult-line local keymap.

#+begin_src elisp (defvar my-consult-line-map (let ((map (make-sparse-keymap))) (define-key map "\C-s" #'previous-history-element) map))

(consult-customize consult-line :keymap my-consult-line-map) #+end_src

** Add category-specific minibuffer keybindings

As a less fine-grained alternative to the above method, the following function creates minibuffer keybindings that take effect only for specific categories of items.

#+begin_src emacs-lisp (defun define-minibuffer-key (key &rest defs) "Define KEY conditionally in the minibuffer. DEFS is a plist associating completion categories to commands." (define-key minibuffer-local-map key (list 'menu-item nil defs :filter (lambda (d) (plist-get d (completion-metadata-get (completion-metadata (minibuffer-contents) minibuffer-completion-table minibuffer-completion-predicate) 'category)))))) #+end_src

For instance, the following binds =C-s= to =previous-history-element= not only in =consult-line=, but also in =consult-oultine=, =consult-mark=, etc; moreover, it binds =C-s= to =consult-find-for-minibuffer= (defined [[#using-find-in-the-minibuffer][below]] in this wiki) whenever the minibuffer is reading a file name.

#+begin_src emacs-lisp (define-minibuffer-key "\C-s" 'consult-location #'previous-history-element 'file #'consult-find-for-minibuffer) #+end_src

** consult-outline support for eshell prompts

In order to support quick jumping to prompts in eshell via =consult-outline= we can set the =outline-regexp= appropriately in the =eshell-mode= ([[https://github.com/minad/consult/issues/130][#130]]).

#+begin_src elisp (add-hook 'eshell-mode-hook (lambda () (setq outline-regexp eshell-prompt-regexp))) #+end_src

** Hide all sources, except normal buffers in consult-buffer by default

Makes only the open buffers list visible when calling =consult-buffer= command by hiding the other sources, but still allowing the narrowing to recent files (by typing =f SPC=), bookmarks (=m SPC=) and project buffer and/or files (=p SPC=).

See issue [[https://github.com/minad/consult/issues/203][#203]] for more context and use case example.

#+begin_src elisp (dolist (src consult-buffer-sources) (unless (eq src 'consult--source-buffer) (set src (plist-put (symbol-value src) :hidden t)))) #+end_src

** Start command with initial narrowing

Start with initial narrowing ([[https://github.com/minad/consult/issues/203][#203]]). Note that there is also the possibility to mark sources as initially =:hidden=. This is an alternative to initial narrowing.

#+begin_src elisp ;; Configure initial narrowing per command (defvar consult-initial-narrow-config '((consult-buffer . ?b)))

;; Add initial narrowing hook (defun consult-initial-narrow () (when-let (key (alist-get this-command consult-initial-narrow-config)) (setq unread-command-events (append unread-command-events (list key 32))))) (add-hook 'minibuffer-setup-hook #'consult-initial-narrow) #+end_src

** Cycle through narrowing keys

You may want to cycle through all the narrowing keys with convenient left and right key bindings. See issue [[https://github.com/minad/consult/issues/337][#337]] for more context. Note that Vertico provides the commands =vertico-next-group= and =vertico-previous-group= which allows to cycle through the groups. The Vertico group cycling is an alternative to the commands described here.

#+begin_src emacs-lisp (define-key consult-narrow-map [C-left] #'consult-narrow-cycle-backward) (define-key consult-narrow-map [C-right] #'consult-narrow-cycle-forward)

(defun consult-narrow-cycle-backward () "Cycle backward through the narrowing keys." (interactive) (when consult--narrow-keys (consult-narrow (if consult--narrow (let ((idx (seq-position consult--narrow-keys (assq consult--narrow consult--narrow-keys)))) (unless (eq idx 0) (car (nth (1- idx) consult--narrow-keys)))) (caar (last consult--narrow-keys))))))

(defun consult-narrow-cycle-forward () "Cycle forward through the narrowing keys." (interactive) (when consult--narrow-keys (consult-narrow (if consult--narrow (let ((idx (seq-position consult--narrow-keys (assq consult--narrow consult--narrow-keys)))) (unless (eq idx (1- (length consult--narrow-keys))) (car (nth (1+ idx) consult--narrow-keys)))) (caar consult--narrow-keys))))) #+end_src

** Previewing files in find-file

We can enable preview in find-file by providing a custom read-file-name-function. A similar approach could work for read-buffer-function such that all commands reading a buffer name would preview the buffer.

#+begin_src emacs-lisp (setq read-file-name-function #'consult-find-file-with-preview)

(defun consult-find-file-with-preview (prompt &optional dir default mustmatch initial pred) (interactive) (let ((default-directory (or dir default-directory)) (minibuffer-completing-file-name t)) (consult--read #'read-file-name-internal :state (consult--file-preview) :prompt prompt :initial initial :require-match mustmatch :predicate pred))) #+end_src

** Orderless style dispatchers (Ensure that the $ regexp works with consult-buffer)

Unfortunately $ does not work out of the box with consult-buffer and consult-line since these commands add disambiguation suffixes to the candidate strings. The problem can be fixed by adjusting the filter regular expressions accordingly. See [[https://www.reddit.com/r/emacs/comments/n43mdo/problem_with_narrowing_minibuffer_items_with/][this reddit post]] for more context.

#+begin_src emacs-lisp (defun fix-dollar (args) (if (string-suffix-p "$" (car args)) (list (format "%s[%c-%c]*$" (substring (car args) 0 -1) consult--tofu-char (+ consult--tofu-char consult--tofu-range -1))) args)) (advice-add #'orderless-regexp :filter-args #'fix-dollar) (advice-add #'prescient-regexp-regexp :filter-args #'fix-dollar) #+end_src

I recommend to use the Orderless style dispatchers for a more robust solution. See the next section for a sophisticated Orderless configuration.

** @minad's orderless configuration

#+begin_src emacs-lisp (use-package orderless :demand t :config

(defun +orderless--consult-suffix ()
  "Regexp which matches the end of string with Consult tofu support."
  (if (and (boundp 'consult--tofu-char) (boundp 'consult--tofu-range))
      (format "[%c-%c]*$"
              consult--tofu-char
              (+ consult--tofu-char consult--tofu-range -1))
    "$"))

;; Recognizes the following patterns:
;; * .ext (file extension)
;; * regexp$ (regexp matching at end)
(defun +orderless-consult-dispatch (word _index _total)
  (cond
   ;; Ensure that $ works with Consult commands, which add disambiguation suffixes
   ((string-suffix-p "$" word)
    `(orderless-regexp . ,(concat (substring word 0 -1) (+orderless--consult-suffix))))
   ;; File extensions
   ((and (or minibuffer-completing-file-name
             (derived-mode-p 'eshell-mode))
         (string-match-p "\\`\\.." word))
    `(orderless-regexp . ,(concat "\\." (substring word 1) (+orderless--consult-suffix))))))

;; Define orderless style with initialism by default
(orderless-define-completion-style +orderless-with-initialism
  (orderless-matching-styles '(orderless-initialism orderless-literal orderless-regexp)))

;; You may want to combine the `orderless` style with `substring` and/or `basic`.
;; There are many details to consider, but the following configurations all work well.
;; Personally I (@minad) use option 3 currently. Also note that you may want to configure
;; special styles for special completion categories, e.g., partial-completion for files.
;;
;; 1. (setq completion-styles '(orderless))
;; This configuration results in a very coherent completion experience,
;; since orderless is used always and exclusively. But it may not work
;; in all scenarios. Prefix expansion with TAB is not possible.
;;
;; 2. (setq completion-styles '(substring orderless))
;; By trying substring before orderless, TAB expansion is possible.
;; The downside is that you can observe the switch from substring to orderless
;; during completion, less coherent.
;;
;; 3. (setq completion-styles '(orderless basic))
;; Certain dynamic completion tables (completion-table-dynamic)
;; do not work properly with orderless. One can add basic as a fallback.
;; Basic will only be used when orderless fails, which happens only for
;; these special tables.
;;
;; 4. (setq completion-styles '(substring orderless basic))
;; Combine substring, orderless and basic.
;;
(setq completion-styles '(orderless basic)
      completion-category-defaults nil
      ;;; Enable partial-completion for files.
      ;;; Either give orderless precedence or partial-completion.
      ;;; Note that completion-category-overrides is not really an override,
      ;;; but rather prepended to the default completion-styles.
      ;; completion-category-overrides '((file (styles orderless partial-completion))) ;; orderless is tried first
      completion-category-overrides '((file (styles partial-completion)) ;; partial-completion is tried first
                                      ;; enable initialism by default for symbols
                                      (command (styles +orderless-with-initialism))
                                      (variable (styles +orderless-with-initialism))
                                      (symbol (styles +orderless-with-initialism)))
      orderless-component-separator #'orderless-escapable-split-on-space ;; allow escaping space with backslash!
      orderless-style-dispatchers (list #'+orderless-consult-dispatch
                                        #'orderless-affix-dispatch)))

#+end_src

** Use Orderless as pattern compiler for consult-grep/ripgrep/find

consult-ripgrep and the other commands use Emacs regular expressions by default, which are translated to the PCRE/ERE regular expression syntax. It is possible to plug-in Orderless as pattern compiler. See issue [[https://github.com/minad/consult/issues/380][#380]] and [[https://github.com/minad/consult/issues/381][#381]] for more information.

#+begin_src emacs-lisp (defun consult--orderless-regexp-compiler (input type &rest _config) (setq input (orderless-pattern-compiler input)) (cons (mapcar (lambda (r) (consult--convert-regexp r type)) input) (lambda (str) (orderless--highlight input t str))))

;; OPTION 1: Activate globally for all consult-grep/ripgrep/find/... ;; (setq consult--regexp-compiler #'consult--orderless-regexp-compiler)

;; OPTION 2: Activate only for some commands, e.g., consult-ripgrep! (defun consult--with-orderless (&rest args) (minibuffer-with-setup-hook (lambda () (setq-local consult--regexp-compiler #'consult--orderless-regexp-compiler)) (apply args))) (advice-add #'consult-ripgrep :around #'consult--with-orderless) #+end_src

** Skipping directories when using consult-find

If you find consult-find slow and would like to skip some directories, consider specifying the directories to skip using consult-find-args:

#+begin_src emacs-lisp (setq consult-find-args "find . -not ( -wholename /. -prune -o -name node_modules -prune )") #+end_src

Default consult-find-args:

#+begin_src emacs-lisp (setq consult-find-args "find . -not ( -wholename /. -prune )") #+end_src

** Use consult-ripgrep instead of project-find-regexp in project.el

By default, project-find-regexp uses grep and also it does not offer the convenient and beloved interface of Consult. You might want to use consult-ripgrep in place of it, when using project.el.

#+begin_src elisp (require 'keymap) ;; keymap-substitute requires emacs version 29.1? (require 'cl-seq)

(keymap-substitute project-prefix-map #'project-find-regexp #'consult-ripgrep) (cl-nsubstitute-if '(consult-ripgrep "Find regexp") (pcase-lambda (`(,cmd _)) (eq cmd #'project-find-regexp)) project-switch-commands) #+end_src

  • Unstable hacks and advices ** Make consult-imenu ignore group titles when searching with orderless

consult-imenu moves the top-most imenu names to the group title via the group-function and adds narrowing if configured in consult-imenu-config. This grouping only makes sense for certain major modes, e.g., elisp where the topmost menu name corresponds to fixed imenu item categories, e.g., "Functions", "Variables", etc. In contrast, for org-mode using the top menu names as group titles does not make sense, since they depend on the buffer content.

The top most menu name is only moved visually to the group title, but the title is still included with the candidate text, and remains searchable. This means that searching for, e.g., ion in an emacs-lisp buffer will by default match /all functions/ and not just function names containing those letters.

If you don't want the group titles for modes configured in consult-imenu-config to be searchable, it is possible to advise orderless, such that it ignores the top most menu names/group titles when searching candidates. The following implements this, while also allowing embark collect to visit a single imenu entry from the collect buffer.

#+begin_src emacs-lisp (defun my/consult-imenu-around-advice (ci-orig &rest r) "Patch orderless to inhibit matching group categories in consult-imenu." (if-let* ((config (cdr (seq-find (lambda (x) (derived-mode-p (car x))) consult-imenu-config))) (types (plist-get config :types)) (types-regex (rx-to-string (and line-start (or ,@(mapcar #'cadr types)) ? )))) (cl-letf* ((of-orig (symbol-function 'orderless-filter)) ((symbol-function 'orderless-filter) ;patch pattern compiler within filter (lambda (&rest r) (cl-letf* ((opc-orig (symbol-function 'orderless-pattern-compiler)) ((symbol-function 'orderless-pattern-compiler) (lambda (&rest r) (if (and (eq (length r) 1) ;single match string starts (string-match-p types-regex (car r))) (apply opc-orig r) (mapcar (lambda (x) ;replace beginning-of-string (if (string-match (regexp-quote "\\" ) x) (concat types-regex (replace-match "\b" nil t x)) (concat types-regex ".?" x))) (apply opc-orig r)))))) (apply of-orig r)))) (oh-orig (symbol-function 'orderless--highlight)) ((symbol-function 'orderless--highlight) ; patch highlighter to skip type (lambda (regexps string) (if-let ((pref (next-single-property-change 0 'consult--type string))) (cl-letf* ((sm-orig (symbol-function 'string-match)) ((symbol-function 'string-match) (lambda (re str) (funcall sm-orig re str (1+ pref))))) (funcall oh-orig regexps string)) (funcall oh-orig regexps string))))) (apply ci-orig r)) (apply ci-orig r)))

(advice-add #'consult-imenu :around #'my/consult-imenu-around-advice)) #+end_src

** Pre-select nearest heading for consult-org-heading and consult-outline using vertico

When looking at an outline of the current buffer, it can be nice to see the current and surrounding headings for context and to jump to nearby headings. These pieces of code will make the nearest heading or imenu item be automatically selected when those minibuffer commands are run.

First, using advice, save the current location in the buffer.

#+begin_src emacs-lisp (defvar consult--previous-point nil "Location of point before entering minibuffer. Used to preselect nearest headings and imenu items.")

(defun consult--set-previous-point (&optional arg1 arg2) "Save location of point. Used before entering the minibuffer." (setq consult--previous-point (point)))

(advice-add #'consult-org-heading :before #'consult--set-previous-point) (advice-add #'consult-outline :before #'consult--set-previous-point) #+end_src

Advise vertico--update to select the nearest candidate if applicable. #+begin_src emacs-lisp (advice-add #'vertico--update :after #'consult-vertico--update-choose)

(defun consult-vertico--update-choose (&rest _) "Pick the nearest candidate rather than the first after updating candidates." (when (and consult--previous-point (memq current-minibuffer-command '(consult-org-heading consult-outline))) (setq vertico--index (max 0 ; if none above, choose the first below (1- (or (seq-position vertico--candidates consult--previous-point (lambda (cand point-pos) ; counts on candidate list being sorted (> (cl-case current-minibuffer-command (consult-outline (car (consult--get-location cand))) (consult-org-heading (get-text-property 0 'consult--candidate cand))) point-pos))) (length vertico--candidates)))))) (setq consult--previous-point nil)) #+end_src

** Temporarily override consult-ripgrep-args

I am really not sure about the reason, maybe because the builder is called asynchronously, but to temporarily override consult-ripgrep-args, you may need to wrap the entire consult--ripgrep-builder. Here is an example that temporarily adds --no-ignore-vcs flag to the builder. Using advice-add and advice-remove to override consult--ripgrep-builder seems also OK, but I haven't tried it.

#+begin_src elisp (defun consult--ripgrep-noignore-builder (input) "consult--ripgrep-builder with INPUT, but ignores .gitignore." (let ((consult-ripgrep-args (if (string-match-p "--no-ignore-vcs" consult-ripgrep-args) consult-ripgrep-args (concat consult-ripgrep-args "--no-ignore-vcs .")))) (consult--make-ripgrep-builder input)))

(defun consult-ripgrep-noignore (&optional dir initial) "Do consult-ripgrep with DIR and INITIAL, but without ignoring." (interactive "P") (consult--grep "Ripgrep" #'consult--ripgrep-noignore-builder (if dir dir t) ;; Here the directory prompt is called by default to avoid searching from the project root initial)) #+end_src

** Narrowing which-key help without delay

After pressing =consult-narrow-key=, the which-key menu should appear immediately ([[https://github.com/minad/consult/issues/191][#191]]).

#+begin_src elisp (defun immediate-which-key-for-narrow (fun &rest args) (let* ((refresh t) (timer (and consult-narrow-key (memq :narrow args) (run-at-time 0.05 0.05 (lambda () (if (eq last-input-event (elt consult-narrow-key 0)) (when refresh (setq refresh nil) (which-key--update)) (setq refresh t))))))) (unwind-protect (apply fun args) (when timer (cancel-timer timer))))) (advice-add #'consult--read :around #'immediate-which-key-for-narrow) #+end_src

** Shorten recent files in consult-buffer

NOTE: The [[https://github.com/jdtsmith/vertico-truncate][vertico-truncate]] package provides this functionality in a more robust way. Using a package is recommended over copying large snippets to your Emacs configuration.

The recent files list which comprise one of the sources of consult-buffer are presented with full (abbreviated) path for completion by default. While this grants us a simple source and precise matching of the candidates, it comes with some drawbacks: i) candidates may get long enough so that the candidate gets truncated out of the window width; ii) even if truncation does not occur, =marginalia= annotations tend to get pushed away; iii) the full paths may match more than we'd like (depending on the use case). So one might prefer shortening the candidates from this source. See discussion at [[https://github.com/minad/consult/issues/713][#713]].

One first approach to that would be to simply use the file name of the candidate (disregarding the path). This makes for a simple and cheap shortening, with the disadvantage that some candidates may occur in duplicity, in which case the duplicates get shadowed in the completion. But, depending on the use case and preferences, it may be a valid option.

#+begin_src emacs-lisp (defun my-consult--source-recentf-items () (let ((ht (consult--buffer-file-hash)) file-name-handler-alist ;; No Tramp slowdown please. items) (dolist (file recentf-list (nreverse items)) ;; Emacs 29 abbreviates file paths by default, see ;; recentf-filename-handlers'. (unless (eq (aref file 0) ?/) (setq file (expand-file-name file))) (unless (gethash file ht) (push (propertize (file-name-nondirectory file) 'multi-category (file . ,file)) items)))))

(plist-put consult--source-recent-file :items #'my-consult--source-recentf-items) #+end_src

A more polished approach, albeit more expensive, is to uniquify the candidates with non-common path parts.

#+begin_src emacs-lisp (defun my-consult--source-recentf-items-uniq () (let ((ht (consult--buffer-file-hash)) file-name-handler-alist ;; No Tramp slowdown please. items) (dolist (file (my-recentf-list-uniq) (nreverse items)) ;; Emacs 29 abbreviates file paths by default, see ;; recentf-filename-handlers'. (unless (eq (aref (cdr file) 0) ?/) (setcdr file (expand-file-name (cdr file)))) (unless (gethash (cdr file) ht) (push (propertize (car file) 'multi-category (file . ,(cdr file))) items)))))

(plist-put consult--source-recent-file :items #'my-consult--source-recentf-items-uniq)

(defun my-recentf-list-uniq () (let* ((proposed (mapcar (lambda (f) (cons (file-name-nondirectory f) f)) recentf-list)) (recentf-uniq proposed) conflicts resol file) ;; collect conflicts (while proposed (setq file (pop proposed)) (if (assoc (car file) conflicts) (push (cdr file) (cdr (assoc (car file) conflicts))) (if (assoc (car file) proposed) (push (list (car file) (cdr file)) conflicts)))) ;; resolve conflicts (dolist (name conflicts) (let* ((files (mapcar (lambda (f) ;; data structure: ;; (file remaining-path curr-propos) (list f (file-name-directory f) (file-name-nondirectory f))) (cdr name))) (curr-step (mapcar (lambda (f) (file-name-nondirectory (directory-file-name (cadr f)))) files))) ;; Quick check, if there are no duplicates, we are done. (if (eq (length curr-step) (length (seq-uniq curr-step))) (setq resol (append resol (mapcar (lambda (f) (cons (car f) (file-name-concat (file-name-nondirectory (directory-file-name (cadr f))) (file-name-nondirectory (car f))))) files))) (while files (let (files-remain) (dolist (file files) (let ((curr-propos (caddr file)) (curr-part (file-name-nondirectory (directory-file-name (cadr file)))) (rest-path (file-name-directory (directory-file-name (cadr file)))) (curr-step (mapcar (lambda (f) (file-name-nondirectory (directory-file-name (cadr f)))) files))) (cond ((length= (seq-uniq curr-step) 1) ;; If all elements of curr-step are equal, we skip ;; this path part. (push (list (car file) rest-path curr-propos) files-remain)) ((member curr-part (cdr (member curr-part curr-step))) ;; There is more than one curr-part in curr-step ;; for this candidate. (push (list (car file) rest-path (file-name-concat curr-part curr-propos)) files-remain)) (t ;; There is no repetition of curr-part in curr-step ;; for this candidate. (push (cons (car file) (file-name-concat curr-part curr-propos)) resol))))) (setq files files-remain)))))) ;; apply resolved conflicts (let (items) (dolist (file recentf-uniq (nreverse items)) (let ((curr-resol (assoc (cdr file) resol))) (if curr-resol (push (cons (cdr curr-resol) (cdr file)) items) (push file items))))))) #+end_src

** Do not preview EXWM windows or Tramp buffers

NOTE: This section relates to the issues [[https://github.com/minad/consult/issues/178][#178]], [[https://github.com/minad/consult/issues/186][#186]], and [[https://github.com/minad/consult/issues/204][#204]]. The recommended solution is to define an [[#exwm-buffers][custom EXWM-specific buffer source]] without preview as is described in another section of the wiki.

Consult's buffer preview functionality causes issues when used with EXWM. Because EXWM can only display an X buffer in one window at a time, previewing the buffer removes it from the original window. If the buffer is shown in another frame, it will also fail to restore the X buffer after finishing buffer selection.

Please take a look first at the [[#exwm-buffers][custom EXWM-specific buffer source]], which works around this problem in a clean way. Alternatively we can define a new buffer state function which checks whether the buffer is an EXWM buffer before deciding whether it should preview the contents or not.

#+BEGIN_SRC emacs-lisp (defun consult-buffer-state-no-exwm () "Buffer state function that doesn't preview X buffers." (let ((orig-state (consult--buffer-state)) (filter (lambda (action cand) (and (or (eq action 'return) (and-let* ((cand) (buffer (get-buffer cand))) (not (eq 'exwm-mode (buffer-local-value 'major-mode buffer))))) cand)))) (lambda (action cand) (funcall orig-state action (funcall filter action cand)))))

(setq consult--source-buffer (plist-put consult--source-buffer :state #'consult-buffer-state-no-exwm)) #+END_SRC

This snippet defines a new :state function consult-buffer-state-no-exwm that we then use to replace the current :state function of consult--source-buffer.

Alternatively, if you wish to keep using previews with EXWM then add the following workaround to keep the minibuffer focused:

#+BEGIN_SRC emacs-lisp (defun consult-exwm-preview-fix (&rest _args) "Kludge to stop EXWM buffers from stealing focus during Consult previews." (when (derived-mode-p 'exwm-mode) (when-let ((mini (active-minibuffer-window))) (select-window (active-minibuffer-window)))))

(advice-add #'consult--buffer-preview :after #'consult-exwm-preview-fix) #+END_SRC

Similarly one can exclude Tramp buffers from preview. This helps if one uses Tramp over an unstable connections where Tramp buffer switching can be slow or can hang ([[https://github.com/minad/consult/issues/224][#224]]).

#+BEGIN_SRC emacs-lisp (defun consult-buffer-state-no-tramp () "Buffer state function that doesn't preview Tramp buffers." (let ((orig-state (consult--buffer-state)) (filter (lambda (action cand) (if (and cand (or (eq action 'return) (let ((buffer (get-buffer cand))) (and buffer (not (file-remote-p (buffer-local-value 'default-directory buffer))))))) cand nil)))) (lambda (action cand) (funcall orig-state action (funcall filter action cand)))))

(setq consult--source-buffer (plist-put consult--source-buffer :state #'consult-buffer-state-no-tramp)) #+END_SRC

  • Consult Buffer Sources ** Point register source

We can add a register source to consult-buffer. It will show registers containing markers to specific places in buffers.

#+begin_src emacs-lisp (defun consult--point-register-p (reg) "Return non-nil if REG is a point register." (markerp (cdr reg)))

(defvar-keymap consult-source-point-register `(:name "Point Register" :narrow (?r . "Register") :category consult-location :state ,(lambda () (let ((state (consult--jump-state))) (lambda (action cand) (funcall state action (and cand (car (consult--get-location cand))))))) :enabled ,(lambda () (seq-some #'consult--point-register-p register-alist)) :items ,(lambda () (consult-register--candidates #'consult--point-register-p))) "Point register source.")

(add-to-list 'consult-buffer-sources 'consult-source-point-register 'append) #+end_src

** Source for files in current directory

#+begin_src emacs-lisp (defvar +consult-source-neighbor-file (:name "File in current directory" :narrow ?. :category file :face consult-file :history file-name-history :state ,#'consult--file-state :new ,#'consult--file-action :items ,(lambda () (let ((ht (consult--buffer-file-hash)) items) (dolist (file (completion-pcm--filename-try-filter (directory-files "." 'full "\\[^.]" nil 100)) (nreverse items)) (unless (or (gethash file ht) (not (file-regular-p file))) (push (file-name-nondirectory file) items)))))) "Neighboring file source for `consult-buffer'.")

(unless (memq '+consult-source-neighbor-file consult-buffer-sources) (let ((p (member 'consult--source-buffer consult-buffer-sources))) (setcdr p (cons '+consult-source-neighbor-file (cdr p))))) #+end_src

** ERC Buffers

ERC is an IRC client. You can define a source containing only ERC buffers ([[https://github.com/minad/consult/issues/290][#290]]).

#+begin_src elisp (autoload 'erc-buffer-list "erc")

(defvar erc-buffer-source `(:name "ERC" :hidden t :narrow ?e :category buffer :state ,#'consult--buffer-state :items ,(lambda () (mapcar #'buffer-name (erc-buffer-list)))))

(add-to-list 'consult-buffer-sources 'erc-buffer-source 'append) #+end_src

If like me you have a dedicated tab for ERC using the built-in tab-bar-mode (starting 27.1), you can use this function to have initial narrowing under the "ERC" tab solely, so as to display the ERC related candidates ([[https://github.com/minad/consult/issues/290][#290]]).

#+begin_src elisp (defun consult-initial-narrow () (when (and (eq this-command #'consult-buffer) (string-equal "ERC" (alist-get 'name (alist-get 'current-tab (tab-bar-tabs))))) (setq unread-command-events (append unread-command-events (list ?e 32)))))

(add-hook 'minibuffer-setup-hook #'consult-initial-narrow) #+end_src

** Circe Buffers

Circe is an alternative IRC client. Similar to ERC, Consult buffer sources can be defined. Because of the way that circe separates chat and server buffers, the :items function is a bit more involved:

#+begin_src elisp (require 'cl-lib) (autoload 'circe-server-buffers "circe") (autoload 'circe-server-chat-buffers "circe")

(defun circe-all-buffers () (cl-loop with servers = (circe-server-buffers) for server in servers collect server nconc (with-current-buffer server (cl-loop for buf in (circe-server-chat-buffers) collect buf))))

(defvar circe-buffer-source `(:name "circe" :hidden t :narrow ?c :category buffer :state ,#'consult--buffer-state :items ,(lambda () (mapcar #'buffer-name (circe-all-buffers)))))

(add-to-list 'consult-buffer-sources 'circe-buffer-source 'append) #+end_src

** Eww Bookmarks

Since Emacs 28, Eww makes use of the standard Emacs bookmark infrastructure. The old-style Eww bookmarks can be integrated with Consult as follows. See discussion in [[https://github.com/minad/consult/issues/347][#347]].

#+begin_src emacs-lisp (require 'eww)

(defvar consult--source-eww (list :name "Eww" :narrow ?e :action (lambda (bm) (eww-browse-url (get-text-property 0 'url bm))) :items (lambda () (eww-read-bookmarks) (mapcar (lambda (bm) (propertize (format "%s (%s)" (plist-get bm :url) (plist-get bm :title)) 'url (plist-get bm :url))) eww-bookmarks))))

(add-to-list 'consult-buffer-sources 'consult--source-eww 'append) #+end_src

** EXWM Buffers

To group all EXWM windows together, we can create an +consult-source-exwm and add it to the list of buffer sources. Preview is disabled for the EXWM buffers here since X11 buffers cannot be duplicated. We also hide EXWM buffers from the other buffer sources.

#+begin_src emacs-lisp (defvar +consult-exwm-filter "\`\*EXWM") (add-to-list 'consult-buffer-filter +consult-exwm-filter)

(defvar +consult-source-exwm `(:name "EXWM" :narrow ?x ;; :hidden t :category buffer :face consult-buffer :history buffer-name-history ;; Specify either :action or :state :action ,#'consult--buffer-action ;; No preview ;; :state ,#'consult--buffer-state ;; Preview :items ,(lambda () (consult--buffer-query :sort 'visibility :as #'buffer-name :exclude (remq +consult-exwm-filter consult-buffer-filter) :mode 'exwm-mode))) "EXWM buffer source.") #+end_src

** Bufler Add a consult-buffer source to group buffers from the current [[https://github.com/alphapapa/bufler.el][bufler]] workspace.

#+begin_src emacs-lisp (defvar consult--bufler-workspace+ (:name "Workspace" :narrow ?w :category buffer :face consult-buffer :history buffer-name-history :state ,#'consult--buffer-state :enabled ,(lambda () (frame-parameter nil 'bufler-workspace-path)) :items ,(lambda () (let ((bufler-vc-state nil)) (mapcar #'buffer-name (mapcar #'cdr (bufler-buffer-alist-at (frame-parameter nil 'bufler-workspace-path) :filter-fns bufler-filter-buffer-fns)))))) "Bufler workspace buffers source for consult-buffer'.")

(with-eval-after-load 'consult (push #'consult--bufler-workspace+ consult-buffer-sources)) #+end_src

** Dogears

Dogears source for Consult. See [[https://github.com/minad/consult/issues/430][#430]].

#+begin_src emacs-lisp (defvar consult--source-dogears (list :name "Dogears" :narrow ?d :category 'dogears :items (lambda () (mapcar (lambda (place) (propertize (dogears--format-record place) 'consult--candidate place)) dogears-list)) :action (lambda (cand) (dogears-go (get-text-property 0 'consult--candidate cand)))))

(defun consult-dogears () (interactive) (consult--multi '(consult--source-dogears))) #+end_src

** Perspective

Use consult-buffer with [[https://github.com/nex3/perspective-el][perspective-el]]. This would hide the default consult--source-buffer, and show the list of perspective buffers on the top

#+begin_src emacs-lisp (consult-customize consult--source-buffer :hidden t :default nil) (consult-customize consult--source-buffer :hidden t :default nil)

(defvar consult--source-perspective (list :name "Perspective" :narrow ?s :category 'buffer :state #'consult--buffer-state :default t :items #'persp-get-buffer-names))

(push consult--source-perspective consult-buffer-sources) #+end_src

** Bookmark views

Sources can be added directly to the =consult-buffer-source= list for convenience. For example views/perspectives can be added to the list of virtual buffers from a library like [[https://github.com/minad/bookmark-view/][bookmark-view]].

#+begin_src emacs-lisp ;; Configure new bookmark-view source (add-to-list 'consult-buffer-sources (list :name "View" :narrow ?v :category 'bookmark :face 'font-lock-keyword-face :history 'bookmark-view-history :action #'consult--bookmark-action :items #'bookmark-view-names) 'append)

;; Modify bookmark source, such that views are hidden (setq consult--source-bookmark (plist-put consult--source-bookmark :items (lambda () (bookmark-maybe-load-default-file) (mapcar #'car (seq-remove (lambda (x) (eq #'bookmark-view-handler (alist-get 'handler (cdr x)))) bookmark-alist))))) #+end_src

  • Commands ** consult-line-literal which matches only literally

consult-line uses the completion-styles for matching. One can write a wrapper around consult-line which adjusts the completion styles to the desired configuration.

#+begin_src emacs-lisp ;; Use the substring completion style (defun consult-line-literal () (interactive) (let ((completion-styles '(substring)) (completion-category-defaults nil) (completion-category-overrides nil)) (consult-line)))

;; Use the orderless completion style, restricted to orderless-literal (defun consult-line-literal () (interactive) (let ((completion-styles '(orderless)) (orderless-matching-styles '(orderless-literal)) (completion-category-defaults nil) (completion-category-overrides nil)) (consult-line))) #+end_src

** Start consult-line search with symbol at point

The symbol at point can be passed as initial argument to consult-line.

#+begin_src elisp (defun consult-line-symbol-at-point () (interactive) (consult-line (thing-at-point 'symbol))) #+end_src

** Start consult-ripgrep search with active region

Conditionally use the active region as the =initial= parameter value for consult-ripgrep.

#+begin_src elisp (defun wrapper/consult-ripgrep (&optional dir given-initial) "Pass the region to consult-ripgrep if available.

DIR and GIVEN-INITIAL match the method signature of `consult-wrapper'." (interactive "P") (let ((initial (or given-initial (when (use-region-p) (buffer-substring-no-properties (region-beginning) (region-end)))))) (consult-ripgrep dir initial))) #+end_src

** Restart consult-ripgrep in parent directory

We can define a command to restart the current consult-ripgrep search in the parent directory. See [[https://github.com/minad/consult/issues/596][issue 596]] for the background.

#+begin_src emacs-lisp (defun consult-ripgrep-up-directory () (interactive) (let ((parent-dir (file-name-directory (directory-file-name default-directory)))) (when parent-dir (run-at-time 0 nil #'consult-ripgrep parent-dir (ignore-errors (buffer-substring-no-properties (1+ (minibuffer-prompt-end)) (point-max)))))) (minibuffer-quit-recursive-edit))

(consult-customize consult-ripgrep :keymap (let ((map (make-sparse-keymap))) (define-key map (kbd "M-l") #'consult-ripgrep-up-directory) map)) #+end_src

** consult-ripgrep-or-line (counsel-grep-or-swiper equivalent)

If consult-line is slow in large buffers, this may be useful. Without native compilation, I can increase the limit significantly. With native compilation, consult-line is near instant in my largest org file already.

#+begin_src elisp (defcustom my/consult-ripgrep-or-line-limit 300000 "Buffer size threshold for my/consult-ripgrep-or-line'. When the number of characters in a buffer exceeds this threshold, consult-ripgrep' will be used instead of `consult-line'." :type 'integer)

(defun my/consult-ripgrep-or-line () "Call consult-line' for small buffers or consult-ripgrep' for large files." (interactive) (if (or (not buffer-file-name) (buffer-narrowed-p) (ignore-errors (file-remote-p buffer-file-name)) (jka-compr-get-compression-info buffer-file-name) (<= (buffer-size) (/ my/consult-ripgrep-or-line-limit (if (eq major-mode 'org-mode) 4 1)))) (consult-line) (when (file-writable-p buffer-file-name) (save-buffer)) (let ((consult-ripgrep-args (concat consult-ripgrep-args ;; filter to desired filename " -g " (shell-quote-argument (file-name-nondirectory buffer-file-name)) " "))) (consult-ripgrep)))) #+end_src

** Access directories of text notes files

hrm-notes is a command to access text file notes from several directories easily. It uses consult-multi and includes embark integration. It's a simple deft-like command.

** Using =find= in the minibuffer

The following command, meant to be called in the minibuffer when it is reading a file name, switches from the usual hierarchical browsing of the file system to a =consult-find= session.

#+begin_src emacs-lisp (defun consult-find-for-minibuffer () "Search file with find, enter the result in the minibuffer." (interactive) (let* ((enable-recursive-minibuffers t) (default-directory (file-name-directory (minibuffer-contents))) (file (consult--find (replace-regexp-in-string "\s-[:([]." (format " (via find in %s): " default-directory) (minibuffer-prompt)) (consult--find-make-builder) (file-name-nondirectory (minibuffer-contents))))) (delete-minibuffer-contents) (insert (expand-file-name file default-directory)) (exit-minibuffer))) #+end_src

It is convenient to add a [[#add-category-specific-minibuffer-keybindings][category-specific keybinding]] to this command.

** Including file recently used by other programs

If you find yourself using other programs with Emacs, it can be helpful to include files used by other programs in the candidate lists of commands like consult-recent-file and consult-buffer. That way, you never have any mental hiccups when trying to open files in Emacs that you recently opened in a different program. Instead, you simply use the same interface with which you are already familiar.

/I put this code into [[https://github.com/hrehfeld/consult-xdg-recent-files][consult-xdg-recent-files]], which should be installable with straight. Maybe this saves someone the maintenance./

The way to access this information is generally specific to each system. Please update this section for other systems, if you find this feature useful.

In Linux (or, more specifically, on systems that comply with the [[https://www.freedesktop.org/wiki/Specifications/desktop-bookmark-spec/][XDG specification]]), these files are listed in the file =recently-used.xbel=, which is found in the directory =~/.local/share= or the location described by the environment variable =XDG_DATA_HOME=.

We can access the data in this file using libraries built-in with Emacs, namely =url-util.el=, =dom.el=, and one of =xml.c= or =xml.el=.

#+begin_src emacs-lisp (require 'dom) (require 'url-util) (require 'xml)

(defun consult--xdg-recent-file-list () "Get a list of recently used files on XDG-compliant systems.

This function extracts a list of files from the file recently-used.xbel' in the folder xdg-data-home'.

For more information on this specification, see https://www.freedesktop.org/wiki/Specifications/desktop-bookmark-spec/" (let ((data-file (expand-file-name "recently-used.xbel" (xdg-data-home))) (xml-parsing-func (if (libxml-available-p) #'libxml-parse-xml-region #'xml-parse-region))) (if (file-readable-p data-file) (delq nil (mapcar (lambda (bookmark-node) (when-let ((local-path (string-remove-prefix "file://" (dom-attr bookmark-node 'href)))) (let ((full-file-name (decode-coding-string (url-unhex-string local-path) 'utf-8))) (when (file-exists-p full-file-name) full-file-name)))) (nreverse (dom-by-tag (with-temp-buffer (insert-file-contents data-file) (funcall xml-parsing-func (point-min) (point-max))) 'bookmark)))) (message "consult: List of XDG recent files not found") '()))) #+end_src

If using multiple systems, then it is good to wrap such a feature in a dispatching function.

#+begin_src emacs-lisp (require 'cl-lib)

(defun consult--recent-system-files () "Return a list of files recently used by the system." (cl-case system-type (gnu/linux (consult--xdg-recent-file-list)) (t (message "consult-recent-file: "%s" currently unsupported" system-type) '()))) #+end_src

Generally, one would want to sort these files from most recently used to least recently used. A file's modification time works well for this, and isn't disturbed when Emacs accesses the file.

#+begin_src emacs-lisp (defun consult--recent-files-sort (file-list) "Sort the FILE-LIST by modification time, from most recent to least recent." (thread-last file-list ;; Use modification time, since getting file access time seems to count as ;; accessing the file, ruining future uses. (mapcar (lambda (f) (cons f (file-attribute-modification-time (file-attributes f))))) (seq-sort (pcase-lambda ((,f1 . ,t1) (,f2 . ,t2)) ;; Want existing, most recent, local files first. (cond ((or (not (file-exists-p f1)) (file-remote-p f1)) nil) ((or (not (file-exists-p f2)) (file-remote-p f2)) t) (t (time-less-p t2 t1))))) (mapcar #'car))) #+end_src

To mix these candidates with those found in the variable recentf-list, we only need to filter according the function recentf-include-p.

#+begin_src emacs-lisp (defun consult--recent-files-mixed-candidates () "Return a list of files recently used by Emacs and the system.

These files are sorted by modification time, from most recent to least." (thread-last (consult--recent-system-files) (seq-filter #'recentf-include-p) (append (mapcar #'substring-no-properties recentf-list)) delete-dups (consult--recent-files-sort))) #+end_src

To include the mixed candidates in consult-recent-file, we can slightly modify its definition.

#+begin_src emacs-lisp (defcustom consult-include-system-recent-files nil "Whether to include files used by other programs in `consult-recent-file'." :type 'boolean :group 'consult)

;;;###autoload (defun consult-recent-file () "Find recent using `completing-read'." (interactive) (find-file (consult--read (or (mapcar #'abbreviate-file-name (if consult-include-system-recent-files (consult--recent-files-mixed-candidates) recentf-list)) (user-error "No recent files")) :prompt "Find recent file: " :sort nil :require-match t :category 'file :state (consult--file-preview) :history 'file-name-history))) #+end_src

To include these candidates in consult-buffer, we can add a source to the variable consult-buffer-sources.

#+begin_src emacs-lisp (defvar consult--source-system-file (:name "System file" :narrow ?F :category file :face consult-file :history file-name-history :action ,#'consult--file-action :items ,(lambda () (let ((ht (consult--buffer-file-hash))) (mapcar #'abbreviate-file-name (seq-remove (lambda (x) (gethash x ht)) (consult--recent-system-files)))))) "Recent system file candidate source for consult-buffer'.")

(defvar consult--source-mixed-file (:name "File" :narrow ?f :category file :face consult-file :history file-name-history :action ,#'consult--file-action :items ,(lambda () (let ((ht (consult--buffer-file-hash))) (mapcar #'abbreviate-file-name (seq-remove (lambda (x) (gethash x ht)) (consult--recent-files-mixed-candidates)))))) "File candidate source for consult-buffer', including system files. This is meant as a replacement for `consult--source-file'.")

;; Example: using the "mixed" source in `consult-buffer': (setq consult-buffer-sources '( consult--source-hidden-buffer consult--source-buffer consult--source-mixed-file consult--source-bookmark consult--source-project-buffer consult--source-project-file)) #+end_src

** Org clock

The following is a simple command to select and clock into an agenda entry.

#+begin_src emacs-lisp (defun consult-clock-in () "Clock into an Org agenda heading." (interactive) (save-window-excursion (consult-org-agenda) (org-clock-in)))

(consult-customize consult-clock-in :prompt "Clock in: " :preview-key "M-.") #+end_src

Below is a fancier version with the following perks, which you may pick and choose:

  • Instead of offering agenda entries, offer headings from all files that have a recent clock entry.
  • Sort recent clock entries separately under a =Recent= group.
  • With a prefix argument, resolve dangling clocks and ask for a time to clock into the selected task.

#+begin_src emacs-lisp (setq org-clock-persist t) (with-eval-after-load 'org (org-clock-persistence-insinuate))

(defun consult-clock-in (&optional match scope resolve) "Clock into an Org heading." (interactive (list nil nil current-prefix-arg)) (require 'org-clock) (org-clock-load) (save-window-excursion (consult-org-heading match (or scope (thread-last org-clock-history (mapcar 'marker-buffer) (mapcar 'buffer-file-name) (delete-dups) (delq nil)) (user-error "No recent clocked tasks"))) (org-clock-in nil (when resolve (org-resolve-clocks) (org-read-date t t)))))

(consult-customize consult-clock-in :prompt "Clock in: " :preview-key "M-." :group (lambda (cand transform) (let* ((marker (get-text-property 0 'consult--candidate cand)) (name (if (member marker org-clock-history) "Recent" (buffer-name (marker-buffer marker))))) (if transform (substring cand (1+ (length name))) name)))) #+end_src

Note that these commands can also be used as an Embark action.

** Org capture

Normally, an Org capture target specifies a fixed file or heading within a file as its target. The following example shows how to define a capture target that first queries an agenda entry using =consult-org-headline=, and then places the capture directly beneath it.

#+begin_src elisp (defun consult-org-capture-target (scope) "Choose a capture target interactively. This function returns a value suitable for use as the target' entry of org-capture-templates'. SCOPE is as in org-map-entries'." (list 'function (lambda () (let ((consult--read-config ((,this-command :prompt "Capture target: " :preview-key "M-.")))) (set-buffer (save-window-excursion (consult-org-heading nil scope) (current-buffer)))))))

(add-to-list 'org-capture-templates `(("c" "Consult..." entry ,(consult-org-capture-target 'agenda) "* TODO %?\n %i" :prepend t))) #+end_src

You may also wish to have a direct keybinding to this capture type, instead of going though the =M-x org-capture= menu. In this case, use the following:

#+begin_src elisp (defun consult-org-capture () (interactive) (org-capture nil "c")) #+end_src

** =isearch=-like backward/forward =consult-line=

Type =C-s= to search forward and =C-r= to search backward. Requires the =vertico-reverse= extension to be enabled.

#+begin_src elisp (defun my/consult-line-forward () "Search for a matching line forward." (interactive) (consult-line))

(defun my/consult-line-backward () "Search for a matching line backward." (interactive) (advice-add 'consult--line-candidates :filter-return 'reverse) (vertico-reverse-mode +1) (unwind-protect (consult-line) (vertico-reverse-mode -1) (advice-remove 'consult--line-candidates 'reverse)))

(with-eval-after-load 'consult (consult-customize my/consult-line-backward :prompt "Go to line backward: ") (consult-customize my/consult-line-forward :prompt "Go to line forward: "))

(global-set-key (kbd "C-s") 'my/consult-line-forward) (global-set-key (kbd "C-r") 'my/consult-line-backward) #+end_src

** Completing a hierarchical outline

This is based on the command selectrum-outline, minimally modified to use consult--read.

#+begin_src emacs-lisp (defvar my-consult-outline-path-history nil "History of chosen headings for ‘my-consult-outline-path’.")

(defcustom my-consult-outline-path-formats ;; Groups: (1) level determinant, (2) heading text. ;; The top level is 0, for a zero-length determinant. ((emacs-lisp-mode . "^;;;\\(?1:;*\\)[[:blank:]]*\\(?2:[[:alnum:]][^z-a]*\\)\\'") (diff-mode ;; We just need a zero-length thing to match the file line ;; and a one-length thing to match the section line. ;; This works even with diff-font-lock-prettify' enabled. . ,(rx (or (seq line-start (group-n 1 "") "diff " (0+ nonl) "b/" (group-n 2 (+? nonl)) string-end) (seq line-start "@" (group-n 1 "@") " " (group-n 2 (0+ nonl)) string-end)))) (lisp-mode . "^;;;\(?1:;\):blank:\(?2::alnum:[^z-a]\)\'") (lua-mode . "^---\(?1:-\):blank:\(?2::alnum:[^z-a]\)\'") (gfm-mode ; Github Flavored Markdown . "^#\(?1:#\):blank:\(?2::alnum:[^z-a]\)\'") (markdown-mode . "^#\(?1:#\):blank:\(?2::alnum:[^z-a]\)\'") (outline-mode . "^\\(?1:\**\):blank:\(?2::alnum:[^z-a]\)\'") ;; For Org, see also ‘org-goto’. (org-mode . "^\\(?1:\\):blank:+\(?2::alnum:[^z-a]*\)\'") (python-mode . "^##\(?1:\\|#\):blank:\(?2::alnum:[^z-a]\)\'") (shortdoc-mode . "^\(?1:\)\(?2:[A-Z].$\)")) "Alist of regexps used for identifying outline headings in each major mode.

The ‘car’ of an item in the list should be a symbol of the major mode. The ‘cdr’ should be a regular expression with two required match groups:

  1. Match group 1, whose length determines the outline level of that heading. For best formatting, the top level should be level 0 for zero length.
  2. Match group 2, which is the actual heading text.

A heading is assumed to be on only one line." :group 'consult :type '(alist :key-type (symbol :tag "Major mode symbol") :value-type (string :tag "Regexp")))

;;;###autoload (defun my-consult-outline-path () "Jump to a heading. Regexps are pre-defined. Obeys narrowing." (interactive) (if-let ((heading-regexp (alist-get major-mode my-consult-outline-path-formats))) (let ((candidates) (default-heading) (initial-line-number (line-number-at-pos (point)))) (save-excursion (goto-char (point-min)) (let* ((line-number (line-number-at-pos (point))) (point-max (point-max)) (beg (point)) (end (line-end-position))

               (backwards-prefix-list)
               (prev-heading-text)
               (prev-heading-level)
               (heading-text)
               (heading-level)

               (formatted-heading))
          (save-match-data
            (while (< end point-max)
              (let ((text-line (buffer-substring beg end)))
                (when (string-match heading-regexp text-line)
                  (setq prev-heading-text  heading-text
                        prev-heading-level heading-level
                        heading-text  (match-string-no-properties 2 text-line)
                        heading-level (- (match-end 1) (match-beginning  1)))

                  ;; Decide whether to update the prefix list and the previous
                  ;; heading level.
                  (let ((prev-heading-level (or prev-heading-level heading-level)))
                    (cond
                     ;; If we've moved to a greater level (further down the tree),
                     ;; add the previous heading to the heading prefix list so
                     ;; that we can prepend it to the current heading when
                     ;; formatting.
                     ((> heading-level prev-heading-level)
                      (push prev-heading-text backwards-prefix-list))
                     ((< heading-level prev-heading-level)
                      ;; Otherwise, if we've moved to a lower level (higher up the
                      ;; tree), and need to remove the most recently added prefix
                      ;; from the list (i.e., go from '(c b a) back to '(b a)).
                      (cl-callf2 nthcdr (- prev-heading-level heading-level)
                                 backwards-prefix-list))))

                  ;; If needed, set default candidate.
                  (when (and (null default-heading)
                             (> line-number initial-line-number))
                    (setq default-heading formatted-heading))

                  (setq formatted-heading
                        (propertize
                         (concat (string-join (reverse backwards-prefix-list) "/")
                                 (and backwards-prefix-list "/")
                                 heading-text)
                         'line-number line-number))

                  (push formatted-heading candidates)))
              (cl-incf line-number)
              (forward-line 1)
              (setq beg (point)
                    end (line-end-position))))

          (unless default-heading
            (setq default-heading formatted-heading))))

      (cl-flet ((ln (str) (get-text-property 0 'line-number str)))
        (let* ((line-number-format
                (format "L%%0%dd: "
                        (length (number-to-string (ln (car candidates))))))
               (affixate-func
                (lambda (cand)
                  (list cand (propertize (format line-number-format (ln cand))
                                         'face 'completions-annotations)
                        "")))
               (lookup-fn (lambda (selected candidates &rest _)
                            (consult--lookup-prop 'line-number
                                                  selected candidates)))
               (chosen-line (consult--read (nreverse candidates)
                                           :prompt "Jump to heading: "
                                           :require-match t
                                           :history 'my-consult-outline-path-history
                                           ;; TODO: Want to select default
                                           ;;       without moving it to the
                                           ;;       top of the list.
                                           ;; :default default-heading
                                           :annotate affixate-func
                                           :lookup lookup-fn
                                           :sort nil)))

          ;; Push mark, in case we want to return to current location.  This
          ;; needs to happen /after/ the user has made it clear that they
          ;; want to go somewhere.
          (push-mark (point) t)
          ;; Move to beginning of chosen line.
          (forward-line (- chosen-line initial-line-number))
          (beginning-of-line-text 1)
          ;; Return non-nil for advice combinator `after-while'.
          t)))
  (call-interactively #'consult-outline)))

#+end_src

** Completing tabs from tab-bar with preview

This can be useful if you want to set =tab-bar-show= to =nil=

#+begin_src emacs-lisp

(defun +tab-bar--make-completion-list (tab-list) "Return completion list of strings formatted from TAB-LIST." (mapcar (lambda (tab) (let ((index (1+ (tab-bar--tab-index tab))) (name (alist-get 'name tab))) (format "%d %s" index name))) tab-list))

(defun +tab-bar--completion-list-recent () "Return completion list of recent tabs (current not included)." (+tab-bar--make-completion-list (tab-bar--tabs-recent)))

(defun +tab-bar--index-from-candidate (cand) "Return prefix index of CAND." (let ((match (string-match "^:digit:+" cand))) (when match (string-to-number (match-string match cand)))))

(defun +tab-bar--tab-from-index (index) "Return tab from `(tab-bar-tabs)' by index of CAND." (when index (nth (1- index) (tab-bar-tabs))))

(defun +consult--tab-preview () "Preview function for tabs." (let ((orig-wc (current-window-configuration))) (lambda (action cand) (if (eq action 'exit) (set-window-configuration orig-wc nil t) (when cand (let* ((index (+tab-bar--index-from-candidate cand)) (tab (+tab-bar--tab-from-index index))) (when tab (if (eq (car tab) 'current-tab) (set-window-configuration orig-wc nil t) (set-window-configuration (alist-get 'wc tab) nil t)))))))))

(defun +consult--tab-annotate (cand) "Annotate current tab." (when (equal (car (+tab-bar--tab-from-index (+tab-bar--index-from-candidate cand))) 'current-tab) "Current"))

(defun +consult--tab-action-select (cand) "Select tab from CAND." (tab-bar-select-tab (+tab-bar--index-from-candidate cand)))

(defvar +consult--tab-history "History of tab completion selections.")

(defvar +consult--source-tab-recent (list :name "Tab" :category 'tab :narrow ?t :default t :history '+consult--tab-history :items #'+tab-bar--completion-list-recent :annotate #'+consult--tab-annotate :action #'+consult--tab-action-select :state #'+consult--tab-preview))

(defun +consult-tab () "Select tab with completion and preview." (interactive) (consult--multi '(+consult--source-tab-recent) :prompt "Select tab: "))

(defun +consult-tab-close () "Select tab to close it." (interactive) (tab-bar-close-tab (+tab-bar--index-from-candidate (car (consult--multi '(+consult--source-tab-recent) :prompt "Close tab: ")))))

#+end_src

** Another approach to tab-bar completion with marginalia

Due to the fact that it is almost impossible to properly complete tabs from a tab bar by their name if their names are not unique, the only thing left is to complete their indexes. This approach is an improvement on the code from the previous section. Only now we use marginalia to display the name.

Here is the result

#+begin_src emacs-lisp ;; -- lexical-binding: t; --

(require 'consult) (require 'marginalia)

(defvar +consult--tab-index-current-tab-name nil "The name of the current tab. Needed for marginalia annotations when previewing tabs. Because we are changing the current window configuration when previewing tabs, we are also changing the name of the current tab unless it's not an explicit name. To prevent this, we can store the name of the current tab before calling consult command and use this saved name in marginalia annotations of the current tab.")

(defvar +consult--tab-index-current-tab-bufs nil "List of current tab buffer names. Needed for marginalia annotations when previewing tabs. Because we are changing the current window configuration when previewing tabs, we need to save the current list of buffers displayed in windows before calling consult command and use this saved list in marginalia annotations of the current tab.")

(defun +marginalia-annotate-tab-index (cand) "Modified version of `marginalia-annotate-tab' suited for tab-index completion." (let* ((tab (nth (1- (string-to-number cand)) (tab-bar-tabs))) (current-p (memq 'current-tab tab)) (ws (alist-get 'ws tab)) (bufs (if current-p +consult--tab-index-current-tab-bufs (window-state-buffers ws)))) ;; NOTE: When the buffer key is present in the window state ;; it is added in front of the window buffer list and gets duplicated. (unless current-p (when (cadr (assq 'buffer ws)) (pop bufs))) (marginalia--fields ;; Tab name ((if current-p +consult--tab-index-current-tab-name (alist-get 'name tab)) :face (if current-p 'marginalia-on 'marginalia-key) :width 15 :truncate 15) ;; Window count ((if (cdr bufs) (format "%d windows" (length bufs)) "1 window ") :face 'marginalia-size :width 15) ;; List of buffers ((string-join bufs " \t ") :face 'marginalia-documentation))))

(add-to-list 'marginalia-annotator-registry '(tab-index +marginalia-annotate-tab-index))

(defun +consult--tab-index-preview () "Preview function for tab-index." (let ((orig-wc (current-window-configuration))) (lambda (action cand) (if (eq action 'exit) (set-window-configuration orig-wc nil t) (when cand (set-window-configuration (alist-get 'wc (nth (1- (string-to-number cand)) (tab-bar-tabs)) ;; default to original wc if ;; there is no tab wc (usually current tab) orig-wc) nil t))))))

(defvar +consult--source-tab-index (list :name "Tab" :category 'tab-index :default t :narrow ?t :state #'+consult--tab-index-preview :items (lambda () (mapcar #'number-to-string (number-sequence 1 (length (tab-bar-tabs)))))) "Source of all tab indexes starting from 1.")

(defun +consult--tab-index (&optional prompt) "Prompt for tab selection and return selected candidate as number. Replace prompt with PROMPT if specified." ;; Marginalia integration (let (;; Align annotations as close to index as possible (marginalia-align-offset -18) ;; Save curret tab name (+consult--tab-index-current-tab-name (alist-get 'name (tab-bar--current-tab))) ;; Save current window buffer list (+consult--tab-index-current-tab-bufs (mapcar #'buffer-name (mapcar #'window-buffer (window-list))))) (string-to-number (car (consult--multi '(+consult--source-tab-index) ;; disable sorting :sort nil :require-match t :prompt (or prompt "Select tab: "))))))

;;;###autoload (defun +consult-tab () "Select tab and switch to it." (interactive) (tab-bar-select-tab (+consult--tab-index))) #+end_src

*** Additions

This approach is very extensible, here are some examples

**** Preselect recent tab with =vertico--goto=

#+begin_src emacs-lisp (defvar +consult--tab-index-commands '(+tab-bar-dwim +consult-tab +consult-tab-close*) "List of commands that will trigger +consult--tab-index-preselect' and +consult--tab-index-refresh'")

(defun +consult--tab-index-preselect () "Preselect recent tab if this-command' in +consult--tab-index-commands'." (when (memq this-command +consult--tab-index-commands) (vertico--goto (or (tab-bar--tab-index-recent 1) (tab-bar--current-tab-index)))))

(add-hook 'minibuffer-setup-hook #'+consult--tab-index-preselect)

(defun +consult--tab-index-refresh () "Run consult-vertico--refresh' if this-command' in `+consult--tab-index-commands'." (when (memq this-command +consult--tab-index-commands) (consult-vertico--refresh)))

(advice-add #'vertico--setup :after #'+consult--tab-index-refresh) #+end_src

**** DWIM function that i took from Prot and adapted

#+begin_src emacs-lisp ;;;###autoload (defun +tab-bar-dwim (&optional arg) "Do-What-I-Mean function for tabs. If optional prefix argument is specified, then switch to `ARG'th tab.

If no other tab exists, create one and switch to it.

If there is one other tab (two in total), switch to it.

If there are more than two tabs, select tab with `+consult-tab'." (interactive "P") (if arg (tab-bar-select-tab arg) (pcase (length (tab-bar-tabs)) (1 (tab-bar-new-tab)) (2 (tab-bar-switch-to-next-tab)) (_ (+consult-tab))))) #+end_src

**** Close multiple tabs

#+begin_src emacs-lisp ;;;###autoload (defun +consult-tab-close* () "Close multiple tabs." (interactive) (let (index) (while (setq index (+consult--tab-index "Close tab: ")) (tab-bar-close-tab index)))) #+end_src

**** Embark integration

#+begin_src emacs-lisp (require 'embark)

(defun +embark-tab-close (tab-index) "Close tab." (tab-bar-close-tab (1- (string-to-number tab-index))))

(defun +embark-tab-rename (tab-index) "Rename tab." (setq current-prefix-arg (string-to-number tab-index)) (call-interactively #'tab-bar-rename-tab))

(defvar-keymap +embark-tab-index-map :doc "Keymap for tab-index." "k" #'+embark-tab-close "r" #'+embark-tab-rename) (add-to-list 'embark-keymap-alist '(tab-index . +embark-tab-index-map)) #+end_src

** Emacs and web colors list.

Functions similar to =counsel-colors-emacs= and =counsel-colors-web=. Insert color name from the list of supported colors or, via embark actions, insert RGB or HEX values.

*** Consult colors functions #+begin_src elisp (defvar consult-colors-history nil "History for consult-colors-emacs' and consult-colors-web'.")

;; No longer preloaded in Emacs 28. (autoload 'list-colors-duplicates "facemenu") ;; No preloaded in consult.el (autoload 'consult--read "consult")

(defun consult-colors-emacs (color) "Show a list of all supported colors for a particular frame.\

You can insert the name (default), or insert or kill the hexadecimal or RGB value of the selected color." (interactive (list (consult--read (list-colors-duplicates (defined-colors)) :prompt "Emacs color: " :require-match t :category 'color :history '(:input consult-colors-history) ))) (insert color))

;; Adapted from counsel.el to get web colors. (defun counsel-colors--web-list nil "Return list of CSS colors for `counsult-colors-web'." (require 'shr-color) (sort (mapcar #'downcase (mapcar #'car shr-color-html-colors-alist)) #'string-lessp))

(defun consult-colors-web (color) "Show a list of all CSS colors.\

You can insert the name (default), or insert or kill the hexadecimal or RGB value of the selected color." (interactive (list (consult--read (counsel-colors--web-list) :prompt "Color: " :require-match t :category 'color :history '(:input consult-colors-history) ))) (insert color)) #+end_src

*** Embark integration **** Convert color's names

#+begin_src elisp (defun rounding-numbers (list-of-num decimal-points) "Return (as a float) the list of nearest integers to each number of list-of-num." (let ((rounding (expt 10 decimal-points))) (mapcar (lambda (x) (/ (fround (* rounding x)) rounding)) list-of-num)))

(defun numbers-to-string (list-of-num SEPARATOR) "Converts a list of numbers to a string "num1,num2,num3,..."." (mapconcat #'number-to-string list-of-num SEPARATOR))

;; Colors RGB number as string (defvar color-rgb-round-decimal-points 2 "Number of decimal points to round RGB colors.") (defvar color-rgb-string-separator "," "SEPARATOR between numbers for RGB strings.")

(defun color-name-to-rgb-string (NAME) "Return the RGB value of color NAME as string "num1,num2,num3", with num between 0 and 1. Return nil if NAME does not designate a valid color." (when-let ((rgb (color-name-to-rgb NAME))) (numbers-to-string rgb color-rgb-string-separator)))

(defun color-name-to-round-rgb-string (NAME) "Returns the rounded RGB value of color as string "num1,num2,num3", with num between 0 and 1. Return nil if NAME does not designate a valid color." (when-let ((rgb (color-name-to-rgb NAME))) (numbers-to-string (rounding-numbers rgb color-rgb-round-decimal-points) color-rgb-string-separator)))

;; Adapted from counsel.el to conver color name to hex. (defun counsel-colors--hex (NAME) "Return hexadecimal value of color with NAME. Return nil if NAME does not designate a valid color." (when-let* ((rgb (color-name-to-rgb NAME)) ;; Sets 2 digits per component. (hex (apply #'color-rgb-to-hex (append rgb '(2))))) hex)) #+end_src

**** Embark config

Config using Doom emacs. For doomed users not using Doom, evaluate code in =(after! embark ... )= using =define-key= instead of =map!=.

#+begin_src elisp (after! embark (defvar-keymap embark-consult-color-action-map :doc "Keymap for embark actions in the `color' category of marginalia.")

;; Kill and insert versions (defvar embark-consult-color-functions-alist '(((color-name-to-round-rgb-string . "rRGB") . ("r" . "k")) ((color-name-to-rgb-string . "RGB") . ("R" . "K")) ((counsel-colors--hex . "hex") . ("h" . "H"))) "Cons list of ((fun . desc) . (bind_insert . bind_kill)) of functions converting a color name to some value. Used to define their insert' and kill-new' versions for embark actions.")

;; Define insert' versions (cl-loop for fun in embark-consult-color-functions-alist do ;; (message "dir %s, name %s" (car dirname) (cdr dirname)) (let* ((sym (caar fun)) (bind (cadr fun)) (desc (format "Insert %s" (cdar fun))) (newname (intern (format "%s-insert" (symbol-name sym))))) ;; (lambda (color) (insert (apply ',fun (list color)))) (fset newname `(lambda (color) (insert (,sym color)) (pushnew! consult-colors-history color))) ;; (define-key embark-consult-color-action-map (kbd bind) (cons desc newname)) (map! :map embark-consult-color-action-map :desc desc bind newname)))

;; Define kill-new' versions (cl-loop for fun in embark-consult-color-functions-alist do (let* ((sym (caar fun)) (bind (cddr fun)) (desc (format "Insert %s" (cdar fun))) (newname (intern (format "%s-kill" (symbol-name sym))))) ;; (lambda (color) (kill-new (apply ',fun (list color)))) (fset newname `(lambda (color) (kill-new (,sym color)) (pushnew! consult-colors-history color))) ;; (define-key embark-consult-color-action-map (kbd bind) (cons desc newname)) (map! :map embark-consult-color-action-map :desc desc bind newname)))

(add-to-list 'embark-keymap-alist '(color . embark-consult-color-action-map))) #+end_src

*** Marginalized

Compatibility with Marginalized comes for free, since it already defines the category =color=, for which uses =marginalia-annotate-color=.

Clone this wiki locally