]> git.ipfire.org Git - thirdparty/git.git/blame - contrib/emacs/git.el
git.el: Don't use font-lock-compile-keywords
[thirdparty/git.git] / contrib / emacs / git.el
CommitLineData
711fc8f6
AJ
1;;; git.el --- A user interface for git
2
5a7b3bf5 3;; Copyright (C) 2005, 2006, 2007, 2008, 2009 Alexandre Julliard <julliard@winehq.org>
711fc8f6
AJ
4
5;; Version: 1.0
6
7;; This program is free software; you can redistribute it and/or
8;; modify it under the terms of the GNU General Public License as
9;; published by the Free Software Foundation; either version 2 of
10;; the License, or (at your option) any later version.
11;;
12;; This program is distributed in the hope that it will be
13;; useful, but WITHOUT ANY WARRANTY; without even the implied
14;; warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15;; PURPOSE. See the GNU General Public License for more details.
16;;
17;; You should have received a copy of the GNU General Public
18;; License along with this program; if not, write to the Free
19;; Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
20;; MA 02111-1307 USA
21
22;;; Commentary:
23
24;; This file contains an interface for the git version control
25;; system. It provides easy access to the most frequently used git
26;; commands. The user interface is as far as possible identical to
27;; that of the PCL-CVS mode.
28;;
29;; To install: put this file on the load-path and place the following
30;; in your .emacs file:
31;;
32;; (require 'git)
33;;
34;; To start: `M-x git-status'
35;;
36;; TODO
711fc8f6
AJ
37;; - diff against other branch
38;; - renaming files from the status buffer
711fc8f6
AJ
39;; - creating tags
40;; - fetch/pull
711fc8f6
AJ
41;; - revlist browser
42;; - git-show-branch browser
711fc8f6
AJ
43;;
44
5a7b3bf5
AJ
45;;; Compatibility:
46;;
47;; This file works on GNU Emacs 21 or later. It may work on older
48;; versions but this is not guaranteed.
49;;
50;; It may work on XEmacs 21, provided that you first install the ewoc
51;; and log-edit packages.
52;;
53
711fc8f6
AJ
54(eval-when-compile (require 'cl))
55(require 'ewoc)
9fa77a51 56(require 'log-edit)
18ff365f 57(require 'easymenu)
711fc8f6
AJ
58
59
a79656e6 60;;;; Customizations
711fc8f6
AJ
61;;;; ------------------------------------------------------------
62
a79656e6 63(defgroup git nil
5df52584
AJ
64 "A user interface for the git versioning system."
65 :group 'tools)
a79656e6
AJ
66
67(defcustom git-committer-name nil
68 "User name to use for commits.
1b3a6674
JN
69The default is to fall back to the repository config,
70then to `add-log-full-name' and then to `user-full-name'."
a79656e6
AJ
71 :group 'git
72 :type '(choice (const :tag "Default" nil)
73 (string :tag "Name")))
74
75(defcustom git-committer-email nil
76 "Email address to use for commits.
1b3a6674
JN
77The default is to fall back to the git repository config,
78then to `add-log-mailing-address' and then to `user-mail-address'."
a79656e6
AJ
79 :group 'git
80 :type '(choice (const :tag "Default" nil)
81 (string :tag "Email")))
82
14b4f2db 83(defcustom git-commits-coding-system nil
a79656e6
AJ
84 "Default coding system for the log message of git commits."
85 :group 'git
14b4f2db
AJ
86 :type '(choice (const :tag "From repository config" nil)
87 (coding-system)))
a79656e6
AJ
88
89(defcustom git-append-signed-off-by nil
90 "Whether to append a Signed-off-by line to the commit message before editing."
91 :group 'git
92 :type 'boolean)
93
73389f12
AJ
94(defcustom git-reuse-status-buffer t
95 "Whether `git-status' should try to reuse an existing buffer
96if there is already one that displays the same directory."
97 :group 'git
98 :type 'boolean)
99
a79656e6
AJ
100(defcustom git-per-dir-ignore-file ".gitignore"
101 "Name of the per-directory ignore file."
102 :group 'git
103 :type 'string)
104
98acc3fa
AJ
105(defcustom git-show-uptodate nil
106 "Whether to display up-to-date files."
107 :group 'git
108 :type 'boolean)
109
110(defcustom git-show-ignored nil
111 "Whether to display ignored files."
112 :group 'git
113 :type 'boolean)
114
115(defcustom git-show-unknown t
116 "Whether to display unknown files."
117 :group 'git
118 :type 'boolean)
119
1b3a6674 120
711fc8f6 121(defface git-status-face
1ff55ff2
DK
122 '((((class color) (background light)) (:foreground "purple"))
123 (((class color) (background dark)) (:foreground "salmon")))
a79656e6
AJ
124 "Git mode face used to highlight added and modified files."
125 :group 'git)
711fc8f6
AJ
126
127(defface git-unmerged-face
1ff55ff2
DK
128 '((((class color) (background light)) (:foreground "red" :bold t))
129 (((class color) (background dark)) (:foreground "red" :bold t)))
a79656e6
AJ
130 "Git mode face used to highlight unmerged files."
131 :group 'git)
711fc8f6
AJ
132
133(defface git-unknown-face
1ff55ff2
DK
134 '((((class color) (background light)) (:foreground "goldenrod" :bold t))
135 (((class color) (background dark)) (:foreground "goldenrod" :bold t)))
a79656e6
AJ
136 "Git mode face used to highlight unknown files."
137 :group 'git)
711fc8f6
AJ
138
139(defface git-uptodate-face
1ff55ff2
DK
140 '((((class color) (background light)) (:foreground "grey60"))
141 (((class color) (background dark)) (:foreground "grey40")))
a79656e6
AJ
142 "Git mode face used to highlight up-to-date files."
143 :group 'git)
711fc8f6
AJ
144
145(defface git-ignored-face
1ff55ff2
DK
146 '((((class color) (background light)) (:foreground "grey60"))
147 (((class color) (background dark)) (:foreground "grey40")))
a79656e6
AJ
148 "Git mode face used to highlight ignored files."
149 :group 'git)
711fc8f6
AJ
150
151(defface git-mark-face
1ff55ff2
DK
152 '((((class color) (background light)) (:foreground "red" :bold t))
153 (((class color) (background dark)) (:foreground "tomato" :bold t)))
a79656e6
AJ
154 "Git mode face used for the file marks."
155 :group 'git)
711fc8f6
AJ
156
157(defface git-header-face
1ff55ff2
DK
158 '((((class color) (background light)) (:foreground "blue"))
159 (((class color) (background dark)) (:foreground "blue")))
a79656e6
AJ
160 "Git mode face used for commit headers."
161 :group 'git)
711fc8f6
AJ
162
163(defface git-separator-face
1ff55ff2
DK
164 '((((class color) (background light)) (:foreground "brown"))
165 (((class color) (background dark)) (:foreground "brown")))
a79656e6
AJ
166 "Git mode face used for commit separator."
167 :group 'git)
711fc8f6
AJ
168
169(defface git-permission-face
1ff55ff2
DK
170 '((((class color) (background light)) (:foreground "green" :bold t))
171 (((class color) (background dark)) (:foreground "green" :bold t)))
a79656e6
AJ
172 "Git mode face used for permission changes."
173 :group 'git)
711fc8f6
AJ
174
175
176;;;; Utilities
177;;;; ------------------------------------------------------------
178
a79656e6
AJ
179(defconst git-log-msg-separator "--- log message follows this line ---")
180
9fa77a51 181(defvar git-log-edit-font-lock-keywords
6fb20426 182 `(("^\\(Author:\\|Date:\\|Merge:\\|Signed-off-by:\\)\\(.*\\)$"
9fa77a51
AJ
183 (1 font-lock-keyword-face)
184 (2 font-lock-function-name-face))
185 (,(concat "^\\(" (regexp-quote git-log-msg-separator) "\\)$")
186 (1 font-lock-comment-face))))
187
711fc8f6
AJ
188(defun git-get-env-strings (env)
189 "Build a list of NAME=VALUE strings from a list of environment strings."
190 (mapcar (lambda (entry) (concat (car entry) "=" (cdr entry))) env))
191
9ddf6d7c 192(defun git-call-process (buffer &rest args)
711fc8f6 193 "Wrapper for call-process that sets environment strings."
9ddf6d7c 194 (apply #'call-process "git" nil buffer nil args))
711fc8f6 195
0520e215
AJ
196(defun git-call-process-display-error (&rest args)
197 "Wrapper for call-process that displays error messages."
198 (let* ((dir default-directory)
199 (buffer (get-buffer-create "*Git Command Output*"))
200 (ok (with-current-buffer buffer
201 (let ((default-directory dir)
202 (buffer-read-only nil))
203 (erase-buffer)
9ddf6d7c 204 (eq 0 (apply #'git-call-process (list buffer t) args))))))
0520e215
AJ
205 (unless ok (display-message-or-buffer buffer))
206 ok))
207
9ddf6d7c
AJ
208(defun git-call-process-string (&rest args)
209 "Wrapper for call-process that returns the process output as a string,
210or nil if the git command failed."
9de83169 211 (with-temp-buffer
9ddf6d7c 212 (and (eq 0 (apply #'git-call-process t args))
9de83169
AJ
213 (buffer-string))))
214
36d2078f
AJ
215(defun git-call-process-string-display-error (&rest args)
216 "Wrapper for call-process that displays error message and returns
217the process output as a string, or nil if the git command failed."
218 (with-temp-buffer
9ddf6d7c 219 (if (eq 0 (apply #'git-call-process (list t t) args))
36d2078f
AJ
220 (buffer-string)
221 (display-message-or-buffer (current-buffer))
222 nil)))
223
711fc8f6
AJ
224(defun git-run-process-region (buffer start end program args)
225 "Run a git process with a buffer region as input."
226 (let ((output-buffer (current-buffer))
227 (dir default-directory))
228 (with-current-buffer buffer
229 (cd dir)
230 (apply #'call-process-region start end program
a7da5c42 231 nil (list output-buffer t) nil args))))
711fc8f6
AJ
232
233(defun git-run-command-buffer (buffer-name &rest args)
234 "Run a git command, sending the output to a buffer named BUFFER-NAME."
235 (let ((dir default-directory)
236 (buffer (get-buffer-create buffer-name)))
237 (message "Running git %s..." (car args))
238 (with-current-buffer buffer
239 (let ((default-directory dir)
240 (buffer-read-only nil))
241 (erase-buffer)
9ddf6d7c 242 (apply #'git-call-process buffer args)))
711fc8f6
AJ
243 (message "Running git %s...done" (car args))
244 buffer))
245
711fc8f6
AJ
246(defun git-run-command-region (buffer start end env &rest args)
247 "Run a git command with specified buffer region as input."
a7da5c42
AJ
248 (with-temp-buffer
249 (if (eq 0 (if env
711fc8f6 250 (git-run-process-region
a7da5c42
AJ
251 buffer start end "env"
252 (append (git-get-env-strings env) (list "git") args))
253 (git-run-process-region buffer start end "git" args)))
254 (buffer-string)
255 (display-message-or-buffer (current-buffer))
256 nil)))
711fc8f6 257
d55552f6
AJ
258(defun git-run-hook (hook env &rest args)
259 "Run a git hook and display its output if any."
260 (let ((dir default-directory)
261 (hook-name (expand-file-name (concat ".git/hooks/" hook))))
262 (or (not (file-executable-p hook-name))
263 (let (status (buffer (get-buffer-create "*Git Hook Output*")))
264 (with-current-buffer buffer
265 (erase-buffer)
266 (cd dir)
267 (setq status
3db4723e
KW
268 (if env
269 (apply #'call-process "env" nil (list buffer t) nil
270 (append (git-get-env-strings env) (list hook-name) args))
d55552f6
AJ
271 (apply #'call-process hook-name nil (list buffer t) nil args))))
272 (display-message-or-buffer buffer)
273 (eq 0 status)))))
274
711fc8f6
AJ
275(defun git-get-string-sha1 (string)
276 "Read a SHA1 from the specified string."
9de83169
AJ
277 (and string
278 (string-match "[0-9a-f]\\{40\\}" string)
279 (match-string 0 string)))
711fc8f6
AJ
280
281(defun git-get-committer-name ()
282 "Return the name to use as GIT_COMMITTER_NAME."
283 ; copied from log-edit
284 (or git-committer-name
e0d10e1c 285 (git-config "user.name")
711fc8f6
AJ
286 (and (boundp 'add-log-full-name) add-log-full-name)
287 (and (fboundp 'user-full-name) (user-full-name))
288 (and (boundp 'user-full-name) user-full-name)))
289
290(defun git-get-committer-email ()
291 "Return the email address to use as GIT_COMMITTER_EMAIL."
292 ; copied from log-edit
293 (or git-committer-email
e0d10e1c 294 (git-config "user.email")
711fc8f6
AJ
295 (and (boundp 'add-log-mailing-address) add-log-mailing-address)
296 (and (fboundp 'user-mail-address) (user-mail-address))
297 (and (boundp 'user-mail-address) user-mail-address)))
298
14b4f2db
AJ
299(defun git-get-commits-coding-system ()
300 "Return the coding system to use for commits."
301 (let ((repo-config (git-config "i18n.commitencoding")))
302 (or git-commits-coding-system
303 (and repo-config
304 (fboundp 'locale-charset-to-coding-system)
305 (locale-charset-to-coding-system repo-config))
306 'utf-8)))
307
b704e589
AJ
308(defun git-get-logoutput-coding-system ()
309 "Return the coding system used for git-log output."
310 (let ((repo-config (or (git-config "i18n.logoutputencoding")
311 (git-config "i18n.commitencoding"))))
312 (or git-commits-coding-system
313 (and repo-config
314 (fboundp 'locale-charset-to-coding-system)
315 (locale-charset-to-coding-system repo-config))
316 'utf-8)))
317
711fc8f6
AJ
318(defun git-escape-file-name (name)
319 "Escape a file name if necessary."
320 (if (string-match "[\n\t\"\\]" name)
321 (concat "\""
322 (mapconcat (lambda (c)
323 (case c
324 (?\n "\\n")
325 (?\t "\\t")
326 (?\\ "\\\\")
327 (?\" "\\\"")
328 (t (char-to-string c))))
329 name "")
330 "\"")
331 name))
332
9f5599b9
AJ
333(defun git-success-message (text files)
334 "Print a success message after having handled FILES."
335 (let ((n (length files)))
336 (if (equal n 1)
337 (message "%s %s" text (car files))
338 (message "%s %d files" text n))))
339
711fc8f6
AJ
340(defun git-get-top-dir (dir)
341 "Retrieve the top-level directory of a git tree."
342 (let ((cdup (with-output-to-string
343 (with-current-buffer standard-output
344 (cd dir)
9ddf6d7c 345 (unless (eq 0 (git-call-process t "rev-parse" "--show-cdup"))
711fc8f6
AJ
346 (error "cannot find top-level git tree for %s." dir))))))
347 (expand-file-name (concat (file-name-as-directory dir)
348 (car (split-string cdup "\n"))))))
349
350;stolen from pcl-cvs
351(defun git-append-to-ignore (file)
352 "Add a file name to the ignore file in its directory."
353 (let* ((fullname (expand-file-name file))
354 (dir (file-name-directory fullname))
b23761d9
AJ
355 (name (file-name-nondirectory fullname))
356 (ignore-name (expand-file-name git-per-dir-ignore-file dir))
357 (created (not (file-exists-p ignore-name))))
711fc8f6 358 (save-window-excursion
b23761d9 359 (set-buffer (find-file-noselect ignore-name))
711fc8f6
AJ
360 (goto-char (point-max))
361 (unless (zerop (current-column)) (insert "\n"))
9f56a7fd 362 (insert "/" name "\n")
711fc8f6 363 (sort-lines nil (point-min) (point-max))
b23761d9
AJ
364 (save-buffer))
365 (when created
9ddf6d7c 366 (git-call-process nil "update-index" "--add" "--" (file-relative-name ignore-name)))
433ee03f 367 (git-update-status-files (list (file-relative-name ignore-name)))))
711fc8f6 368
03d311ed
AJ
369; propertize definition for XEmacs, stolen from erc-compat
370(eval-when-compile
371 (unless (fboundp 'propertize)
372 (defun propertize (string &rest props)
373 (let ((string (copy-sequence string)))
374 (while props
375 (put-text-property 0 (length string) (nth 0 props) (nth 1 props) string)
376 (setq props (cddr props)))
377 string))))
711fc8f6
AJ
378
379;;;; Wrappers for basic git commands
380;;;; ------------------------------------------------------------
381
382(defun git-rev-parse (rev)
383 "Parse a revision name and return its SHA1."
384 (git-get-string-sha1
9ddf6d7c 385 (git-call-process-string "rev-parse" rev)))
711fc8f6 386
e0d10e1c 387(defun git-config (key)
75a8180d 388 "Retrieve the value associated to KEY in the git repository config file."
9ddf6d7c 389 (let ((str (git-call-process-string "config" key)))
75a8180d
AJ
390 (and str (car (split-string str "\n")))))
391
711fc8f6
AJ
392(defun git-symbolic-ref (ref)
393 "Wrapper for the git-symbolic-ref command."
9ddf6d7c 394 (let ((str (git-call-process-string "symbolic-ref" ref)))
9de83169 395 (and str (car (split-string str "\n")))))
711fc8f6 396
413689d3 397(defun git-update-ref (ref newval &optional oldval reason)
711fc8f6 398 "Update a reference by calling git-update-ref."
413689d3 399 (let ((args (and oldval (list oldval))))
db18a182 400 (when newval (push newval args))
413689d3
AJ
401 (push ref args)
402 (when reason
403 (push reason args)
404 (push "-m" args))
db18a182 405 (unless newval (push "-d" args))
0520e215 406 (apply 'git-call-process-display-error "update-ref" args)))
711fc8f6 407
c375e9d0
AJ
408(defun git-for-each-ref (&rest specs)
409 "Return a list of refs using git-for-each-ref.
410Each entry is a cons of (SHORT-NAME . FULL-NAME)."
411 (let (refs)
412 (with-temp-buffer
413 (apply #'git-call-process t "for-each-ref" "--format=%(refname)" specs)
414 (goto-char (point-min))
415 (while (re-search-forward "^[^/\n]+/[^/\n]+/\\(.+\\)$" nil t)
416 (push (cons (match-string 1) (match-string 0)) refs)))
417 (nreverse refs)))
418
711fc8f6
AJ
419(defun git-read-tree (tree &optional index-file)
420 "Read a tree into the index file."
36d2078f
AJ
421 (let ((process-environment
422 (append (and index-file (list (concat "GIT_INDEX_FILE=" index-file))) process-environment)))
423 (apply 'git-call-process-display-error "read-tree" (if tree (list tree)))))
711fc8f6
AJ
424
425(defun git-write-tree (&optional index-file)
426 "Call git-write-tree and return the resulting tree SHA1 as a string."
36d2078f
AJ
427 (let ((process-environment
428 (append (and index-file (list (concat "GIT_INDEX_FILE=" index-file))) process-environment)))
429 (git-get-string-sha1
430 (git-call-process-string-display-error "write-tree"))))
711fc8f6 431
8918f5cf
DK
432(defun git-commit-tree (buffer tree parent)
433 "Create a commit and possibly update HEAD.
434Create a commit with the message in BUFFER using the tree with hash TREE.
435Use PARENT as the parent of the new commit. If PARENT is the current \"HEAD\",
436update the \"HEAD\" reference to the new commit."
711fc8f6
AJ
437 (let ((author-name (git-get-committer-name))
438 (author-email (git-get-committer-email))
413689d3 439 (subject "commit (initial): ")
14b4f2db 440 author-date log-start log-end args coding-system-for-write)
8918f5cf 441 (when parent
413689d3 442 (setq subject "commit: ")
711fc8f6 443 (push "-p" args)
8918f5cf 444 (push parent args))
711fc8f6
AJ
445 (with-current-buffer buffer
446 (goto-char (point-min))
447 (if
a79656e6 448 (setq log-start (re-search-forward (concat "^" (regexp-quote git-log-msg-separator) "\n") nil t))
711fc8f6
AJ
449 (save-restriction
450 (narrow-to-region (point-min) log-start)
451 (goto-char (point-min))
452 (when (re-search-forward "^Author: +\\(.*?\\) *<\\(.*\\)> *$" nil t)
453 (setq author-name (match-string 1)
454 author-email (match-string 2)))
455 (goto-char (point-min))
456 (when (re-search-forward "^Date: +\\(.*\\)$" nil t)
457 (setq author-date (match-string 1)))
458 (goto-char (point-min))
6fb20426
AJ
459 (when (re-search-forward "^Merge: +\\(.*\\)" nil t)
460 (setq subject "commit (merge): ")
461 (dolist (parent (split-string (match-string 1) " +" t))
711fc8f6 462 (push "-p" args)
6fb20426 463 (push parent args))))
711fc8f6 464 (setq log-start (point-min)))
14b4f2db 465 (setq log-end (point-max))
413689d3
AJ
466 (goto-char log-start)
467 (when (re-search-forward ".*$" nil t)
468 (setq subject (concat subject (match-string 0))))
14b4f2db 469 (setq coding-system-for-write buffer-file-coding-system))
413689d3
AJ
470 (let ((commit
471 (git-get-string-sha1
a7da5c42
AJ
472 (let ((env `(("GIT_AUTHOR_NAME" . ,author-name)
473 ("GIT_AUTHOR_EMAIL" . ,author-email)
474 ("GIT_COMMITTER_NAME" . ,(git-get-committer-name))
475 ("GIT_COMMITTER_EMAIL" . ,(git-get-committer-email)))))
476 (when author-date (push `("GIT_AUTHOR_DATE" . ,author-date) env))
477 (apply #'git-run-command-region
478 buffer log-start log-end env
479 "commit-tree" tree (nreverse args))))))
8918f5cf 480 (when commit (git-update-ref "HEAD" commit parent subject))
a7da5c42 481 commit)))
711fc8f6
AJ
482
483(defun git-empty-db-p ()
484 "Check if the git db is empty (no commit done yet)."
9ddf6d7c 485 (not (eq 0 (git-call-process nil "rev-parse" "--verify" "HEAD"))))
711fc8f6
AJ
486
487(defun git-get-merge-heads ()
488 "Retrieve the merge heads from the MERGE_HEAD file if present."
489 (let (heads)
490 (when (file-readable-p ".git/MERGE_HEAD")
491 (with-temp-buffer
492 (insert-file-contents ".git/MERGE_HEAD" nil nil nil t)
493 (goto-char (point-min))
494 (while (re-search-forward "[0-9a-f]\\{40\\}" nil t)
495 (push (match-string 0) heads))))
496 (nreverse heads)))
497
b704e589
AJ
498(defun git-get-commit-description (commit)
499 "Get a one-line description of COMMIT."
500 (let ((coding-system-for-read (git-get-logoutput-coding-system)))
9ddf6d7c 501 (let ((descr (git-call-process-string "log" "--max-count=1" "--pretty=oneline" commit)))
b704e589
AJ
502 (if (and descr (string-match "\\`\\([0-9a-f]\\{40\\}\\) *\\(.*\\)$" descr))
503 (concat (substring (match-string 1 descr) 0 10) " - " (match-string 2 descr))
504 descr))))
505
711fc8f6
AJ
506;;;; File info structure
507;;;; ------------------------------------------------------------
508
509; fileinfo structure stolen from pcl-cvs
510(defstruct (git-fileinfo
511 (:copier nil)
512 (:constructor git-create-fileinfo (state name &optional old-perm new-perm rename-state orig-name marked))
513 (:conc-name git-fileinfo->))
514 marked ;; t/nil
515 state ;; current state
516 name ;; file name
517 old-perm new-perm ;; permission flags
518 rename-state ;; rename or copy state
519 orig-name ;; original name for renames or copies
433ee03f 520 needs-update ;; whether file needs to be updated
711fc8f6
AJ
521 needs-refresh) ;; whether file needs to be refreshed
522
523(defvar git-status nil)
524
72dc52bf
AJ
525(defun git-set-fileinfo-state (info state)
526 "Set the state of a file info."
527 (unless (eq (git-fileinfo->state info) state)
528 (setf (git-fileinfo->state info) state
40f162b0 529 (git-fileinfo->new-perm info) (git-fileinfo->old-perm info)
72dc52bf
AJ
530 (git-fileinfo->rename-state info) nil
531 (git-fileinfo->orig-name info) nil
433ee03f 532 (git-fileinfo->needs-update info) nil
72dc52bf 533 (git-fileinfo->needs-refresh info) t)))
711fc8f6 534
b9b7bab4 535(defun git-status-filenames-map (status func files &rest args)
21ba0e84
AJ
536 "Apply FUNC to the status files names in the FILES list.
537The list must be sorted."
1b655040 538 (when files
1b655040
AJ
539 (let ((file (pop files))
540 (node (ewoc-nth status 0)))
541 (while (and file node)
433ee03f
AJ
542 (let* ((info (ewoc-data node))
543 (name (git-fileinfo->name info)))
544 (if (string-lessp name file)
b9b7bab4 545 (setq node (ewoc-next status node))
433ee03f 546 (if (string-equal name file)
b9b7bab4
AJ
547 (apply func info args))
548 (setq file (pop files))))))))
549
550(defun git-set-filenames-state (status files state)
21ba0e84 551 "Set the state of a list of named files. The list must be sorted"
b9b7bab4 552 (when files
72dc52bf 553 (git-status-filenames-map status #'git-set-fileinfo-state files state)
1b655040
AJ
554 (unless state ;; delete files whose state has been set to nil
555 (ewoc-filter status (lambda (info) (git-fileinfo->state info))))))
556
711fc8f6
AJ
557(defun git-state-code (code)
558 "Convert from a string to a added/deleted/modified state."
559 (case (string-to-char code)
560 (?M 'modified)
561 (?? 'unknown)
562 (?A 'added)
563 (?D 'deleted)
564 (?U 'unmerged)
40f162b0 565 (?T 'modified)
711fc8f6
AJ
566 (t nil)))
567
568(defun git-status-code-as-string (code)
569 "Format a git status code as string."
570 (case code
571 ('modified (propertize "Modified" 'face 'git-status-face))
572 ('unknown (propertize "Unknown " 'face 'git-unknown-face))
573 ('added (propertize "Added " 'face 'git-status-face))
574 ('deleted (propertize "Deleted " 'face 'git-status-face))
575 ('unmerged (propertize "Unmerged" 'face 'git-unmerged-face))
576 ('uptodate (propertize "Uptodate" 'face 'git-uptodate-face))
577 ('ignored (propertize "Ignored " 'face 'git-ignored-face))
578 (t "? ")))
579
ef40b3ef
AJ
580(defun git-file-type-as-string (old-perm new-perm)
581 "Return a string describing the file type based on its permissions."
582 (let* ((old-type (lsh (or old-perm 0) -9))
583 (new-type (lsh (or new-perm 0) -9))
40f162b0 584 (str (case new-type
6c4f70d5 585 (64 ;; file
40f162b0 586 (case old-type
6c4f70d5
AJ
587 (64 nil)
588 (80 " (type change symlink -> file)")
589 (112 " (type change subproject -> file)")))
590 (80 ;; symlink
40f162b0 591 (case old-type
6c4f70d5
AJ
592 (64 " (type change file -> symlink)")
593 (112 " (type change subproject -> symlink)")
40f162b0 594 (t " (symlink)")))
6c4f70d5 595 (112 ;; subproject
40f162b0 596 (case old-type
6c4f70d5
AJ
597 (64 " (type change file -> subproject)")
598 (80 " (type change symlink -> subproject)")
40f162b0 599 (t " (subproject)")))
6c4f70d5
AJ
600 (72 nil) ;; directory (internal, not a real git state)
601 (0 ;; deleted or unknown
40f162b0 602 (case old-type
6c4f70d5
AJ
603 (80 " (symlink)")
604 (112 " (subproject)")))
40f162b0 605 (t (format " (unknown type %o)" new-type)))))
3f3d564a 606 (cond (str (propertize str 'face 'git-status-face))
6c4f70d5 607 ((eq new-type 72) "/")
3f3d564a 608 (t ""))))
40f162b0 609
711fc8f6
AJ
610(defun git-rename-as-string (info)
611 "Return a string describing the copy or rename associated with INFO, or an empty string if none."
612 (let ((state (git-fileinfo->rename-state info)))
613 (if state
614 (propertize
615 (concat " ("
616 (if (eq state 'copy) "copied from "
c530c5aa
AJ
617 (if (eq (git-fileinfo->state info) 'added) "renamed from "
618 "renamed to "))
711fc8f6
AJ
619 (git-escape-file-name (git-fileinfo->orig-name info))
620 ")") 'face 'git-status-face)
621 "")))
622
623(defun git-permissions-as-string (old-perm new-perm)
624 "Format a permission change as string."
625 (propertize
626 (if (or (not old-perm)
627 (not new-perm)
18e3e99e 628 (eq 0 (logand ?\111 (logxor old-perm new-perm))))
711fc8f6 629 " "
18e3e99e 630 (if (eq 0 (logand ?\111 old-perm)) "+x" "-x"))
711fc8f6
AJ
631 'face 'git-permission-face))
632
633(defun git-fileinfo-prettyprint (info)
634 "Pretty-printer for the git-fileinfo structure."
ef40b3ef
AJ
635 (let ((old-perm (git-fileinfo->old-perm info))
636 (new-perm (git-fileinfo->new-perm info)))
637 (insert (concat " " (if (git-fileinfo->marked info) (propertize "*" 'face 'git-mark-face) " ")
638 " " (git-status-code-as-string (git-fileinfo->state info))
639 " " (git-permissions-as-string old-perm new-perm)
640 " " (git-escape-file-name (git-fileinfo->name info))
641 (git-file-type-as-string old-perm new-perm)
642 (git-rename-as-string info)))))
711fc8f6 643
433ee03f
AJ
644(defun git-update-node-fileinfo (node info)
645 "Update the fileinfo of the specified node. The names are assumed to match already."
646 (let ((data (ewoc-data node)))
647 (setf
648 ;; preserve the marked flag
649 (git-fileinfo->marked info) (git-fileinfo->marked data)
650 (git-fileinfo->needs-update data) nil)
651 (when (not (equal info data))
652 (setf (git-fileinfo->needs-refresh info) t
653 (ewoc-data node) info))))
654
655(defun git-insert-info-list (status infolist files)
656 "Insert a sorted list of file infos in the status buffer, replacing existing ones if any."
657 (let* ((info (pop infolist))
658 (node (ewoc-nth status 0))
659 (name (and info (git-fileinfo->name info)))
660 remaining)
1b655040 661 (while info
433ee03f
AJ
662 (let ((nodename (and node (git-fileinfo->name (ewoc-data node)))))
663 (while (and files (string-lessp (car files) name))
664 (push (pop files) remaining))
665 (when (and files (string-equal (car files) name))
666 (setq files (cdr files)))
667 (cond ((not nodename)
668 (setq node (ewoc-enter-last status info))
669 (setq info (pop infolist))
670 (setq name (and info (git-fileinfo->name info))))
671 ((string-lessp nodename name)
672 (setq node (ewoc-next status node)))
673 ((string-equal nodename name)
674 ;; preserve the marked flag
675 (git-update-node-fileinfo node info)
676 (setq info (pop infolist))
677 (setq name (and info (git-fileinfo->name info))))
678 (t
679 (setq node (ewoc-enter-before status node info))
680 (setq info (pop infolist))
681 (setq name (and info (git-fileinfo->name info)))))))
682 (nconc (nreverse remaining) files)))
93c22eeb
AJ
683
684(defun git-run-diff-index (status files)
685 "Run git-diff-index on FILES and parse the results into STATUS.
686Return the list of files that haven't been handled."
433ee03f 687 (let (infolist)
93c22eeb 688 (with-temp-buffer
9ddf6d7c 689 (apply #'git-call-process t "diff-index" "-z" "-M" "HEAD" "--" files)
93c22eeb
AJ
690 (goto-char (point-min))
691 (while (re-search-forward
40f162b0 692 ":\\([0-7]\\{6\\}\\) \\([0-7]\\{6\\}\\) [0-9a-f]\\{40\\} [0-9a-f]\\{40\\} \\(\\([ADMUT]\\)\0\\([^\0]+\\)\\|\\([CR]\\)[0-9]*\0\\([^\0]+\\)\0\\([^\0]+\\)\\)\0"
93c22eeb
AJ
693 nil t 1)
694 (let ((old-perm (string-to-number (match-string 1) 8))
695 (new-perm (string-to-number (match-string 2) 8))
696 (state (or (match-string 4) (match-string 6)))
697 (name (or (match-string 5) (match-string 7)))
698 (new-name (match-string 8)))
699 (if new-name ; copy or rename
700 (if (eq ?C (string-to-char state))
1b655040
AJ
701 (push (git-create-fileinfo 'added new-name old-perm new-perm 'copy name) infolist)
702 (push (git-create-fileinfo 'deleted name 0 0 'rename new-name) infolist)
703 (push (git-create-fileinfo 'added new-name old-perm new-perm 'rename name) infolist))
433ee03f
AJ
704 (push (git-create-fileinfo (git-state-code state) name old-perm new-perm) infolist)))))
705 (setq infolist (sort (nreverse infolist)
706 (lambda (info1 info2)
707 (string-lessp (git-fileinfo->name info1)
708 (git-fileinfo->name info2)))))
709 (git-insert-info-list status infolist files)))
711fc8f6
AJ
710
711(defun git-find-status-file (status file)
712 "Find a given file in the status ewoc and return its node."
713 (let ((node (ewoc-nth status 0)))
714 (while (and node (not (string= file (git-fileinfo->name (ewoc-data node)))))
715 (setq node (ewoc-next status node)))
716 node))
717
93c22eeb
AJ
718(defun git-run-ls-files (status files default-state &rest options)
719 "Run git-ls-files on FILES and parse the results into STATUS.
720Return the list of files that haven't been handled."
1b655040 721 (let (infolist)
93c22eeb 722 (with-temp-buffer
9ddf6d7c 723 (apply #'git-call-process t "ls-files" "-z" (append options (list "--") files))
93c22eeb 724 (goto-char (point-min))
3f3d564a 725 (while (re-search-forward "\\([^\0]*?\\)\\(/?\\)\0" nil t 1)
1b655040 726 (let ((name (match-string 1)))
3f3d564a
AJ
727 (push (git-create-fileinfo default-state name 0
728 (if (string-equal "/" (match-string 2)) (lsh ?\110 9) 0))
433ee03f
AJ
729 infolist))))
730 (setq infolist (nreverse infolist)) ;; assume it is sorted already
731 (git-insert-info-list status infolist files)))
93c22eeb 732
5e3cb7e5
AJ
733(defun git-run-ls-files-cached (status files default-state)
734 "Run git-ls-files -c on FILES and parse the results into STATUS.
735Return the list of files that haven't been handled."
433ee03f 736 (let (infolist)
5e3cb7e5 737 (with-temp-buffer
9ddf6d7c 738 (apply #'git-call-process t "ls-files" "-z" "-s" "-c" "--" files)
5e3cb7e5 739 (goto-char (point-min))
87e3d812 740 (while (re-search-forward "\\([0-7]\\{6\\}\\) [0-9a-f]\\{40\\} 0\t\\([^\0]+\\)\0" nil t)
5e3cb7e5
AJ
741 (let* ((new-perm (string-to-number (match-string 1) 8))
742 (old-perm (if (eq default-state 'added) 0 new-perm))
743 (name (match-string 2)))
433ee03f
AJ
744 (push (git-create-fileinfo default-state name old-perm new-perm) infolist))))
745 (setq infolist (nreverse infolist)) ;; assume it is sorted already
746 (git-insert-info-list status infolist files)))
5e3cb7e5 747
93c22eeb
AJ
748(defun git-run-ls-unmerged (status files)
749 "Run git-ls-files -u on FILES and parse the results into STATUS."
750 (with-temp-buffer
9ddf6d7c 751 (apply #'git-call-process t "ls-files" "-z" "-u" "--" files)
93c22eeb
AJ
752 (goto-char (point-min))
753 (let (unmerged-files)
754 (while (re-search-forward "[0-7]\\{6\\} [0-9a-f]\\{40\\} [123]\t\\([^\0]+\\)\0" nil t)
1b655040 755 (push (match-string 1) unmerged-files))
21ba0e84 756 (setq unmerged-files (nreverse unmerged-files)) ;; assume it is sorted already
1b655040 757 (git-set-filenames-state status unmerged-files 'unmerged))))
93c22eeb 758
274e13e0
AJ
759(defun git-get-exclude-files ()
760 "Get the list of exclude files to pass to git-ls-files."
761 (let (files
762 (config (git-config "core.excludesfile")))
763 (when (file-readable-p ".git/info/exclude")
764 (push ".git/info/exclude" files))
765 (when (and config (file-readable-p config))
766 (push config files))
767 files))
768
98acc3fa
AJ
769(defun git-run-ls-files-with-excludes (status files default-state &rest options)
770 "Run git-ls-files on FILES with appropriate --exclude-from options."
771 (let ((exclude-files (git-get-exclude-files)))
21a2d69b 772 (apply #'git-run-ls-files status files default-state "--directory" "--no-empty-directory"
98acc3fa
AJ
773 (concat "--exclude-per-directory=" git-per-dir-ignore-file)
774 (append options (mapcar (lambda (f) (concat "--exclude-from=" f)) exclude-files)))))
775
c4e8b72f 776(defun git-update-status-files (&optional files mark-files)
21ba0e84
AJ
777 "Update the status of FILES from the index.
778The FILES list must be sorted."
711fc8f6 779 (unless git-status (error "Not in git-status buffer."))
433ee03f 780 ;; set the needs-update flag on existing files
21ba0e84 781 (if files
433ee03f
AJ
782 (git-status-filenames-map
783 git-status (lambda (info) (setf (git-fileinfo->needs-update info) t)) files)
784 (ewoc-map (lambda (info) (setf (git-fileinfo->needs-update info) t) nil) git-status)
785 (git-call-process nil "update-index" "--refresh")
786 (when git-show-uptodate
787 (git-run-ls-files-cached git-status nil 'uptodate)))
21ba0e84 788 (let ((remaining-files
93c22eeb 789 (if (git-empty-db-p) ; we need some special handling for an empty db
5e3cb7e5 790 (git-run-ls-files-cached git-status files 'added)
98acc3fa
AJ
791 (git-run-diff-index git-status files))))
792 (git-run-ls-unmerged git-status files)
793 (when (or remaining-files (and git-show-unknown (not files)))
794 (setq remaining-files (git-run-ls-files-with-excludes git-status remaining-files 'unknown "-o")))
795 (when (or remaining-files (and git-show-ignored (not files)))
796 (setq remaining-files (git-run-ls-files-with-excludes git-status remaining-files 'ignored "-o" "-i")))
433ee03f
AJ
797 (unless files
798 (setq remaining-files (git-get-filenames (ewoc-collect git-status #'git-fileinfo->needs-update))))
799 (when remaining-files
800 (setq remaining-files (git-run-ls-files-cached git-status remaining-files 'uptodate)))
801 (git-set-filenames-state git-status remaining-files nil)
c4e8b72f 802 (when mark-files (git-mark-files git-status files))
93c22eeb 803 (git-refresh-files)
98acc3fa 804 (git-refresh-ewoc-hf git-status)))
711fc8f6 805
76127b3a
AJ
806(defun git-mark-files (status files)
807 "Mark all the specified FILES, and unmark the others."
76127b3a
AJ
808 (let ((file (and files (pop files)))
809 (node (ewoc-nth status 0)))
810 (while node
811 (let ((info (ewoc-data node)))
812 (if (and file (string-equal (git-fileinfo->name info) file))
813 (progn
814 (unless (git-fileinfo->marked info)
815 (setf (git-fileinfo->marked info) t)
816 (setf (git-fileinfo->needs-refresh info) t))
817 (setq file (pop files))
818 (setq node (ewoc-next status node)))
819 (when (git-fileinfo->marked info)
820 (setf (git-fileinfo->marked info) nil)
821 (setf (git-fileinfo->needs-refresh info) t))
822 (if (and file (string-lessp file (git-fileinfo->name info)))
823 (setq file (pop files))
824 (setq node (ewoc-next status node))))))))
825
711fc8f6
AJ
826(defun git-marked-files ()
827 "Return a list of all marked files, or if none a list containing just the file at cursor position."
828 (unless git-status (error "Not in git-status buffer."))
829 (or (ewoc-collect git-status (lambda (info) (git-fileinfo->marked info)))
830 (list (ewoc-data (ewoc-locate git-status)))))
831
832(defun git-marked-files-state (&rest states)
21ba0e84 833 "Return a sorted list of marked files that are in the specified states."
711fc8f6
AJ
834 (let ((files (git-marked-files))
835 result)
836 (dolist (info files)
837 (when (memq (git-fileinfo->state info) states)
838 (push info result)))
21ba0e84 839 (nreverse result)))
711fc8f6
AJ
840
841(defun git-refresh-files ()
842 "Refresh all files that need it and clear the needs-refresh flag."
843 (unless git-status (error "Not in git-status buffer."))
844 (ewoc-map
845 (lambda (info)
846 (let ((refresh (git-fileinfo->needs-refresh info)))
847 (setf (git-fileinfo->needs-refresh info) nil)
848 refresh))
849 git-status)
850 ; move back to goal column
851 (when goal-column (move-to-column goal-column)))
852
853(defun git-refresh-ewoc-hf (status)
854 "Refresh the ewoc header and footer."
855 (let ((branch (git-symbolic-ref "HEAD"))
856 (head (if (git-empty-db-p) "Nothing committed yet"
b704e589 857 (git-get-commit-description "HEAD")))
711fc8f6
AJ
858 (merge-heads (git-get-merge-heads)))
859 (ewoc-set-hf status
860 (format "Directory: %s\nBranch: %s\nHead: %s%s\n"
861 default-directory
ef08c149
AJ
862 (if branch
863 (if (string-match "^refs/heads/" branch)
864 (substring branch (match-end 0))
865 branch)
866 "none (detached HEAD)")
711fc8f6
AJ
867 head
868 (if merge-heads
869 (concat "\nMerging: "
b704e589 870 (mapconcat (lambda (str) (git-get-commit-description str)) merge-heads "\n "))
711fc8f6
AJ
871 ""))
872 (if (ewoc-nth status 0) "" " No changes."))))
873
874(defun git-get-filenames (files)
875 (mapcar (lambda (info) (git-fileinfo->name info)) files))
876
877(defun git-update-index (index-file files)
878 "Run git-update-index on a list of files."
36d2078f
AJ
879 (let ((process-environment (append (and index-file (list (concat "GIT_INDEX_FILE=" index-file)))
880 process-environment))
711fc8f6
AJ
881 added deleted modified)
882 (dolist (info files)
883 (case (git-fileinfo->state info)
884 ('added (push info added))
885 ('deleted (push info deleted))
886 ('modified (push info modified))))
36d2078f
AJ
887 (and
888 (or (not added) (apply #'git-call-process-display-error "update-index" "--add" "--" (git-get-filenames added)))
889 (or (not deleted) (apply #'git-call-process-display-error "update-index" "--remove" "--" (git-get-filenames deleted)))
890 (or (not modified) (apply #'git-call-process-display-error "update-index" "--" (git-get-filenames modified))))))
711fc8f6 891
d55552f6
AJ
892(defun git-run-pre-commit-hook ()
893 "Run the pre-commit hook if any."
894 (unless git-status (error "Not in git-status buffer."))
895 (let ((files (git-marked-files-state 'added 'deleted 'modified)))
896 (or (not files)
897 (not (file-executable-p ".git/hooks/pre-commit"))
898 (let ((index-file (make-temp-file "gitidx")))
899 (unwind-protect
900 (let ((head-tree (unless (git-empty-db-p) (git-rev-parse "HEAD^{tree}"))))
901 (git-read-tree head-tree index-file)
902 (git-update-index index-file files)
903 (git-run-hook "pre-commit" `(("GIT_INDEX_FILE" . ,index-file))))
904 (delete-file index-file))))))
905
711fc8f6
AJ
906(defun git-do-commit ()
907 "Perform the actual commit using the current buffer as log message."
908 (interactive)
909 (let ((buffer (current-buffer))
910 (index-file (make-temp-file "gitidx")))
911 (with-current-buffer log-edit-parent-buffer
912 (if (git-marked-files-state 'unmerged)
913 (message "You cannot commit unmerged files, resolve them first.")
914 (unwind-protect
915 (let ((files (git-marked-files-state 'added 'deleted 'modified))
36d2078f 916 head tree head-tree)
711fc8f6
AJ
917 (unless (git-empty-db-p)
918 (setq head (git-rev-parse "HEAD")
919 head-tree (git-rev-parse "HEAD^{tree}")))
1905a866
AJ
920 (message "Running git commit...")
921 (when
922 (and
923 (git-read-tree head-tree index-file)
924 (git-update-index nil files) ;update both the default index
925 (git-update-index index-file files) ;and the temporary one
926 (setq tree (git-write-tree index-file)))
927 (if (or (not (string-equal tree head-tree))
928 (yes-or-no-p "The tree was not modified, do you really want to perform an empty commit? "))
929 (let ((commit (git-commit-tree buffer tree head)))
930 (when commit
931 (condition-case nil (delete-file ".git/MERGE_HEAD") (error nil))
932 (condition-case nil (delete-file ".git/MERGE_MSG") (error nil))
933 (with-current-buffer buffer (erase-buffer))
934 (git-update-status-files (git-get-filenames files))
935 (git-call-process nil "rerere")
936 (git-call-process nil "gc" "--auto")
937 (message "Committed %s." commit)
938 (git-run-hook "post-commit" nil)))
939 (message "Commit aborted."))))
711fc8f6
AJ
940 (delete-file index-file))))))
941
942
943;;;; Interactive functions
944;;;; ------------------------------------------------------------
945
946(defun git-mark-file ()
947 "Mark the file that the cursor is on and move to the next one."
948 (interactive)
949 (unless git-status (error "Not in git-status buffer."))
950 (let* ((pos (ewoc-locate git-status))
951 (info (ewoc-data pos)))
952 (setf (git-fileinfo->marked info) t)
953 (ewoc-invalidate git-status pos)
954 (ewoc-goto-next git-status 1)))
955
956(defun git-unmark-file ()
957 "Unmark the file that the cursor is on and move to the next one."
958 (interactive)
959 (unless git-status (error "Not in git-status buffer."))
960 (let* ((pos (ewoc-locate git-status))
961 (info (ewoc-data pos)))
962 (setf (git-fileinfo->marked info) nil)
963 (ewoc-invalidate git-status pos)
964 (ewoc-goto-next git-status 1)))
965
966(defun git-unmark-file-up ()
967 "Unmark the file that the cursor is on and move to the previous one."
968 (interactive)
969 (unless git-status (error "Not in git-status buffer."))
970 (let* ((pos (ewoc-locate git-status))
971 (info (ewoc-data pos)))
972 (setf (git-fileinfo->marked info) nil)
973 (ewoc-invalidate git-status pos)
974 (ewoc-goto-prev git-status 1)))
975
976(defun git-mark-all ()
977 "Mark all files."
978 (interactive)
979 (unless git-status (error "Not in git-status buffer."))
2f6e86a8
AJ
980 (ewoc-map (lambda (info) (unless (git-fileinfo->marked info)
981 (setf (git-fileinfo->marked info) t))) git-status)
711fc8f6
AJ
982 ; move back to goal column after invalidate
983 (when goal-column (move-to-column goal-column)))
984
985(defun git-unmark-all ()
986 "Unmark all files."
987 (interactive)
988 (unless git-status (error "Not in git-status buffer."))
2f6e86a8
AJ
989 (ewoc-map (lambda (info) (when (git-fileinfo->marked info)
990 (setf (git-fileinfo->marked info) nil)
991 t)) git-status)
711fc8f6
AJ
992 ; move back to goal column after invalidate
993 (when goal-column (move-to-column goal-column)))
994
995(defun git-toggle-all-marks ()
996 "Toggle all file marks."
997 (interactive)
998 (unless git-status (error "Not in git-status buffer."))
999 (ewoc-map (lambda (info) (setf (git-fileinfo->marked info) (not (git-fileinfo->marked info))) t) git-status)
1000 ; move back to goal column after invalidate
1001 (when goal-column (move-to-column goal-column)))
1002
1003(defun git-next-file (&optional n)
1004 "Move the selection down N files."
1005 (interactive "p")
1006 (unless git-status (error "Not in git-status buffer."))
1007 (ewoc-goto-next git-status n))
1008
1009(defun git-prev-file (&optional n)
1010 "Move the selection up N files."
1011 (interactive "p")
1012 (unless git-status (error "Not in git-status buffer."))
1013 (ewoc-goto-prev git-status n))
1014
8a078c3f
AJ
1015(defun git-next-unmerged-file (&optional n)
1016 "Move the selection down N unmerged files."
1017 (interactive "p")
1018 (unless git-status (error "Not in git-status buffer."))
1019 (let* ((last (ewoc-locate git-status))
1020 (node (ewoc-next git-status last)))
1021 (while (and node (> n 0))
1022 (when (eq 'unmerged (git-fileinfo->state (ewoc-data node)))
1023 (setq n (1- n))
1024 (setq last node))
1025 (setq node (ewoc-next git-status node)))
1026 (ewoc-goto-node git-status last)))
1027
1028(defun git-prev-unmerged-file (&optional n)
1029 "Move the selection up N unmerged files."
1030 (interactive "p")
1031 (unless git-status (error "Not in git-status buffer."))
1032 (let* ((last (ewoc-locate git-status))
1033 (node (ewoc-prev git-status last)))
1034 (while (and node (> n 0))
1035 (when (eq 'unmerged (git-fileinfo->state (ewoc-data node)))
1036 (setq n (1- n))
1037 (setq last node))
1038 (setq node (ewoc-prev git-status node)))
1039 (ewoc-goto-node git-status last)))
1040
b0a53e9e
AJ
1041(defun git-insert-file (file)
1042 "Insert file(s) into the git-status buffer."
1043 (interactive "fInsert file: ")
1044 (git-update-status-files (list (file-relative-name file))))
1045
711fc8f6
AJ
1046(defun git-add-file ()
1047 "Add marked file(s) to the index cache."
1048 (interactive)
aaa68dd5 1049 (let ((files (git-get-filenames (git-marked-files-state 'unknown 'ignored 'unmerged))))
3f3d564a 1050 ;; FIXME: add support for directories
711fc8f6 1051 (unless files
93c22eeb 1052 (push (file-relative-name (read-file-name "File to add: " nil nil t)) files))
0520e215 1053 (when (apply 'git-call-process-display-error "update-index" "--add" "--" files)
433ee03f 1054 (git-update-status-files files)
0520e215 1055 (git-success-message "Added" files))))
711fc8f6
AJ
1056
1057(defun git-ignore-file ()
1058 "Add marked file(s) to the ignore list."
1059 (interactive)
93c22eeb 1060 (let ((files (git-get-filenames (git-marked-files-state 'unknown))))
711fc8f6 1061 (unless files
93c22eeb
AJ
1062 (push (file-relative-name (read-file-name "File to ignore: " nil nil t)) files))
1063 (dolist (f files) (git-append-to-ignore f))
433ee03f 1064 (git-update-status-files files)
9f5599b9 1065 (git-success-message "Ignored" files)))
711fc8f6
AJ
1066
1067(defun git-remove-file ()
1068 "Remove the marked file(s)."
1069 (interactive)
568d2cde 1070 (let ((files (git-get-filenames (git-marked-files-state 'added 'modified 'unknown 'uptodate 'ignored))))
711fc8f6 1071 (unless files
93c22eeb 1072 (push (file-relative-name (read-file-name "File to remove: " nil nil t)) files))
711fc8f6 1073 (if (yes-or-no-p
5b4e4410
AJ
1074 (if (cdr files)
1075 (format "Remove %d files? " (length files))
1076 (format "Remove %s? " (car files))))
711fc8f6 1077 (progn
93c22eeb 1078 (dolist (name files)
3f3d564a
AJ
1079 (ignore-errors
1080 (if (file-directory-p name)
1081 (delete-directory name)
1082 (delete-file name))))
0520e215 1083 (when (apply 'git-call-process-display-error "update-index" "--remove" "--" files)
433ee03f 1084 (git-update-status-files files)
0520e215 1085 (git-success-message "Removed" files)))
711fc8f6
AJ
1086 (message "Aborting"))))
1087
1088(defun git-revert-file ()
1089 "Revert changes to the marked file(s)."
1090 (interactive)
3f3d564a 1091 (let ((files (git-marked-files-state 'added 'deleted 'modified 'unmerged))
711fc8f6
AJ
1092 added modified)
1093 (when (and files
1094 (yes-or-no-p
5b4e4410
AJ
1095 (if (cdr files)
1096 (format "Revert %d files? " (length files))
1097 (format "Revert %s? " (git-fileinfo->name (car files))))))
711fc8f6
AJ
1098 (dolist (info files)
1099 (case (git-fileinfo->state info)
93c22eeb
AJ
1100 ('added (push (git-fileinfo->name info) added))
1101 ('deleted (push (git-fileinfo->name info) modified))
1102 ('unmerged (push (git-fileinfo->name info) modified))
1103 ('modified (push (git-fileinfo->name info) modified))))
928323af 1104 ;; check if a buffer contains one of the files and isn't saved
0520e215 1105 (dolist (file modified)
928323af
AJ
1106 (let ((buffer (get-file-buffer file)))
1107 (when (and buffer (buffer-modified-p buffer))
1108 (error "Buffer %s is modified. Please kill or save modified buffers before reverting." (buffer-name buffer)))))
0520e215
AJ
1109 (let ((ok (and
1110 (or (not added)
1111 (apply 'git-call-process-display-error "update-index" "--force-remove" "--" added))
1112 (or (not modified)
21ba0e84
AJ
1113 (apply 'git-call-process-display-error "checkout" "HEAD" modified))))
1114 (names (git-get-filenames files)))
1115 (git-update-status-files names)
0520e215
AJ
1116 (when ok
1117 (dolist (file modified)
1118 (let ((buffer (get-file-buffer file)))
1119 (when buffer (with-current-buffer buffer (revert-buffer t t t)))))
21ba0e84 1120 (git-success-message "Reverted" names))))))
711fc8f6 1121
711fc8f6
AJ
1122(defun git-remove-handled ()
1123 "Remove handled files from the status list."
1124 (interactive)
1125 (ewoc-filter git-status
1126 (lambda (info)
98acc3fa
AJ
1127 (case (git-fileinfo->state info)
1128 ('ignored git-show-ignored)
1129 ('uptodate git-show-uptodate)
1130 ('unknown git-show-unknown)
1131 (t t))))
711fc8f6
AJ
1132 (unless (ewoc-nth git-status 0) ; refresh header if list is empty
1133 (git-refresh-ewoc-hf git-status)))
1134
98acc3fa
AJ
1135(defun git-toggle-show-uptodate ()
1136 "Toogle the option for showing up-to-date files."
1137 (interactive)
1138 (if (setq git-show-uptodate (not git-show-uptodate))
1139 (git-refresh-status)
1140 (git-remove-handled)))
1141
1142(defun git-toggle-show-ignored ()
1143 "Toogle the option for showing ignored files."
1144 (interactive)
1145 (if (setq git-show-ignored (not git-show-ignored))
1146 (progn
9f5599b9 1147 (message "Inserting ignored files...")
98acc3fa
AJ
1148 (git-run-ls-files-with-excludes git-status nil 'ignored "-o" "-i")
1149 (git-refresh-files)
9f5599b9
AJ
1150 (git-refresh-ewoc-hf git-status)
1151 (message "Inserting ignored files...done"))
98acc3fa
AJ
1152 (git-remove-handled)))
1153
1154(defun git-toggle-show-unknown ()
1155 "Toogle the option for showing unknown files."
1156 (interactive)
1157 (if (setq git-show-unknown (not git-show-unknown))
1158 (progn
9f5599b9 1159 (message "Inserting unknown files...")
98acc3fa
AJ
1160 (git-run-ls-files-with-excludes git-status nil 'unknown "-o")
1161 (git-refresh-files)
9f5599b9
AJ
1162 (git-refresh-ewoc-hf git-status)
1163 (message "Inserting unknown files...done"))
98acc3fa
AJ
1164 (git-remove-handled)))
1165
3f3d564a
AJ
1166(defun git-expand-directory (info)
1167 "Expand the directory represented by INFO to list its files."
1168 (when (eq (lsh (git-fileinfo->new-perm info) -9) ?\110)
1169 (let ((dir (git-fileinfo->name info)))
1170 (git-set-filenames-state git-status (list dir) nil)
1171 (git-run-ls-files-with-excludes git-status (list (concat dir "/")) 'unknown "-o")
1172 (git-refresh-files)
1173 (git-refresh-ewoc-hf git-status)
1174 t)))
1175
711fc8f6
AJ
1176(defun git-setup-diff-buffer (buffer)
1177 "Setup a buffer for displaying a diff."
8fdc3972
AJ
1178 (let ((dir default-directory))
1179 (with-current-buffer buffer
1180 (diff-mode)
1181 (goto-char (point-min))
1182 (setq default-directory dir)
1183 (setq buffer-read-only t)))
711fc8f6 1184 (display-buffer buffer)
8b30aa50
AJ
1185 ; shrink window only if it displays the status buffer
1186 (when (eq (window-buffer) (current-buffer))
1187 (shrink-window-if-larger-than-buffer)))
711fc8f6
AJ
1188
1189(defun git-diff-file ()
1190 "Diff the marked file(s) against HEAD."
1191 (interactive)
1192 (let ((files (git-marked-files)))
1193 (git-setup-diff-buffer
1194 (apply #'git-run-command-buffer "*git-diff*" "diff-index" "-p" "-M" "HEAD" "--" (git-get-filenames files)))))
1195
2b1c0ef2
AJ
1196(defun git-diff-file-merge-head (arg)
1197 "Diff the marked file(s) against the first merge head (or the nth one with a numeric prefix)."
1198 (interactive "p")
1199 (let ((files (git-marked-files))
1200 (merge-heads (git-get-merge-heads)))
1201 (unless merge-heads (error "No merge in progress"))
1202 (git-setup-diff-buffer
1203 (apply #'git-run-command-buffer "*git-diff*" "diff-index" "-p" "-M"
1204 (or (nth (1- arg) merge-heads) "HEAD") "--" (git-get-filenames files)))))
1205
711fc8f6
AJ
1206(defun git-diff-unmerged-file (stage)
1207 "Diff the marked unmerged file(s) against the specified stage."
1208 (let ((files (git-marked-files)))
1209 (git-setup-diff-buffer
1210 (apply #'git-run-command-buffer "*git-diff*" "diff-files" "-p" stage "--" (git-get-filenames files)))))
1211
1212(defun git-diff-file-base ()
1213 "Diff the marked unmerged file(s) against the common base file."
1214 (interactive)
1215 (git-diff-unmerged-file "-1"))
1216
1217(defun git-diff-file-mine ()
1218 "Diff the marked unmerged file(s) against my pre-merge version."
1219 (interactive)
1220 (git-diff-unmerged-file "-2"))
1221
1222(defun git-diff-file-other ()
1223 "Diff the marked unmerged file(s) against the other's pre-merge version."
1224 (interactive)
1225 (git-diff-unmerged-file "-3"))
1226
1227(defun git-diff-file-combined ()
1228 "Do a combined diff of the marked unmerged file(s)."
1229 (interactive)
1230 (git-diff-unmerged-file "-c"))
1231
1232(defun git-diff-file-idiff ()
1233 "Perform an interactive diff on the current file."
1234 (interactive)
09afcd69
AJ
1235 (let ((files (git-marked-files-state 'added 'deleted 'modified)))
1236 (unless (eq 1 (length files))
1237 (error "Cannot perform an interactive diff on multiple files."))
1238 (let* ((filename (car (git-get-filenames files)))
1239 (buff1 (find-file-noselect filename))
1240 (buff2 (git-run-command-buffer (concat filename ".~HEAD~") "cat-file" "blob" (concat "HEAD:" filename))))
1241 (ediff-buffers buff1 buff2))))
711fc8f6
AJ
1242
1243(defun git-log-file ()
1244 "Display a log of changes to the marked file(s)."
1245 (interactive)
1246 (let* ((files (git-marked-files))
1247 (coding-system-for-read git-commits-coding-system)
1248 (buffer (apply #'git-run-command-buffer "*git-log*" "rev-list" "--pretty" "HEAD" "--" (git-get-filenames files))))
1249 (with-current-buffer buffer
1250 ; (git-log-mode) FIXME: implement log mode
1251 (goto-char (point-min))
1252 (setq buffer-read-only t))
1253 (display-buffer buffer)))
1254
1255(defun git-log-edit-files ()
1256 "Return a list of marked files for use in the log-edit buffer."
1257 (with-current-buffer log-edit-parent-buffer
1258 (git-get-filenames (git-marked-files-state 'added 'deleted 'modified))))
1259
8b30aa50
AJ
1260(defun git-log-edit-diff ()
1261 "Run a diff of the current files being committed from a log-edit buffer."
1262 (with-current-buffer log-edit-parent-buffer
1263 (git-diff-file)))
1264
38448147
AJ
1265(defun git-append-sign-off (name email)
1266 "Append a Signed-off-by entry to the current buffer, avoiding duplicates."
1267 (let ((sign-off (format "Signed-off-by: %s <%s>" name email))
1268 (case-fold-search t))
1269 (goto-char (point-min))
1270 (unless (re-search-forward (concat "^" (regexp-quote sign-off)) nil t)
1271 (goto-char (point-min))
1272 (unless (re-search-forward "^Signed-off-by: " nil t)
1273 (setq sign-off (concat "\n" sign-off)))
1274 (goto-char (point-max))
1275 (insert sign-off "\n"))))
1276
ef5133df 1277(defun git-setup-log-buffer (buffer &optional merge-heads author-name author-email subject date msg)
60fa08ed 1278 "Setup the log buffer for a commit."
711fc8f6 1279 (unless git-status (error "Not in git-status buffer."))
ef5133df 1280 (let ((dir default-directory)
38448147
AJ
1281 (committer-name (git-get-committer-name))
1282 (committer-email (git-get-committer-email))
45033ad9 1283 (sign-off git-append-signed-off-by))
711fc8f6 1284 (with-current-buffer buffer
60fa08ed
AJ
1285 (cd dir)
1286 (erase-buffer)
1287 (insert
1288 (propertize
1289 (format "Author: %s <%s>\n%s%s"
1290 (or author-name committer-name)
1291 (or author-email committer-email)
1292 (if date (format "Date: %s\n" date) "")
1293 (if merge-heads
6fb20426
AJ
1294 (format "Merge: %s\n"
1295 (mapconcat 'identity merge-heads " "))
60fa08ed
AJ
1296 ""))
1297 'face 'git-header-face)
1298 (propertize git-log-msg-separator 'face 'git-separator-face)
1299 "\n")
1300 (when subject (insert subject "\n\n"))
1301 (cond (msg (insert msg "\n"))
51ef1daa
JS
1302 ((file-readable-p ".git/rebase-apply/msg")
1303 (insert-file-contents ".git/rebase-apply/msg"))
60fa08ed
AJ
1304 ((file-readable-p ".git/MERGE_MSG")
1305 (insert-file-contents ".git/MERGE_MSG")))
1306 ; delete empty lines at end
1307 (goto-char (point-min))
1308 (when (re-search-forward "\n+\\'" nil t)
1309 (replace-match "\n" t t))
76127b3a
AJ
1310 (when sign-off (git-append-sign-off committer-name committer-email)))
1311 buffer))
60fa08ed 1312
485cdb9b
LM
1313(define-derived-mode git-log-edit-mode log-edit-mode "Git-Log-Edit"
1314 "Major mode for editing git log messages.
1315
1316Set up git-specific `font-lock-keywords' for `log-edit-mode'."
1317 (set (make-local-variable 'font-lock-defaults)
1318 '(git-log-edit-font-lock-keywords t t)))
1319
60fa08ed
AJ
1320(defun git-commit-file ()
1321 "Commit the marked file(s), asking for a commit message."
1322 (interactive)
1323 (unless git-status (error "Not in git-status buffer."))
d55552f6
AJ
1324 (when (git-run-pre-commit-hook)
1325 (let ((buffer (get-buffer-create "*git-commit*"))
1326 (coding-system (git-get-commits-coding-system))
1327 author-name author-email subject date)
1328 (when (eq 0 (buffer-size buffer))
51ef1daa 1329 (when (file-readable-p ".git/rebase-apply/info")
d55552f6 1330 (with-temp-buffer
51ef1daa 1331 (insert-file-contents ".git/rebase-apply/info")
d55552f6
AJ
1332 (goto-char (point-min))
1333 (when (re-search-forward "^Author: \\(.*\\)\nEmail: \\(.*\\)$" nil t)
1334 (setq author-name (match-string 1))
1335 (setq author-email (match-string 2)))
1336 (goto-char (point-min))
1337 (when (re-search-forward "^Subject: \\(.*\\)$" nil t)
1338 (setq subject (match-string 1)))
1339 (goto-char (point-min))
1340 (when (re-search-forward "^Date: \\(.*\\)$" nil t)
1341 (setq date (match-string 1)))))
ef5133df 1342 (git-setup-log-buffer buffer (git-get-merge-heads) author-name author-email subject date))
8b30aa50
AJ
1343 (if (boundp 'log-edit-diff-function)
1344 (log-edit 'git-do-commit nil '((log-edit-listfun . git-log-edit-files)
485cdb9b
LM
1345 (log-edit-diff-function . git-log-edit-diff)) buffer 'git-log-edit-mode)
1346 (log-edit 'git-do-commit nil 'git-log-edit-files buffer
1347 'git-log-edit-mode))
efd49f50 1348 (setq paragraph-separate (concat (regexp-quote git-log-msg-separator) "$\\|Author: \\|Date: \\|Merge: \\|Signed-off-by: \\|\f\\|[ ]*$"))
d55552f6
AJ
1349 (setq buffer-file-coding-system coding-system)
1350 (re-search-forward (regexp-quote (concat git-log-msg-separator "\n")) nil t))))
711fc8f6 1351
76127b3a
AJ
1352(defun git-setup-commit-buffer (commit)
1353 "Setup the commit buffer with the contents of COMMIT."
ef5133df 1354 (let (parents author-name author-email subject date msg)
76127b3a
AJ
1355 (with-temp-buffer
1356 (let ((coding-system (git-get-logoutput-coding-system)))
ef5133df 1357 (git-call-process t "log" "-1" "--pretty=medium" "--abbrev=40" commit)
76127b3a 1358 (goto-char (point-min))
ef5133df
AJ
1359 (when (re-search-forward "^Merge: *\\(.*\\)$" nil t)
1360 (setq parents (cdr (split-string (match-string 1) " +"))))
76127b3a
AJ
1361 (when (re-search-forward "^Author: *\\(.*\\) <\\(.*\\)>$" nil t)
1362 (setq author-name (match-string 1))
1363 (setq author-email (match-string 2)))
1364 (when (re-search-forward "^Date: *\\(.*\\)$" nil t)
1365 (setq date (match-string 1)))
1366 (while (re-search-forward "^ \\(.*\\)$" nil t)
1367 (push (match-string 1) msg))
1368 (setq msg (nreverse msg))
1369 (setq subject (pop msg))
1370 (while (and msg (zerop (length (car msg))) (pop msg)))))
1371 (git-setup-log-buffer (get-buffer-create "*git-commit*")
ef5133df 1372 parents author-name author-email subject date
76127b3a
AJ
1373 (mapconcat #'identity msg "\n"))))
1374
1375(defun git-get-commit-files (commit)
21ba0e84 1376 "Retrieve a sorted list of files modified by COMMIT."
76127b3a
AJ
1377 (let (files)
1378 (with-temp-buffer
db18a182 1379 (git-call-process t "diff-tree" "-m" "-r" "-z" "--name-only" "--no-commit-id" "--root" commit)
76127b3a
AJ
1380 (goto-char (point-min))
1381 (while (re-search-forward "\\([^\0]*\\)\0" nil t 1)
1382 (push (match-string 1) files)))
21ba0e84 1383 (sort files #'string-lessp)))
76127b3a 1384
c375e9d0
AJ
1385(defun git-read-commit-name (prompt &optional default)
1386 "Ask for a commit name, with completion for local branch, remote branch and tag."
1387 (completing-read prompt
1388 (list* "HEAD" "ORIG_HEAD" "FETCH_HEAD" (mapcar #'car (git-for-each-ref)))
1389 nil nil nil nil default))
1390
1391(defun git-checkout (branch &optional merge)
1392 "Checkout a branch, tag, or any commit.
1393Use a prefix arg if git should merge while checking out."
1394 (interactive
1395 (list (git-read-commit-name "Checkout: ")
1396 current-prefix-arg))
1397 (unless git-status (error "Not in git-status buffer."))
1398 (let ((args (list branch "--")))
1399 (when merge (push "-m" args))
1400 (when (apply #'git-call-process-display-error "checkout" args)
1401 (git-update-status-files))))
1402
811b10c7
AJ
1403(defun git-branch (branch)
1404 "Create a branch from the current HEAD and switch to it."
1405 (interactive (list (git-read-commit-name "Branch: ")))
1406 (unless git-status (error "Not in git-status buffer."))
1407 (if (git-rev-parse (concat "refs/heads/" branch))
1408 (if (yes-or-no-p (format "Branch %s already exists, replace it? " branch))
1409 (and (git-call-process-display-error "branch" "-f" branch)
1410 (git-call-process-display-error "checkout" branch))
1411 (message "Canceled."))
1412 (git-call-process-display-error "checkout" "-b" branch))
1413 (git-refresh-ewoc-hf git-status))
1414
76127b3a
AJ
1415(defun git-amend-commit ()
1416 "Undo the last commit on HEAD, and set things up to commit an
1417amended version of it."
1418 (interactive)
1419 (unless git-status (error "Not in git-status buffer."))
1420 (when (git-empty-db-p) (error "No commit to amend."))
1421 (let* ((commit (git-rev-parse "HEAD"))
1422 (files (git-get-commit-files commit)))
db18a182
AJ
1423 (when (if (git-rev-parse "HEAD^")
1424 (git-call-process-display-error "reset" "--soft" "HEAD^")
1425 (and (git-update-ref "ORIG_HEAD" commit)
1426 (git-update-ref "HEAD" nil commit)))
c4e8b72f 1427 (git-update-status-files files t)
0520e215
AJ
1428 (git-setup-commit-buffer commit)
1429 (git-commit-file))))
76127b3a 1430
ab69e3e4
AJ
1431(defun git-cherry-pick-commit (arg)
1432 "Cherry-pick a commit."
1433 (interactive (list (git-read-commit-name "Cherry-pick commit: ")))
1434 (unless git-status (error "Not in git-status buffer."))
1435 (let ((commit (git-rev-parse (concat arg "^0"))))
1436 (unless commit (error "Not a valid commit '%s'." arg))
1437 (when (git-rev-parse (concat commit "^2"))
1438 (error "Cannot cherry-pick a merge commit."))
1439 (let ((files (git-get-commit-files commit))
1440 (ok (git-call-process-display-error "cherry-pick" "-n" commit)))
1441 (git-update-status-files files ok)
1442 (with-current-buffer (git-setup-commit-buffer commit)
1443 (goto-char (point-min))
1444 (if (re-search-forward "^\n*Signed-off-by:" nil t 1)
1445 (goto-char (match-beginning 0))
1446 (goto-char (point-max)))
1447 (insert "(cherry picked from commit " commit ")\n"))
1448 (when ok (git-commit-file)))))
1449
1450(defun git-revert-commit (arg)
1451 "Revert a commit."
1452 (interactive (list (git-read-commit-name "Revert commit: ")))
1453 (unless git-status (error "Not in git-status buffer."))
1454 (let ((commit (git-rev-parse (concat arg "^0"))))
1455 (unless commit (error "Not a valid commit '%s'." arg))
1456 (when (git-rev-parse (concat commit "^2"))
1457 (error "Cannot revert a merge commit."))
1458 (let ((files (git-get-commit-files commit))
1459 (subject (git-get-commit-description commit))
1460 (ok (git-call-process-display-error "revert" "-n" commit)))
1461 (git-update-status-files files ok)
1462 (when (string-match "^[0-9a-f]+ - \\(.*\\)$" subject)
1463 (setq subject (match-string 1 subject)))
1464 (git-setup-log-buffer (get-buffer-create "*git-commit*")
1465 (git-get-merge-heads) nil nil (format "Revert \"%s\"" subject) nil
1466 (format "This reverts commit %s.\n" commit))
1467 (when ok (git-commit-file)))))
1468
711fc8f6
AJ
1469(defun git-find-file ()
1470 "Visit the current file in its own buffer."
1471 (interactive)
1472 (unless git-status (error "Not in git-status buffer."))
1473 (let ((info (ewoc-data (ewoc-locate git-status))))
3f3d564a
AJ
1474 (unless (git-expand-directory info)
1475 (find-file (git-fileinfo->name info))
1476 (when (eq 'unmerged (git-fileinfo->state info))
1477 (smerge-mode 1)))))
711fc8f6 1478
b8ee5181
AJ
1479(defun git-find-file-other-window ()
1480 "Visit the current file in its own buffer in another window."
1481 (interactive)
1482 (unless git-status (error "Not in git-status buffer."))
1483 (let ((info (ewoc-data (ewoc-locate git-status))))
1484 (find-file-other-window (git-fileinfo->name info))
1485 (when (eq 'unmerged (git-fileinfo->state info))
1486 (smerge-mode))))
1487
711fc8f6
AJ
1488(defun git-find-file-imerge ()
1489 "Visit the current file in interactive merge mode."
1490 (interactive)
1491 (unless git-status (error "Not in git-status buffer."))
1492 (let ((info (ewoc-data (ewoc-locate git-status))))
1493 (find-file (git-fileinfo->name info))
1494 (smerge-ediff)))
1495
1496(defun git-view-file ()
1497 "View the current file in its own buffer."
1498 (interactive)
1499 (unless git-status (error "Not in git-status buffer."))
1500 (let ((info (ewoc-data (ewoc-locate git-status))))
1501 (view-file (git-fileinfo->name info))))
1502
1503(defun git-refresh-status ()
1504 "Refresh the git status buffer."
1505 (interactive)
433ee03f
AJ
1506 (unless git-status (error "Not in git-status buffer."))
1507 (message "Refreshing git status...")
1508 (git-update-status-files)
1509 (message "Refreshing git status...done"))
711fc8f6
AJ
1510
1511(defun git-status-quit ()
1512 "Quit git-status mode."
1513 (interactive)
1514 (bury-buffer))
1515
1516;;;; Major Mode
1517;;;; ------------------------------------------------------------
1518
1519(defvar git-status-mode-hook nil
1520 "Run after `git-status-mode' is setup.")
1521
1522(defvar git-status-mode-map nil
1523 "Keymap for git major mode.")
1524
1525(defvar git-status nil
1526 "List of all files managed by the git-status mode.")
1527
1528(unless git-status-mode-map
1529 (let ((map (make-keymap))
76127b3a 1530 (commit-map (make-sparse-keymap))
98acc3fa
AJ
1531 (diff-map (make-sparse-keymap))
1532 (toggle-map (make-sparse-keymap)))
711fc8f6 1533 (suppress-keymap map)
5716e794
JN
1534 (define-key map "?" 'git-help)
1535 (define-key map "h" 'git-help)
711fc8f6
AJ
1536 (define-key map " " 'git-next-file)
1537 (define-key map "a" 'git-add-file)
1538 (define-key map "c" 'git-commit-file)
76127b3a 1539 (define-key map "\C-c" commit-map)
711fc8f6
AJ
1540 (define-key map "d" diff-map)
1541 (define-key map "=" 'git-diff-file)
1542 (define-key map "f" 'git-find-file)
18e3e99e 1543 (define-key map "\r" 'git-find-file)
711fc8f6
AJ
1544 (define-key map "g" 'git-refresh-status)
1545 (define-key map "i" 'git-ignore-file)
b0a53e9e 1546 (define-key map "I" 'git-insert-file)
711fc8f6
AJ
1547 (define-key map "l" 'git-log-file)
1548 (define-key map "m" 'git-mark-file)
1549 (define-key map "M" 'git-mark-all)
1550 (define-key map "n" 'git-next-file)
8a078c3f 1551 (define-key map "N" 'git-next-unmerged-file)
b8ee5181 1552 (define-key map "o" 'git-find-file-other-window)
711fc8f6 1553 (define-key map "p" 'git-prev-file)
8a078c3f 1554 (define-key map "P" 'git-prev-unmerged-file)
711fc8f6
AJ
1555 (define-key map "q" 'git-status-quit)
1556 (define-key map "r" 'git-remove-file)
98acc3fa 1557 (define-key map "t" toggle-map)
711fc8f6
AJ
1558 (define-key map "T" 'git-toggle-all-marks)
1559 (define-key map "u" 'git-unmark-file)
1560 (define-key map "U" 'git-revert-file)
1561 (define-key map "v" 'git-view-file)
1562 (define-key map "x" 'git-remove-handled)
1563 (define-key map "\C-?" 'git-unmark-file-up)
1564 (define-key map "\M-\C-?" 'git-unmark-all)
76127b3a
AJ
1565 ; the commit submap
1566 (define-key commit-map "\C-a" 'git-amend-commit)
811b10c7 1567 (define-key commit-map "\C-b" 'git-branch)
c375e9d0 1568 (define-key commit-map "\C-o" 'git-checkout)
ab69e3e4
AJ
1569 (define-key commit-map "\C-p" 'git-cherry-pick-commit)
1570 (define-key commit-map "\C-v" 'git-revert-commit)
711fc8f6
AJ
1571 ; the diff submap
1572 (define-key diff-map "b" 'git-diff-file-base)
1573 (define-key diff-map "c" 'git-diff-file-combined)
1574 (define-key diff-map "=" 'git-diff-file)
1575 (define-key diff-map "e" 'git-diff-file-idiff)
1576 (define-key diff-map "E" 'git-find-file-imerge)
2b1c0ef2 1577 (define-key diff-map "h" 'git-diff-file-merge-head)
711fc8f6
AJ
1578 (define-key diff-map "m" 'git-diff-file-mine)
1579 (define-key diff-map "o" 'git-diff-file-other)
98acc3fa
AJ
1580 ; the toggle submap
1581 (define-key toggle-map "u" 'git-toggle-show-uptodate)
1582 (define-key toggle-map "i" 'git-toggle-show-ignored)
1583 (define-key toggle-map "k" 'git-toggle-show-unknown)
1584 (define-key toggle-map "m" 'git-toggle-all-marks)
18ff365f
AJ
1585 (setq git-status-mode-map map))
1586 (easy-menu-define git-menu git-status-mode-map
1587 "Git Menu"
1588 `("Git"
1589 ["Refresh" git-refresh-status t]
1590 ["Commit" git-commit-file t]
c375e9d0 1591 ["Checkout..." git-checkout t]
811b10c7 1592 ["New Branch..." git-branch t]
ab69e3e4
AJ
1593 ["Cherry-pick Commit..." git-cherry-pick-commit t]
1594 ["Revert Commit..." git-revert-commit t]
18ff365f
AJ
1595 ("Merge"
1596 ["Next Unmerged File" git-next-unmerged-file t]
1597 ["Prev Unmerged File" git-prev-unmerged-file t]
18ff365f
AJ
1598 ["Interactive Merge File" git-find-file-imerge t]
1599 ["Diff Against Common Base File" git-diff-file-base t]
1600 ["Diff Combined" git-diff-file-combined t]
1601 ["Diff Against Merge Head" git-diff-file-merge-head t]
1602 ["Diff Against Mine" git-diff-file-mine t]
1603 ["Diff Against Other" git-diff-file-other t])
1604 "--------"
1605 ["Add File" git-add-file t]
1606 ["Revert File" git-revert-file t]
1607 ["Ignore File" git-ignore-file t]
1608 ["Remove File" git-remove-file t]
b0a53e9e 1609 ["Insert File" git-insert-file t]
18ff365f
AJ
1610 "--------"
1611 ["Find File" git-find-file t]
1612 ["View File" git-view-file t]
1613 ["Diff File" git-diff-file t]
1614 ["Interactive Diff File" git-diff-file-idiff t]
1615 ["Log" git-log-file t]
1616 "--------"
1617 ["Mark" git-mark-file t]
1618 ["Mark All" git-mark-all t]
1619 ["Unmark" git-unmark-file t]
1620 ["Unmark All" git-unmark-all t]
1621 ["Toggle All Marks" git-toggle-all-marks t]
1622 ["Hide Handled Files" git-remove-handled t]
1623 "--------"
1624 ["Show Uptodate Files" git-toggle-show-uptodate :style toggle :selected git-show-uptodate]
1625 ["Show Ignored Files" git-toggle-show-ignored :style toggle :selected git-show-ignored]
1626 ["Show Unknown Files" git-toggle-show-unknown :style toggle :selected git-show-unknown]
1627 "--------"
1628 ["Quit" git-status-quit t])))
1629
711fc8f6
AJ
1630
1631;; git mode should only run in the *git status* buffer
1632(put 'git-status-mode 'mode-class 'special)
1633
1634(defun git-status-mode ()
1635 "Major mode for interacting with Git.
1636Commands:
1637\\{git-status-mode-map}"
1638 (kill-all-local-variables)
1639 (buffer-disable-undo)
1640 (setq mode-name "git status"
1641 major-mode 'git-status-mode
1642 goal-column 17
1643 buffer-read-only t)
1644 (use-local-map git-status-mode-map)
1645 (let ((buffer-read-only nil))
1646 (erase-buffer)
1647 (let ((status (ewoc-create 'git-fileinfo-prettyprint "" "")))
1648 (set (make-local-variable 'git-status) status))
a944652c 1649 (set (make-local-variable 'list-buffers-directory) default-directory)
98acc3fa
AJ
1650 (make-local-variable 'git-show-uptodate)
1651 (make-local-variable 'git-show-ignored)
1652 (make-local-variable 'git-show-unknown)
711fc8f6
AJ
1653 (run-hooks 'git-status-mode-hook)))
1654
73389f12
AJ
1655(defun git-find-status-buffer (dir)
1656 "Find the git status buffer handling a specified directory."
1657 (let ((list (buffer-list))
1658 (fulldir (expand-file-name dir))
1659 found)
1660 (while (and list (not found))
1661 (let ((buffer (car list)))
1662 (with-current-buffer buffer
1663 (when (and list-buffers-directory
1664 (string-equal fulldir (expand-file-name list-buffers-directory))
a1eebfb3 1665 (eq major-mode 'git-status-mode))
73389f12
AJ
1666 (setq found buffer))))
1667 (setq list (cdr list)))
1668 found))
1669
711fc8f6
AJ
1670(defun git-status (dir)
1671 "Entry point into git-status mode."
1672 (interactive "DSelect directory: ")
1673 (setq dir (git-get-top-dir dir))
1674 (if (file-directory-p (concat (file-name-as-directory dir) ".git"))
73389f12
AJ
1675 (let ((buffer (or (and git-reuse-status-buffer (git-find-status-buffer dir))
1676 (create-file-buffer (expand-file-name "*git-status*" dir)))))
711fc8f6 1677 (switch-to-buffer buffer)
711fc8f6 1678 (cd dir)
a944652c 1679 (git-status-mode)
711fc8f6 1680 (git-refresh-status)
0365d885
AJ
1681 (goto-char (point-min))
1682 (add-hook 'after-save-hook 'git-update-saved-file))
711fc8f6
AJ
1683 (message "%s is not a git working tree." dir)))
1684
0365d885
AJ
1685(defun git-update-saved-file ()
1686 "Update the corresponding git-status buffer when a file is saved.
1687Meant to be used in `after-save-hook'."
1688 (let* ((file (expand-file-name buffer-file-name))
6df02388 1689 (dir (condition-case nil (git-get-top-dir (file-name-directory file)) (error nil)))
0365d885
AJ
1690 (buffer (and dir (git-find-status-buffer dir))))
1691 (when buffer
1692 (with-current-buffer buffer
1693 (let ((filename (file-relative-name file dir)))
1694 ; skip files located inside the .git directory
1695 (unless (string-match "^\\.git/" filename)
9ddf6d7c 1696 (git-call-process nil "add" "--refresh" "--" filename)
433ee03f 1697 (git-update-status-files (list filename))))))))
0365d885 1698
5716e794
JN
1699(defun git-help ()
1700 "Display help for Git mode."
1701 (interactive)
1702 (describe-function 'git-status-mode))
1703
711fc8f6
AJ
1704(provide 'git)
1705;;; git.el ends here