2017-01-16 23:15:48 -05:00
|
|
|
;;; core-packages.el
|
|
|
|
|
2017-01-31 19:42:41 -05:00
|
|
|
;; Emacs package management is opinionated. Unfortunately, so am I. So I
|
|
|
|
;; combined `use-package`, quelpa and package.el to manage my plugins.
|
2017-01-16 23:15:48 -05:00
|
|
|
;;
|
2017-01-28 02:02:16 -05:00
|
|
|
;; Why all the trouble? Because:
|
|
|
|
;; 1. Scriptability: I want my plugins managable from the command line (as well
|
2017-01-31 19:42:41 -05:00
|
|
|
;; as an `doom/packages-update' command within emacs to update my plugins
|
|
|
|
;; automatically, rather than through package.el's interface).
|
2017-01-28 02:02:16 -05:00
|
|
|
;; 2. Flexibility: I want to install packages from sources other than ELPA
|
|
|
|
;; repositories. Such as github or the Emacs wiki. Some plugins are out of
|
|
|
|
;; date through official channels, have changed hands unofficially, or simply
|
|
|
|
;; haven't been submitted to an ELPA repo yet.
|
2017-02-01 00:29:39 -05:00
|
|
|
;; 3. Stability: I don't want to worry that each time I use my package
|
|
|
|
;; manager something might inexplicably go wrong. This was the case with
|
|
|
|
;; Cask, which I used previously. package.el and quelpa appear to be much
|
|
|
|
;; more stable.
|
2017-01-28 02:02:16 -05:00
|
|
|
;; 4. No external dependencies (e.g. Cask) for plugin management.
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defvar doom-enabled-modules nil
|
2017-02-01 00:29:39 -05:00
|
|
|
"List of enabled modules; each element is a cons cell (MODULE . SUBMODULE),
|
|
|
|
where MODULE is the module's property symbol, e.g. :lang, and SUBMODULE is the
|
|
|
|
submodule symbol, e.g. 'evil.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defvar doom-packages nil
|
|
|
|
"A list of enabled packages.")
|
|
|
|
|
|
|
|
(defvar doom-protected-packages '(quelpa-use-package f s dash)
|
|
|
|
"A list of packages that shouldn't be deleted.")
|
|
|
|
|
2017-02-02 19:15:43 -05:00
|
|
|
(defvar doom-installed-packages nil
|
|
|
|
"A list of packages that were installed during the current session.")
|
|
|
|
|
2017-02-01 00:29:39 -05:00
|
|
|
(defvar doom--init nil
|
|
|
|
"Non-nil if doom's package system has been initialized or not. It may not be
|
|
|
|
if you have byte-compiled your configuration (as intended).")
|
2017-01-31 04:31:14 -05:00
|
|
|
|
2017-02-01 00:29:39 -05:00
|
|
|
(defvar doom--auto-install-p nil
|
|
|
|
"If non-nil, install missing packages. Otherwise, strip :ensure and :quelpa
|
|
|
|
from `package!' calls.")
|
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defvar doom--prefer-el-p (or noninteractive doom--auto-install-p)
|
|
|
|
"If non-nil, load uncompiled .el config files.")
|
2017-01-31 18:47:59 -05:00
|
|
|
|
2017-01-31 04:31:14 -05:00
|
|
|
(defvar doom--load-path (append (list doom-core-dir
|
2017-02-01 00:29:39 -05:00
|
|
|
doom-modules-dir)
|
2017-01-31 04:31:14 -05:00
|
|
|
load-path)
|
|
|
|
"A backup of `load-path', used as a bare-bones foundation for
|
|
|
|
`doom/packages-reload' or `doom-initialize'.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
(setq load-prefer-newer nil
|
|
|
|
package--init-file-ensured t
|
|
|
|
package-user-dir (expand-file-name "elpa" doom-packages-dir)
|
|
|
|
package-enable-at-startup nil
|
|
|
|
package-archives
|
|
|
|
'(("gnu" . "http://elpa.gnu.org/packages/")
|
|
|
|
("melpa" . "http://melpa.org/packages/")
|
|
|
|
("org" . "http://orgmode.org/elpa/"))
|
|
|
|
|
|
|
|
use-package-always-defer t
|
2017-01-28 02:02:16 -05:00
|
|
|
use-package-always-ensure nil
|
2017-02-02 04:37:59 -05:00
|
|
|
use-package-expand-minimally t
|
2017-01-28 02:02:16 -05:00
|
|
|
use-package-debug doom-debug-mode
|
2017-01-31 04:31:14 -05:00
|
|
|
use-package-verbose doom-debug-mode
|
2017-01-16 23:15:48 -05:00
|
|
|
quelpa-checkout-melpa-p nil
|
|
|
|
quelpa-update-melpa-p nil
|
2017-01-28 02:02:16 -05:00
|
|
|
quelpa-use-package-inhibit-loading-quelpa t
|
2017-01-31 18:59:58 -05:00
|
|
|
quelpa-dir (expand-file-name "quelpa" doom-packages-dir)
|
|
|
|
;; ssh, no tears. Only compiling.
|
|
|
|
byte-compile-warnings
|
|
|
|
'(unresolved callargs obsolete noruntime cl-functions make-local constants suspicious))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
|
|
|
|
;;
|
2017-01-31 04:31:14 -05:00
|
|
|
;; Bootstrap function
|
2017-01-16 23:15:48 -05:00
|
|
|
;;
|
|
|
|
|
2017-01-31 18:47:59 -05:00
|
|
|
(defmacro doom! (&rest packages)
|
|
|
|
"DOOM Emacs bootstrap macro. List the modules to load. Benefits from
|
|
|
|
byte-compilation."
|
|
|
|
(let (mode)
|
|
|
|
(dolist (p packages)
|
|
|
|
(cond ((string-prefix-p ":" (symbol-name p))
|
|
|
|
(setq mode p))
|
|
|
|
((not mode)
|
|
|
|
(error "No namespace specified on `doom!' for %s" p))
|
|
|
|
(t
|
2017-02-02 04:37:59 -05:00
|
|
|
(setq doom-enabled-modules (append doom-enabled-modules (list (cons mode p))))))))
|
|
|
|
`(unless noninteractive
|
|
|
|
(let (file-name-handler-alist)
|
2017-02-02 19:16:12 -05:00
|
|
|
,@(mapcar (lambda (pkg) `(load! ,(car pkg) ,(cdr pkg)))
|
2017-02-02 04:37:59 -05:00
|
|
|
doom-enabled-modules)
|
2017-01-31 18:47:59 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(when (display-graphic-p)
|
|
|
|
(require 'server)
|
|
|
|
(unless (server-running-p)
|
|
|
|
(server-start)))
|
2017-01-31 18:47:59 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
;; Prevent any auto-displayed text + benchmarking
|
|
|
|
(advice-add 'display-startup-echo-area-message :override 'ignore)
|
|
|
|
(message "Loaded %s packages in %s"
|
|
|
|
(- (length load-path) (length doom--load-path))
|
|
|
|
(emacs-init-time)))))
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
(defun doom-initialize (&optional force-p)
|
2017-02-02 04:37:59 -05:00
|
|
|
"Initialize installed packages (using package.el) and ensure the core packages
|
|
|
|
are installed. If you byte compile core/core.el, calls to `package.el' are
|
|
|
|
avoided to speed up startup."
|
2017-01-31 04:31:14 -05:00
|
|
|
(unless (or doom--init force-p)
|
|
|
|
(setq load-path doom--load-path
|
|
|
|
package-activated-list nil)
|
2017-01-16 23:15:48 -05:00
|
|
|
(package-initialize)
|
2017-02-02 04:37:59 -05:00
|
|
|
(unless (and (file-exists-p doom-packages-dir)
|
|
|
|
(require 'quelpa-use-package nil t))
|
2017-01-31 04:31:14 -05:00
|
|
|
(package-refresh-contents)
|
2017-02-02 04:37:59 -05:00
|
|
|
;; Ensure core packages are installed
|
|
|
|
(mapc 'package-install '(quelpa-use-package dash f s)))
|
|
|
|
(unless (require 'quelpa-use-package nil t)
|
|
|
|
(delete-directory doom-packages-dir t)
|
|
|
|
(error "There was an error initializing DOOM. Try running it again"))
|
|
|
|
(quelpa-use-package-activate-advice)
|
|
|
|
;; Move :ensure to after conditional properties
|
|
|
|
(delq :ensure use-package-keywords)
|
|
|
|
(push :ensure (cdr (memq :unless use-package-keywords)))
|
2017-01-31 04:31:14 -05:00
|
|
|
(setq doom--init t)))
|
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defun doom-reload-modules ()
|
|
|
|
"Reload `doom-modules'."
|
|
|
|
(setq doom-modules nil)
|
|
|
|
(let ((doom--prefer-el-p t)
|
|
|
|
(noninteractive t))
|
|
|
|
(load (concat doom-emacs-dir "init.el") nil :nomessage :nosuffix)))
|
|
|
|
|
|
|
|
(defun doom-reload-packages (&optional install-p)
|
|
|
|
"Reload `doom-packages'."
|
|
|
|
(doom-initialize)
|
|
|
|
(doom-reload-modules)
|
|
|
|
(let ((before-packages-n (length package-alist))
|
|
|
|
(doom--auto-install-p (and install-p t))
|
|
|
|
(doom--prefer-el-p t)
|
|
|
|
(after-packages-n 0)
|
|
|
|
noninteractive)
|
|
|
|
(when doom--auto-install-p
|
|
|
|
(package-refresh-contents))
|
|
|
|
(load (f-expand "core.el" doom-core-dir) nil (not doom-debug-mode) :nosuffix)
|
|
|
|
(mapc (lambda (pkg)
|
|
|
|
(let ((path (f-expand "packages.el" (doom-module-path (car pkg) (cdr pkg)))))
|
|
|
|
(when (f-exists-p path)
|
|
|
|
(load path nil (not doom-debug-mode) :nosuffix))))
|
|
|
|
doom-enabled-modules)
|
|
|
|
(- (length package-alist) before-packages-n)))
|
|
|
|
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
;;
|
|
|
|
;; Macros
|
|
|
|
;;
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defvar __DIR__ nil "The directory of the currently loaded file (set by `load!')")
|
|
|
|
(defvar __FILE__ nil "The full path of the currently loaded file (set by `load!')")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defmacro use-package! (name &rest plist)
|
|
|
|
"A `use-package' wrapper, to adhere to the naming conventions of DOOM emacs
|
|
|
|
and let-bind `package-name' for the containing forms. Note that packages are
|
|
|
|
deferred by default."
|
2017-02-02 19:16:12 -05:00
|
|
|
`(let ((package-name ',name))
|
|
|
|
(use-package ,name ,@plist)))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-01 00:29:39 -05:00
|
|
|
(defmacro package! (name &rest plist)
|
|
|
|
"Wraps around `use-package' (with `quelpa-use-package') and takes the same
|
|
|
|
arguments. Ensures the package named NAME is installed and available. If
|
|
|
|
`doom--auto-install-p' is nil, then strip out :ensure and :quelpa properties,
|
2017-02-02 04:37:59 -05:00
|
|
|
which is the case if you've byte-compiled DOOM Emacs.
|
|
|
|
|
|
|
|
Note that packages are deferred by default."
|
2017-01-16 23:15:48 -05:00
|
|
|
(declare (indent defun))
|
2017-02-01 00:29:39 -05:00
|
|
|
(let ((use-package-always-ensure doom--auto-install-p)
|
2017-01-31 19:41:07 -05:00
|
|
|
(recipe (plist-get plist :quelpa)))
|
|
|
|
;; prepend NAME to quelpa recipe, if none is specified, to avoid local
|
|
|
|
;; MELPA lookups by quelpa.
|
|
|
|
(when (and recipe (= 0 (mod (length recipe) 2)))
|
|
|
|
(push name recipe)
|
2017-02-02 04:37:59 -05:00
|
|
|
(setq plist (plist-put plist :quelpa recipe)))
|
2017-02-02 19:15:43 -05:00
|
|
|
(if doom--auto-install-p
|
|
|
|
(unless (package-installed-p name)
|
|
|
|
(pushnew name doom-installed-packages))
|
2017-01-31 19:41:07 -05:00
|
|
|
(setq plist (use-package-plist-delete plist :ensure))
|
|
|
|
(setq plist (use-package-plist-delete plist :quelpa)))
|
2017-02-02 04:37:59 -05:00
|
|
|
;; (package--save-selected-packages (cons name package-selected-packages))
|
|
|
|
(pushnew name doom-packages)
|
|
|
|
(macroexpand `(use-package ,name ,@plist))))
|
2017-01-31 04:31:14 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defmacro load! (file-or-module-sym &optional submodule file)
|
2017-01-28 02:02:16 -05:00
|
|
|
"Load a module from `doom-modules-dir'. Plays the same role as
|
2017-02-02 04:37:59 -05:00
|
|
|
`load-relative', but is specific to DOOM emacs modules and submodules. If
|
|
|
|
`doom--prefer-el-p' is non-nil, prefer the un-compiled elisp file.
|
2017-01-28 02:02:16 -05:00
|
|
|
|
|
|
|
Examples:
|
2017-02-02 04:37:59 -05:00
|
|
|
(load! :lang emacs-lisp)
|
2017-02-01 00:29:39 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
Loads modules/lang/emacs-lisp/FILE.el (defaults to config.el).
|
2017-02-01 00:29:39 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(load! +local-module)
|
2017-01-31 04:31:14 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
Loads +local-module.el relative to `__DIR__' or `doom-core-dir'."
|
2017-01-31 19:42:11 -05:00
|
|
|
(let (path file)
|
2017-01-31 04:31:14 -05:00
|
|
|
(cond ((null submodule)
|
2017-02-02 04:37:59 -05:00
|
|
|
(setq path (or __DIR__ doom-core-dir)
|
|
|
|
file (concat (if (symbolp file-or-module-sym)
|
|
|
|
(symbol-name file-or-module-sym)
|
|
|
|
file-or-module-sym)
|
|
|
|
".el")))
|
2017-01-31 04:31:14 -05:00
|
|
|
(t
|
2017-02-02 04:37:59 -05:00
|
|
|
(setq path (doom-module-path file-or-module-sym submodule)
|
|
|
|
file (or file "config.el"))))
|
2017-01-31 04:31:14 -05:00
|
|
|
(setq path (f-slash path)
|
|
|
|
file (concat path file))
|
2017-02-02 04:37:59 -05:00
|
|
|
`(let ((__FILE__ ,file)
|
|
|
|
(__DIR__ ,path))
|
|
|
|
(load (if doom--prefer-el-p ,file ,(f-no-ext file))
|
|
|
|
nil (not doom-debug-mode) doom--prefer-el-p))))
|
2017-01-28 02:02:16 -05:00
|
|
|
|
2017-01-31 19:42:11 -05:00
|
|
|
(defun doom-module-path (module submodule &optional file)
|
2017-02-02 04:37:59 -05:00
|
|
|
"Get the full path to a module: e.g. :lang emacs-lisp maps to
|
|
|
|
~/.emacs.d/modules/lang/emacs-lisp/. Will append FILE if non-nil."
|
2017-02-02 19:17:02 -05:00
|
|
|
(setq module
|
|
|
|
(cond ((keywordp module) (substring (symbol-name module) 1))
|
|
|
|
((symbolp module) (symbol-name module))
|
|
|
|
((stringp module) module)
|
|
|
|
(t (error "Not a valid module name: %s" module))))
|
2017-02-02 04:37:59 -05:00
|
|
|
(when (symbolp submodule)
|
|
|
|
(setq submodule (symbol-name submodule)))
|
|
|
|
(f-expand (concat module "/" submodule "/" file)
|
2017-01-31 19:42:11 -05:00
|
|
|
doom-modules-dir))
|
|
|
|
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
;;
|
2017-02-02 04:37:59 -05:00
|
|
|
;; Defuns
|
2017-01-31 04:31:14 -05:00
|
|
|
;;
|
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defun doom/refresh-autoloads ()
|
|
|
|
"Refreshes the autoloads.el file, which tells Emacs where to find all the
|
|
|
|
autoloaded functions in the modules you use or among the core libraries.
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
Rerun this whenever you modify your init.el (or use `make autoloads` from the
|
|
|
|
command line)."
|
2017-01-16 23:15:48 -05:00
|
|
|
(interactive)
|
2017-02-02 04:37:59 -05:00
|
|
|
(unless doom--init
|
|
|
|
(doom-reload-modules))
|
|
|
|
(let ((generated-autoload-file (concat doom-local-dir "autoloads.el"))
|
|
|
|
(autoload-files
|
|
|
|
(append (-flatten (mapcar (lambda (dir)
|
|
|
|
(let ((auto-dir (f-expand "autoload" dir))
|
|
|
|
(auto-file (f-expand "autoload.el" dir)))
|
|
|
|
(cond ((f-directory-p auto-dir)
|
|
|
|
(f-glob "*.el" auto-dir))
|
|
|
|
((f-exists-p auto-file)
|
|
|
|
auto-file))))
|
|
|
|
(--map (doom-module-path (car it) (cdr it)) doom-enabled-modules)))
|
|
|
|
(f-glob "autoload/*.el" doom-core-dir))))
|
|
|
|
(when (f-exists-p generated-autoload-file)
|
|
|
|
(f-delete generated-autoload-file)
|
|
|
|
(message "Deleted old autoloads.el"))
|
|
|
|
(dolist (file autoload-files)
|
|
|
|
(update-file-autoloads file)
|
|
|
|
(message "Detected: %s" (f-relative file doom-emacs-dir)))
|
|
|
|
(with-current-buffer (get-file-buffer generated-autoload-file)
|
|
|
|
(save-buffer))
|
|
|
|
(load generated-autoload-file nil t t)
|
|
|
|
(message "Done!")))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-01-31 19:44:31 -05:00
|
|
|
(defun doom/byte-compile ()
|
|
|
|
"Byte (re)compile the important files in your emacs configuration (i.e.
|
|
|
|
init.el, core/*.el and modules/*/*/config.el) DOOM Emacs was designed to benefit
|
|
|
|
a lot from this."
|
2017-01-16 23:15:48 -05:00
|
|
|
(interactive)
|
2017-01-31 19:44:31 -05:00
|
|
|
(doom-initialize)
|
2017-02-02 04:37:59 -05:00
|
|
|
(doom-reload-modules)
|
2017-01-31 19:44:31 -05:00
|
|
|
(let ((targets (append (list (f-expand "init.el" doom-emacs-dir)
|
|
|
|
(f-expand "core.el" doom-core-dir))
|
|
|
|
(reverse (f-glob "core-*.el" doom-core-dir))
|
|
|
|
(-filter 'f-exists-p
|
|
|
|
(--map (doom-module-path (car it) (cdr it) "config.el")
|
2017-02-02 04:37:59 -05:00
|
|
|
doom-enabled-modules))))
|
|
|
|
(n 0)
|
|
|
|
results
|
2017-01-31 19:44:31 -05:00
|
|
|
use-package-always-ensure
|
2017-01-31 04:31:14 -05:00
|
|
|
file-name-handler-alist)
|
2017-02-02 04:37:59 -05:00
|
|
|
(mapc (lambda (file)
|
|
|
|
(push (cons (f-relative file doom-emacs-dir)
|
|
|
|
(when (byte-compile-file file)
|
|
|
|
(setq n (1+ n))
|
|
|
|
t))
|
|
|
|
results))
|
|
|
|
targets)
|
2017-01-31 19:44:31 -05:00
|
|
|
(when noninteractive
|
|
|
|
(when targets (message "\n"))
|
2017-02-02 04:37:59 -05:00
|
|
|
(message "Compiling %s files:\n%s" n
|
|
|
|
(mapconcat (lambda (file) (concat "+ " (if (cdr file) "SUCCESS" "FAIL") ": " (car file)))
|
|
|
|
(reverse results) "\n")))))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
(provide 'core-packages)
|
|
|
|
;;; core-packages.el ends here
|