doomemacs/core/core-ui.el

453 lines
17 KiB
EmacsLisp
Raw Normal View History

2016-05-07 21:57:43 -04:00
;;; core-ui.el --- interface & mode-line config
2015-06-15 09:05:52 +02:00
2016-05-24 22:15:44 -04:00
(defconst doom-fringe-size '3 "Default fringe width")
2016-05-23 17:13:59 -04:00
2016-05-20 09:23:46 -04:00
;; y/n instead of yes/no
(fset 'yes-or-no-p 'y-or-n-p)
2015-06-04 18:23:21 -04:00
(setq-default
2016-05-23 17:13:59 -04:00
indicate-buffer-boundaries nil ; don't show where buffer starts/ends
2016-05-24 22:15:44 -04:00
indicate-empty-lines nil ; don't show empty lines
2016-05-23 17:13:59 -04:00
fringes-outside-margins t ; switches order of fringe and margin
2016-05-07 21:57:43 -04:00
;; Keep cursors and highlights in current window only
cursor-in-non-selected-windows nil
2016-04-08 16:15:37 -04:00
highlight-nonselected-windows nil
2016-05-23 17:13:59 -04:00
;; Disable bidirectional text support for slight performance bonus
bidi-display-reordering nil
;; Remove continuation arrow on right fringe
fringe-indicator-alist (delq (assq 'continuation fringe-indicator-alist)
fringe-indicator-alist)
2016-04-08 16:15:37 -04:00
2016-05-23 17:13:59 -04:00
blink-matching-paren nil ; don't blink--too distracting
show-paren-delay 0.075
uniquify-buffer-name-style nil
2016-04-08 16:15:37 -04:00
visible-bell nil
visible-cursor nil
x-stretch-cursor t
use-dialog-box nil ; always avoid GUI
redisplay-dont-pause t ; don't pause display on input
split-width-threshold nil ; favor horizontal splits
show-help-function nil ; hide :help-echo text
2016-04-26 02:00:19 -04:00
jit-lock-defer-time nil
2016-04-08 16:15:37 -04:00
jit-lock-stealth-nice 0.1
jit-lock-stealth-time 0.2
jit-lock-stealth-verbose nil
NARF v0.7.0 vcs: + +git-gutter to conf-modes; -git-gutter from evil-insert-state-exit + switch github-browse-file for browse-at-remote + fix <leader>ob; add <leader>d[./sr] vc bindings + vc-annotate bindings and initial state Workgroups2 integration: + don't mess with buffers (speeds up emacs a lot!) + unicode numbers in display + single display function + remember workgroup uid instead (and smarter :tabrename) + clean up after wg update Org-mode + give highlight precedence to links in org-mode + enable encryption + config clean up + use different font for org + exclude attachments in recentf + redo latex and inline-image config + add narf/org-open-notes + update file templates for org CRM Mode-line + polish mode-line + decouple from spaceline-segments.el + refactor narf|spaceline-env-update + add macro-recording and buffer-size indicators to mode-line + python: '2>&1' in env-command + flycheck fringe indicator: change to arrow Aesthetics + update narf-dark-theme + add narf-minibuffer-active face + change writing indicator in writing-mode Misc + fix whitespace in display-startup-echo-area-message + reset fonts for more unicode characters + custom imenu entries + helm-imenu fontification + enable yascroll-bar in REPLs + reorganize my-commands.el + force quit iedit on ESC in normal mode + update snippets submodule + remove ido init (helm handles it all) [EXPERIMENTAL] + back to Terminus(TTF) font + popwin: update config for git-gutter and vc-diff windows + highlight :g[lobal] and :al[ign] matches + decouple narf/get-buffers+narf/get-all-buffers from wg-mess-with-buffer-list + fix narf/helm-buffers-dwim (add interactive form)
2015-12-11 16:51:04 -05:00
;; Minibuffer resizing
2016-05-01 01:05:25 -04:00
resize-mini-windows 'grow-only
max-mini-window-height 0.3
2016-05-23 17:13:59 -04:00
;; Ask for confirmation on exit only if there are real buffers left
confirm-kill-emacs
(lambda (_) (if (doom/get-real-buffers) (y-or-n-p " Quit?") t) t))
2016-05-23 17:13:59 -04:00
;; Initialize UI
2016-05-20 22:37:30 -04:00
(load-theme doom-current-theme t)
2016-05-20 09:23:46 -04:00
(tooltip-mode -1) ; show tooltips in echo area
(if (not window-system)
2016-03-03 15:04:14 -05:00
(menu-bar-mode -1)
2016-03-26 01:19:31 -04:00
(scroll-bar-mode -1) ; no scrollbar
(tool-bar-mode -1) ; no toolbar
2016-03-29 23:48:06 -04:00
;; full filename in frame title
(setq frame-title-format '(buffer-file-name "%f" ("%b")))
;; set fonts
2016-08-21 16:48:09 +02:00
(ignore-errors
(set-frame-font doom-default-font t)
(set-face-attribute 'default t :font doom-current-font))
2016-05-21 00:13:50 -04:00
;; standardize fringe width
2016-05-24 22:15:44 -04:00
(fringe-mode doom-fringe-size)
2016-05-21 00:13:50 -04:00
(push `(left-fringe . ,doom-fringe-size) default-frame-alist)
(push `(right-fringe . ,doom-fringe-size) default-frame-alist)
;; Default frame size on startup
(push '(width . 120) default-frame-alist)
2016-05-30 21:19:10 -04:00
(push '(height . 40) default-frame-alist)
2016-05-20 17:19:59 -04:00
;; no fringe in the minibuffer
2016-05-23 17:13:59 -04:00
(add-hook! (emacs-startup minibuffer-setup)
(set-window-fringes (minibuffer-window) 0 0 nil))
2016-03-29 23:48:06 -04:00
;; Show tilde in margin on empty lines
2016-03-03 15:04:14 -05:00
(define-fringe-bitmap 'tilde [64 168 16] nil nil 'center)
2016-05-20 17:19:59 -04:00
(set-fringe-bitmap-face 'tilde 'fringe)
2016-05-23 20:56:19 -04:00
(setcdr (assq 'empty-line fringe-indicator-alist) 'tilde)
2016-06-05 23:03:47 -04:00
;; Fallback to `doom-unicode-font' for Unicode characters
(set-fontset-font t 'unicode doom-unicode-font))
2015-11-14 02:35:15 -05:00
2016-05-23 20:56:19 -04:00
;; Hide mode-line in help/compile window
(add-hook 'help-mode-hook 'doom-hide-mode-line-mode)
(add-hook 'compilation-mode-hook 'doom-hide-mode-line-mode)
2016-03-03 15:04:14 -05:00
2016-05-23 17:13:59 -04:00
;; On by default in Emacs 25. I'll enable it manually, so disable it globally
(when (and (> emacs-major-version 24) (featurep 'eldoc))
2016-01-30 21:16:10 -05:00
(global-eldoc-mode -1))
2015-11-10 18:01:57 -05:00
2016-05-26 05:23:25 -04:00
(use-package eldoc-eval
:config
(setq eldoc-in-minibuffer-show-fn 'doom/eldoc-show-in-mode-line)
(eldoc-in-minibuffer-mode +1))
2016-03-03 15:04:14 -05:00
;; Highlight TODO/FIXME/NOTE tags
(add-hook! (prog-mode emacs-lisp-mode css-mode)
2016-04-26 02:00:19 -04:00
(font-lock-add-keywords
2016-05-23 17:14:37 -04:00
nil '(("\\<\\(TODO\\(?:(.*)\\)?:?\\)\\>" 1 'warning prepend)
("\\<\\(FIXME\\(?:(.*)\\)?:?\\)\\>" 1 'error prepend)
("\\<\\(NOTE\\(?:(.*)\\)?:?\\)\\>" 1 'success prepend))))
2015-06-04 18:23:21 -04:00
2016-03-03 15:04:14 -05:00
;;
;; Plugins
;;
2016-08-28 22:08:48 +02:00
(use-package beacon
:config
(beacon-mode +1)
(setq beacon-color (face-attribute 'highlight :background nil t)
beacon-blink-when-buffer-changes t
beacon-blink-when-point-moves-vertically 10))
2016-05-01 01:05:25 -04:00
(use-package hl-line
2016-08-28 22:54:48 +02:00
:init (add-hook 'prog-mode-hook 'hl-line-mode)
2016-05-01 01:05:25 -04:00
:config
2016-06-05 00:56:28 -04:00
;; Doesn't seem to play nice in emacs 25+
2016-06-08 14:43:40 -04:00
(setq hl-line-sticky-flag nil
global-hl-line-sticky-flag nil)
2016-05-01 01:05:25 -04:00
2016-06-05 00:56:28 -04:00
(defvar-local doom--hl-line-mode nil)
2016-05-20 22:37:30 -04:00
(defun doom|hl-line-on () (if doom--hl-line-mode (hl-line-mode +1)))
(defun doom|hl-line-off () (if doom--hl-line-mode (hl-line-mode -1)))
(add-hook! hl-line-mode (if hl-line-mode (setq doom--hl-line-mode t)))
2016-05-01 01:05:25 -04:00
;; Disable line highlight in visual mode
2016-05-20 22:37:30 -04:00
(add-hook 'evil-visual-state-entry-hook 'doom|hl-line-off)
(add-hook 'evil-visual-state-exit-hook 'doom|hl-line-on))
2016-05-01 01:05:25 -04:00
2016-04-12 02:59:36 -04:00
(use-package highlight-indentation
:commands (highlight-indentation-mode
highlight-indentation-current-column-mode)
2016-04-12 02:59:36 -04:00
:init
(after! editorconfig
(advice-add 'highlight-indentation-guess-offset
2016-05-20 22:37:30 -04:00
:override 'doom*hl-indent-guess-offset))
2016-04-12 02:59:36 -04:00
;; A long-winded method for ensuring whitespace is maintained (so that
;; highlight-indentation-mode can display them consistently)
(add-hook! highlight-indentation-mode
(if highlight-indentation-mode
(progn
2016-05-20 22:37:30 -04:00
(doom/add-whitespace)
(add-hook 'after-save-hook 'doom/add-whitespace nil t))
2016-05-20 22:37:30 -04:00
(remove-hook 'after-save-hook 'doom/add-whitespace t)
(delete-trailing-whitespace))))
2016-04-12 02:59:36 -04:00
(use-package highlight-numbers :commands (highlight-numbers-mode))
2015-10-26 01:29:38 -04:00
(use-package nlinum
2015-10-28 03:31:51 -04:00
:commands nlinum-mode
2015-06-06 06:40:33 -04:00
:preface
(setq linum-format "%3d ")
(defvar nlinum-format "%4d ")
2016-05-20 22:37:30 -04:00
(defvar doom--hl-nlinum-overlay nil)
(defvar doom--hl-nlinum-line nil)
2015-11-09 15:52:42 -05:00
:init
2015-11-10 18:01:57 -05:00
(add-hook!
(markdown-mode prog-mode scss-mode web-mode conf-mode groovy-mode
nxml-mode snippet-mode php-mode)
2016-04-30 23:13:39 -04:00
'nlinum-mode)
2016-08-28 22:54:48 +02:00
;; FIXME This only works if hl-line is active!
(add-hook! nlinum-mode
2016-05-26 18:51:39 -04:00
(if nlinum-mode-hook
(add-hook 'post-command-hook 'doom|nlinum-hl-line nil t)
(remove-hook 'post-command-hook 'doom|nlinum-hl-line t)))
2015-10-03 04:56:33 -04:00
:config
2016-05-24 22:50:39 -04:00
;; Calculate line number column width
2015-06-15 09:05:52 +02:00
(add-hook! nlinum-mode
2016-05-23 17:13:59 -04:00
(setq nlinum--width (length (save-excursion (goto-char (point-max))
2016-05-24 22:50:39 -04:00
(format-mode-line "%l")))))
2016-08-28 22:54:48 +02:00
2016-05-24 22:50:39 -04:00
;; Disable nlinum when making frames, otherwise we get linum face error
;; messages that prevent frame creation.
(add-hook 'before-make-frame-hook 'doom|nlinum-disable)
(add-hook 'after-make-frame-functions 'doom|nlinum-enable))
2015-10-26 01:29:38 -04:00
2016-08-28 22:54:48 +02:00
(use-package rainbow-delimiters
:commands rainbow-delimiters-mode
:config (setq rainbow-delimiters-max-face-count 3)
:init
(add-hook! (emacs-lisp-mode lisp-mode js-mode css-mode c-mode-common)
'rainbow-delimiters-mode))
;; NOTE hl-line-mode and rainbow-mode don't play well together
(use-package rainbow-mode
:commands rainbow-mode
:init (after! hl-line (add-hook 'rainbow-mode-hook 'doom|hl-line-off)))
(use-package stripe-buffer
:commands stripe-buffer-mode
:init (add-hook 'dired-mode-hook 'stripe-buffer-mode))
(use-package visual-fill-column :defer t
:config
(setq-default visual-fill-column-center-text nil
visual-fill-column-width fill-column
split-window-preferred-function 'visual-line-mode-split-window-sensibly))
2015-06-06 06:40:33 -04:00
2016-04-26 02:03:42 -04:00
;;
;; Mode-line
;;
2015-06-06 06:40:33 -04:00
(defvar mode-line-height 30)
(defvar-local doom--env-version nil)
(defvar-local doom--env-command nil)
(eval-when-compile (require 'powerline))
(defvar mode-line-bar (! (pl/percent-xpm mode-line-height 100 0 100 0 3 "#00B3EF" nil)))
(defvar mode-line-eldoc-bar (! (pl/percent-xpm mode-line-height 100 0 100 0 3 "#B3EF00" nil)))
(defvar mode-line-inactive-bar (! (pl/percent-xpm mode-line-height 100 0 100 0 3 nil nil)))
(defface mode-line-is-modified nil "Face for mode-line modified symbol")
(defface mode-line-buffer-path nil "Face for mode-line buffer file path")
(defface mode-line-highlight nil "")
(defface mode-line-2 nil "")
;; Make certain unicode glyphs bigger for UI purposes
(doom-fix-unicode '("DejaVu Sans Mono" 15) ?✱)
(let ((font "DejaVu Sans Mono for Powerline"))
(doom-fix-unicode (list font 12) ?)
(doom-fix-unicode (list font 16) ?∄)
(doom-fix-unicode (list font 15) ?))
(defvar mode-line-selected-window nil)
(defun doom|set-selected-window (&rest _)
(let ((window (frame-selected-window)))
(unless (minibuffer-window-active-p window)
(setq mode-line-selected-window window))))
(add-hook 'window-configuration-change-hook #'doom|set-selected-window)
(add-hook 'focus-in-hook #'doom|set-selected-window)
(advice-add 'select-window :after 'doom|set-selected-window)
(advice-add 'select-frame :after 'doom|set-selected-window)
2015-09-28 15:55:25 -04:00
;;
;; Mode-line segments
;;
(defun *buffer-path ()
(when buffer-file-name
(propertize
(f-dirname
(let ((buffer-path (file-relative-name buffer-file-name (doom/project-root)))
(max-length (truncate (/ (window-body-width) 1.75))))
(concat (projectile-project-name) "/"
(if (> (length buffer-path) max-length)
(let ((path (reverse (split-string buffer-path "/" t)))
(output ""))
(when (and path (equal "" (car path)))
(setq path (cdr path)))
(while (and path (<= (length output) (- max-length 4)))
(setq output (concat (car path) "/" output))
(setq path (cdr path)))
(when path
(setq output (concat "../" output)))
(when (string-suffix-p "/" output)
(setq output (substring output 0 -1)))
output)
buffer-path))))
'face (if active 'mode-line-buffer-path))))
(defun *buffer-state ()
(when buffer-file-name
(propertize
(concat (if (not (file-exists-p buffer-file-name))
""
(if (buffer-modified-p) ""))
(if buffer-read-only ""))
'face 'mode-line-is-modified)))
(defun *buffer-name ()
"The buffer's name."
(s-trim-left (format-mode-line "%b")))
(defun *buffer-pwd ()
"Displays `default-directory'."
(propertize
(concat "[" (abbreviate-file-name default-directory) "]")
'face 'mode-line-2))
(defun *major-mode ()
"The major mode, including process, environment and text-scale info."
(concat (format-mode-line mode-name)
(if (stringp mode-line-process) mode-line-process)
(if doom--env-version (concat " " doom--env-version))
(and (featurep 'face-remap)
(/= text-scale-mode-amount 0)
(format " (%+d)" text-scale-mode-amount))))
(defun *buffer-encoding-abbrev ()
"The line ending convention used in the buffer."
(if (memq buffer-file-coding-system '(utf-8 utf-8-unix))
""
(symbol-name buffer-file-coding-system)))
(defface mode-line-vcs-info nil '((t (:inherit warning))))
(defface mode-line-vcs-warning nil '((t (:inherit warning))))
(defun *vc ()
"Displays the current branch, colored based on its state."
(when vc-mode
(let ((backend (concat "" (substring vc-mode (+ 2 (length (symbol-name (vc-backend buffer-file-name)))))))
(face (let ((state (vc-state buffer-file-name)))
(cond ((memq state '(edited added))
'mode-line-vcs-info)
((memq state '(removed needs-merge needs-update conflict removed unregistered))
'mode-line-vcs-warning)))))
(if active
(propertize backend 'face face)
backend))))
(defface doom-flycheck-error '((t (:inherit error)))
"Face for flycheck error feedback in the modeline.")
(defface doom-flycheck-warning '((t (:inherit warning)))
"Face for flycheck warning feedback in the modeline.")
(defvar-local doom--flycheck-err-cache nil "")
(defvar-local doom--flycheck-cache nil "")
(defun *flycheck ()
"Persistent and cached flycheck indicators in the mode-line."
(when (and (featurep 'flycheck)
flycheck-mode
(or flycheck-current-errors
(eq 'running flycheck-last-status-change)))
(or (and (or (eq doom--flycheck-err-cache doom--flycheck-cache)
(memq flycheck-last-status-change '(running not-checked)))
doom--flycheck-cache)
(and (setq doom--flycheck-err-cache flycheck-current-errors)
(setq doom--flycheck-cache
(let ((fe (doom/-flycheck-count 'error))
(fw (doom/-flycheck-count 'warning)))
(concat
(if fe (propertize (format " •%d " fe)
'face (if active
'doom-flycheck-error
'mode-line)))
(if fw (propertize (format " •%d " fw)
'face (if active
'doom-flycheck-warning
'mode-line))))))))))
(defun *selection-info ()
"Information about the current selection."
(when (and active (evil-visual-state-p))
(propertize
(let ((reg-beg (region-beginning))
(reg-end (region-end))
(evil (eq 'visual evil-state)))
(let ((lines (count-lines reg-beg (min (1+ reg-end) (point-max))))
(chars (- (1+ reg-end) reg-beg))
(cols (1+ (abs (- (evil-column reg-end)
(evil-column reg-beg))))))
(cond
;; rectangle selection
((or (bound-and-true-p rectangle-mark-mode)
(and evil (eq 'block evil-visual-selection)))
(format " %dx%dB " lines (if evil cols (1- cols))))
;; line selection
((or (> lines 1) (eq 'line evil-visual-selection))
(if (and (eq evil-state 'visual) (eq evil-this-type 'line))
(format " %dL " lines)
(format " %dC %dL " chars lines)))
(t (format " %dC " (if evil chars (1- chars)))))))
'face 'mode-line-highlight)))
(defun *macro-recording ()
"Show when recording macro."
(when (and active defining-kbd-macro)
(propertize
(format " %s ▶ " (char-to-string evil-this-macro))
'face 'mode-line-highlight)))
(defface mode-line-count-face nil "")
(make-variable-buffer-local 'anzu--state)
(defun *anzu ()
"Show the current match number and the total number of matches. Requires anzu
to be enabled."
(when (and (featurep 'evil-anzu) (evil-ex-hl-active-p 'evil-ex-search))
(propertize
(format " %s/%d%s "
anzu--current-position anzu--total-matched
(if anzu--overflow-p "+" ""))
'face (if active 'mode-line-count-face))))
(defun *evil-substitute ()
"Show number of :s matches in real time."
(when (and (evil-ex-p) (evil-ex-hl-active-p 'evil-ex-substitute))
(propertize
(let ((range (if evil-ex-range
(cons (car evil-ex-range) (cadr evil-ex-range))
(cons (line-beginning-position) (line-end-position))))
(pattern (car-safe (evil-delimited-arguments evil-ex-argument 2))))
(if pattern
(format " %s matches "
(count-matches pattern (car range) (cdr range))
evil-ex-argument)
" ... "))
'face (if active 'mode-line-count-face))))
(defun *iedit ()
"Show the number of iedit regions matches + what match you're on."
(when (and (boundp 'iedit-mode) iedit-mode)
(propertize
(let ((this-oc (let (message-log-max) (iedit-find-current-occurrence-overlay)))
(length (or (ignore-errors (length iedit-occurrences-overlays)) 0)))
(format
" %s/%s "
(save-excursion
(unless this-oc
(iedit-prev-occurrence)
(setq this-oc (iedit-find-current-occurrence-overlay)))
(if this-oc
;; NOTE: Not terribly reliable
(- length (-elem-index this-oc iedit-occurrences-overlays))
"-"))
length))
'face (if active 'mode-line-count-face))))
(defun *buffer-position ()
"A more vim-like buffer position."
(let ((start (window-start))
(end (window-end))
(pend (point-max)))
(if (and (= start 1)
(= end pend))
":All"
(cond ((= start 1) ":Top")
((= end pend) ":Bot")
(t (format ":%d%%%%" (/ end 0.01 pend)))))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
(defun doom-mode-line (&optional id)
`(:eval
(let* ((active (eq (selected-window) mode-line-selected-window))
(lhs (list (propertize " " 'display (if active mode-line-bar mode-line-inactive-bar))
(*flycheck)
(*macro-recording)
(*selection-info)
(*anzu)
(*evil-substitute)
(*iedit)
" "
(*buffer-path)
(*buffer-name)
" "
(*buffer-state)
,(if (eq id 'scratch) '(*buffer-pwd))))
(rhs (list (*buffer-encoding-abbrev)
(*vc)
" " (*major-mode) " "
(propertize
(concat "(%l,%c) " (*buffer-position))
'face (if active 'mode-line-2))))
(middle (propertize
" " 'display `((space :align-to (- (+ right right-fringe right-margin)
,(1+ (string-width (format-mode-line rhs)))))))))
(list lhs middle rhs))))
(setq-default mode-line-format (doom-mode-line))
2015-06-04 18:23:21 -04:00
(provide 'core-ui)
;;; core-ui.el ends here