clojure-emacs / clojure-mode

Emacs support for the Clojure(Script) programming language
921 stars 247 forks source link
clojure clojure-mode emacs emacs-lisp emacs-package major-mode

circleci MELPA MELPA Stable NonGNU ELPA Discord License GPL 3

Clojure Mode

clojure-mode is an Emacs major mode that provides font-lock (syntax highlighting), indentation, navigation and refactoring support for the Clojure(Script) programming language.


This documentation tracks the master branch of clojure-mode. Some of the features and settings discussed here might not be available in older releases (including the current stable release). Please, consult the relevant git tag (e.g. 5.19.0) if you need documentation for a specific clojure-mode release.

Installation

Available on the major package.el community maintained repos - MELPA Stable and MELPA repos.

MELPA Stable is the recommended repo as it has the latest stable version. MELPA has a development snapshot for users who don't mind (infrequent) breakage but don't want to run from a git checkout.

You can install clojure-mode using the following command:

M-x package-install [RET] clojure-mode [RET]

or if you'd rather keep it in your dotfiles:

(unless (package-installed-p 'clojure-mode)
  (package-install 'clojure-mode))

If the installation doesn't work try refreshing the package list:

M-x package-refresh-contents

Bundled major modes

The clojure-mode package actually bundles together several major modes:

All the major modes derive from clojure-mode and provide more or less the same functionality. Differences can be found mostly in the font-locking - e.g. ClojureScript has some built-in constructs that are not present in Clojure.

The proper major mode is selected automatically based on the extension of the file you're editing.

Having separate major modes gives you the flexibility to attach different hooks to them and to alter their behavior individually (e.g. add extra font-locking just to clojurescript-mode) .

Note that all modes derive from clojure-mode, so things you add to clojure-mode-hook and clojure-mode-map will affect all the derived modes as well.

Configuration

In the spirit of Emacs, pretty much everything you can think of in clojure-mode is configurable.

To see a list of available configuration options do M-x customize-group RET clojure.

Indentation options

The default indentation rules in clojure-mode are derived from the community Clojure Style Guide. Please, refer to the guide for the general Clojure indentation rules.

If you'd like to use the alternative "fixed/tonsky" indentation style you should update your configuration accordingly:

(setq clojure-indent-style 'always-indent
      clojure-indent-keyword-style 'always-indent
      clojure-enable-indent-specs nil)

Read on for more details on the available indentation-related configuration options.

Indentation of docstrings

By default multi-line docstrings are indented with 2 spaces, as this is a somewhat common standard in the Clojure community. You can however adjust this by modifying clojure-docstring-fill-prefix-width. Set it to 0 if you don't want multi-line docstrings to be indented at all (which is pretty common in most lisps).

Indentation of function forms

The indentation of function forms is configured by the variable clojure-indent-style. It takes three possible values:

(some-function
 10
 1
 2)
(some-function 10
               1
               2)
(some-function
  10
  1
  2)
(some-function 10
  1
  2)
(some-function
  10
  1
  2)
(some-function 10
               1
               2)

Note: Prior to clojure-mode 5.10, the configuration options for clojure-indent-style used to be keywords, but now they are symbols. Keywords will still be supported at least until clojure-mode 6.

Indentation of keywords

Similarly we have the clojure-indent-keyword-style, which works in the following way:

(:require [foo.bar]
          [bar.baz])
(:require
 [foo.bar]
 [bar.baz])
(:require [foo.bar]
   [bar.baz])
(:x
   location
   0)
(:require [foo.bar]
          [bar.baz])
(:x
   location
   0)

Indentation of macro forms

The indentation of special forms and macros with bodies is controlled via put-clojure-indent, define-clojure-indent and clojure-backtracking-indent. Nearly all special forms and built-in macros with bodies have special indentation settings in clojure-mode. You can add/alter the indentation settings in your personal config. Let's assume you want to indent ->> and -> like this:

(->> something
  ala
  bala
  portokala)

You can do so by putting the following in your config:

(put-clojure-indent '-> 1)
(put-clojure-indent '->> 1)

This means that the body of the ->/->> is after the first argument.

A more compact way to do the same thing is:

(define-clojure-indent
  (-> 1)
  (->> 1))

To indent something like a definition (defn) you can do something like:

(put-clojure-indent '>defn :defn)

You can also specify different indentation settings for symbols prefixed with some ns (or ns alias):

(put-clojure-indent 'do 0)
(put-clojure-indent 'my-ns/do 1)

The bodies of certain more complicated macros and special forms (e.g. letfn, deftype, extend-protocol, etc) are indented using a contextual backtracking indentation method, require more sophisticated indent specifications. Here are a few examples:

(define-clojure-indent
  (implement '(1 (1)))
  (letfn     '(1 ((:defn)) nil))
  (proxy     '(2 nil nil (1)))
  (reify     '(:defn (1)))
  (deftype   '(2 nil nil (1)))
  (defrecord '(2 nil nil (1)))
  (specify   '(1 (1)))
  (specify   '(1 (1))))

These follow the same rules as the :style/indent metadata specified by cider-nrepl. For instructions on how to write these specifications, see this document. The only difference is that you're allowed to use lists instead of vectors.

The indentation of special arguments is controlled by clojure-special-arg-indent-factor, which by default indents special arguments a further lisp-body-indent when compared to ordinary arguments.

An example of the default formatting is:

(defrecord MyRecord
    [my-field])

Note that defrecord has two special arguments, followed by the form's body - namely the record's name and its fields vector.

Setting clojure-special-arg-indent-factor to 1, results in:

(defrecord MyRecord
  [my-field])

You can completely disable the effect of indentation specs like this:

(setq clojure-enable-indent-specs nil)

Indentation of Comments

clojure-mode differentiates between comments like ;, ;;, etc. By default clojure-mode treats ; as inline comments and always indents those. You can change this behaviour like this:

(add-hook 'clojure-mode-hook (lambda () (setq-local comment-column 0)))

You might also want to change comment-add to 0 in that way, so that Emacs comment functions (e.g. comment-region) would use ; by default instead of ;;.

Note: Check out this section of the Clojure style guide to understand better the semantics of the different comment levels and why clojure-mode treats them differently by default.

Vertical alignment

You can vertically align sexps with C-c SPC. For instance, typing this combo on the following form:

(def my-map
  {:a-key 1
   :other-key 2})

Leads to the following:

(def my-map
  {:a-key     1
   :other-key 2})

This can also be done automatically (as part of indentation) by turning on clojure-align-forms-automatically. This way it will happen whenever you select some code and hit TAB.

Font-locking

clojure-mode features static font-locking (syntax highlighting) that you can extend yourself if needed. As typical for Emacs, it's based on regular expressions. You can find the default font-locking rules in clojure-font-lock-keywords. Here's how you can add font-locking for built-in Clojure functions and vars:

(defvar clojure-built-in-vars
  '(;; clojure.core
    "accessor" "aclone"
    "agent" "agent-errors" "aget" "alength" "alias"
    "all-ns" "alter" "alter-meta!" "alter-var-root" "amap"
    ;; omitted for brevity
    ))

(defvar clojure-built-in-dynamic-vars
  '(;; clojure.test
    "*initial-report-counters*" "*load-tests*" "*report-counters*"
    "*stack-trace-depth*" "*test-out*" "*testing-contexts*" "*testing-vars*"
    ;; clojure.xml
    "*current*" "*sb*" "*stack*" "*state*"
    ))

(font-lock-add-keywords 'clojure-mode
                        `((,(concat "(\\(?:\.*/\\)?"
                                    (regexp-opt clojure-built-in-vars t)
                                    "\\>")
                           1 font-lock-builtin-face)))

(font-lock-add-keywords 'clojure-mode
                        `((,(concat "\\<"
                                    (regexp-opt clojure-built-in-dynamic-vars t)
                                    "\\>")
                           0 font-lock-builtin-face)))

Note: The package clojure-mode-extra-font-locking provides such additional font-locking for Clojure built-ins.

As you might imagine one problem with this font-locking approach is that because it's based on regular expressions you'll get some false positives here and there (there's no namespace information, and no way for clojure-mode to know what var a symbol resolves to). That's why clojure-mode's font-locking defaults are conservative and minimalistic.

Precise font-locking requires additional data that can obtained from a running REPL (that's how CIDER's dynamic font-locking works) or from static code analysis.

When it comes to non built-in definitions, clojure-mode needs to be manually instructed how to handle the docstrings and highlighting. Here's an example:

(put '>defn 'clojure-doc-string-elt 2)

(font-lock-add-keywords 'clojure-mode
                        `((,(concat "(\\(?:" clojure--sym-regexp "/\\)?"
                                    "\\(>defn\\)\\>")
                           1 font-lock-keyword-face)))

Note: The clojure-doc-string-elt attribute is processed by the function clojure-font-lock-syntactic-face-function.

Refactoring support

The available refactorings were originally created and maintained by the clj-refactor.el team. The ones implemented in Elisp only are gradually migrated to clojure-mode.

Threading macros related features

clojure-thread: Thread another form into the surrounding thread. Both ->> and -> variants are supported.

clojure-unwind: Unwind a threaded expression. Supports both ->> and ->.

clojure-thread-first-all: Introduce the thread first macro (->) and rewrite the entire form. With a prefix argument do not thread the last form.

clojure-thread-last-all: Introduce the thread last macro and rewrite the entire form. With a prefix argument do not thread the last form.

clojure-unwind-all: Fully unwind a threaded expression removing the threading macro.

Cycling things

clojure-cycle-privacy: Cycle privacy of defs or defns. Use metadata explicitly with setting clojure-use-metadata-for-privacy to t for defns too.

clojure-cycle-not: Add or remove a not form around the current form.

clojure-cycle-when: Find the closest when or when-not up the syntax tree and toggle it.

clojure-cycle-if: Find the closest if or if-not up the syntax tree and toggle it. Also transpose the else and then branches, keeping the semantics the same as before.

Convert collection

Convert any given collection at point to list, quoted list, map, vector or set.

Let expression

clojure-introduce-let: Introduce a new let form. Put the current form into its binding form with a name provided by the user as a bound name. If called with a numeric prefix put the let form Nth level up in the form hierarchy.

clojure-move-to-let: Move the current form to the closest let's binding form. Replace all occurrences of the form in the body of the let.

clojure-let-forward-slurp-sexp: Slurp the next form after the let into the let. Replace all occurrences of the bound forms in the form added to the let form. If called with a prefix argument slurp the next n forms.

clojure-let-backward-slurp-sexp: Slurp the form before the let into the let. Replace all occurrences of the bound forms in the form added to the let form. If called with a prefix argument slurp the previous n forms.

paredit-convolute-sexp is advised to replace occurrences of bound forms with their bound names when convolute is used on a let form.

Rename ns alias

clojure-rename-ns-alias: Rename an alias inside a namespace declaration, and all of its usages in the buffer

If there is an active selected region, only rename usages of aliases within the region, without affecting the namespace declaration.

Add arity to a function

clojure-add-arity: Add a new arity to an existing single-arity or multi-arity function.

Related packages

(require 'clojure-mode-extra-font-locking)

The code in clojure-mode-font-locking used to be bundled with clojure-mode before version 3.0.

You can also use the code in this package as a basis for extending the font-locking further (e.g. functions/macros from more namespaces). Generally you should avoid adding special font-locking for things that don't have fairly unique names, as this will result in plenty of incorrect font-locking. CIDER users should avoid this package, as CIDER does its own dynamic font-locking, which is namespace-aware and doesn't produce almost any false positives.

(add-hook 'clojure-mode-hook #'subword-mode)
(add-hook 'clojure-mode-hook #'paredit-mode)
(add-hook 'clojure-mode-hook #'smartparens-strict-mode)
(add-hook 'clojure-mode-hook #'rainbow-delimiters-mode)
(add-hook 'clojure-mode-hook #'aggressive-indent-mode)

REPL Interaction

One of the fundamental aspects of Lisps in general, and Clojure in particular, is the notion of interactive programming - building your programs by continuously changing the state of the running Lisp program (as opposed to doing something more traditional like making a change and re-running the program afterwards to see the changes in action). To get the most of clojure-mode you'll have to combine it with some tool which will allow you to interact with your Clojure program (a.k.a. process/REPL).

A number of options exist for connecting to a running Clojure process and evaluating code interactively.

Basic REPL

inf-clojure provides basic interaction with a Clojure REPL process. It's very similar in nature and supported functionality to inferior-lisp-mode for Common Lisp.

CIDER

CIDER is a powerful Clojure interactive development environment, similar to SLIME for Common Lisp.

If you're into Clojure and Emacs you should definitely check it out.

Tutorials

Tutorials, targeting Emacs beginners, are available at clojure-doc.org and Clojure for the Brave and the True. Keep in mind, however, that they might be out-of-date.

Caveats

clojure-mode is a capable tool, but it's certainly not perfect. This section lists a couple of general design problems/limitations that might affect your experience negatively.

General Issues

clojure-mode derives a lot of functionality directly from lisp-mode (an Emacs major mode for Common Lisp), which simplified the initial implementation, but also made it harder to implement certain functionality. Down the road it'd be nice to fully decouple clojure-mode from lisp-mode.

See this ticket for a bit more details.

Indentation Performance

clojure-mode's indentation engine is a bit slow. You can speed things up significantly by disabling clojure-use-backtracking-indent, but this will break the indentation of complex forms like deftype, defprotocol, reify, letfn, etc.

We should look into ways to optimize the performance of the backtracking indentation logic. See this ticket for more details.

Font-locking Implementation

As mentioned above, the font-locking is implemented in terms of regular expressions which makes it both slow and inaccurate.

Changelog

An extensive changelog is available here.

License

Copyright © 2007-2024 Jeffrey Chu, Lennart Staflin, Phil Hagelberg, Bozhidar Batsov, Artur Malabarba, Magnar Sveen and contributors.

Distributed under the GNU General Public License; type C-h C-c to view it.