core-projects: add def-project-mode! macro & doom-project-hook
This commit is contained in:
parent
c037c325a1
commit
cc6dee6c99
1 changed files with 75 additions and 11 deletions
|
@ -4,6 +4,10 @@
|
||||||
;; digging through project files and exposing an API I can use to make other
|
;; digging through project files and exposing an API I can use to make other
|
||||||
;; plugins/features project-aware.
|
;; plugins/features project-aware.
|
||||||
|
|
||||||
|
(defvar doom-project-hook nil
|
||||||
|
"Hook run when a project is enabled. The name of the project's mode and its
|
||||||
|
state are passed in.")
|
||||||
|
|
||||||
(def-package! projectile :demand t
|
(def-package! projectile :demand t
|
||||||
:init
|
:init
|
||||||
(setq projectile-cache-file (concat doom-cache-dir "projectile.cache")
|
(setq projectile-cache-file (concat doom-cache-dir "projectile.cache")
|
||||||
|
@ -55,17 +59,77 @@
|
||||||
(let ((projectile-require-project-root strict-p))
|
(let ((projectile-require-project-root strict-p))
|
||||||
(ignore-errors (projectile-project-root))))
|
(ignore-errors (projectile-project-root))))
|
||||||
|
|
||||||
(defun doom-project-has-files (files &optional root)
|
(defmacro doom-project-has! (files)
|
||||||
"Return non-nil if FILES exist in the project root."
|
"Checks if the project has the specified FILES, relative to the project root,
|
||||||
(let ((root (or root (doom-project-root)))
|
unless the path begins with ./ or ../, in which case it's relative to
|
||||||
(files (if (listp files) files (list files)))
|
`default-directory'. Recognizes (and ...) and/or (or ...) forms."
|
||||||
(found-p (if files t)))
|
(doom--resolve-paths files (doom-project-root)))
|
||||||
(while (and found-p files)
|
|
||||||
(let ((file (expand-file-name (pop files) root)))
|
|
||||||
(setq found-p (if (string-suffix-p "/" file)
|
;;
|
||||||
(file-directory-p file)
|
;; Projects
|
||||||
(file-exists-p file)))))
|
;;
|
||||||
found-p))
|
|
||||||
|
(defvar-local doom-project nil
|
||||||
|
"A list of project mode symbols to enable. Used for .dir-locals.el.")
|
||||||
|
|
||||||
|
(defun doom|autoload-project-mode ()
|
||||||
|
"Auto-enable projects listed in `doom-project', which is meant to be set from
|
||||||
|
.dir-locals.el files."
|
||||||
|
(dolist (mode doom-project)
|
||||||
|
(funcall mode)))
|
||||||
|
(add-hook 'after-change-major-mode-hook 'doom|autoload-project-mode)
|
||||||
|
|
||||||
|
(defmacro def-project-mode! (name &rest plist)
|
||||||
|
"Define a project minor-mode named NAME, and declare where and how it is
|
||||||
|
activated. Project modes allow for project-specific settings, keymaps, hooks &
|
||||||
|
custom configuration without having to litter the project with .dir-locals.el
|
||||||
|
files.
|
||||||
|
|
||||||
|
This creates NAME-hook and NAME-map as well.
|
||||||
|
|
||||||
|
A project can be enabled through .dir-locals.el however, if `doom-project' is set
|
||||||
|
to the name of the project mode(s) to enable.
|
||||||
|
|
||||||
|
PLIST should contain any or all of these properties, which each are checked to
|
||||||
|
see if NAME should be activated.
|
||||||
|
|
||||||
|
:modes MODES -- if buffers are derived from MODES (one or a list of symbols).
|
||||||
|
|
||||||
|
:files FILES -- if project contains FILES (relative to project root); takes a
|
||||||
|
solitary string or a form comprised of (and ...) and/or (or ...) forms.
|
||||||
|
|
||||||
|
:match REGEXP -- if file name matches REGEXP
|
||||||
|
|
||||||
|
:when PREDICATE -- if PREDICATE returns true (can be a form or the symbol of a
|
||||||
|
function)
|
||||||
|
|
||||||
|
:init FORM -- FORM will be run the first time this project mode is enabled."
|
||||||
|
(declare (indent 1))
|
||||||
|
(let ((modes (plist-get plist :modes))
|
||||||
|
(files (plist-get plist :files))
|
||||||
|
(when (plist-get plist :when))
|
||||||
|
(match (plist-get plist :match))
|
||||||
|
(init-form (plist-get plist :init))
|
||||||
|
(keymap-sym (intern (format "%s-map" name))))
|
||||||
|
`(progn
|
||||||
|
(defvar ,keymap-sym (make-sparse-keymap)
|
||||||
|
,(concat "Keymap for `" (symbol-name name) "'"))
|
||||||
|
(define-minor-mode ,name
|
||||||
|
"A project minor mode."
|
||||||
|
:init-value nil
|
||||||
|
:keymap ,keymap-sym)
|
||||||
|
,(when (or modes match files when)
|
||||||
|
`(associate! ,name
|
||||||
|
:modes ,modes
|
||||||
|
:match ,match
|
||||||
|
:files ,files
|
||||||
|
:when ,when))
|
||||||
|
(add-hook! ,name
|
||||||
|
(run-hook-with-args doom-project-hook ',name))
|
||||||
|
,(when init-form
|
||||||
|
`(add-transient-hook! ',(intern-soft (format "%s-hook" name))
|
||||||
|
,init-form)))))
|
||||||
|
|
||||||
(provide 'core-projects)
|
(provide 'core-projects)
|
||||||
;;; core-projects.el ends here
|
;;; core-projects.el ends here
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue