;;; bookmark.el --- Saved files and directories on my filesystem -*- lexical-binding: t -*- ;; Author: William Carroll ;;; Commentary: ;; After enjoying and relying on Emacs's builtin `jump-to-register' command, I'd ;; like to recreate this functionality with a few extensions. ;; ;; Everything herein will mimmick my previous KBDs for `jump-to-register', which ;; were -j-. If the `bookmark-path' is a file, Emacs will ;; open a buffer with that file. If the `bookmark-path' is a directory, Emacs ;; will open an ivy window searching that directory. ;;; Code: ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Dependencies ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (require 'f) (require 'buffer) (require 'list) (require 'string) (require 'set) (require 'constants) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; Constants ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (cl-defstruct bookmark label path kbd) (defconst bookmark/install-kbds? t "When t, install keybindings.") ;; TODO: Consider hosting this function somewhere other than here, since it ;; feels useful above of the context of bookmarks. ;; TODO: Assess whether it'd be better to use the existing function: ;; `counsel-projectile-switch-project-action'. See the noise I made on GH for ;; more context: https://github.com/ericdanan/counsel-projectile/issues/137 (defun bookmark/handle-directory-dwim (path) "Open PATH as either a project directory or a regular directory. If PATH is `projectile-project-p', open with `counsel-projectile-find-file'. Otherwise, open with `counsel-find-file'." (if (projectile-project-p path) (with-temp-buffer (cd (projectile-project-p path)) (call-interactively #'counsel-projectile-find-file)) (let ((ivy-extra-directories nil)) (counsel-find-file path)))) (defconst bookmark/handle-directory #'bookmark/handle-directory-dwim "Function to call when a bookmark points to a directory.") (defconst bookmark/handle-file #'counsel-find-file-action "Function to call when a bookmark points to a file.") (defconst bookmark/whitelist (list (make-bookmark :label "briefcase" :path constants/briefcase :kbd "b") (make-bookmark :label "current project" :path constants/current-project :kbd "p")) "List of registered bookmarks.") ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;; API ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (defun bookmark/open (b) "Open bookmark, B, in a new buffer or an ivy minibuffer." (let ((path (bookmark-path b))) (cond ((f-directory? path) (funcall bookmark/handle-directory path)) ((f-file? path) (funcall bookmark/handle-file path))))) (when bookmark/install-kbds? (->> bookmark/whitelist (list/map (lambda (b) (general-define-key :prefix "" :states '(normal) (string-concat "j" (bookmark-kbd b)) ;; TODO: Consider `cl-labels' so `which-key' minibuffer is more ;; helpful. (lambda () (interactive) (bookmark/open b))))))) (provide 'bookmark) ;;; bookmark.el ends here