buffer-layers.el 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. ;;; buffer-layers.el --- Layered Buffers for Buffer Management
  2. ;; Copyright (C) 2016 Samuel Flint
  3. ;; Author: Samuel W. Flint <swflint@flintfam.org>
  4. ;; Version: 1.7
  5. ;; Package-Requires: ((cl-lib "0.5"))
  6. ;; Keywords: buffer-management
  7. ;; URL: http://github.com/swflint/buffer-layers
  8. ;;; Commentary:
  9. ;;
  10. ;; This is Buffer Layers, a simple, layer-based buffer management system.
  11. ;;
  12. ;; It works by defining buffer layers using a fairly simple macro. The following would be put in a file called `org.layer`
  13. ;;
  14. ;; ```elisp
  15. ;; ;; -*- emacs-lisp -*-
  16. ;; (define-buffer-layer org
  17. ;; :files ("~/org/"
  18. ;; "~/org/main.org")
  19. ;; :buffer-to-select "main.org"
  20. ;; :run-on-apply ((my/find-current-notes-file)))
  21. ;; ```
  22. ;;
  23. ;; It can be loaded with `(load-buffer-layer "/path/to/org.layer" nil)`. If the final `nil` is changed to `t`, it will load and apply the layer.
  24. ;;
  25. ;; Buffer Layer Definitions take the following arguments:
  26. ;;
  27. ;; - `:files`: A list. This is the list of files that are loaded when the buffer layer is applied.
  28. ;; - `:buffer-to-select`: The buffer to select after files are loaded, and the given forms to run on application are executed.
  29. ;; - `:run-on-apply`: This is a list of forms to be executed in between finding files and selecting the given buffer.
  30. ;; - `:run-on-remove`: This is a list of forms to be executed after killing the buffers that have been loaded.
  31. ;;
  32. ;; To manipulate buffer layers, execute `buffer-layer-mode`, and then you can use the following keybindings:
  33. ;;
  34. ;; - `C-x L l`: Load a buffer layer, if defined, otherwise, load from the given file.
  35. ;; - `C-x L u`: Unload a loaded buffer layer.
  36. ;; - `C-x L U`: Unload all loaded buffer layers.
  37. ;; - `C-x L L`: List defined buffer layers, noting if they've been applied.
  38. ;;
  39. ;; The following are the user-facing functions:
  40. ;;
  41. ;; - `define-buffer-layer`
  42. ;; - `buffer-layer-load-buffer-layer`, also known as `load-buffer-layer`
  43. ;; - `buffer-layers-unload-buffer-layer`
  44. ;; - `buffer-layers-list`
  45. ;; - `buffer-layers-unload-all-buffer-layers`
  46. ;; - `buffer-layers-mode`
  47. ;;
  48. ;; On enabling `buffer-layer-mode`, the map is placed onto `C-x L`, and `buffer-layers-unload-all-buffer-layers` is added to the `kill-emacs-hook`, and on disabling the mode, they are removed.
  49. ;;; Code:
  50. (require 'cl-lib)
  51. (defvar *buffer-layers* nil
  52. "List of all defined buffer layers.")
  53. (defvar *buffer-layers-applied* nil
  54. "List of applied buffer-layers.")
  55. (defvar *buffer-layer-definitions* nil
  56. "List of all buffer layer definitions.")
  57. (defvar *buffer-layer-buffers* nil
  58. "List of buffers in loaded buffer layers.")
  59. (defvar *buffer-layer-file* "~/.emacs.d/buffer-layer-definitions.el"
  60. "The file to store buffer layer definitions in.")
  61. (defun buffer-layers-applied-p (layer)
  62. "Returns true if LAYER is applied."
  63. (member layer *buffer-layers-applied*))
  64. (defun buffer-layers--applier-name (name)
  65. "Generate name to apply a buffer layer based on NAME."
  66. (intern (format "apply-buffer-layers-%s" name)))
  67. (defun buffer-layers--remover-name (name)
  68. "Generate name to remove a buffer layer based on NAME."
  69. (intern (format "remove-buffer-layers-%s" name)))
  70. (defun buffer-layers--buffer-list-name (name)
  71. "Generate name to contain buffer layer buffer list based on NAME."
  72. (intern (format "*buffer-layers-%s-buffers*" name)))
  73. (cl-defmacro define-buffer-layer (name &key files run-on-apply run-on-remove buffer-to-select)
  74. "Define a buffer layer named NAME, taking FILES, RUN-ON-APPLY, RUN-ON-REMOVE and BUFFER-TO-SELECT as keyword arguments."
  75. (let ((applier (buffer-layers--applier-name name))
  76. (remover (buffer-layers--remover-name name))
  77. (buffers-list (buffer-layers--buffer-list-name name))
  78. (files-list (cons 'list
  79. (when (not (null files))
  80. (mapcar #'(lambda (name)
  81. (format "%s" name)) files)))))
  82. `(progn
  83. (add-to-list '*buffer-layers* ',name)
  84. (defvar ,buffers-list nil)
  85. (defun ,applier ()
  86. ,(format "Apply buffer-layer %s." name)
  87. (interactive)
  88. (mapcar #'(lambda (file)
  89. (add-to-list ',buffers-list (find-file file)))
  90. ,files-list)
  91. ,@run-on-apply
  92. (when (not (null ,buffer-to-select))
  93. (switch-to-buffer ,buffer-to-select))
  94. (add-to-list '*buffer-layers-applied* ',name)
  95. (message "Applied Buffer Layer %s" ',name))
  96. (defun ,remover ()
  97. ,(format "Remove buffer-layer %s." name)
  98. (interactive)
  99. (mapc #'(lambda (buffer)
  100. (when (buffer-live-p buffer)
  101. (with-current-buffer buffer
  102. (save-buffer)
  103. (kill-buffer))))
  104. ,buffers-list)
  105. ,@run-on-remove
  106. (setq ,buffers-list nil)
  107. (setq *buffer-layers-applied* (delq ',name *buffer-layers-applied*))
  108. (message "Removed Buffer Layer %s" ',name))
  109. (setq current-buffer-applier ',applier)
  110. (list ',applier ',remover))))
  111. (cl-defmacro define-buffer-layer-new (name &key files run-on-apply run-on-remove buffer-to-select)
  112. `(progn
  113. (cl-pushnew '(,name
  114. (:files ,@files)
  115. (:on-apply ,@run-on-apply)
  116. (:on-remove ,@run-on-remove)
  117. ,@(when buffer-to-select
  118. (list :select-buffer buffer-to-select))
  119. ,@(when run-on-apply
  120. (list :on-apply-lambda (eval `(lambda () ,@run-on-apply))))
  121. ,@(when run-on-remove
  122. (list :on-remove-lambda (eval `(lambda () ,@run-on-remove)))))
  123. *buffer-layer-definitions* :key #'car)
  124. (cl-pushnew ',name *buffer-layers*)
  125. ',name))
  126. (defun buffer-layers-load-layer (name)
  127. (interactive (list (completing-read "Layer Name: " *buffer-layers* ;; (cl-remove-if #'(lambda (layer) (member layer *buffer-layers-applied*)) *buffer-layers*)
  128. nil t)))
  129. (let* ((record (rest (assoc name *buffer-layer-definitions*)))
  130. (files (rest (assoc :files record)))
  131. (on-apply (rest (assoc :on-apply-lambda record)))
  132. (buffer-select (rest (assoc :select-buffer record))))
  133. (let ((buffer-list nil))
  134. (dolist (file files)
  135. (cl-pushnew (find-file file) buffer-list))
  136. (add-to-list '*buffer-layer-buffers* `(,name ,@buffer-list)))
  137. (when (functionp on-apply)
  138. (funcall on-apply))
  139. (when buffer-select
  140. (switch-to-buffer buffer-select))
  141. (message "Applied Buffer Layer %s." name)))
  142. (defun buffer-layers-load-buffer-layer (name-or-path load-it-p)
  143. "Load a buffer named NAME-OR-PATH, and if a file, apply if LOAD-IT-P is true."
  144. (interactive (list (completing-read "Buffer Layer Name or Path: " (cl-remove-if #'(lambda (layer)
  145. (member layer *buffer-layers-applied*))
  146. *buffer-layers*))
  147. nil))
  148. (if (functionp (buffer-layers--applier-name name-or-path))
  149. (funcall (buffer-layers--applier-name name-or-path))
  150. (load name-or-path)
  151. (when load-it-p
  152. (funcall current-buffer-applier))))
  153. (defalias 'load-buffer-layer 'buffer-layers-load-buffer-layer)
  154. (defun buffer-layers-unload-buffer-layer (name)
  155. "Unload Buffer Layer named NAME."
  156. (interactive (list (completing-read "Buffer Layer Name: " *buffer-layers-applied*)))
  157. (funcall (buffer-layers--remover-name name)))
  158. (defun buffer-layers-list ()
  159. "Produce a list of defined buffer layers."
  160. (interactive)
  161. (when (buffer-live-p "*Buffer Layers*")
  162. (kill-buffer "*Buffer Layers*"))
  163. (with-help-window "*Buffer Layers*"
  164. (with-current-buffer "*Buffer Layers*"
  165. (insert "Defined Buffer Layers:\n\n")
  166. (dolist (layer *buffer-layers*)
  167. (if (not (buffer-layers-applied-p layer))
  168. (insert (format " - %s\n" layer))
  169. (insert (format " - %s (Applied)\n" layer)))
  170. (dolist (buffer (symbol-value (buffer-layers--buffer-list-name layer)))
  171. (if (null (get-buffer-window-list buffer nil t))
  172. (insert (format " - %s\n" (buffer-name buffer)))
  173. (insert (format " - %s (visible)\n" (buffer-name buffer)))))))))
  174. (defun buffer-layers-unload-all-buffer-layers ()
  175. "Unload all loaded buffer layers."
  176. (interactive)
  177. (dolist (buffer-layer *buffer-layers-applied*)
  178. (buffer-layers-unload-buffer-layer buffer-layer)))
  179. (defun buffer-layers-create-layer (name)
  180. "Create a new layer."
  181. (interactive "SNew Layer Name: "))
  182. (defun buffer-layers-add-file-to-layer (name file)
  183. "Add a file to the layer."
  184. (interactive (list
  185. (completing-read "Layer: " *buffer-layers* nil t)
  186. (read-file-name "File Name: "))))
  187. (defun buffer-layers-add-buffer-to-layer (name buffer)
  188. "Add a buffer to the given layer."
  189. (interactive (list
  190. (completing-read "Layer: " *buffer-layers* nil t)
  191. (read-buffer "Buffer: " (current-buffer)))))
  192. (defun buffer-layers-edit-load-actions (layer)
  193. "Edit the actions to be preformed on buffer layer load."
  194. (interactive (list (completing-read "Layer: " *buffer-layers* nil t))))
  195. (defun buffer-layers-edit-remove-actions (layer)
  196. "Edit the actions to be preformed on buffer layer removal."
  197. (interactive (list (completing-read "Layer: " *buffer-layers* nil t))))
  198. (defun buffer-layers-set-buffer-to-select (layer)
  199. "Set the buffer to automatically select."
  200. (interactive (list (completing-read "Layer: " *buffer-layers* nil t))))
  201. (defun buffer-layers-save ()
  202. "Save defined buffer layers."
  203. (interactive))
  204. (defvar buffer-layers-map (make-keymap)
  205. "Keymap for buffer-layer commands.")
  206. (define-key buffer-layers-map (kbd "l") #'buffer-layers-load-buffer-layer)
  207. (define-key buffer-layers-map (kbd "L") #'buffer-layers-list)
  208. (define-key buffer-layers-map (kbd "u") #'buffer-layers-unload-buffer-layer)
  209. (define-key buffer-layers-map (kbd "U") #'buffer-layers-unload-all-buffer-layers)
  210. (define-key buffer-layers-map (kbd "c") #'buffer-layers-create-layer)
  211. (define-key buffer-layers-map (kbd "f") #'buffer-layers-add-file-to-layer)
  212. (define-key buffer-layers-map (kbd "b") #'buffer-layers-add-buffer-to-layer)
  213. (define-key buffer-layers-map (kbd "a") #'buffer-layers-edit-load-actions)
  214. (define-key buffer-layers-map (kbd "r") #'buffer-layers-edit-remove-actions)
  215. (define-key buffer-layers-map (kbd "s") #'buffer-layers-set-buffer-to-select)
  216. (define-key buffer-layers-map (kbd "C-s") #'buffer-layers-save)
  217. (define-minor-mode buffer-layers-mode
  218. "A mode for managing layers of buffers."
  219. :lighter " BLM" :global t :variable buffer-layers-mode-p
  220. (if buffer-layers-mode-p
  221. (progn
  222. (define-key ctl-x-map (kbd "L") buffer-layers-map)
  223. (add-hook 'kill-emacs-hook #'buffer-layers-unload-all-buffer-layers))
  224. (progn
  225. (define-key ctl-x-map (kbd "L") nil)
  226. (remove-hook 'kill-emacs-hook #'buffer-layers-unload-all-buffer-layers))))
  227. (provide 'buffer-layers)
  228. ;;; buffer-layers.el ends here