You may think buffers are just for editing text files, but you can wield Emacs Lisp to do so much more:
- Automate the editing (or creation) of files
- Create a custom display of information for a major mode
- Communicate with external programs
- Provide interactive interfaces like REPLs and more
In Emacs, buffers are the primary user interface, so they’re really important!
In this episode we’re going to focus on the basic APIs for Emacs buffers and then show how to read and edit buffer contents.
We’re going to use a concrete example as the basis for everything we learn!
As we talked about in previous videos in this series, we are working on a dotfiles management package for Emacs.
I created a repo for it! https://github.com/daviwil/dotcrafter.el. Check it out and give it a star!
In this video, we’ll update our dotfiles management package code to add the following capabilities:
- Automatically detect tangled configuration output files from our Org Mode files
- Update the dotfiles repository’s
.gitignore
file to add the tangled output files so that they don’t get checked in
The goal of this package is to manage configuration files that are generated from Org Mode code blocks.
The pattern is to generate these files into the dotfiles folder and then symbolically link them into your home directory.
We might not want to check in those files since we will have to keep them in sync! Instead, we add them to .gitignore
in the dotfiles repo.
The code we will write today will automatically update the .gitignore
file with the generated configuration file paths.
In Emacs Lisp, a buffer is an object that contains text to be displayed, edited, or manipulated. The content of buffer may not come from a file, it could be generated from within Emacs!
The text isn’t just plain letters, it can also contain additional properties that control font, color, size, and other factors!
In Emacs, a buffer is usually displayed by a window, but you can also work with buffers without displaying them.
As we learned in the video about variables, a buffer can have it’s own set of variables!
Emacs’ buffer manipulation functions work on the current buffer.
In Emacs Lisp, you can get the current buffer with the aptly-named current-buffer
function:
(current-buffer)
The current buffer in this case is not necessarily the one visible in the selected window!
If you want to retrieve a buffer by its name, use get-buffer
:
(get-buffer "*scratch*")
You can also create the buffer if it doesn’t already exist with get-buffer-create
:
(get-buffer-create "Hello System Crafters!")
You can now see this in the buffer list!
You can change the current buffer with set-buffer
. This allows you to set the current buffer that the buffer manipulation functions will operate on:
(progn
(set-buffer (get-buffer "*scratch*"))
(current-buffer))
;; You can also pass the name of the buffer directly
(progn
(set-buffer "*scratch*")
(current-buffer))
However, this sets the current buffer until the current command in the command loop is finished, so it might affect other code that runs after yours!
You can avoid causing trouble by wrapping your buffer-setting code in the save-current-buffer
form:
(progn
(save-current-buffer
(set-buffer "*scratch*")
(message "Current buffer: %s" (current-buffer)))
(current-buffer))
For an even shorter solution, you can use the with-current-buffer
macro:
(progn
(with-current-buffer "*scratch*"
(message "Current buffer: %s" (current-buffer)))
(current-buffer))
If you use set-buffer
in your code, you almost always want to use save-current-buffer
or with-current-buffer
so that you don’t cause weird things to happen in Emacs! These functions also handle errors correctly and ensure that the previous buffer is set back to what it was before you changed it.
Most often in Emacs, you’ll be working with buffers that contain text loaded from a file. If Emacs created the buffer (using find-file
or a similar function), you can use the buffer-file-name
function to get full file path for the file that the buffer represents:
(buffer-file-name)
You can also find a buffer that represents a particular file (or file path) using the get-file-buffer
function:
(get-file-buffer "Emacs-Lisp-04.org")
(get-file-buffer "~/Notes/Streams/Emacs-Lisp-05.org")
(get-file-buffer "~/Notes2/Streams/Emacs-Lisp-05.org")
NOTE: This function will convert the path you provide into its absolute file path before searching for it. Path expansion takes the default-directory
variable into account, so make sure to provide as much path context as you can to make sure the right file is found!
(setq dotcrafter-org-files '("Emacs.org" "Desktop.org" "Systems.org"))
(dolist (org-file dotcrafter-org-files)
(with-current-buffer (get-file-buffer (expand-file-name org-file
dotcrafter-dotfiles-folder))
(message "File: %s" (buffer-file-name))))
Emacs provides a function for “visiting” a file without displaying it, mainly for opening a file into a buffer. This function is called find-file-noselect
:
(find-file-noselect "Emacs-Lisp-01.org")
A couple of interesting details:
- Running this more than once for the same file will return the same buffer
- If the buffer for that file is modified and not saved, the user might be prompted before opening a new buffer for the file
- To prevent the user from being prompted, send
t
as the second argument:
(find-file-noselect "Emacs-Lisp-01.org" t)
We can use find-file-noselect
to create a buffer for the file if it doesn’t already exist:
(dolist (org-file dotcrafter-org-files)
(let ((file-path (expand-file-name org-file
dotcrafter-dotfiles-folder)))
(with-current-buffer (or (get-file-buffer file-path)
(find-file-noselect file-path))
(message "File: %s" (buffer-file-name)))))
The point of talking about all of this is so that we can talk about the point!
The “point” is the location of the cursor in the buffer. It is the location from which all hand editing commands operate.
The point is represented by an integer (whole number) which starts at 1 and increases for every character in the buffer.
(point)
NOTE: The same buffer can be displayed in multiple windows and the point can be different in those two windows!
You can also check the minimum and maximum point locations of the buffer using point-min
and point-max
:
(point-min)
(point-max)
“Narrowing” may affect these positions, but we’ll talk about that another time.
You can use the following motion commands to move the point:
goto-char
- Move the point to a specific position (integer)forward-char
- Move the point forward by a number of positions (1 by default)backward-char
- Move the point backward by a number of positions (1 by default)beginning-of-buffer
- Go to the beginning of the bufferend-of-buffer
- Go to the end of the buffer
You can also move based on larger textual units in the buffer:
forward-word
- Move forward by one “word”backward-word
- Move backward by one “word”
(goto-char 1)
(goto-char (point-min))
(goto-char (point-max))
(beginning-of-buffer)
(end-of-buffer)
(forward-char)
(forward-char 5)
(backward-char)
(backward-char 10)
(forward-word)
(backward-word)
There are many more forward
and backward
functions, just check the function list (describe-function
, C-h f
) to find them!
The cool thing about these functions is that they’re the same ones you use with Emacs default movement keybindings. You can automate buffer editing with the same functions you use for typing!
Similarly to setting the current buffer, you might want to preserve the current point location in a buffer before you move it for another purpose. You can use the save-excursion
special form for this purpose:
(save-excursion
(goto-char (point-max))
(point))
This is useful when you need to do an operation in the current buffer which might be displayed in the user’s current window!
Now that we understand the point, we can talk about how to look at text in the buffer. The simplest thing you can do is read the character at a location with char-after
:
(char-after)
(char-after (point))
(char-after (point-min))
You can also get a substring of text in the buffer between two points using buffer-substring
and buffer-substring-no-properties
:
(buffer-substring 9328 9349)
(buffer-substring-no-properties 9328 9349)
The thing-at-point
function is very useful for grabbing the text at the point if it matches a particular type of “thing”:
word
,sentence
, andline
sexp
,list
, anddefun
- Lisp expressionsurl
,email
filename
(thing-at-point 'word)
(thing-at-point 'sentence)
(thing-at-point 'sentence t)
(thing-at-point 'sexp)
Try it on this: ~/Projects/Code/emacs-from-scratch/Emacs.org
Emacs Lisp Manual: Examining Buffer Contents
Sometimes it can be useful to search for text inside of the buffer and move the point to where the match was found. You can use the search-forward
and search-backward
functions for this:
(search-forward "ways")
(search-backward "I just searched myself")
(search-backward "inside" nil t 1)
(search-backward "inside" nil t 3)
Keep in mind that search-forward
will put the point after the match and search-backward
will put the point before the match!
The other parameters can be useful too:
bound
(param 2) - A pair (cons) of positions restricting the search within those two positionsnoerror
- Ift
, don’t signal an error when no match is foundcount
- Find the “nth” result wherecount
isn
There are other ways to search inside of buffers, including the use of regular expressions to extract text from matches. We’ll cover this in another episode!
Now we can finally do something useful in our code! Let’s write some Emacs Lisp to search for :tangle
properties on Org source blocks so that we can extract the file path:
(defun dotcrafter--scan-for-output-files (org-file)
(let ((output-files '())
(current-match t))
(with-current-buffer (or (get-file-buffer org-file)
(find-file-noselect org-file))
(save-excursion
(goto-char (point-min)) ;; Or (beginning-of-buffer)
(while current-match
(setq current-match (search-forward ":tangle " nil t))
(when current-match
(let ((output-file (thing-at-point 'filename t)))
;; If a file path was found, add it to the list
(unless (or (not output-file)
(string-equal output-file "no"))
(setq output-files (cons output-file
output-files))))))))
output-files))
(let ((output-files '()))
(dolist (org-file dotcrafter-org-files)
(setq output-files
(append output-files
(dotcrafter--scan-for-output-files
(expand-file-name org-file
dotcrafter-dotfiles-folder)))))
output-files)
You can insert text into the buffer at the current point using the insert
and insert-char
functions. insert
will insert the arbitrary list of strings or characters at point and insert-char
will insert the specified character with an optional repeat count:
(insert " 0_o")
(insert "\n" "This is" ?\s ?\n "Sparta!")
(insert-char ?\- 20)
Emacs Lisp Manual: Inserting Text Emacs Lisp Manual: Basic Char Syntax
(defvar dotcrafter-gitignore-marker "\n# -- Generated by dotcrafter.el! --\n\n"
"The marker string to be placed in the .gitignore file of the
dotfiles repo to indicate where the auto-generated list of ignored
files begins.")
(defun dotcrafter--update-gitignore ()
(let ((output-files '()))
(dolist (org-file dotcrafter-org-files)
(setq output-files
(append output-files
(dotcrafter--scan-for-output-files
(expand-file-name org-file
dotcrafter-dotfiles-folder)))))
(let ((gitignore-file (expand-file-name ".gitignore"
dotcrafter-dotfiles-folder)))
(with-current-buffer (or (get-file-buffer gitignore-file)
(find-file-noselect gitignore-file))
(save-excursion
(beginning-of-buffer)
(or (progn
(search-forward dotcrafter-gitignore-marker nil t))
(progn
(end-of-buffer)
(insert "\n" dotcrafter-gitignore-marker)))
(dolist (output-file output-files)
(insert output-file "\n")))))))
Notice that running this multiple times appends the list again and again! We need to delete the old list before adding the new one.
You can delete a region of text in a buffer using the delete-region
function. It takes two parameters, the start
point and the end
point.
(with-current-buffer ".gitignore"
(delete-region (point) (point-max)))
To save the contents of a buffer back to the file it is associated with, you can use the save-buffer
function:
(save-buffer)
Let’s finish the job of automatically managing the .gitignore
file by cleaning up its contents and saving it:
(defun dotcrafter--update-gitignore ()
(let ((output-files '()))
(dolist (org-file dotcrafter-org-files)
(setq output-files
(append output-files
(dotcrafter--scan-for-output-files
(expand-file-name org-file
dotcrafter-dotfiles-folder)))))
(let ((gitignore-file (expand-file-name ".gitignore"
dotcrafter-dotfiles-folder)))
(with-current-buffer (or (get-file-buffer gitignore-file)
(find-file-noselect gitignore-file))
(save-excursion
(beginning-of-buffer)
(or (progn
(search-forward dotcrafter-gitignore-marker nil t))
(progn
(end-of-buffer)
(insert "\n" dotcrafter-gitignore-marker)))
(delete-region (point) (point-max))
(dolist (output-file output-files)
(insert output-file "\n"))
(save-buffer))))))
In the next episode, we’ll cover how to manage files and directories in Emacs Lisp. We’ll also extend dotcrafter
to create symbolic links to their target locations in the home directory!
You can check out the final code here at the dotcrafter.el
repository!
I’ve also included the code below for posterity:
(defvar dotcrafter-gitignore-marker "\n# -- Generated by dotcrafter.el! --\n\n"
"The marker string to be placed in the .gitignore file of the
dotfiles repo to indicate where the auto-generated list of ignored
files begins.")
(defun dotcrafter--scan-for-output-files (org-file)
(let ((output-files '())
(current-match t))
;; Get a buffer for the file, either one that is
;; already open or open a new one
(with-current-buffer (or (get-file-buffer org-file)
(find-file-noselect org-file))
;; Save the current buffer position
(save-excursion
;; Go back to the beginning of the buffer
(goto-char (point-min))
;; Loop until no more matches are found
(while current-match
;; Search for blocks with a :tangle property
(setq current-match (search-forward ":tangle " nil t))
(when current-match
(let ((output-file (thing-at-point 'filename t)))
;; If a file path was found, add it to the list
(unless (or (not output-file)
(string-equal output-file "no"))
(setq output-files (cons output-file
output-files))))))))
output-files))
(defun dotcrafter--update-gitignore ()
(let ((output-files '()))
;; Gather the list of output files from all Org files
(dolist (org-file dotcrafter-org-files)
(setq output-files
(append output-files
(dotcrafter--scan-for-output-files
(expand-file-name org-file dotcrafter-dotfiles-folder)))))
;; Now that we have the output files, update the .gitignore file
(let ((gitignore-file (expand-file-name ".gitignore"
dotcrafter-dotfiles-folder)))
;; Find the .gitignore buffer and prepare for editing
(with-current-buffer (or (get-file-buffer gitignore-file)
(find-file-noselect gitignore-file))
(save-excursion
;; Find or insert the dotcrafter-gitignore-marker
(beginning-of-buffer)
(or (progn
(search-forward dotcrafter-gitignore-marker nil t))
(progn
(end-of-buffer)
(insert "\n" dotcrafter-gitignore-marker)))
;; Delete the rest of the buffer after the marker
(delete-region (point) (point-max))
;; Insert a line for each output file
(dolist (output-file output-files)
(insert output-file "\n"))
;; Make sure the buffer is saved
(save-buffer))))))