slackbuilds/magit-key-mode.el

455 lines
17 KiB
EmacsLisp
Raw Normal View History

2010-09-02 19:22:13 +02:00
(require 'assoc)
(defvar magit-key-mode-key-maps '()
"This will be filled lazily with proper `define-key' built
keymaps as they're reqeusted.")
2010-08-31 09:42:21 +02:00
(defvar magit-key-mode-buf-name "*magit-key*"
"Name of the buffer.")
(defvar magit-key-mode-current-args '()
"Will contain the arguments to be passed to git.")
(defvar magit-key-mode-current-options '()
"Will contain the arguments to be passed to git.")
(defvar magit-log-mode-window-conf nil
"Will hold the pre-menu configuration of magit.")
(defvar magit-key-mode-groups
'((logging
2010-09-17 10:21:02 +02:00
(man-page "git-log")
(actions
2010-09-02 00:53:23 +02:00
("l" "Short" magit-log)
("L" "Long" magit-log-long)
("h" "Reflog" magit-reflog)
2010-08-31 09:42:21 +02:00
("H" "Reflog on head" magit-reflog-head))
(switches
2010-09-01 00:51:58 +02:00
("-m" "Only merge commits" "--merges")
2010-08-31 13:16:23 +02:00
("-f" "First parent" "--first-parent")
("-i" "Case insesnitive patterns" "-i")
2010-09-15 22:28:01 +02:00
("-pr" "Pickaxe regex" "--pickaxe-regex")
("-n" "Name only" "--name-only")
("-am" "All match" "--all-match")
2010-09-20 20:20:14 +02:00
("-al" "All" "--all"))
2010-08-31 19:53:37 +02:00
(arguments
2010-09-15 22:28:01 +02:00
("=r" "Relative" "--relative=" read-directory-name)
2010-09-23 17:45:25 +02:00
("=c" "Committer" "--committer=" read-from-minibuffer)
2010-09-15 22:28:01 +02:00
("=>" "Since" "--since=" read-from-minibuffer)
("=<" "Before" "--before=" read-from-minibuffer)
2010-09-15 14:52:10 +02:00
("=s" "Pickaxe search" "-S" read-from-minibuffer)
("=a" "Author" "--author=" read-from-minibuffer)
("=g" "Grep" "--grep=" read-from-minibuffer)))
2010-09-01 11:17:21 +02:00
2010-09-07 20:35:16 +02:00
(running
(actions
("!" "Command from root" magit-shell-command)
(":" "Git command" magit-git-command)))
2010-09-02 01:00:42 +02:00
(fetching
2010-09-17 15:41:35 +02:00
(man-page "git-fetch")
2010-09-02 01:00:42 +02:00
(actions
("f" "Current" magit-fetch-current)
("a" "All" magit-remote-update)
("o" "Other" magit-fetch)))
2010-09-02 01:00:42 +02:00
2010-09-01 23:28:33 +02:00
(pushing
2010-09-17 15:41:35 +02:00
(man-page "git-push")
2010-09-01 23:28:33 +02:00
(actions
2010-09-28 18:00:19 +02:00
("p" "Push" magit-push)
("t" "Push tags" magit-push-tags))
(switches
2010-09-17 15:33:30 +02:00
("-f" "Force" "--force")
("-d" "Dry run" "-n")))
2010-09-01 23:28:33 +02:00
(pulling
2010-09-17 15:41:35 +02:00
(man-page "git-pull")
2010-09-01 23:28:33 +02:00
(actions
("p" "Pull" magit-pull))
(switches
("-r" "Rebase" "--rebase")))
2010-09-01 22:49:58 +02:00
(branching
2010-09-17 15:41:35 +02:00
(man-page "git-branch")
2010-09-01 22:49:58 +02:00
(actions
2010-09-01 23:13:08 +02:00
("V" "Branch manager" magit-show-branches)
("B" "Create" magit-create-branch)
("m" "Move" magit-move-branch)
("d" "Delete" magit-delete-branch)
2010-09-01 22:49:58 +02:00
("c" "Checkout" magit-checkout)))
2010-09-01 11:45:09 +02:00
(tagging
2010-09-17 15:41:35 +02:00
(man-page "git-tag")
2010-09-01 11:45:09 +02:00
(actions
("t" "Lightweight" magit-tag)
("T" "Annotated" magit-annotated-tag))
(switches
("-f" "Force" "-f")))
2010-09-03 19:50:39 +02:00
(stashing
2010-09-17 15:41:35 +02:00
(man-page "git-stash")
2010-09-03 19:50:39 +02:00
(actions
("s" "Save" magit-stash)
("S" "Snapshot" magit-stash-snapshot))
(switches
("-k" "Keep index" "--keep-index")))
2010-09-03 19:34:37 +02:00
(merging
2010-09-17 15:41:35 +02:00
(man-page "git-merge")
2010-09-03 19:34:37 +02:00
(actions
("m" "Merge" magit-merge))
2010-09-03 19:34:37 +02:00
(switches
("-nf" "No fast-forward" "--no-ff")
("-nc" "No commit" "--no-commit")
2010-09-05 01:27:55 +02:00
("-sq" "Squash" "--squash"))
2010-09-03 19:34:37 +02:00
(arguments
("-st" "Strategy" "--strategy" read-from-minibuffer)))
2010-09-01 11:17:21 +02:00
(rewriting
(actions
("b" "Begin" magit-rewrite-start)
("s" "Stop" magit-rewrite-stop)
("a" "Abort" magit-rewrite-abort)
("f" "Finish" magit-rewrite-finish)
("*" "Set unused" magit-rewrite-set-unused)
("." "Set used" magit-rewrite-set-used))))
2010-08-31 00:06:29 +02:00
"Holds the key, help, function mapping for the log-mode. If you
modify this make sure you reset `magit-key-mode-key-maps' to
nil.")
(defun magit-key-mode-delete-group (group)
"Add a new group (GROUP) to `magit-key-mode-key-maps'."
(let ((items (assoc group magit-key-mode-groups)))
(when items
;; reset the cache
(setq magit-key-mode-key-maps nil)
;; delete the whole group
(setq magit-key-mode-groups
(delq items magit-key-mode-groups))
;; unbind the defun
(magit-key-mode-de-generate group))
magit-key-mode-groups))
(defun magit-key-mode-add-group (group)
"Add a new group to `magit-key-mode-key-maps'. If there's
already a group of that name then this will completely remove it
and put in its place an empty one of the same name."
(when (assoc group magit-key-mode-groups)
(magit-key-mode-delete-group group))
(setq magit-key-mode-groups
(cons (list group '(actions)) magit-key-mode-groups)))
(defun magit-key-mode-key-defined-p (for-group key)
"If KEY is defined as any of switch, argument or action within
FOR-GROUP then return t"
(catch 'result
(let ((options (magit-key-mode-options-for-group for-group)))
(dolist (type '(actions switches arguments))
(when (assoc key (assoc type options))
(throw 'result t))))))
2010-09-10 18:32:57 +02:00
(defun magit-key-mode-update-group (for-group thing &rest args)
"Abstraction for setting values in `magit-key-mode-key-maps'."
(let* ((options (magit-key-mode-options-for-group for-group))
(things (assoc thing options))
(key (car args)))
(if (cdr things)
(if (magit-key-mode-key-defined-p for-group key)
(error "%s is already defined in the %s group." key for-group)
(setcdr (cdr things) (cons args (cddr things))))
(setcdr things (list args)))
(setq magit-key-mode-key-maps nil)
things))
(defun magit-key-mode-insert-argument (for-group key desc arg read-func)
"Add a new binding (KEY) in FOR-GROUP which will use READ-FUNC
to receive input to apply to argument ARG git is run. DESC should
be a brief description of the binding."
(magit-key-mode-update-group for-group 'arguments key desc arg read-func))
(defun magit-key-mode-insert-switch (for-group key desc switch)
"Add a new binding (KEY) in FOR-GROUP which will add SWITCH to git's
commandline when it runs. DESC should be a brief description of
the binding."
(magit-key-mode-update-group for-group 'switches key desc switch))
(defun magit-key-mode-insert-action (for-group key desc func)
"Add a new binding (KEY) in FOR-GROUP which will run command
FUNC. DESC should be a brief description of the binding."
(magit-key-mode-update-group for-group 'actions key desc func))
2010-08-31 00:28:52 +02:00
(defun magit-key-mode-options-for-group (for-group)
2010-09-10 10:30:05 +02:00
"Retrieve the options (switches, commands and arguments) for
the group FOR-GROUP."
2010-08-31 00:28:52 +02:00
(or (cdr (assoc for-group magit-key-mode-groups))
(error "Unknown group '%s'" for-group)))
2010-09-17 10:21:02 +02:00
(defun magit-key-mode-help (for-group)
2010-09-17 19:22:04 +02:00
"Provide help for a key (which the user is prompted for) within
FOR-GROUP."
2010-09-17 10:21:02 +02:00
(let* ((opts (magit-key-mode-options-for-group for-group))
(seq (read-key-sequence "Enter command prefix: "))
(actions (cdr (assoc 'actions opts))))
;; is it an action? If so popup the help for the to-be-run
;; function
(if (assoc seq actions)
(describe-function (nth 2 (assoc seq actions)))
;; otherwise give the user a man page
(man (or (cadr (assoc 'man-page opts))
(error "No help associated with %s" seq))))))
(defun magit-key-mode-exec-at-point ()
"Run action/args/option at point."
(interactive)
(let* ((key (or (get-text-property (point) 'key-group-executor)
(error "Nothing at point to do.")))
(def (lookup-key (current-local-map) key)))
(call-interactively def)))
(defun magit-key-mode-build-keymap (for-group)
"Construct a normal looking keymap for the key mode to use and
put it in magit-key-mode-key-maps for fast lookup."
2010-08-31 00:28:52 +02:00
(let* ((options (magit-key-mode-options-for-group for-group))
(actions (cdr (assoc 'actions options)))
2010-08-31 09:42:21 +02:00
(switches (cdr (assoc 'switches options)))
2010-08-31 19:53:37 +02:00
(arguments (cdr (assoc 'arguments options))))
2010-08-30 23:50:57 +02:00
(let ((map (make-sparse-keymap)))
;; ret dwim
(define-key map (kbd "RET") 'magit-key-mode-exec-at-point)
2010-08-31 00:28:52 +02:00
;; all maps should 'quit' with C-g
(define-key map (kbd "C-g") (lambda ()
(interactive)
2010-09-01 00:41:03 +02:00
(magit-key-mode-command nil)))
;; run help
2010-09-17 10:21:02 +02:00
(define-key map (kbd "?") `(lambda ()
(interactive)
(magit-key-mode-help ',for-group)))
2010-08-30 23:50:57 +02:00
(when actions
(dolist (k actions)
2010-08-31 09:42:21 +02:00
(define-key map (car k) `(lambda ()
(interactive)
2010-09-01 00:34:46 +02:00
(magit-key-mode-command ',(nth 2 k))))))
2010-08-31 09:42:21 +02:00
(when switches
(dolist (k switches)
(define-key map (car k) `(lambda ()
2010-08-31 13:16:23 +02:00
(interactive)
(magit-key-mode-add-option
',for-group
,(nth 2 k))))))
2010-08-31 19:53:37 +02:00
(when arguments
(dolist (k arguments)
(define-key map (car k) `(lambda ()
(interactive)
(magit-key-mode-add-argument
',for-group
2010-08-31 23:04:36 +02:00
,(nth 2 k)
',(nth 3 k))))))
2010-08-31 09:42:21 +02:00
(aput 'magit-key-mode-key-maps for-group map)
2010-08-30 23:50:57 +02:00
map)))
2010-09-01 00:34:46 +02:00
(defun magit-key-mode-command (func)
(let ((args '()))
;; why can't maphash return a list?!
(maphash (lambda (k v)
2010-09-15 14:52:10 +02:00
(push (concat k (shell-quote-argument v)) args))
2010-09-01 00:34:46 +02:00
magit-key-mode-current-args)
(let ((magit-custom-options (append args magit-key-mode-current-options)))
(set-window-configuration magit-log-mode-window-conf)
2010-09-01 00:41:03 +02:00
(when func
2010-09-01 11:17:21 +02:00
(call-interactively func))
2010-09-01 00:34:46 +02:00
(magit-key-mode-kill-buffer))))
2010-08-31 00:45:51 +02:00
2010-08-31 23:04:36 +02:00
(defvar magit-key-mode-current-args nil
"A hash-table of current argument set (which will eventually
make it to the git command-line).")
(defun magit-key-mode-add-argument (for-group arg-name input-func)
(let ((input (funcall input-func (concat arg-name ": "))))
(puthash arg-name input magit-key-mode-current-args)
2010-09-01 00:34:46 +02:00
(magit-key-mode-redraw for-group)))
2010-08-31 23:04:36 +02:00
(defvar magit-key-mode-current-options '()
"Current option set (which will eventually make it to the git
command-line).")
2010-08-31 19:53:37 +02:00
2010-08-31 09:42:21 +02:00
(defun magit-key-mode-add-option (for-group option-name)
"Toggles the appearance of OPTION-NAME in
`magit-key-mode-current-options'."
(if (not (member option-name magit-key-mode-current-options))
(add-to-list 'magit-key-mode-current-options option-name)
(setq magit-key-mode-current-options
(delete option-name magit-key-mode-current-options)))
(magit-key-mode-redraw for-group))
2010-08-31 00:45:51 +02:00
2010-08-31 09:42:21 +02:00
(defun magit-key-mode-kill-buffer ()
2010-08-31 19:15:05 +02:00
(interactive)
2010-08-31 09:42:21 +02:00
(kill-buffer magit-key-mode-buf-name))
(defvar magit-log-mode-window-conf nil
"Pre-popup window configuration.")
2010-08-31 09:42:21 +02:00
(defun magit-key-mode (for-group &optional original-opts)
2010-09-17 19:22:04 +02:00
"Mode for magit key selection. All commands, switches and
options can be toggled/actioned with the key combination
highlighed before the description."
(interactive)
2010-08-31 22:48:06 +02:00
;; save the window config to restore it as was (no need to make this
;; buffer local)
(setq magit-log-mode-window-conf
(current-window-configuration))
;; setup the mode, draw the buffer
2010-08-31 09:42:21 +02:00
(let ((buf (get-buffer-create magit-key-mode-buf-name)))
(delete-other-windows)
(split-window-vertically)
(other-window 1)
(switch-to-buffer buf)
(kill-all-local-variables)
(set (make-local-variable
'magit-key-mode-current-options)
original-opts)
(set (make-local-variable
2010-08-31 23:04:36 +02:00
'magit-key-mode-current-args)
(make-hash-table))
(magit-key-mode-redraw for-group))
2010-09-17 10:21:02 +02:00
(message "Bindings prefixing options action them. '?' for help"))
2010-08-31 09:42:21 +02:00
(defun magit-key-mode-get-key-map (for-group)
"Get or build the keymap for FOR-GROUP."
(or (cdr (assoc for-group magit-key-mode-key-maps))
(magit-key-mode-build-keymap for-group)))
2010-08-31 09:42:21 +02:00
(defun magit-key-mode-redraw (for-group)
2010-08-31 19:38:44 +02:00
"(re)draw the magit key buffer."
(let ((buffer-read-only nil)
(old-point (point)))
2010-08-31 09:42:21 +02:00
(erase-buffer)
(make-local-variable 'font-lock-defaults)
(use-local-map (magit-key-mode-get-key-map for-group))
2010-08-31 09:42:21 +02:00
(magit-key-mode-draw for-group)
2010-09-10 09:53:07 +02:00
(delete-trailing-whitespace)
(setq mode-name "magit-key-mode" major-mode 'magit-key-mode)
(goto-char old-point))
(setq buffer-read-only t)
(fit-window-to-buffer))
2010-09-01 20:32:10 +02:00
(defun magit-key-mode-draw-header (header)
2010-09-01 22:35:09 +02:00
"Draw a header with the correct face."
2010-09-01 20:32:10 +02:00
(insert (propertize header 'face 'font-lock-keyword-face)))
2010-09-15 21:09:57 +02:00
(defvar magit-key-mode-args-in-cols nil
"When true, draw arguments in columns as with switches and
options.")
2010-09-01 20:37:08 +02:00
(defun magit-key-mode-draw-args (args)
2010-09-01 22:35:09 +02:00
"Draw the args part of the menu."
2010-09-01 20:37:08 +02:00
(when args
2010-09-15 21:09:57 +02:00
(let ((strs (mapcar
(lambda (argument)
(propertize
(format " %s: %s (%s) %s"
(propertize
(car argument)
'face 'font-lock-builtin-face)
(nth 1 argument)
(nth 2 argument)
(propertize
(gethash (nth 2 argument)
magit-key-mode-current-args
"")
'face 'widget-field))
'key-group-executor (car argument)))
2010-09-15 21:09:57 +02:00
args)))
(magit-key-mode-draw-header "Args\n")
(magit-key-mode-draw-in-cols strs (not magit-key-mode-args-in-cols)))))
2010-09-01 20:37:08 +02:00
2010-09-01 20:39:58 +02:00
(defun magit-key-mode-draw-switches (switches)
2010-09-01 22:35:09 +02:00
"Draw the switches part of the menu."
2010-09-01 20:39:58 +02:00
(when switches
2010-09-01 22:14:46 +02:00
(let ((switch-strs (mapcar
(lambda (s)
(let ((option (nth 2 s)))
(propertize
(format " %s: %s (%s)"
(propertize (car s)
'face 'font-lock-builtin-face)
(nth 1 s)
(if (member option magit-key-mode-current-options)
(propertize
option
'face 'font-lock-warning-face)
option))
'key-group-executor (car s))))
switches)))
2010-09-01 22:14:46 +02:00
(magit-key-mode-draw-header "Switches\n")
(magit-key-mode-draw-in-cols switch-strs))))
2010-09-01 20:27:49 +02:00
(defun magit-key-mode-draw-actions (actions)
2010-09-01 22:35:09 +02:00
"Draw the actions part of the menu."
2010-09-01 20:27:49 +02:00
(when actions
2010-09-01 22:14:46 +02:00
(let ((action-strs (mapcar
(lambda (a)
(propertize
(format
" %s: %s"
(propertize (car a)
'face 'font-lock-builtin-face)
(nth 1 a))
'key-group-executor (car a)))
actions)))
(magit-key-mode-draw-header "Actions\n")
(magit-key-mode-draw-in-cols action-strs))))
2010-09-01 22:14:46 +02:00
2010-09-15 21:09:57 +02:00
(defun magit-key-mode-draw-in-cols (strings &optional one-col-each)
"Given a list of strings, print in columns (using `insert'). If
ONE-COL-EACH is true then don't columify, but rather, draw each
item on one line."
2010-09-02 00:43:57 +02:00
(let ((longest-act (apply 'max (mapcar 'length strings))))
(while strings
(let ((str (car strings)))
2010-09-01 23:13:08 +02:00
(let ((padding (make-string (- (+ longest-act 3) (length str)) ? )))
(insert str)
2010-09-15 21:09:57 +02:00
(if (or one-col-each
(and (> (+ (length padding) ;
(current-column)
longest-act)
(window-width))
(cdr strings)))
(insert "\n")
(insert padding))))
(setq strings (cdr strings))))
(insert "\n"))
2010-09-01 20:27:49 +02:00
2010-08-31 00:28:52 +02:00
(defun magit-key-mode-draw (for-group)
2010-08-31 00:45:51 +02:00
"Function used to draw actions, switches and parameters."
2010-08-31 00:28:52 +02:00
(let* ((options (magit-key-mode-options-for-group for-group))
2010-08-31 09:42:21 +02:00
(switches (cdr (assoc 'switches options)))
2010-08-31 19:53:37 +02:00
(arguments (cdr (assoc 'arguments options)))
2010-08-31 00:28:52 +02:00
(actions (cdr (assoc 'actions options))))
2010-09-01 20:39:58 +02:00
(magit-key-mode-draw-switches switches)
(magit-key-mode-draw-args arguments)
(magit-key-mode-draw-actions actions)))
2010-08-31 00:28:52 +02:00
(defun magit-key-mode-de-generate (group)
"Unbind the function for GROUP."
(fmakunbound
(intern (concat "magit-key-mode-popup-" (symbol-name group)))))
(defun magit-key-mode-generate (group)
"Generate the key-group menu for GROUP"
(let ((opts (magit-key-mode-options-for-group group)))
(eval
`(defun ,(intern (concat "magit-key-mode-popup-" (symbol-name group))) nil
,(concat "Key menu for " (symbol-name group))
(interactive)
(magit-key-mode (quote ,group))))))
2010-09-17 15:50:18 +02:00
;; create the interactive functions for the key mode popups (which are
;; applied in the top-level key maps)
(mapc (lambda (g)
(magit-key-mode-generate (car g)))
magit-key-mode-groups)
2010-08-31 09:42:21 +02:00
(provide 'magit-key-mode)