2023-03-05 12:03:36 -05:00
|
|
|
;;; completion/corfu/config.el -*- lexical-binding: t; -*-
|
|
|
|
|
2023-10-19 11:23:45 -04:00
|
|
|
(defvar +corfu-completion-styles '(basic partial-completion flex)
|
|
|
|
"Completion styles for corfu to use.
|
|
|
|
|
|
|
|
If the user enables +orderless, `orderless' is automatically appended to this
|
|
|
|
list before fowarding to `completion-styles'.")
|
|
|
|
|
|
|
|
(defvar +corfu-icon-mapping
|
|
|
|
`((array ,(nerd-icons-codicon "nf-cod-symbol_array") :face font-lock-type-face)
|
|
|
|
(boolean ,(nerd-icons-codicon "nf-cod-symbol_boolean") :face font-lock-builtin-face)
|
|
|
|
(class ,(nerd-icons-codicon "nf-cod-symbol_class") :face font-lock-type-face)
|
|
|
|
(color ,(nerd-icons-codicon "nf-cod-symbol_color") :face success)
|
|
|
|
(command ,(nerd-icons-codicon "nf-cod-terminal") :face default)
|
|
|
|
(constant ,(nerd-icons-codicon "nf-cod-symbol_constant") :face font-lock-constant-face)
|
|
|
|
(constructor ,(nerd-icons-codicon "nf-cod-triangle_right") :face font-lock-function-name-face)
|
|
|
|
(enummember ,(nerd-icons-codicon "nf-cod-symbol_enum_member") :face font-lock-builtin-face)
|
|
|
|
(enum-member ,(nerd-icons-codicon "nf-cod-symbol_enum_member") :face font-lock-builtin-face)
|
|
|
|
(enum ,(nerd-icons-codicon "nf-cod-symbol_enum") :face font-lock-builtin-face)
|
|
|
|
(event ,(nerd-icons-codicon "nf-cod-symbol_event") :face font-lock-warning-face)
|
|
|
|
(field ,(nerd-icons-codicon "nf-cod-symbol_field") :face font-lock-variable-name-face)
|
|
|
|
(file ,(nerd-icons-codicon "nf-cod-symbol_file") :face font-lock-string-face)
|
|
|
|
(folder ,(nerd-icons-codicon "nf-cod-folder") :face font-lock-doc-face)
|
|
|
|
(interface ,(nerd-icons-codicon "nf-cod-symbol_interface") :face font-lock-type-face)
|
|
|
|
(keyword ,(nerd-icons-codicon "nf-cod-symbol_keyword") :face font-lock-keyword-face)
|
|
|
|
(macro ,(nerd-icons-codicon "nf-cod-symbol_misc") :face font-lock-keyword-face)
|
|
|
|
(magic ,(nerd-icons-codicon "nf-cod-wand") :face font-lock-builtin-face)
|
|
|
|
(method ,(nerd-icons-codicon "nf-cod-symbol_method") :face font-lock-function-name-face)
|
|
|
|
(function ,(nerd-icons-codicon "nf-cod-symbol_method") :face font-lock-function-name-face)
|
|
|
|
(module ,(nerd-icons-codicon "nf-cod-file_submodule") :face font-lock-preprocessor-face)
|
|
|
|
(numeric ,(nerd-icons-codicon "nf-cod-symbol_numeric") :face font-lock-builtin-face)
|
|
|
|
(operator ,(nerd-icons-codicon "nf-cod-symbol_operator") :face font-lock-comment-delimiter-face)
|
|
|
|
(param ,(nerd-icons-codicon "nf-cod-symbol_parameter") :face default)
|
|
|
|
(property ,(nerd-icons-codicon "nf-cod-symbol_property") :face font-lock-variable-name-face)
|
|
|
|
(reference ,(nerd-icons-codicon "nf-cod-references") :face font-lock-variable-name-face)
|
|
|
|
(snippet ,(nerd-icons-codicon "nf-cod-symbol_snippet") :face font-lock-string-face)
|
|
|
|
(string ,(nerd-icons-codicon "nf-cod-symbol_string") :face font-lock-string-face)
|
|
|
|
(struct ,(nerd-icons-codicon "nf-cod-symbol_structure") :face font-lock-variable-name-face)
|
|
|
|
(text ,(nerd-icons-codicon "nf-cod-text_size") :face font-lock-doc-face)
|
|
|
|
(typeparameter ,(nerd-icons-codicon "nf-cod-list_unordered") :face font-lock-type-face)
|
|
|
|
(type-parameter ,(nerd-icons-codicon "nf-cod-list_unordered") :face font-lock-type-face)
|
|
|
|
(unit ,(nerd-icons-codicon "nf-cod-symbol_ruler") :face font-lock-constant-face)
|
|
|
|
(value ,(nerd-icons-codicon "nf-cod-symbol_field") :face font-lock-builtin-face)
|
|
|
|
(variable ,(nerd-icons-codicon "nf-cod-symbol_variable") :face font-lock-variable-name-face)
|
|
|
|
(t ,(nerd-icons-codicon "nf-cod-code") :face font-lock-warning-face))
|
|
|
|
"Mapping of completion kinds to icons.
|
|
|
|
|
|
|
|
It should be a list of elements with the form (KIND ICON-TXT [:face FACE]).
|
|
|
|
KIND is a symbol determining what the completion is, and comes from calling the
|
|
|
|
`:company-kind' property of the completion. ICON-TXT is a string with the icon
|
|
|
|
to use, usually as a character from the `nerd-icons' symbol font. See that
|
|
|
|
package for how to get these. Note that it can be simple text if that is
|
|
|
|
preferred. FACE, if present, is applied to the icon, mainly for its color. The
|
|
|
|
special `t' symbol should be used for KIND to represent the default icon, and
|
|
|
|
must be present.")
|
|
|
|
|
2023-03-29 17:03:22 -04:00
|
|
|
;;
|
|
|
|
;;; Packages
|
2023-03-05 12:03:36 -05:00
|
|
|
(use-package! corfu
|
2023-03-29 17:03:22 -04:00
|
|
|
:hook (doom-first-buffer . global-corfu-mode)
|
2023-10-19 11:23:45 -04:00
|
|
|
:hook (org-mode . corfu-mode)
|
2023-03-05 12:03:36 -05:00
|
|
|
:init
|
2023-03-29 17:03:22 -04:00
|
|
|
;; Auto-completion settings, must be set before calling `global-corfu-mode'.
|
2023-09-06 10:06:14 -04:00
|
|
|
;; Due to lazy-loading, setting them in config.el works too.
|
2023-03-29 17:03:22 -04:00
|
|
|
(setq corfu-auto t
|
2023-09-06 10:06:14 -04:00
|
|
|
corfu-auto-delay 0.1
|
|
|
|
corfu-auto-prefix 2
|
2023-03-29 17:03:22 -04:00
|
|
|
corfu-excluded-modes '(erc-mode
|
|
|
|
circe-mode
|
|
|
|
help-mode
|
|
|
|
gud-mode
|
|
|
|
vterm-mode))
|
2023-03-05 12:03:36 -05:00
|
|
|
:config
|
2023-03-29 17:03:22 -04:00
|
|
|
(setq corfu-cycle t
|
2023-09-06 10:06:14 -04:00
|
|
|
corfu-separator (when (modulep! +orderless) ?\s)
|
2023-03-29 17:03:22 -04:00
|
|
|
corfu-preselect t
|
|
|
|
corfu-count 16
|
|
|
|
corfu-max-width 120
|
|
|
|
corfu-preview-current 'insert
|
2023-09-06 10:06:14 -04:00
|
|
|
corfu-on-exact-match nil
|
|
|
|
corfu-quit-at-boundary (if (modulep! +orderless) 'separator t)
|
|
|
|
corfu-quit-no-match (if (modulep! +orderless) 'separator t)
|
2023-03-29 17:03:22 -04:00
|
|
|
;; In the case of +tng, TAB should be smart regarding completion;
|
|
|
|
;; However, it should otherwise behave like normal, whatever normal was.
|
|
|
|
tab-always-indent (if (modulep! +tng) 'complete tab-always-indent))
|
2023-10-19 11:23:45 -04:00
|
|
|
|
2023-09-06 10:06:14 -04:00
|
|
|
(when (modulep! +orderless)
|
2023-10-19 11:23:45 -04:00
|
|
|
(after! 'lsp-mode
|
|
|
|
(add-to-list 'completion-category-overrides
|
|
|
|
`(lsp-capf (styles ,@+corfu-completion-styles ,(when (modulep! +orderless) 'orderless)))))
|
|
|
|
(after! 'eglot
|
|
|
|
(add-to-list 'completion-category-overrides
|
|
|
|
`(eglot (styles ,@+corfu-completion-styles ,(when (modulep! +orderless) 'orderless))))))
|
|
|
|
|
|
|
|
;; For the icons, we use a custom margin formatter, which simply reads the
|
|
|
|
;; mapping in `+corfu-icon-mapping'.
|
|
|
|
(when (modulep! +icons)
|
|
|
|
(defun icon-margin-formatter (metadata)
|
|
|
|
(when-let ((kindfunc (or (plist-get completion-extra-properties :company-kind)
|
|
|
|
(assq 'company-kind metadata))))
|
|
|
|
(lambda (cand)
|
|
|
|
(let* ((kind (funcall kindfunc cand))
|
|
|
|
(icon-entry (assq (or kind t) +corfu-icon-mapping))
|
|
|
|
(str (cadr icon-entry))
|
|
|
|
(props (cddr icon-entry))
|
|
|
|
(extra-face (plist-get props :face))
|
|
|
|
(space (propertize " " 'display '(space :width 1)))
|
|
|
|
(str (concat " " str space)))
|
|
|
|
(when extra-face
|
|
|
|
(put-text-property 0 3 'face extra-face str))
|
|
|
|
str))))
|
|
|
|
(setq corfu-margin-formatters '(icon-margin-formatter)))
|
|
|
|
|
|
|
|
;; This is to decouple the use of `completion-styles' in corfu from other
|
|
|
|
;; completion packages, such as vertico. That way, the user can leave the
|
|
|
|
;; global value of the variable alone, say, to be used by the default
|
|
|
|
;; front-end or consult. The vertico module also does something similar with
|
|
|
|
;; `+vertico-company-completion-styles'.
|
|
|
|
(defadvice! +corfu--completion-styles (orig &rest args)
|
|
|
|
"Try default completion styles before orderless.
|
|
|
|
|
|
|
|
Meant as :around advice for `corfu--recompute'."
|
|
|
|
:around #'corfu--recompute
|
|
|
|
(let ((completion-styles
|
|
|
|
(append +corfu-completion-styles (when (modulep! +orderless)
|
|
|
|
'(orderless))))
|
|
|
|
completion-category-overrides completion-category-defaults)
|
|
|
|
(apply orig args)))
|
|
|
|
|
2023-03-29 17:03:22 -04:00
|
|
|
(map! (:unless (modulep! +tng)
|
2023-09-06 10:06:14 -04:00
|
|
|
"C-SPC" #'completion-at-point)
|
2023-03-29 17:03:22 -04:00
|
|
|
(:map 'corfu-map
|
2023-09-06 10:06:14 -04:00
|
|
|
(:when (modulep! +orderless)
|
|
|
|
"C-SPC" #'corfu-insert-separator)
|
2023-03-29 17:03:22 -04:00
|
|
|
(:when (modulep! +tng)
|
2023-09-06 10:06:14 -04:00
|
|
|
[tab] #'corfu-next
|
|
|
|
[backtab] #'corfu-previous
|
|
|
|
"TAB" #'corfu-next
|
|
|
|
"S-TAB" #'corfu-previous)))
|
2023-10-19 11:23:45 -04:00
|
|
|
(after! evil-collection-corfu
|
2023-09-06 10:06:14 -04:00
|
|
|
(evil-collection-define-key 'insert 'corfu-map
|
|
|
|
(kbd "RET") #'corfu-insert
|
|
|
|
[return] #'corfu-insert))
|
|
|
|
|
|
|
|
(after! vertico
|
|
|
|
;; Taken from corfu's README.
|
|
|
|
;; TODO: extend this to other completion front-ends.
|
|
|
|
(defun corfu-move-to-minibuffer ()
|
|
|
|
(interactive)
|
|
|
|
(let ((completion-extra-properties corfu--extra)
|
|
|
|
(completion-cycle-threshold completion-cycling))
|
|
|
|
(apply #'consult-completion-in-region completion-in-region--data)))
|
|
|
|
(map! :map 'corfu-map "s-<down>" #'corfu-move-to-minibuffer
|
|
|
|
(:when (modulep! :editor evil) "s-j" #'corfu-move-to-minibuffer))))
|
2023-03-05 12:03:36 -05:00
|
|
|
|
|
|
|
(use-package! cape
|
2023-03-29 17:03:22 -04:00
|
|
|
:after corfu
|
|
|
|
:config
|
2023-10-19 11:23:45 -04:00
|
|
|
(add-hook! prog-mode (add-to-list 'completion-at-point-functions #'cape-file))
|
|
|
|
(add-hook! (org-mode markdown-mode) (add-to-list 'completion-at-point-functions #'cape-elisp-block))
|
|
|
|
(advice-add #'lsp-completion-at-point :around #'cape-wrap-noninterruptible))
|
2023-09-06 10:06:14 -04:00
|
|
|
|
|
|
|
(use-package! yasnippet-capf
|
|
|
|
:after corfu
|
|
|
|
:config
|
|
|
|
(add-hook 'yas-minor-mode-hook
|
|
|
|
(lambda () (add-to-list 'completion-at-point-functions #'yasnippet-capf))))
|
2023-03-05 12:03:36 -05:00
|
|
|
|
2023-03-29 17:03:22 -04:00
|
|
|
(use-package! corfu-terminal
|
2023-09-06 10:06:14 -04:00
|
|
|
:when (not (display-graphic-p))
|
2023-03-29 17:03:22 -04:00
|
|
|
:hook (corfu-mode . corfu-terminal-mode))
|
2023-03-05 12:03:36 -05:00
|
|
|
|
2023-03-29 17:03:22 -04:00
|
|
|
;;
|
|
|
|
;;; Extensions
|
|
|
|
(use-package! corfu-history
|
|
|
|
:after savehist
|
|
|
|
:hook (corfu-mode . corfu-history-mode)
|
|
|
|
:config
|
|
|
|
(add-to-list 'savehist-additional-variables 'corfu-history))
|
2023-03-05 12:03:36 -05:00
|
|
|
(use-package! corfu-popupinfo
|
2023-03-29 17:03:22 -04:00
|
|
|
:hook (corfu-mode . corfu-popupinfo-mode)
|
|
|
|
:config
|
|
|
|
(setq corfu-popupinfo-delay '(0.5 . 1.0))
|
|
|
|
(map! (:map 'corfu-map
|
2023-09-06 10:06:14 -04:00
|
|
|
"C-<up>" #'corfu-popupinfo-scroll-down
|
|
|
|
"C-<down>" #'corfu-popupinfo-scroll-up
|
|
|
|
"C-S-p" #'corfu-popupinfo-scroll-down
|
|
|
|
"C-S-n" #'corfu-popupinfo-scroll-up
|
|
|
|
"C-h" #'corfu-popupinfo-toggle)
|
2023-03-29 17:03:22 -04:00
|
|
|
(:map 'corfu-popupinfo-map
|
|
|
|
:when (modulep! :editor evil)
|
|
|
|
;; Reversed because popupinfo assumes opposite of what feels intuitive
|
|
|
|
;; with evil.
|
2023-09-06 10:06:14 -04:00
|
|
|
"C-S-k" #'corfu-popupinfo-scroll-down
|
|
|
|
"C-S-j" #'corfu-popupinfo-scroll-up)))
|