slackbuilds/magit.el

472 lines
14 KiB
EmacsLisp
Raw Normal View History

2008-08-06 01:41:05 +02:00
;;; magit -- control git from Emacs.
2008-07-31 22:11:46 +02:00
;; Copyright (C) 2008 Marius Vollmer
;;
2008-08-06 01:41:05 +02:00
;; Magit is free software; you can redistribute it and/or modify it
2008-08-05 18:09:47 +02:00
;; under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 3, or (at your option)
;; any later version.
2008-07-31 22:11:46 +02:00
;;
2008-08-06 01:41:05 +02:00
;; Magit is distributed in the hope that it will be useful, but WITHOUT
2008-08-05 18:09:47 +02:00
;; ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
;; or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
;; License for more details.
2008-07-31 22:11:46 +02:00
;;
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to
;; the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
;;; Introduction
2008-08-06 01:41:05 +02:00
;; Invoking the magit-status function will show a buffer with the
2008-07-31 22:11:46 +02:00
;; current status of the current git repository and its checkout.
;; That buffer offers key bindings for manipulating the status in
;; simple ways.
;;
;; The status buffer mainly shows the difference between the working
;; tree and the index, and the difference between the index and the
2008-08-04 01:05:02 +02:00
;; current HEAD. You can add individual hunks from the working tree
;; to the index, and you can commit the index.
2008-08-01 13:37:54 +02:00
2008-08-03 20:41:24 +02:00
;;; TODO
2008-08-06 02:47:26 +02:00
;; - Documentation, major mode, menu, ...
2008-08-03 20:50:00 +02:00
;; - History browsing
2008-08-06 01:44:41 +02:00
;; - Detect and handle renames and copies.
2008-08-03 20:41:24 +02:00
2008-08-01 13:37:54 +02:00
;;; Utilities
2008-08-06 01:41:05 +02:00
(defun magit-shell (cmd &rest args)
2008-08-01 13:37:54 +02:00
(let ((str (shell-command-to-string (apply 'format cmd args))))
(if (string= str "")
nil
(if (equal (elt str (- (length str) 1)) ?\n)
(substring str 0 (- (length str) 1))
str))))
2008-08-06 01:41:05 +02:00
(defun magit-shell-lines (cmd &rest args)
2008-08-06 01:21:29 +02:00
(let ((str (shell-command-to-string (apply 'format cmd args))))
(if (string= str "")
nil
(let ((lines (nreverse (split-string str "\n"))))
(if (string= (car lines) "")
(setq lines (cdr lines)))
(nreverse lines)))))
(defun magit-file-lines (file)
(if (file-exists-p file)
(magit-shell-lines "cat '%s'" file)
nil))
2008-08-06 01:21:29 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-concat-with-delim (delim seqs)
2008-08-03 20:33:25 +02:00
(cond ((null seqs)
nil)
((null (cdr seqs))
(car seqs))
(t
2008-08-06 01:41:05 +02:00
(concat (car seqs) delim (magit-concat-with-delim delim (cdr seqs))))))
2008-08-03 20:33:25 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-get (&rest keys)
(magit-shell "git-config %s" (magit-concat-with-delim "." keys)))
2008-08-03 20:33:25 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-set (val &rest keys)
2008-08-04 01:05:02 +02:00
(if val
2008-08-06 01:41:05 +02:00
(magit-shell "git-config %s %s" (magit-concat-with-delim "." keys) val)
(magit-shell "git-config --unset %s" (magit-concat-with-delim "." keys))))
2008-08-04 01:05:02 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-get-top-dir (cwd)
2008-08-01 13:37:54 +02:00
(let* ((cwd (expand-file-name cwd))
2008-08-06 01:41:05 +02:00
(magit-dir (magit-shell "cd '%s' && git-rev-parse --git-dir 2>/dev/null"
2008-08-01 13:37:54 +02:00
cwd)))
2008-08-06 01:41:05 +02:00
(if magit-dir
(file-name-as-directory (or (file-name-directory magit-dir) cwd))
2008-08-01 13:37:54 +02:00
nil)))
2008-08-06 01:41:05 +02:00
(defun magit-get-ref (ref)
(magit-shell "git-symbolic-ref -q %s" ref))
2008-08-03 20:33:25 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-get-current-branch ()
(let* ((head (magit-get-ref "HEAD"))
2008-08-03 20:33:25 +02:00
(pos (and head (string-match "^refs/heads/" head))))
(if pos
(substring head 11)
nil)))
2008-08-06 01:41:05 +02:00
(defun magit-read-top-dir (prefix)
(let ((dir (magit-get-top-dir default-directory)))
2008-08-03 20:33:25 +02:00
(if prefix
2008-08-06 01:41:05 +02:00
(magit-get-top-dir (read-directory-name "Git repository: " dir))
2008-08-03 20:33:25 +02:00
dir)))
2008-08-01 13:37:54 +02:00
(defun magit-name-rev (rev)
(magit-shell "git name-rev --always --name-only %s" rev))
2008-08-06 01:41:05 +02:00
(defun magit-insert-output (title washer cmd &rest args)
2008-08-05 21:21:51 +02:00
(if title
2008-08-05 23:47:07 +02:00
(insert (propertize title 'face 'bold) "\n"))
2008-08-04 02:52:16 +02:00
(let* ((beg (point))
(status (apply 'call-process cmd nil t nil args))
(end (point)))
(if washer
(save-restriction
(narrow-to-region beg (point))
(funcall washer status)
(goto-char (point-max))
(insert "\n")))))
2008-08-03 21:50:39 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-put-line-property (prop val)
2008-08-04 01:05:02 +02:00
(put-text-property (line-beginning-position) (line-end-position)
prop val))
2008-08-01 15:04:52 +02:00
;;; Running asynchronous commands
2008-08-06 01:41:05 +02:00
(defvar magit-process nil)
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-run (cmd &rest args)
(or (not magit-process)
2008-08-01 15:04:52 +02:00
(error "Git is already running."))
(let ((dir default-directory)
(buf (get-buffer-create "*git-process*")))
(save-excursion
(set-buffer buf)
(setq default-directory dir)
(erase-buffer)
2008-08-06 01:41:05 +02:00
(insert "$ " (magit-concat-with-delim " " (cons cmd args)) "\n")
(setq magit-process (apply 'start-process "git" buf cmd args))
(set-process-sentinel magit-process 'magit-process-sentinel))))
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-revert-files ()
(let ((files (magit-shell-lines "git ls-files")))
2008-08-06 01:21:29 +02:00
(dolist (file files)
(let ((buffer (find-buffer-visiting file)))
(when (and buffer
(not (verify-visited-file-modtime buffer))
(not (buffer-modified-p buffer)))
(with-current-buffer buffer
(ignore-errors
(revert-buffer t t t))))))))
2008-08-06 01:41:05 +02:00
(defun magit-process-sentinel (process event)
2008-08-01 15:04:52 +02:00
(cond ((string= event "finished\n")
(message "Git finished.")
2008-08-06 01:41:05 +02:00
(setq magit-process nil))
2008-08-01 15:04:52 +02:00
((string= event "killed\n")
(message "Git was killed.")
2008-08-06 01:41:05 +02:00
(setq magit-process nil))
2008-08-01 15:04:52 +02:00
((string-match "exited abnormally" event)
(message "Git failed.")
2008-08-06 01:41:05 +02:00
(setq magit-process nil))
2008-08-01 15:04:52 +02:00
(t
(message "Git is weird.")))
2008-08-06 01:41:05 +02:00
(magit-revert-files)
(magit-update-status))
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-display-process ()
2008-08-03 21:29:08 +02:00
(interactive)
(display-buffer "*git-process*"))
2008-08-03 21:10:00 +02:00
;;; Keymap
2008-08-06 01:41:05 +02:00
(defvar magit-keymap nil)
(when (not magit-keymap)
(setq magit-keymap (make-keymap))
(suppress-keymap magit-keymap)
(define-key magit-keymap (kbd "g") 'magit-status)
(define-key magit-keymap (kbd "A") 'magit-stage-all)
(define-key magit-keymap (kbd "a") 'magit-stage-thing-at-point)
(define-key magit-keymap (kbd "u") 'magit-unstage-thing-at-point)
(define-key magit-keymap (kbd "i") 'magit-ignore-thing-at-point)
(define-key magit-keymap (kbd "RET") 'magit-visit-thing-at-point)
(define-key magit-keymap (kbd "b") 'magit-switch-branch)
(define-key magit-keymap (kbd "B") 'magit-create-branch)
(define-key magit-keymap (kbd "m") 'magit-manual-merge)
(define-key magit-keymap (kbd "M") 'magit-automatic-merge)
(define-key magit-keymap (kbd "U") 'magit-pull)
(define-key magit-keymap (kbd "P") 'magit-push)
(define-key magit-keymap (kbd "c") 'magit-log-edit)
(define-key magit-keymap (kbd "p") 'magit-display-process))
2008-08-03 21:10:00 +02:00
2008-08-01 15:04:52 +02:00
;;; Status
2008-08-06 01:41:05 +02:00
(defun magit-wash-other-files (status)
2008-08-04 01:05:02 +02:00
(goto-char (point-min))
(while (not (eobp))
(let ((filename (buffer-substring (point) (line-end-position))))
(insert " ")
2008-08-06 01:41:05 +02:00
(magit-put-line-property 'face '(:foreground "red"))
(magit-put-line-property 'magit-info (list 'other-file filename)))
2008-08-04 01:05:02 +02:00
(forward-line)
(beginning-of-line)))
2008-08-06 01:41:05 +02:00
(defun magit-wash-diff-propertize-diff (head-beg head-end)
(let ((head-end (or head-end (point))))
(when head-beg
(put-text-property head-beg head-end
2008-08-06 01:41:05 +02:00
'magit-info (list 'diff
head-beg (point))))))
2008-08-06 01:41:05 +02:00
(defun magit-wash-diff-propertize-hunk (head-beg head-end hunk-beg)
(when hunk-beg
(put-text-property hunk-beg (point)
2008-08-06 01:41:05 +02:00
'magit-info (list 'hunk
head-beg head-end
hunk-beg (point)))))
2008-08-04 14:54:08 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-wash-diff (status)
2008-08-04 02:26:34 +02:00
(goto-char (point-min))
(let ((n-files 1)
2008-08-04 02:52:16 +02:00
(head-beg nil)
(head-end nil)
(hunk-beg nil))
2008-08-04 02:26:34 +02:00
(while (not (eobp))
(let ((prefix (buffer-substring-no-properties
(point) (+ (point) n-files))))
2008-08-04 02:52:16 +02:00
(cond ((looking-at "^diff")
2008-08-06 01:41:05 +02:00
(magit-wash-diff-propertize-diff head-beg head-end)
2008-08-04 02:52:16 +02:00
(setq head-beg (point))
(setq head-end nil))
((looking-at "^@+")
2008-08-04 02:26:34 +02:00
(setq n-files (- (length (match-string 0)) 1))
(if (null head-end)
(setq head-end (point)))
2008-08-06 01:41:05 +02:00
(magit-wash-diff-propertize-hunk head-beg head-end hunk-beg)
2008-08-04 02:52:16 +02:00
(setq hunk-beg (point)))
2008-08-04 02:26:34 +02:00
((string-match "\\+" prefix)
2008-08-06 01:41:05 +02:00
(magit-put-line-property 'face '(:foreground "blue1")))
2008-08-04 02:26:34 +02:00
((string-match "-" prefix)
2008-08-06 01:41:05 +02:00
(magit-put-line-property 'face '(:foreground "red")))))
2008-08-04 02:26:34 +02:00
(forward-line)
(beginning-of-line))
2008-08-06 01:41:05 +02:00
(magit-wash-diff-propertize-diff head-beg head-end)
(magit-wash-diff-propertize-hunk head-beg head-end hunk-beg)))
2008-08-04 02:26:34 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-update-status ()
2008-08-01 15:04:52 +02:00
(let ((buf (get-buffer "*git-status*")))
(save-excursion
(set-buffer buf)
2008-08-03 21:10:00 +02:00
(setq buffer-read-only t)
2008-08-06 01:41:05 +02:00
(use-local-map magit-keymap)
2008-08-03 21:10:00 +02:00
(let ((inhibit-read-only t))
(erase-buffer)
2008-08-06 01:41:05 +02:00
(let* ((branch (magit-get-current-branch))
(remote (and branch (magit-get "branch" branch "remote"))))
2008-08-05 18:09:47 +02:00
(if remote
(insert (format "Remote: %s %s\n"
2008-08-06 01:41:05 +02:00
remote (magit-get "remote" remote "url"))))
2008-08-03 22:04:18 +02:00
(insert (format "Local: %s %s\n"
2008-08-05 23:47:07 +02:00
(propertize (or branch "(detached)") 'face 'bold)
2008-08-05 18:09:47 +02:00
(abbreviate-file-name default-directory)))
(let ((merge-heads (magit-file-lines ".git/MERGE_HEAD")))
(if merge-heads
(insert (format "Merging: %s\n"
(magit-concat-with-delim
", "
(mapcar 'magit-name-rev merge-heads))))))
2008-08-05 18:09:47 +02:00
(insert "\n")
2008-08-06 01:41:05 +02:00
(magit-insert-output "Untracked files:" 'magit-wash-other-files
2008-08-05 18:09:47 +02:00
"git" "ls-files" "--others" "--exclude-standard")
2008-08-06 01:41:05 +02:00
(magit-insert-output "Unstaged changes:" 'magit-wash-diff
2008-08-05 18:09:47 +02:00
"git" "diff")
2008-08-06 01:41:05 +02:00
(magit-insert-output "Staged changes:" 'magit-wash-diff
2008-08-05 18:09:47 +02:00
"git" "diff" "--cached")
(if remote
2008-08-06 02:22:01 +02:00
(magit-insert-output "Unpushed changes:" 'nil
2008-08-05 18:09:47 +02:00
"git" "diff" "--stat"
(format "%s/%s..HEAD" remote branch))))))))
2008-08-05 18:09:47 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-status (dir)
(interactive (list (magit-read-top-dir current-prefix-arg)))
(save-some-buffers)
2008-08-03 20:33:25 +02:00
(let ((buf (get-buffer-create "*git-status*")))
(switch-to-buffer buf)
(setq default-directory dir)
2008-08-06 01:41:05 +02:00
(magit-update-status)))
2008-08-01 13:37:54 +02:00
2008-08-04 01:05:02 +02:00
;;; Staging
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-write-diff-patch (info file)
(write-region (elt info 1) (elt info 2) file))
2008-08-06 01:41:05 +02:00
(defun magit-write-hunk-patch (info file)
2008-08-04 03:10:48 +02:00
(write-region (elt info 1) (elt info 2) file)
(write-region (elt info 3) (elt info 4) file t))
2008-08-06 01:41:05 +02:00
(defun magit-hunk-is-conflict-p (info)
2008-08-05 18:09:47 +02:00
(save-excursion
(goto-char (elt info 1))
(looking-at-p "^diff --cc")))
2008-08-06 01:41:05 +02:00
(defun magit-diff-conflict-file (info)
2008-08-05 18:09:47 +02:00
(save-excursion
(goto-char (elt info 1))
(if (looking-at "^diff --cc +\\(.*\\)$")
(match-string 1)
nil)))
2008-08-05 23:30:37 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-diff-info-file (info)
2008-08-05 23:30:37 +02:00
(save-excursion
(goto-char (elt info 1))
(cond ((looking-at "^diff --git a/\\(.*\\) b/\\(.*\\)$")
(match-string 2))
((looking-at "^diff --cc +\\(.*\\)$")
(match-string 1))
(t
nil))))
2008-08-06 01:41:05 +02:00
(defun magit-diff-info-position (info)
2008-08-05 23:30:37 +02:00
(save-excursion
(cond ((eq (car info) 'hunk)
(goto-char (elt info 3))
(if (looking-at "@@+ .* \\+\\([0-9]+\\),[0-9]+ @@+")
(parse-integer (match-string 1))
nil))
(t nil))))
2008-08-05 18:09:47 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-stage-thing-at-point ()
2008-08-01 15:04:52 +02:00
(interactive)
2008-08-06 01:41:05 +02:00
(let ((info (get-char-property (point) 'magit-info)))
2008-08-04 01:05:02 +02:00
(if info
(case (car info)
((other-file)
2008-08-06 01:41:05 +02:00
(magit-run "git" "add" (cadr info)))
2008-08-04 03:10:48 +02:00
((hunk)
2008-08-06 01:41:05 +02:00
(if (magit-hunk-is-conflict-p info)
2008-08-05 18:09:47 +02:00
(error
"Can't stage individual resolution hunks. Please stage the whole file."))
2008-08-06 01:41:05 +02:00
(magit-write-hunk-patch info ".git/magit-tmp")
(magit-run "git" "apply" "--cached" ".git/magit-tmp"))
((diff)
2008-08-06 01:41:05 +02:00
(magit-run "git" "add" (magit-diff-info-file info)))))))
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-unstage-thing-at-point ()
2008-08-04 03:13:23 +02:00
(interactive)
2008-08-06 01:41:05 +02:00
(let ((info (get-char-property (point) 'magit-info)))
2008-08-04 03:13:23 +02:00
(if info
(case (car info)
((hunk)
2008-08-06 01:41:05 +02:00
(magit-write-hunk-patch info ".git/magit-tmp")
(magit-run "git" "apply" "--cached" "--reverse" ".git/magit-tmp"))
((diff)
2008-08-06 01:41:05 +02:00
(magit-run "git" "reset" "HEAD" (magit-diff-info-file info)))))))
2008-08-04 03:13:23 +02:00
2008-08-06 02:22:22 +02:00
(defun magit-stage-all ()
2008-08-04 02:26:34 +02:00
(interactive)
2008-08-06 02:22:22 +02:00
(magit-run "git-add" "-u" "."))
2008-08-04 02:26:34 +02:00
;;; Branches
2008-08-06 01:41:05 +02:00
(defun magit-list-branches ()
(magit-shell-lines "git branch -a | cut -c3-"))
2008-08-06 01:41:05 +02:00
(defun magit-read-other-branch (prompt)
(completing-read prompt (delete (magit-get-current-branch)
(magit-list-branches))
2008-08-06 00:17:34 +02:00
nil t))
2008-08-06 01:41:05 +02:00
(defun magit-switch-branch (branch)
(interactive (list (magit-read-other-branch "Switch to branch: ")))
(if (and branch (not (string= branch "")))
2008-08-06 01:41:05 +02:00
(magit-run "git" "checkout" branch)))
2008-08-06 01:41:05 +02:00
(defun magit-read-create-branch-args ()
(let* ((branches (magit-list-branches))
(cur-branch (magit-get-current-branch))
(branch (read-string "Create branch: "))
(parent (completing-read "Parent: " branches nil t cur-branch)))
(list branch parent)))
2008-08-06 01:41:05 +02:00
(defun magit-create-branch (branch parent)
(interactive (magit-read-create-branch-args))
(if (and branch (not (string= branch ""))
parent (not (string= parent "")))
2008-08-06 01:41:05 +02:00
(magit-run "git" "checkout" "-b" branch parent)))
2008-08-06 00:17:34 +02:00
;;; Merging
2008-08-06 01:41:05 +02:00
(defun magit-manual-merge (branch)
(interactive (list (magit-read-other-branch "Manually merge from branch: ")))
(magit-run "git" "merge" "--no-ff" "--no-commit" branch))
2008-08-06 00:17:34 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-automatic-merge (branch)
(interactive (list (magit-read-other-branch "Merge from branch: ")))
(magit-run "git" "merge" branch))
2008-08-06 00:17:34 +02:00
2008-08-04 01:05:02 +02:00
;;; Push and pull
2008-08-06 01:41:05 +02:00
(defun magit-pull ()
(interactive)
2008-08-06 01:41:05 +02:00
(magit-run "git" "pull" "-v"))
2008-08-04 01:05:02 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-push ()
(interactive)
2008-08-06 01:41:05 +02:00
(magit-run "git" "push" "-v"))
2008-08-04 01:05:02 +02:00
;;; Commit
2008-08-01 15:04:52 +02:00
2008-08-06 01:41:05 +02:00
(defvar magit-log-edit-map nil)
2008-08-04 14:35:25 +02:00
2008-08-06 01:41:05 +02:00
(when (not magit-log-edit-map)
(setq magit-log-edit-map (make-sparse-keymap))
(define-key magit-log-edit-map (kbd "C-c C-c") 'magit-log-edit-commit))
2008-08-04 14:35:25 +02:00
2008-08-06 01:41:05 +02:00
(defvar magit-pre-log-edit-window-configuration nil)
2008-08-04 14:35:25 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-log-edit-cleanup ()
(save-excursion
(goto-char (point-min))
(flush-lines "^#")
(goto-char (point-min))
(replace-regexp "[ \t\n]*\\'" "\n")))
2008-08-06 01:41:05 +02:00
(defun magit-log-edit-commit ()
2008-08-04 14:35:25 +02:00
(interactive)
2008-08-06 01:41:05 +02:00
(magit-log-edit-cleanup)
(if (> (buffer-size) 0)
2008-08-06 01:41:05 +02:00
(write-region (point-min) (point-max) ".git/magit-log")
(write-region "(Empty description)" nil ".git/magit-log"))
(erase-buffer)
2008-08-06 01:41:05 +02:00
(magit-run "git-commit" "-F" ".git/magit-log")
2008-08-04 14:35:25 +02:00
(bury-buffer)
2008-08-06 01:41:05 +02:00
(when magit-pre-log-edit-window-configuration
(set-window-configuration magit-pre-log-edit-window-configuration)
(setq magit-pre-log-edit-window-configuration nil)))
2008-08-04 14:35:25 +02:00
2008-08-06 01:41:05 +02:00
(defun magit-log-edit ()
2008-08-01 15:04:52 +02:00
(interactive)
2008-08-04 14:35:25 +02:00
(let ((dir default-directory)
(buf (get-buffer-create "*git-log-edit*")))
(setq magit-pre-log-edit-window-configuration
(current-window-configuration))
2008-08-04 14:35:25 +02:00
(pop-to-buffer buf)
(setq default-directory dir)
2008-08-06 01:41:05 +02:00
(use-local-map magit-log-edit-map)
(message "Type C-c C-c to commit.")))
2008-08-04 01:05:02 +02:00
2008-08-06 02:22:22 +02:00
;;; Miscellaneous
2008-08-04 01:05:02 +02:00
2008-08-06 02:22:22 +02:00
(defun magit-ignore-thing-at-point ()
2008-08-04 01:05:02 +02:00
(interactive)
2008-08-06 02:22:22 +02:00
(let ((info (get-char-property (point) 'magit-info)))
(if info
(case (car info)
((other-file)
(append-to-file (concat (cadr info) "\n") nil ".gitignore")
(magit-update-status))))))
(defun magit-visit-thing-at-point ()
(interactive)
(let ((info (get-char-property (point) 'magit-info)))
(if info
(case (car info)
((other-file)
(find-file (cadr info)))
((diff hunk)
(let ((file (magit-diff-info-file info))
(position (magit-diff-info-position info)))
(find-file file)
(if position
(goto-line position))))))))