Starter Kit Lisp
This is part of the Emacs Starter Kit.
Starter Kit Lisp
Support for editing list dialects including Emacs Lisp, Scheme, Common Lisp, and Clojure.
Define keys
(define-key read-expression-map (kbd "TAB") 'lisp-complete-symbol) ;; (define-key lisp-mode-shared-map (kbd "C-c l") "lambda") (define-key lisp-mode-shared-map (kbd "RET") 'reindent-then-newline-and-indent) (define-key lisp-mode-shared-map (kbd "C-\\") 'lisp-complete-symbol) ;; (define-key lisp-mode-shared-map (kbd "C-c v") 'eval-buffer)
Paredit
Paredit might seem weird at first, but it really makes writing lisp a
much more comfortable experience. This is especially useful in
combination with the sexp movement functions (C-M-f
forward, C-M-b
back, C-M-u
up, C-M-d
down)
(starter-kit-install-if-needed 'paredit)
(require 'paredit) (defun turn-on-paredit () (paredit-mode +1))
;; (eval-after-load 'paredit ;; ;; Not sure why paredit behaves this way with comments; it's annoying ;; '(define-key paredit-mode-map (kbd ";") 'self-insert-command))
Non-obtrusive parenthesis faces
(defface starter-kit-paren-face '((((class color) (background dark)) (:foreground "grey50")) (((class color) (background light)) (:foreground "grey55"))) "Face used to dim parentheses." :group 'starter-kit-faces)
Emacs Lisp
(add-hook 'emacs-lisp-mode-hook 'turn-on-eldoc-mode) (add-hook 'emacs-lisp-mode-hook 'run-starter-kit-coding-hook) (add-hook 'emacs-lisp-mode-hook 'starter-kit-remove-elc-on-save) ;; (add-hook 'emacs-lisp-mode-hook 'idle-highlight) (add-hook 'emacs-lisp-mode-hook 'turn-on-paredit) (define-key emacs-lisp-mode-map (kbd "C-c v") 'eval-buffer) (defun starter-kit-remove-elc-on-save () "If you're saving an elisp file, likely the .elc is no longer valid." (make-local-variable 'after-save-hook) (add-hook 'after-save-hook (lambda () (if (file-exists-p (concat buffer-file-name "c")) (delete-file (concat buffer-file-name "c")))))) (font-lock-add-keywords 'emacs-lisp-mode '(("(\\|)" . 'starter-kit-paren-face)))
Clojure
(starter-kit-install-if-needed 'clojure-mode)
(require 'clojure-mode) (add-hook 'clojure-mode-hook 'run-starter-kit-coding-hook) (font-lock-add-keywords 'clojure-mode '(("(\\|)" . 'starter-kit-paren-face))) (define-key clojure-mode-map (kbd "C-c v") 'slime-eval-buffer) (define-key clojure-mode-map (kbd "C-c C-v") 'slime-eval-buffer) (defface starter-kit-clojure-trace-face '((((class color) (background dark)) (:foreground "grey50")) (((class color) (background light)) (:foreground "grey55"))) "Face used to dim parentheses." :group 'starter-kit-faces) (setq starter-kit-clojure-trace-face 'starter-kit-clojure-trace-face) ;; This will make relevant lines stand out more in stack traces (defun sldb-font-lock () (font-lock-add-keywords nil '(("[0-9]+: \\(clojure\.\\(core\\|lang\\).*\\)" 1 starter-kit-clojure-trace-face) ("[0-9]+: \\(java.*\\)" 1 starter-kit-clojure-trace-face) ("[0-9]+: \\(swank.*\\)" 1 starter-kit-clojure-trace-face) ("\\[\\([A-Z]+\\)\\]" 1 font-lock-function-name-face)))) (add-hook 'sldb-mode-hook 'sldb-font-lock) (defun slime-jump-to-trace (&optional on) "Jump to the file/line that the current stack trace line references. Only works with files in your project root's src/, not in dependencies." (interactive) (save-excursion (beginning-of-line) (search-forward-regexp "[0-9]: \\([^$(]+\\).*?\\([0-9]*\\))") (let ((line (string-to-number (match-string 2))) (ns-path (split-string (match-string 1) "\\.")) (project-root (locate-dominating-file default-directory "src/"))) (find-file (format "%s/src/%s.clj" project-root (mapconcat 'identity ns-path "/"))) (goto-line line)))) (eval-after-load 'slime '(progn (defalias 'sldb-toggle-details 'slime-jump-to-trace) (defun sldb-prune-initial-frames (frames) "Show all stack trace lines by default." frames))) (eval-after-load 'find-file-in-project '(add-to-list 'ffip-patterns "*.clj")) ;; You might like this, but it's a bit disorienting at first: (add-hook 'clojure-mode-hook 'turn-on-paredit) (defun clojure-project (path) "Setup classpaths for a clojure project and starts a new SLIME session. Kills existing SLIME session, if any." (interactive (list (ido-read-directory-name "Project root: " (locate-dominating-file default-directory "pom.xml")))) (when (get-buffer "*inferior-lisp*") (kill-buffer "*inferior-lisp*")) (add-to-list 'swank-clojure-extra-vm-args (format "-Dclojure.compile.path=%s" (expand-file-name "target/classes/" path))) (setq swank-clojure-binary nil swank-clojure-jar-path (expand-file-name "target/dependency/" path) swank-clojure-extra-classpaths (append (mapcar (lambda (d) (expand-file-name d path)) '("src/" "target/classes/" "test/")) (let ((lib (expand-file-name "lib" path))) (if (file-exists-p lib) (directory-files lib t ".jar$")))) slime-lisp-implementations (cons `(clojure ,(swank-clojure-cmd) :init swank-clojure-init) (remove-if #'(lambda (x) (eq (car x) 'clojure)) slime-lisp-implementations))) (save-window-excursion (slime)))
pretty λ's in clojure
(eval-after-load 'clojure-mode '(font-lock-add-keywords 'clojure-mode `(("(\\(fn\\>\\)" (0 (progn (compose-region (match-beginning 1) (match-end 1) "ƒ") nil))))))