There are some subtle issues with using the unicode, which depends largely on the font you use. It is also slightly faster to use a space. If you use whitespace-mode's space-mark feature, then change this to ?\u2002. Addresses #109
464 lines
18 KiB
EmacsLisp
464 lines
18 KiB
EmacsLisp
;;; core-ui.el -*- lexical-binding: t; -*-
|
||
|
||
(defvar doom-fringe-size '4
|
||
"Default fringe width.")
|
||
|
||
(defvar doom-theme nil
|
||
"A symbol representing the color theme to load.")
|
||
|
||
(defvar doom-font nil
|
||
"The default font to use. Expects a FONT-SPEC (`font-spec').")
|
||
|
||
(defvar doom-big-font nil
|
||
"The default font to use. Expects a FONT-SPEC (`font-spec').")
|
||
|
||
(defvar doom-variable-pitch-font nil
|
||
"The default font to use for variable-pitch text. Expects a FONT-SPEC (`font-spec').")
|
||
|
||
(defvar doom-unicode-font nil
|
||
"Fallback font for unicode glyphs. Is ignored if :feature unicode is active.")
|
||
|
||
(defvar doom-major-mode-names
|
||
'((sh-mode . "sh")
|
||
(emacs-lisp-mode . "Elisp"))
|
||
"An alist mapping major modes symbols to strings (or functions that will
|
||
return a string). This changes the 'long' name of a major-mode, allowing for
|
||
shorter major mode name in the mode-line. See `doom|set-mode-name'.")
|
||
|
||
;; Line numbers
|
||
(defvar doom-line-number-lpad 4
|
||
"How much padding to place before line numbers.")
|
||
|
||
(defvar doom-line-number-rpad 1
|
||
"How much padding to place after line numbers.")
|
||
|
||
(defvar doom-line-number-pad-char 32
|
||
"Character to use for padding line numbers.
|
||
|
||
By default, this is a space key. If you use `whitespace-mode' with `space-mark',
|
||
the whitespace in line numbers will be affected (this can look ugly). In this
|
||
case, you can change this to ?\u2002, which is a unicode character that looks
|
||
like a space that `whitespace-mode' won't affect.")
|
||
|
||
|
||
;; Hook(s)
|
||
(defvar doom-init-ui-hook nil
|
||
"List of hooks to run when the theme and font is initialized (or reloaded with
|
||
`doom/reload-theme').")
|
||
|
||
|
||
;; Settings
|
||
(def-setting! :theme (theme)
|
||
`(unless doom-theme
|
||
(setq doom-theme ,theme)))
|
||
|
||
(def-setting! :font (family &rest spec)
|
||
`(unless doom-font
|
||
(setq doom-font (font-spec :family ,family ,@spec))))
|
||
|
||
(def-setting! :variable-pitch-font (family &rest spec)
|
||
`(unless doom-variable-pitch-font
|
||
(setq doom-variable-pitch-font (font-spec :family ,family ,@spec))))
|
||
|
||
(def-setting! :big-font (family &rest spec)
|
||
`(unless doom-big-font
|
||
(setq doom-big-font (font-spec :family ,family ,@spec))))
|
||
|
||
(def-setting! :unicode-font (family &rest spec)
|
||
`(unless doom-unicode-font
|
||
(setq doom-unicode-font (font-spec :family ,family ,@spec))))
|
||
|
||
|
||
(setq-default
|
||
bidi-display-reordering nil ; disable bidirectional text for tiny performance boost
|
||
blink-matching-paren nil ; don't blink--too distracting
|
||
cursor-in-non-selected-windows nil ; hide cursors in other windows
|
||
frame-inhibit-implied-resize t
|
||
;; remove continuation arrow on right fringe
|
||
fringe-indicator-alist (delq (assq 'continuation fringe-indicator-alist)
|
||
fringe-indicator-alist)
|
||
highlight-nonselected-windows nil
|
||
image-animate-loop t
|
||
indicate-buffer-boundaries nil
|
||
indicate-empty-lines nil
|
||
max-mini-window-height 0.3
|
||
mode-line-default-help-echo nil ; disable mode-line mouseovers
|
||
mouse-yank-at-point t ; middle-click paste at point, not at click
|
||
resize-mini-windows 'grow-only ; Minibuffer resizing
|
||
show-help-function nil ; hide :help-echo text
|
||
split-width-threshold 160 ; favor horizontal splits
|
||
uniquify-buffer-name-style 'forward
|
||
use-dialog-box nil ; always avoid GUI
|
||
visible-cursor nil
|
||
x-stretch-cursor nil
|
||
;; defer jit font locking slightly to [try to] improve Emacs performance
|
||
jit-lock-defer-time nil
|
||
jit-lock-stealth-nice 0.1
|
||
jit-lock-stealth-time 0.2
|
||
jit-lock-stealth-verbose nil
|
||
;; `pos-tip' defaults
|
||
pos-tip-internal-border-width 6
|
||
pos-tip-border-width 1
|
||
;; no beeping or blinking please
|
||
ring-bell-function #'ignore
|
||
visible-bell nil)
|
||
|
||
(fset #'yes-or-no-p #'y-or-n-p) ; y/n instead of yes/no
|
||
|
||
(defun doom-quit-p (&optional prompt)
|
||
"Return t if this session should be killed. Prompts the user for
|
||
confirmation."
|
||
(interactive)
|
||
(if (ignore-errors (doom-real-buffer-list))
|
||
(or (yes-or-no-p (format "››› %s" (or prompt "Quit Emacs?")))
|
||
(ignore (message "Aborted")))
|
||
t))
|
||
(setq confirm-kill-emacs nil)
|
||
(add-hook 'kill-emacs-query-functions #'doom-quit-p)
|
||
|
||
;; show typed keystrokes in minibuffer
|
||
(defun doom|enable-ui-keystrokes () (setq echo-keystrokes 0.02))
|
||
(defun doom|disable-ui-keystrokes () (setq echo-keystrokes 0))
|
||
(doom|enable-ui-keystrokes)
|
||
;; ...but hide them while isearch is active
|
||
(add-hook 'isearch-mode-hook #'doom|disable-ui-keystrokes)
|
||
(add-hook 'isearch-mode-end-hook #'doom|enable-ui-keystrokes)
|
||
|
||
;; A minor mode for toggling the mode-line
|
||
(defvar-local doom--modeline-format nil
|
||
"The modeline format to use when `doom-hide-modeline-mode' is active. Don't
|
||
set this directly. Let-bind it instead.")
|
||
(defvar-local doom--old-modeline-format nil
|
||
"The old modeline format, so `doom-hide-modeline-mode' can revert when it's
|
||
disabled.")
|
||
(define-minor-mode doom-hide-modeline-mode
|
||
"Minor mode to hide the mode-line in the current buffer."
|
||
:init-value nil
|
||
:global nil
|
||
(if doom-hide-modeline-mode
|
||
(setq doom--old-modeline-format mode-line-format
|
||
mode-line-format doom--modeline-format)
|
||
(setq mode-line-format doom--old-modeline-format
|
||
doom--old-modeline-format nil))
|
||
(force-mode-line-update))
|
||
;; Ensure major-mode or theme changes don't overwrite these variables
|
||
(put 'doom--modeline-format 'permanent-local t)
|
||
(put 'doom--old-modeline-format 'permanent-local t)
|
||
(put 'doom-hide-modeline-mode 'permanent-local t)
|
||
|
||
(defun doom|hide-modeline-mode-reset ()
|
||
"Sometimes, a major-mode is activated after `doom-hide-modeline-mode' is
|
||
activated, thus disabling it (because changing major modes invokes
|
||
`kill-all-local-variables' and specifically seems to kill `mode-line-format's
|
||
local value, whether or not it's permanent-local. Therefore, we cycle
|
||
`doom-hide-modeline-mode' to fix this."
|
||
(when doom-hide-modeline-mode
|
||
(doom-hide-modeline-mode -1)
|
||
(doom-hide-modeline-mode +1)))
|
||
(add-hook 'after-change-major-mode-hook #'doom|hide-modeline-mode-reset)
|
||
|
||
;; no modeline in completion popups
|
||
(add-hook 'completion-list-mode-hook #'doom-hide-modeline-mode)
|
||
|
||
;; undo/redo changes to Emacs' window layout
|
||
(defvar winner-dont-bind-my-keys t) ; I'll bind keys myself
|
||
(autoload 'winner-mode "winner" nil t)
|
||
(add-hook 'doom-init-hook #'winner-mode)
|
||
|
||
;; highlight matching delimiters
|
||
(setq show-paren-delay 0.1
|
||
show-paren-highlight-openparen t
|
||
show-paren-when-point-inside-paren t)
|
||
(add-hook 'doom-init-hook #'show-paren-mode)
|
||
|
||
;;; More reliable inter-window border
|
||
;; The native border "consumes" a pixel of the fringe on righter-most splits,
|
||
;; `window-divider' does not. Available since Emacs 25.1.
|
||
(setq-default window-divider-default-places t
|
||
window-divider-default-bottom-width 0
|
||
window-divider-default-right-width 1)
|
||
(add-hook 'doom-init-hook #'window-divider-mode)
|
||
|
||
;; like diminish, but for major-modes. [pedantry intensifies]
|
||
(defun doom|set-mode-name ()
|
||
"Set the major mode's `mode-name', as dictated by `doom-major-mode-names'."
|
||
(when-let (name (cdr (assq major-mode doom-major-mode-names)))
|
||
(setq mode-name
|
||
(cond ((functionp name)
|
||
(funcall name))
|
||
((stringp name)
|
||
name)
|
||
(t
|
||
(error "'%s' isn't a valid name for %s" name major-mode))))))
|
||
(add-hook 'after-change-major-mode-hook #'doom|set-mode-name)
|
||
|
||
|
||
;;
|
||
;; Themes & fonts
|
||
;;
|
||
|
||
;; Getting themes to remain consistent across GUI Emacs, terminal Emacs and
|
||
;; daemon Emacs is hairy.
|
||
;;
|
||
;; + Running `+doom|init-ui' directly sorts out the initial GUI frame.
|
||
;; + Attaching it to `after-make-frame-functions' sorts out daemon Emacs.
|
||
;; + Waiting for 0.1s in `doom|reload-ui-in-daemon' fixes daemon Emacs started
|
||
;; with `server-start' in an interactive session of Emacs AND in tty Emacs.
|
||
(defun doom|init-ui (&optional frame)
|
||
"Set the theme and load the font, in that order."
|
||
(when doom-theme
|
||
(load-theme doom-theme t))
|
||
(when (display-graphic-p)
|
||
(with-demoted-errors "FONT ERROR: %s"
|
||
(when (fontp doom-font)
|
||
(set-frame-font doom-font nil (if frame (list frame) t)))
|
||
;; Fallback to `doom-unicode-font' for Unicode characters
|
||
(when (fontp doom-unicode-font)
|
||
(set-fontset-font t 'unicode doom-unicode-font frame))
|
||
;; ...and for variable-pitch-mode:
|
||
(when (fontp doom-variable-pitch-font)
|
||
(set-face-attribute 'variable-pitch frame :font doom-variable-pitch-font))))
|
||
(run-hooks 'doom-init-ui-hook))
|
||
|
||
(defun doom|reload-ui-in-daemon (frame)
|
||
"Reload the theme (and font) in an daemon frame."
|
||
(when (or (daemonp) (not (display-graphic-p)))
|
||
(with-selected-frame frame
|
||
(run-with-timer 0.1 nil #'doom|init-ui))))
|
||
|
||
;; register UI init hooks
|
||
(add-hook 'doom-init-hook #'doom|init-ui)
|
||
(add-hook! 'after-make-frame-functions #'(doom|init-ui doom|reload-ui-in-daemon))
|
||
|
||
|
||
;;
|
||
;; Bootstrap
|
||
;;
|
||
|
||
;; prompts the user for confirmation when deleting a non-empty frame
|
||
(define-key global-map [remap delete-frame] #'doom/delete-frame)
|
||
;; buffer name in frame title
|
||
(setq-default frame-title-format '("DOOM Emacs"))
|
||
;; auto-enabled in Emacs 25+; I'll do it myself
|
||
(global-eldoc-mode -1)
|
||
;; a good indicator that Emacs isn't frozen
|
||
(add-hook 'doom-post-init-hook #'blink-cursor-mode)
|
||
;; standardize default fringe width
|
||
(if (fboundp 'fringe-mode) (fringe-mode doom-fringe-size))
|
||
;; draw me like one of your French editors
|
||
(tooltip-mode -1) ; relegate tooltips to echo area only
|
||
(menu-bar-mode -1)
|
||
(if (fboundp 'tool-bar-mode) (tool-bar-mode -1))
|
||
(if (fboundp 'scroll-bar-mode) (scroll-bar-mode -1))
|
||
|
||
(defun doom|no-fringes-in-minibuffer ()
|
||
"Disable fringes in the minibuffer window."
|
||
(set-window-fringes (minibuffer-window) 0 0 nil))
|
||
(add-hook! '(doom-post-init-hook minibuffer-setup-hook)
|
||
#'doom|no-fringes-in-minibuffer)
|
||
|
||
|
||
;;
|
||
;; Plugins
|
||
;;
|
||
|
||
(def-package! all-the-icons
|
||
:commands (all-the-icons-octicon all-the-icons-faicon all-the-icons-fileicon
|
||
all-the-icons-wicon all-the-icons-material all-the-icons-alltheicon
|
||
all-the-icons-install-fonts)
|
||
:init
|
||
(defun doom*disable-all-the-icons-in-tty (orig-fn &rest args)
|
||
(when (display-graphic-p)
|
||
(apply orig-fn args)))
|
||
|
||
;; all-the-icons doesn't work in the terminal, so we "disable" it.
|
||
(advice-add #'all-the-icons-octicon :around #'doom*disable-all-the-icons-in-tty)
|
||
(advice-add #'all-the-icons-material :around #'doom*disable-all-the-icons-in-tty)
|
||
(advice-add #'all-the-icons-faicon :around #'doom*disable-all-the-icons-in-tty)
|
||
(advice-add #'all-the-icons-fileicon :around #'doom*disable-all-the-icons-in-tty)
|
||
(advice-add #'all-the-icons-wicon :around #'doom*disable-all-the-icons-in-tty)
|
||
(advice-add #'all-the-icons-alltheicon :around #'doom*disable-all-the-icons-in-tty))
|
||
|
||
(def-package! fringe-helper
|
||
:commands (fringe-helper-define fringe-helper-convert)
|
||
:init
|
||
(unless (fboundp 'define-fringe-bitmap)
|
||
(defun define-fringe-bitmap (&rest _))))
|
||
|
||
(def-package! hideshow ; built-in
|
||
:commands (hs-minor-mode hs-toggle-hiding hs-already-hidden-p)
|
||
:config
|
||
(setq hs-hide-comments-when-hiding-all nil))
|
||
|
||
;; Show uninterrupted indentation markers with some whitespace voodoo.
|
||
(def-package! highlight-indentation
|
||
:commands (highlight-indentation-mode highlight-indentation-current-column-mode))
|
||
|
||
;; For modes that don't adequately highlight numbers
|
||
(def-package! highlight-numbers :commands highlight-numbers-mode)
|
||
|
||
;; Line highlighting
|
||
(def-package! hl-line ; built-in
|
||
:init (add-hook! (linum-mode nlinum-mode) #'hl-line-mode)
|
||
:config
|
||
;; I don't need hl-line showing in other windows. This also offers a small
|
||
;; speed boost when buffer is displayed in multiple windows.
|
||
(setq hl-line-sticky-flag nil
|
||
global-hl-line-sticky-flag nil)
|
||
|
||
;; Fix lingering hl-line overlays
|
||
(add-hook! 'hl-line-mode-hook
|
||
(remove-overlays (point-min) (point-max) 'face 'hl-line))
|
||
|
||
(after! evil
|
||
;; Disable `hl-line' in evil-visual mode (temporarily). `hl-line' can make
|
||
;; the selection region harder to see while in evil visual mode.
|
||
(defun doom|turn-off-hl-line () (hl-line-mode -1))
|
||
|
||
(add-hook 'evil-visual-state-entry-hook #'doom|turn-off-hl-line)
|
||
(add-hook 'evil-visual-state-exit-hook #'hl-line-mode)))
|
||
|
||
;; Line number column. A faster (or equivalent, in the worst case) line number
|
||
;; plugin than the built-in `linum'.
|
||
(def-package! nlinum
|
||
:commands nlinum-mode
|
||
:init
|
||
(defun doom|init-nlinum-mode ()
|
||
"Turn on `nlinum-mode', except in org-mode"
|
||
(unless (eq major-mode 'org-mode)
|
||
(nlinum-mode +1)))
|
||
(add-hook! (prog-mode text-mode) #'doom|init-nlinum-mode)
|
||
:config
|
||
(setq nlinum-highlight-current-line t)
|
||
|
||
(defun doom-nlinum-format-fn (line _width)
|
||
"A more customizable `nlinum-format-function'. See `doom-line-number-lpad',
|
||
`doom-line-number-rpad' and `doom-line-number-pad-char'. Allows a fix for
|
||
`whitespace-mode' space-marks appearing inside the line number."
|
||
(let ((str (number-to-string line)))
|
||
(setq str (concat (make-string (max 0 (- doom-line-number-lpad (length str)))
|
||
doom-line-number-pad-char)
|
||
str
|
||
(make-string doom-line-number-rpad doom-line-number-pad-char)))
|
||
(put-text-property 0 (length str) 'face
|
||
(if (and nlinum-highlight-current-line
|
||
(= line nlinum--current-line))
|
||
'nlinum-current-line
|
||
'linum)
|
||
str)
|
||
str))
|
||
(setq nlinum-format-function #'doom-nlinum-format-fn)
|
||
|
||
(defun doom|init-nlinum-width ()
|
||
"Calculate line number column width beforehand (optimization)."
|
||
(setq nlinum--width
|
||
(length (save-excursion (goto-char (point-max))
|
||
(format-mode-line "%l")))))
|
||
(add-hook 'nlinum-mode-hook #'doom|init-nlinum-width))
|
||
|
||
;; Fixes disappearing line numbers in nlinum and other quirks
|
||
(def-package! nlinum-hl
|
||
:after nlinum
|
||
:config
|
||
;; With `markdown-fontify-code-blocks-natively' enabled in `markdown-mode',
|
||
;; line numbers tend to vanish next to code blocks.
|
||
(advice-add #'markdown-fontify-code-block-natively
|
||
:after #'nlinum-hl-do-markdown-fontify-region)
|
||
|
||
;; When using `web-mode's code-folding an entire range of line numbers will
|
||
;; vanish in the affected area.
|
||
(advice-add #'web-mode-fold-or-unfold :after #'nlinum-hl-do-generic-flush)
|
||
|
||
;; Changing fonts can leave nlinum line numbers in their original size; this
|
||
;; forces them to resize.
|
||
(advice-add #'set-frame-font :after #'nlinum-hl-flush-all-windows))
|
||
|
||
;; Helps us distinguish stacked delimiter pairs. Especially in parentheses-drunk
|
||
;; languages like Lisp.
|
||
(def-package! rainbow-delimiters
|
||
:commands rainbow-delimiters-mode
|
||
:config (setq rainbow-delimiters-max-face-count 3)
|
||
:init (add-hook 'lisp-mode-hook #'rainbow-delimiters-mode))
|
||
|
||
;; indicators for empty lines past EOF
|
||
(def-package! vi-tilde-fringe
|
||
:commands global-vi-tilde-fringe-mode
|
||
:init (add-hook 'doom-init-hook #'global-vi-tilde-fringe-mode))
|
||
|
||
;; For a distractions-free-like UI, that dynamically resizes margets and can
|
||
;; center a buffer.
|
||
(def-package! visual-fill-column
|
||
:commands visual-fill-column-mode
|
||
:config
|
||
(setq-default visual-fill-column-center-text nil
|
||
visual-fill-column-width fill-column))
|
||
|
||
|
||
;;
|
||
;; Modeline
|
||
;;
|
||
|
||
(defmacro def-modeline-segment! (name &rest forms)
|
||
"Defines a modeline segment and byte compiles it."
|
||
(declare (indent defun) (doc-string 2))
|
||
(let ((sym (intern (format "doom-modeline-segment--%s" name))))
|
||
`(progn
|
||
(defun ,sym () ,@forms)
|
||
,(unless (bound-and-true-p byte-compile-current-file)
|
||
`(let (byte-compile-warnings)
|
||
(byte-compile #',sym))))))
|
||
|
||
(defsubst doom--prepare-modeline-segments (segments)
|
||
(cl-loop for seg in segments
|
||
if (stringp seg)
|
||
collect seg
|
||
else
|
||
collect (list (intern (format "doom-modeline-segment--%s" (symbol-name seg))))))
|
||
|
||
(defmacro def-modeline! (name lhs &optional rhs)
|
||
"Defines a modeline format and byte-compiles it. NAME is a symbol to identify
|
||
it (used by `doom-modeline' for retrieval). LHS and RHS are lists of symbols of
|
||
modeline segments defined with `def-modeline-segment!'.
|
||
|
||
Example:
|
||
(def-modeline! minimal
|
||
(bar matches \" \" buffer-info)
|
||
(media-info major-mode))
|
||
(doom-set-modeline 'minimal t)"
|
||
(let ((sym (intern (format "doom-modeline-format--%s" name)))
|
||
(lhs-forms (doom--prepare-modeline-segments lhs))
|
||
(rhs-forms (doom--prepare-modeline-segments rhs)))
|
||
`(progn
|
||
(defun ,sym ()
|
||
(let ((lhs (list ,@lhs-forms))
|
||
(rhs (list ,@rhs-forms)))
|
||
(let ((rhs-str (format-mode-line rhs)))
|
||
(list lhs
|
||
(propertize
|
||
" " 'display
|
||
`((space :align-to (- (+ right right-fringe right-margin)
|
||
,(+ 1 (string-width rhs-str))))))
|
||
rhs-str))))
|
||
,(unless (bound-and-true-p byte-compile-current-file)
|
||
`(let (byte-compile-warnings)
|
||
(byte-compile #',sym))))))
|
||
|
||
(defun doom-modeline (key)
|
||
"Returns a mode-line configuration associated with KEY (a symbol). Throws an
|
||
error if it doesn't exist."
|
||
(let ((fn (intern (format "doom-modeline-format--%s" key))))
|
||
(when (functionp fn)
|
||
`(:eval (,fn)))))
|
||
|
||
(defun doom-set-modeline (key &optional default)
|
||
"Set the modeline format. Does nothing if the modeline KEY doesn't exist. If
|
||
DEFAULT is non-nil, set the default mode-line for all buffers."
|
||
(let ((modeline (doom-modeline key)))
|
||
(when modeline
|
||
(setf (if default
|
||
(default-value 'mode-line-format)
|
||
(buffer-local-value 'mode-line-format (current-buffer)))
|
||
modeline))))
|
||
|
||
(provide 'core-ui)
|
||
;;; core-ui.el ends here
|