##// END OF EJS Templates
mercurial.el: add hg-cwd
Bryan O'Sullivan -
r3003:78fe7e2c default
parent child Browse files
Show More
@@ -1,1249 +1,1264
1 ;;; mercurial.el --- Emacs support for the Mercurial distributed SCM
1 ;;; mercurial.el --- Emacs support for the Mercurial distributed SCM
2
2
3 ;; Copyright (C) 2005, 2006 Bryan O'Sullivan
3 ;; Copyright (C) 2005, 2006 Bryan O'Sullivan
4
4
5 ;; Author: Bryan O'Sullivan <bos@serpentine.com>
5 ;; Author: Bryan O'Sullivan <bos@serpentine.com>
6
6
7 ;; mercurial.el is free software; you can redistribute it and/or
7 ;; mercurial.el is free software; you can redistribute it and/or
8 ;; modify it under the terms of version 2 of the GNU General Public
8 ;; modify it under the terms of version 2 of the GNU General Public
9 ;; License as published by the Free Software Foundation.
9 ;; License as published by the Free Software Foundation.
10
10
11 ;; mercurial.el is distributed in the hope that it will be useful, but
11 ;; mercurial.el is distributed in the hope that it will be useful, but
12 ;; WITHOUT ANY WARRANTY; without even the implied warranty of
12 ;; WITHOUT ANY WARRANTY; without even the implied warranty of
13 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 ;; General Public License for more details.
14 ;; General Public License for more details.
15
15
16 ;; You should have received a copy of the GNU General Public License
16 ;; You should have received a copy of the GNU General Public License
17 ;; along with mercurial.el, GNU Emacs, or XEmacs; see the file COPYING
17 ;; along with mercurial.el, GNU Emacs, or XEmacs; see the file COPYING
18 ;; (`C-h C-l'). If not, write to the Free Software Foundation, Inc.,
18 ;; (`C-h C-l'). If not, write to the Free Software Foundation, Inc.,
19 ;; 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19 ;; 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
20
20
21 ;;; Commentary:
21 ;;; Commentary:
22
22
23 ;; mercurial.el builds upon Emacs's VC mode to provide flexible
23 ;; mercurial.el builds upon Emacs's VC mode to provide flexible
24 ;; integration with the Mercurial distributed SCM tool.
24 ;; integration with the Mercurial distributed SCM tool.
25
25
26 ;; To get going as quickly as possible, load mercurial.el into Emacs and
26 ;; To get going as quickly as possible, load mercurial.el into Emacs and
27 ;; type `C-c h h'; this runs hg-help-overview, which prints a helpful
27 ;; type `C-c h h'; this runs hg-help-overview, which prints a helpful
28 ;; usage overview.
28 ;; usage overview.
29
29
30 ;; Much of the inspiration for mercurial.el comes from Rajesh
30 ;; Much of the inspiration for mercurial.el comes from Rajesh
31 ;; Vaidheeswarran's excellent p4.el, which does an admirably thorough
31 ;; Vaidheeswarran's excellent p4.el, which does an admirably thorough
32 ;; job for the commercial Perforce SCM product. In fact, substantial
32 ;; job for the commercial Perforce SCM product. In fact, substantial
33 ;; chunks of code are adapted from p4.el.
33 ;; chunks of code are adapted from p4.el.
34
34
35 ;; This code has been developed under XEmacs 21.5, and may not work as
35 ;; This code has been developed under XEmacs 21.5, and may not work as
36 ;; well under GNU Emacs (albeit tested under 21.4). Patches to
36 ;; well under GNU Emacs (albeit tested under 21.4). Patches to
37 ;; enhance the portability of this code, fix bugs, and add features
37 ;; enhance the portability of this code, fix bugs, and add features
38 ;; are most welcome. You can clone a Mercurial repository for this
38 ;; are most welcome. You can clone a Mercurial repository for this
39 ;; package from http://www.serpentine.com/hg/hg-emacs
39 ;; package from http://www.serpentine.com/hg/hg-emacs
40
40
41 ;; Please send problem reports and suggestions to bos@serpentine.com.
41 ;; Please send problem reports and suggestions to bos@serpentine.com.
42
42
43
43
44 ;;; Code:
44 ;;; Code:
45
45
46 (require 'advice)
46 (require 'advice)
47 (require 'cl)
47 (require 'cl)
48 (require 'diff-mode)
48 (require 'diff-mode)
49 (require 'easymenu)
49 (require 'easymenu)
50 (require 'executable)
50 (require 'executable)
51 (require 'vc)
51 (require 'vc)
52
52
53
53
54 ;;; XEmacs has view-less, while GNU Emacs has view. Joy.
54 ;;; XEmacs has view-less, while GNU Emacs has view. Joy.
55
55
56 (condition-case nil
56 (condition-case nil
57 (require 'view-less)
57 (require 'view-less)
58 (error nil))
58 (error nil))
59 (condition-case nil
59 (condition-case nil
60 (require 'view)
60 (require 'view)
61 (error nil))
61 (error nil))
62
62
63
63
64 ;;; Variables accessible through the custom system.
64 ;;; Variables accessible through the custom system.
65
65
66 (defgroup mercurial nil
66 (defgroup mercurial nil
67 "Mercurial distributed SCM."
67 "Mercurial distributed SCM."
68 :group 'tools)
68 :group 'tools)
69
69
70 (defcustom hg-binary
70 (defcustom hg-binary
71 (or (executable-find "hg")
71 (or (executable-find "hg")
72 (dolist (path '("~/bin/hg" "/usr/bin/hg" "/usr/local/bin/hg"))
72 (dolist (path '("~/bin/hg" "/usr/bin/hg" "/usr/local/bin/hg"))
73 (when (file-executable-p path)
73 (when (file-executable-p path)
74 (return path))))
74 (return path))))
75 "The path to Mercurial's hg executable."
75 "The path to Mercurial's hg executable."
76 :type '(file :must-match t)
76 :type '(file :must-match t)
77 :group 'mercurial)
77 :group 'mercurial)
78
78
79 (defcustom hg-mode-hook nil
79 (defcustom hg-mode-hook nil
80 "Hook run when a buffer enters hg-mode."
80 "Hook run when a buffer enters hg-mode."
81 :type 'sexp
81 :type 'sexp
82 :group 'mercurial)
82 :group 'mercurial)
83
83
84 (defcustom hg-commit-mode-hook nil
84 (defcustom hg-commit-mode-hook nil
85 "Hook run when a buffer is created to prepare a commit."
85 "Hook run when a buffer is created to prepare a commit."
86 :type 'sexp
86 :type 'sexp
87 :group 'mercurial)
87 :group 'mercurial)
88
88
89 (defcustom hg-pre-commit-hook nil
89 (defcustom hg-pre-commit-hook nil
90 "Hook run before a commit is performed.
90 "Hook run before a commit is performed.
91 If you want to prevent the commit from proceeding, raise an error."
91 If you want to prevent the commit from proceeding, raise an error."
92 :type 'sexp
92 :type 'sexp
93 :group 'mercurial)
93 :group 'mercurial)
94
94
95 (defcustom hg-log-mode-hook nil
95 (defcustom hg-log-mode-hook nil
96 "Hook run after a buffer is filled with log information."
96 "Hook run after a buffer is filled with log information."
97 :type 'sexp
97 :type 'sexp
98 :group 'mercurial)
98 :group 'mercurial)
99
99
100 (defcustom hg-global-prefix "\C-ch"
100 (defcustom hg-global-prefix "\C-ch"
101 "The global prefix for Mercurial keymap bindings."
101 "The global prefix for Mercurial keymap bindings."
102 :type 'sexp
102 :type 'sexp
103 :group 'mercurial)
103 :group 'mercurial)
104
104
105 (defcustom hg-commit-allow-empty-message nil
105 (defcustom hg-commit-allow-empty-message nil
106 "Whether to allow changes to be committed with empty descriptions."
106 "Whether to allow changes to be committed with empty descriptions."
107 :type 'boolean
107 :type 'boolean
108 :group 'mercurial)
108 :group 'mercurial)
109
109
110 (defcustom hg-commit-allow-empty-file-list nil
110 (defcustom hg-commit-allow-empty-file-list nil
111 "Whether to allow changes to be committed without any modified files."
111 "Whether to allow changes to be committed without any modified files."
112 :type 'boolean
112 :type 'boolean
113 :group 'mercurial)
113 :group 'mercurial)
114
114
115 (defcustom hg-rev-completion-limit 100
115 (defcustom hg-rev-completion-limit 100
116 "The maximum number of revisions that hg-read-rev will offer to complete.
116 "The maximum number of revisions that hg-read-rev will offer to complete.
117 This affects memory usage and performance when prompting for revisions
117 This affects memory usage and performance when prompting for revisions
118 in a repository with a lot of history."
118 in a repository with a lot of history."
119 :type 'integer
119 :type 'integer
120 :group 'mercurial)
120 :group 'mercurial)
121
121
122 (defcustom hg-log-limit 50
122 (defcustom hg-log-limit 50
123 "The maximum number of revisions that hg-log will display."
123 "The maximum number of revisions that hg-log will display."
124 :type 'integer
124 :type 'integer
125 :group 'mercurial)
125 :group 'mercurial)
126
126
127 (defcustom hg-update-modeline t
127 (defcustom hg-update-modeline t
128 "Whether to update the modeline with the status of a file after every save.
128 "Whether to update the modeline with the status of a file after every save.
129 Set this to nil on platforms with poor process management, such as Windows."
129 Set this to nil on platforms with poor process management, such as Windows."
130 :type 'boolean
130 :type 'boolean
131 :group 'mercurial)
131 :group 'mercurial)
132
132
133 (defcustom hg-incoming-repository "default"
133 (defcustom hg-incoming-repository "default"
134 "The repository from which changes are pulled from by default.
134 "The repository from which changes are pulled from by default.
135 This should be a symbolic repository name, since it is used for all
135 This should be a symbolic repository name, since it is used for all
136 repository-related commands."
136 repository-related commands."
137 :type 'string
137 :type 'string
138 :group 'mercurial)
138 :group 'mercurial)
139
139
140 (defcustom hg-outgoing-repository "default-push"
140 (defcustom hg-outgoing-repository "default-push"
141 "The repository to which changes are pushed to by default.
141 "The repository to which changes are pushed to by default.
142 This should be a symbolic repository name, since it is used for all
142 This should be a symbolic repository name, since it is used for all
143 repository-related commands."
143 repository-related commands."
144 :type 'string
144 :type 'string
145 :group 'mercurial)
145 :group 'mercurial)
146
146
147
147
148 ;;; Other variables.
148 ;;; Other variables.
149
149
150 (defconst hg-running-xemacs (string-match "XEmacs" emacs-version)
150 (defconst hg-running-xemacs (string-match "XEmacs" emacs-version)
151 "Is mercurial.el running under XEmacs?")
151 "Is mercurial.el running under XEmacs?")
152
152
153 (defvar hg-mode nil
153 (defvar hg-mode nil
154 "Is this file managed by Mercurial?")
154 "Is this file managed by Mercurial?")
155 (make-variable-buffer-local 'hg-mode)
155 (make-variable-buffer-local 'hg-mode)
156 (put 'hg-mode 'permanent-local t)
156 (put 'hg-mode 'permanent-local t)
157
157
158 (defvar hg-status nil)
158 (defvar hg-status nil)
159 (make-variable-buffer-local 'hg-status)
159 (make-variable-buffer-local 'hg-status)
160 (put 'hg-status 'permanent-local t)
160 (put 'hg-status 'permanent-local t)
161
161
162 (defvar hg-prev-buffer nil)
162 (defvar hg-prev-buffer nil)
163 (make-variable-buffer-local 'hg-prev-buffer)
163 (make-variable-buffer-local 'hg-prev-buffer)
164 (put 'hg-prev-buffer 'permanent-local t)
164 (put 'hg-prev-buffer 'permanent-local t)
165
165
166 (defvar hg-root nil)
166 (defvar hg-root nil)
167 (make-variable-buffer-local 'hg-root)
167 (make-variable-buffer-local 'hg-root)
168 (put 'hg-root 'permanent-local t)
168 (put 'hg-root 'permanent-local t)
169
169
170 (defvar hg-output-buffer-name "*Hg*"
170 (defvar hg-output-buffer-name "*Hg*"
171 "The name to use for Mercurial output buffers.")
171 "The name to use for Mercurial output buffers.")
172
172
173 (defvar hg-file-history nil)
173 (defvar hg-file-history nil)
174 (defvar hg-repo-history nil)
174 (defvar hg-repo-history nil)
175 (defvar hg-rev-history nil)
175 (defvar hg-rev-history nil)
176
176
177
177
178 ;;; Random constants.
178 ;;; Random constants.
179
179
180 (defconst hg-commit-message-start
180 (defconst hg-commit-message-start
181 "--- Enter your commit message. Type `C-c C-c' to commit. ---\n")
181 "--- Enter your commit message. Type `C-c C-c' to commit. ---\n")
182
182
183 (defconst hg-commit-message-end
183 (defconst hg-commit-message-end
184 "--- Files in bold will be committed. Click to toggle selection. ---\n")
184 "--- Files in bold will be committed. Click to toggle selection. ---\n")
185
185
186
186
187 ;;; hg-mode keymap.
187 ;;; hg-mode keymap.
188
188
189 (defvar hg-mode-map (make-sparse-keymap))
189 (defvar hg-mode-map (make-sparse-keymap))
190 (define-key hg-mode-map "\C-xv" 'hg-prefix-map)
190 (define-key hg-mode-map "\C-xv" 'hg-prefix-map)
191
191
192 (defvar hg-prefix-map
192 (defvar hg-prefix-map
193 (let ((map (copy-keymap vc-prefix-map)))
193 (let ((map (copy-keymap vc-prefix-map)))
194 (if (functionp 'set-keymap-name)
194 (if (functionp 'set-keymap-name)
195 (set-keymap-name map 'hg-prefix-map)); XEmacs
195 (set-keymap-name map 'hg-prefix-map)); XEmacs
196 map)
196 map)
197 "This keymap overrides some default vc-mode bindings.")
197 "This keymap overrides some default vc-mode bindings.")
198 (fset 'hg-prefix-map hg-prefix-map)
198 (fset 'hg-prefix-map hg-prefix-map)
199 (define-key hg-prefix-map "=" 'hg-diff)
199 (define-key hg-prefix-map "=" 'hg-diff)
200 (define-key hg-prefix-map "c" 'hg-undo)
200 (define-key hg-prefix-map "c" 'hg-undo)
201 (define-key hg-prefix-map "g" 'hg-annotate)
201 (define-key hg-prefix-map "g" 'hg-annotate)
202 (define-key hg-prefix-map "l" 'hg-log)
202 (define-key hg-prefix-map "l" 'hg-log)
203 (define-key hg-prefix-map "n" 'hg-commit-start)
203 (define-key hg-prefix-map "n" 'hg-commit-start)
204 ;; (define-key hg-prefix-map "r" 'hg-update)
204 ;; (define-key hg-prefix-map "r" 'hg-update)
205 (define-key hg-prefix-map "u" 'hg-revert-buffer)
205 (define-key hg-prefix-map "u" 'hg-revert-buffer)
206 (define-key hg-prefix-map "~" 'hg-version-other-window)
206 (define-key hg-prefix-map "~" 'hg-version-other-window)
207
207
208 (add-minor-mode 'hg-mode 'hg-mode hg-mode-map)
208 (add-minor-mode 'hg-mode 'hg-mode hg-mode-map)
209
209
210
210
211 ;;; Global keymap.
211 ;;; Global keymap.
212
212
213 (global-set-key "\C-xvi" 'hg-add)
213 (global-set-key "\C-xvi" 'hg-add)
214
214
215 (defvar hg-global-map (make-sparse-keymap))
215 (defvar hg-global-map (make-sparse-keymap))
216 (fset 'hg-global-map hg-global-map)
216 (fset 'hg-global-map hg-global-map)
217 (global-set-key hg-global-prefix 'hg-global-map)
217 (global-set-key hg-global-prefix 'hg-global-map)
218 (define-key hg-global-map "," 'hg-incoming)
218 (define-key hg-global-map "," 'hg-incoming)
219 (define-key hg-global-map "." 'hg-outgoing)
219 (define-key hg-global-map "." 'hg-outgoing)
220 (define-key hg-global-map "<" 'hg-pull)
220 (define-key hg-global-map "<" 'hg-pull)
221 (define-key hg-global-map "=" 'hg-diff-repo)
221 (define-key hg-global-map "=" 'hg-diff-repo)
222 (define-key hg-global-map ">" 'hg-push)
222 (define-key hg-global-map ">" 'hg-push)
223 (define-key hg-global-map "?" 'hg-help-overview)
223 (define-key hg-global-map "?" 'hg-help-overview)
224 (define-key hg-global-map "A" 'hg-addremove)
224 (define-key hg-global-map "A" 'hg-addremove)
225 (define-key hg-global-map "U" 'hg-revert)
225 (define-key hg-global-map "U" 'hg-revert)
226 (define-key hg-global-map "a" 'hg-add)
226 (define-key hg-global-map "a" 'hg-add)
227 (define-key hg-global-map "c" 'hg-commit-start)
227 (define-key hg-global-map "c" 'hg-commit-start)
228 (define-key hg-global-map "f" 'hg-forget)
228 (define-key hg-global-map "f" 'hg-forget)
229 (define-key hg-global-map "h" 'hg-help-overview)
229 (define-key hg-global-map "h" 'hg-help-overview)
230 (define-key hg-global-map "i" 'hg-init)
230 (define-key hg-global-map "i" 'hg-init)
231 (define-key hg-global-map "l" 'hg-log-repo)
231 (define-key hg-global-map "l" 'hg-log-repo)
232 (define-key hg-global-map "r" 'hg-root)
232 (define-key hg-global-map "r" 'hg-root)
233 (define-key hg-global-map "s" 'hg-status)
233 (define-key hg-global-map "s" 'hg-status)
234 (define-key hg-global-map "u" 'hg-update)
234 (define-key hg-global-map "u" 'hg-update)
235
235
236
236
237 ;;; View mode keymap.
237 ;;; View mode keymap.
238
238
239 (defvar hg-view-mode-map
239 (defvar hg-view-mode-map
240 (let ((map (copy-keymap (if (boundp 'view-minor-mode-map)
240 (let ((map (copy-keymap (if (boundp 'view-minor-mode-map)
241 view-minor-mode-map
241 view-minor-mode-map
242 view-mode-map))))
242 view-mode-map))))
243 (if (functionp 'set-keymap-name)
243 (if (functionp 'set-keymap-name)
244 (set-keymap-name map 'hg-view-mode-map)); XEmacs
244 (set-keymap-name map 'hg-view-mode-map)); XEmacs
245 map))
245 map))
246 (fset 'hg-view-mode-map hg-view-mode-map)
246 (fset 'hg-view-mode-map hg-view-mode-map)
247 (define-key hg-view-mode-map
247 (define-key hg-view-mode-map
248 (if hg-running-xemacs [button2] [mouse-2])
248 (if hg-running-xemacs [button2] [mouse-2])
249 'hg-buffer-mouse-clicked)
249 'hg-buffer-mouse-clicked)
250
250
251
251
252 ;;; Commit mode keymaps.
252 ;;; Commit mode keymaps.
253
253
254 (defvar hg-commit-mode-map (make-sparse-keymap))
254 (defvar hg-commit-mode-map (make-sparse-keymap))
255 (define-key hg-commit-mode-map "\C-c\C-c" 'hg-commit-finish)
255 (define-key hg-commit-mode-map "\C-c\C-c" 'hg-commit-finish)
256 (define-key hg-commit-mode-map "\C-c\C-k" 'hg-commit-kill)
256 (define-key hg-commit-mode-map "\C-c\C-k" 'hg-commit-kill)
257 (define-key hg-commit-mode-map "\C-xv=" 'hg-diff-repo)
257 (define-key hg-commit-mode-map "\C-xv=" 'hg-diff-repo)
258
258
259 (defvar hg-commit-mode-file-map (make-sparse-keymap))
259 (defvar hg-commit-mode-file-map (make-sparse-keymap))
260 (define-key hg-commit-mode-file-map
260 (define-key hg-commit-mode-file-map
261 (if hg-running-xemacs [button2] [mouse-2])
261 (if hg-running-xemacs [button2] [mouse-2])
262 'hg-commit-mouse-clicked)
262 'hg-commit-mouse-clicked)
263 (define-key hg-commit-mode-file-map " " 'hg-commit-toggle-file)
263 (define-key hg-commit-mode-file-map " " 'hg-commit-toggle-file)
264 (define-key hg-commit-mode-file-map "\r" 'hg-commit-toggle-file)
264 (define-key hg-commit-mode-file-map "\r" 'hg-commit-toggle-file)
265
265
266
266
267 ;;; Convenience functions.
267 ;;; Convenience functions.
268
268
269 (defsubst hg-binary ()
269 (defsubst hg-binary ()
270 (if hg-binary
270 (if hg-binary
271 hg-binary
271 hg-binary
272 (error "No `hg' executable found!")))
272 (error "No `hg' executable found!")))
273
273
274 (defsubst hg-replace-in-string (str regexp newtext &optional literal)
274 (defsubst hg-replace-in-string (str regexp newtext &optional literal)
275 "Replace all matches in STR for REGEXP with NEWTEXT string.
275 "Replace all matches in STR for REGEXP with NEWTEXT string.
276 Return the new string. Optional LITERAL non-nil means do a literal
276 Return the new string. Optional LITERAL non-nil means do a literal
277 replacement.
277 replacement.
278
278
279 This function bridges yet another pointless impedance gap between
279 This function bridges yet another pointless impedance gap between
280 XEmacs and GNU Emacs."
280 XEmacs and GNU Emacs."
281 (if (fboundp 'replace-in-string)
281 (if (fboundp 'replace-in-string)
282 (replace-in-string str regexp newtext literal)
282 (replace-in-string str regexp newtext literal)
283 (replace-regexp-in-string regexp newtext str nil literal)))
283 (replace-regexp-in-string regexp newtext str nil literal)))
284
284
285 (defsubst hg-strip (str)
285 (defsubst hg-strip (str)
286 "Strip leading and trailing blank lines from a string."
286 "Strip leading and trailing blank lines from a string."
287 (hg-replace-in-string (hg-replace-in-string str "[\r\n][ \t\r\n]*\\'" "")
287 (hg-replace-in-string (hg-replace-in-string str "[\r\n][ \t\r\n]*\\'" "")
288 "\\`[ \t\r\n]*[\r\n]" ""))
288 "\\`[ \t\r\n]*[\r\n]" ""))
289
289
290 (defsubst hg-chomp (str)
290 (defsubst hg-chomp (str)
291 "Strip trailing newlines from a string."
291 "Strip trailing newlines from a string."
292 (hg-replace-in-string str "[\r\n]+\\'" ""))
292 (hg-replace-in-string str "[\r\n]+\\'" ""))
293
293
294 (defun hg-run-command (command &rest args)
294 (defun hg-run-command (command &rest args)
295 "Run the shell command COMMAND, returning (EXIT-CODE . COMMAND-OUTPUT).
295 "Run the shell command COMMAND, returning (EXIT-CODE . COMMAND-OUTPUT).
296 The list ARGS contains a list of arguments to pass to the command."
296 The list ARGS contains a list of arguments to pass to the command."
297 (let* (exit-code
297 (let* (exit-code
298 (output
298 (output
299 (with-output-to-string
299 (with-output-to-string
300 (with-current-buffer
300 (with-current-buffer
301 standard-output
301 standard-output
302 (setq exit-code
302 (setq exit-code
303 (apply 'call-process command nil t nil args))))))
303 (apply 'call-process command nil t nil args))))))
304 (cons exit-code output)))
304 (cons exit-code output)))
305
305
306 (defun hg-run (command &rest args)
306 (defun hg-run (command &rest args)
307 "Run the Mercurial command COMMAND, returning (EXIT-CODE . COMMAND-OUTPUT)."
307 "Run the Mercurial command COMMAND, returning (EXIT-CODE . COMMAND-OUTPUT)."
308 (apply 'hg-run-command (hg-binary) command args))
308 (apply 'hg-run-command (hg-binary) command args))
309
309
310 (defun hg-run0 (command &rest args)
310 (defun hg-run0 (command &rest args)
311 "Run the Mercurial command COMMAND, returning its output.
311 "Run the Mercurial command COMMAND, returning its output.
312 If the command does not exit with a zero status code, raise an error."
312 If the command does not exit with a zero status code, raise an error."
313 (let ((res (apply 'hg-run-command (hg-binary) command args)))
313 (let ((res (apply 'hg-run-command (hg-binary) command args)))
314 (if (not (eq (car res) 0))
314 (if (not (eq (car res) 0))
315 (error "Mercurial command failed %s - exit code %s"
315 (error "Mercurial command failed %s - exit code %s"
316 (cons command args)
316 (cons command args)
317 (car res))
317 (car res))
318 (cdr res))))
318 (cdr res))))
319
319
320 (defun hg-sync-buffers (path)
320 (defun hg-sync-buffers (path)
321 "Sync buffers visiting PATH with their on-disk copies.
321 "Sync buffers visiting PATH with their on-disk copies.
322 If PATH is not being visited, but is under the repository root, sync
322 If PATH is not being visited, but is under the repository root, sync
323 all buffers visiting files in the repository."
323 all buffers visiting files in the repository."
324 (let ((buf (find-buffer-visiting path)))
324 (let ((buf (find-buffer-visiting path)))
325 (if buf
325 (if buf
326 (with-current-buffer buf
326 (with-current-buffer buf
327 (vc-buffer-sync))
327 (vc-buffer-sync))
328 (hg-do-across-repo path
328 (hg-do-across-repo path
329 (vc-buffer-sync)))))
329 (vc-buffer-sync)))))
330
330
331 (defun hg-buffer-commands (pnt)
331 (defun hg-buffer-commands (pnt)
332 "Use the properties of a character to do something sensible."
332 "Use the properties of a character to do something sensible."
333 (interactive "d")
333 (interactive "d")
334 (let ((rev (get-char-property pnt 'rev))
334 (let ((rev (get-char-property pnt 'rev))
335 (file (get-char-property pnt 'file))
335 (file (get-char-property pnt 'file))
336 (date (get-char-property pnt 'date))
336 (date (get-char-property pnt 'date))
337 (user (get-char-property pnt 'user))
337 (user (get-char-property pnt 'user))
338 (host (get-char-property pnt 'host))
338 (host (get-char-property pnt 'host))
339 (prev-buf (current-buffer)))
339 (prev-buf (current-buffer)))
340 (cond
340 (cond
341 (file
341 (file
342 (find-file-other-window file))
342 (find-file-other-window file))
343 (rev
343 (rev
344 (hg-diff hg-view-file-name rev rev prev-buf))
344 (hg-diff hg-view-file-name rev rev prev-buf))
345 ((message "I don't know how to do that yet")))))
345 ((message "I don't know how to do that yet")))))
346
346
347 (defsubst hg-event-point (event)
347 (defsubst hg-event-point (event)
348 "Return the character position of the mouse event EVENT."
348 "Return the character position of the mouse event EVENT."
349 (if hg-running-xemacs
349 (if hg-running-xemacs
350 (event-point event)
350 (event-point event)
351 (posn-point (event-start event))))
351 (posn-point (event-start event))))
352
352
353 (defsubst hg-event-window (event)
353 (defsubst hg-event-window (event)
354 "Return the window over which mouse event EVENT occurred."
354 "Return the window over which mouse event EVENT occurred."
355 (if hg-running-xemacs
355 (if hg-running-xemacs
356 (event-window event)
356 (event-window event)
357 (posn-window (event-start event))))
357 (posn-window (event-start event))))
358
358
359 (defun hg-buffer-mouse-clicked (event)
359 (defun hg-buffer-mouse-clicked (event)
360 "Translate the mouse clicks in a HG log buffer to character events.
360 "Translate the mouse clicks in a HG log buffer to character events.
361 These are then handed off to `hg-buffer-commands'.
361 These are then handed off to `hg-buffer-commands'.
362
362
363 Handle frickin' frackin' gratuitous event-related incompatibilities."
363 Handle frickin' frackin' gratuitous event-related incompatibilities."
364 (interactive "e")
364 (interactive "e")
365 (select-window (hg-event-window event))
365 (select-window (hg-event-window event))
366 (hg-buffer-commands (hg-event-point event)))
366 (hg-buffer-commands (hg-event-point event)))
367
367
368 (unless (fboundp 'view-minor-mode)
368 (unless (fboundp 'view-minor-mode)
369 (defun view-minor-mode (prev-buffer exit-func)
369 (defun view-minor-mode (prev-buffer exit-func)
370 (view-mode)))
370 (view-mode)))
371
371
372 (defsubst hg-abbrev-file-name (file)
372 (defsubst hg-abbrev-file-name (file)
373 "Portable wrapper around abbreviate-file-name."
373 "Portable wrapper around abbreviate-file-name."
374 (if hg-running-xemacs
374 (if hg-running-xemacs
375 (abbreviate-file-name file t)
375 (abbreviate-file-name file t)
376 (abbreviate-file-name file)))
376 (abbreviate-file-name file)))
377
377
378 (defun hg-read-file-name (&optional prompt default)
378 (defun hg-read-file-name (&optional prompt default)
379 "Read a file or directory name, or a pattern, to use with a command."
379 "Read a file or directory name, or a pattern, to use with a command."
380 (save-excursion
380 (save-excursion
381 (while hg-prev-buffer
381 (while hg-prev-buffer
382 (set-buffer hg-prev-buffer))
382 (set-buffer hg-prev-buffer))
383 (let ((path (or default
383 (let ((path (or default
384 (buffer-file-name)
384 (buffer-file-name)
385 (expand-file-name default-directory))))
385 (expand-file-name default-directory))))
386 (if (or (not path) current-prefix-arg)
386 (if (or (not path) current-prefix-arg)
387 (expand-file-name
387 (expand-file-name
388 (eval (list* 'read-file-name
388 (eval (list* 'read-file-name
389 (format "File, directory or pattern%s: "
389 (format "File, directory or pattern%s: "
390 (or prompt ""))
390 (or prompt ""))
391 (and path (file-name-directory path))
391 (and path (file-name-directory path))
392 nil nil
392 nil nil
393 (and path (file-name-nondirectory path))
393 (and path (file-name-nondirectory path))
394 (if hg-running-xemacs
394 (if hg-running-xemacs
395 (cons (quote 'hg-file-history) nil)
395 (cons (quote 'hg-file-history) nil)
396 nil))))
396 nil))))
397 path))))
397 path))))
398
398
399 (defun hg-read-number (&optional prompt default)
399 (defun hg-read-number (&optional prompt default)
400 "Read a integer value."
400 "Read a integer value."
401 (save-excursion
401 (save-excursion
402 (if (or (not default) current-prefix-arg)
402 (if (or (not default) current-prefix-arg)
403 (string-to-number
403 (string-to-number
404 (eval (list* 'read-string
404 (eval (list* 'read-string
405 (or prompt "")
405 (or prompt "")
406 (if default (cons (format "%d" default) nil) nil))))
406 (if default (cons (format "%d" default) nil) nil))))
407 default)))
407 default)))
408
408
409 (defun hg-read-config ()
409 (defun hg-read-config ()
410 "Return an alist of (key . value) pairs of Mercurial config data.
410 "Return an alist of (key . value) pairs of Mercurial config data.
411 Each key is of the form (section . name)."
411 Each key is of the form (section . name)."
412 (let (items)
412 (let (items)
413 (dolist (line (split-string (hg-chomp (hg-run0 "debugconfig")) "\n") items)
413 (dolist (line (split-string (hg-chomp (hg-run0 "debugconfig")) "\n") items)
414 (string-match "^\\([^=]*\\)=\\(.*\\)" line)
414 (string-match "^\\([^=]*\\)=\\(.*\\)" line)
415 (let* ((left (substring line (match-beginning 1) (match-end 1)))
415 (let* ((left (substring line (match-beginning 1) (match-end 1)))
416 (right (substring line (match-beginning 2) (match-end 2)))
416 (right (substring line (match-beginning 2) (match-end 2)))
417 (key (split-string left "\\."))
417 (key (split-string left "\\."))
418 (value (hg-replace-in-string right "\\\\n" "\n" t)))
418 (value (hg-replace-in-string right "\\\\n" "\n" t)))
419 (setq items (cons (cons (cons (car key) (cadr key)) value) items))))))
419 (setq items (cons (cons (cons (car key) (cadr key)) value) items))))))
420
420
421 (defun hg-config-section (section config)
421 (defun hg-config-section (section config)
422 "Return an alist of (name . value) pairs for SECTION of CONFIG."
422 "Return an alist of (name . value) pairs for SECTION of CONFIG."
423 (let (items)
423 (let (items)
424 (dolist (item config items)
424 (dolist (item config items)
425 (when (equal (caar item) section)
425 (when (equal (caar item) section)
426 (setq items (cons (cons (cdar item) (cdr item)) items))))))
426 (setq items (cons (cons (cdar item) (cdr item)) items))))))
427
427
428 (defun hg-string-starts-with (sub str)
428 (defun hg-string-starts-with (sub str)
429 "Indicate whether string STR starts with the substring or character SUB."
429 "Indicate whether string STR starts with the substring or character SUB."
430 (if (not (stringp sub))
430 (if (not (stringp sub))
431 (and (> (length str) 0) (equal (elt str 0) sub))
431 (and (> (length str) 0) (equal (elt str 0) sub))
432 (let ((sub-len (length sub)))
432 (let ((sub-len (length sub)))
433 (and (<= sub-len (length str))
433 (and (<= sub-len (length str))
434 (string= sub (substring str 0 sub-len))))))
434 (string= sub (substring str 0 sub-len))))))
435
435
436 (defun hg-complete-repo (string predicate all)
436 (defun hg-complete-repo (string predicate all)
437 "Attempt to complete a repository name.
437 "Attempt to complete a repository name.
438 We complete on either symbolic names from Mercurial's config or real
438 We complete on either symbolic names from Mercurial's config or real
439 directory names from the file system. We do not penalise URLs."
439 directory names from the file system. We do not penalise URLs."
440 (or (if all
440 (or (if all
441 (all-completions string hg-repo-completion-table predicate)
441 (all-completions string hg-repo-completion-table predicate)
442 (try-completion string hg-repo-completion-table predicate))
442 (try-completion string hg-repo-completion-table predicate))
443 (let* ((str (expand-file-name string))
443 (let* ((str (expand-file-name string))
444 (dir (file-name-directory str))
444 (dir (file-name-directory str))
445 (file (file-name-nondirectory str)))
445 (file (file-name-nondirectory str)))
446 (if all
446 (if all
447 (let (completions)
447 (let (completions)
448 (dolist (name (delete "./" (file-name-all-completions file dir))
448 (dolist (name (delete "./" (file-name-all-completions file dir))
449 completions)
449 completions)
450 (let ((path (concat dir name)))
450 (let ((path (concat dir name)))
451 (when (file-directory-p path)
451 (when (file-directory-p path)
452 (setq completions (cons name completions))))))
452 (setq completions (cons name completions))))))
453 (let ((comp (file-name-completion file dir)))
453 (let ((comp (file-name-completion file dir)))
454 (if comp
454 (if comp
455 (hg-abbrev-file-name (concat dir comp))))))))
455 (hg-abbrev-file-name (concat dir comp))))))))
456
456
457 (defun hg-read-repo-name (&optional prompt initial-contents default)
457 (defun hg-read-repo-name (&optional prompt initial-contents default)
458 "Read the location of a repository."
458 "Read the location of a repository."
459 (save-excursion
459 (save-excursion
460 (while hg-prev-buffer
460 (while hg-prev-buffer
461 (set-buffer hg-prev-buffer))
461 (set-buffer hg-prev-buffer))
462 (let (hg-repo-completion-table)
462 (let (hg-repo-completion-table)
463 (if current-prefix-arg
463 (if current-prefix-arg
464 (progn
464 (progn
465 (dolist (path (hg-config-section "paths" (hg-read-config)))
465 (dolist (path (hg-config-section "paths" (hg-read-config)))
466 (setq hg-repo-completion-table
466 (setq hg-repo-completion-table
467 (cons (cons (car path) t) hg-repo-completion-table))
467 (cons (cons (car path) t) hg-repo-completion-table))
468 (unless (hg-string-starts-with directory-sep-char (cdr path))
468 (unless (hg-string-starts-with directory-sep-char (cdr path))
469 (setq hg-repo-completion-table
469 (setq hg-repo-completion-table
470 (cons (cons (cdr path) t) hg-repo-completion-table))))
470 (cons (cons (cdr path) t) hg-repo-completion-table))))
471 (completing-read (format "Repository%s: " (or prompt ""))
471 (completing-read (format "Repository%s: " (or prompt ""))
472 'hg-complete-repo
472 'hg-complete-repo
473 nil
473 nil
474 nil
474 nil
475 initial-contents
475 initial-contents
476 'hg-repo-history
476 'hg-repo-history
477 default))
477 default))
478 default))))
478 default))))
479
479
480 (defun hg-read-rev (&optional prompt default)
480 (defun hg-read-rev (&optional prompt default)
481 "Read a revision or tag, offering completions."
481 "Read a revision or tag, offering completions."
482 (save-excursion
482 (save-excursion
483 (while hg-prev-buffer
483 (while hg-prev-buffer
484 (set-buffer hg-prev-buffer))
484 (set-buffer hg-prev-buffer))
485 (let ((rev (or default "tip")))
485 (let ((rev (or default "tip")))
486 (if current-prefix-arg
486 (if current-prefix-arg
487 (let ((revs (split-string
487 (let ((revs (split-string
488 (hg-chomp
488 (hg-chomp
489 (hg-run0 "-q" "log" "-r"
489 (hg-run0 "-q" "log" "-r"
490 (format "-%d:tip" hg-rev-completion-limit)))
490 (format "-%d:tip" hg-rev-completion-limit)))
491 "[\n:]")))
491 "[\n:]")))
492 (dolist (line (split-string (hg-chomp (hg-run0 "tags")) "\n"))
492 (dolist (line (split-string (hg-chomp (hg-run0 "tags")) "\n"))
493 (setq revs (cons (car (split-string line "\\s-")) revs)))
493 (setq revs (cons (car (split-string line "\\s-")) revs)))
494 (completing-read (format "Revision%s (%s): "
494 (completing-read (format "Revision%s (%s): "
495 (or prompt "")
495 (or prompt "")
496 (or default "tip"))
496 (or default "tip"))
497 (map 'list 'cons revs revs)
497 (map 'list 'cons revs revs)
498 nil
498 nil
499 nil
499 nil
500 nil
500 nil
501 'hg-rev-history
501 'hg-rev-history
502 (or default "tip")))
502 (or default "tip")))
503 rev))))
503 rev))))
504
504
505 (defun hg-parents-for-mode-line (root)
505 (defun hg-parents-for-mode-line (root)
506 "Format the parents of the working directory for the mode line."
506 "Format the parents of the working directory for the mode line."
507 (let ((parents (split-string (hg-chomp
507 (let ((parents (split-string (hg-chomp
508 (hg-run0 "--cwd" root "parents" "--template"
508 (hg-run0 "--cwd" root "parents" "--template"
509 "{rev}\n")) "\n")))
509 "{rev}\n")) "\n")))
510 (mapconcat 'identity parents "+")))
510 (mapconcat 'identity parents "+")))
511
511
512 (defun hg-buffers-visiting-repo (&optional path)
512 (defun hg-buffers-visiting-repo (&optional path)
513 "Return a list of buffers visiting the repository containing PATH."
513 "Return a list of buffers visiting the repository containing PATH."
514 (let ((root-name (hg-root (or path (buffer-file-name))))
514 (let ((root-name (hg-root (or path (buffer-file-name))))
515 bufs)
515 bufs)
516 (save-excursion
516 (save-excursion
517 (dolist (buf (buffer-list) bufs)
517 (dolist (buf (buffer-list) bufs)
518 (set-buffer buf)
518 (set-buffer buf)
519 (let ((name (buffer-file-name)))
519 (let ((name (buffer-file-name)))
520 (when (and hg-status name (equal (hg-root name) root-name))
520 (when (and hg-status name (equal (hg-root name) root-name))
521 (setq bufs (cons buf bufs))))))))
521 (setq bufs (cons buf bufs))))))))
522
522
523 (defun hg-update-mode-lines (path)
523 (defun hg-update-mode-lines (path)
524 "Update the mode lines of all buffers visiting the same repository as PATH."
524 "Update the mode lines of all buffers visiting the same repository as PATH."
525 (let* ((root (hg-root path))
525 (let* ((root (hg-root path))
526 (parents (hg-parents-for-mode-line root)))
526 (parents (hg-parents-for-mode-line root)))
527 (save-excursion
527 (save-excursion
528 (dolist (info (hg-path-status
528 (dolist (info (hg-path-status
529 root
529 root
530 (mapcar
530 (mapcar
531 (function
531 (function
532 (lambda (buf)
532 (lambda (buf)
533 (substring (buffer-file-name buf) (length root))))
533 (substring (buffer-file-name buf) (length root))))
534 (hg-buffers-visiting-repo root))))
534 (hg-buffers-visiting-repo root))))
535 (let* ((name (car info))
535 (let* ((name (car info))
536 (status (cdr info))
536 (status (cdr info))
537 (buf (find-buffer-visiting (concat root name))))
537 (buf (find-buffer-visiting (concat root name))))
538 (when buf
538 (when buf
539 (set-buffer buf)
539 (set-buffer buf)
540 (hg-mode-line-internal status parents)))))))
540 (hg-mode-line-internal status parents)))))))
541
541
542 (defmacro hg-do-across-repo (path &rest body)
542 (defmacro hg-do-across-repo (path &rest body)
543 (let ((root-name (gensym "root-"))
543 (let ((root-name (gensym "root-"))
544 (buf-name (gensym "buf-")))
544 (buf-name (gensym "buf-")))
545 `(let ((,root-name (hg-root ,path)))
545 `(let ((,root-name (hg-root ,path)))
546 (save-excursion
546 (save-excursion
547 (dolist (,buf-name (buffer-list))
547 (dolist (,buf-name (buffer-list))
548 (set-buffer ,buf-name)
548 (set-buffer ,buf-name)
549 (when (and hg-status (equal (hg-root buffer-file-name) ,root-name))
549 (when (and hg-status (equal (hg-root buffer-file-name) ,root-name))
550 ,@body))))))
550 ,@body))))))
551
551
552 (put 'hg-do-across-repo 'lisp-indent-function 1)
552 (put 'hg-do-across-repo 'lisp-indent-function 1)
553
553
554
554
555 ;;; View mode bits.
555 ;;; View mode bits.
556
556
557 (defun hg-exit-view-mode (buf)
557 (defun hg-exit-view-mode (buf)
558 "Exit from hg-view-mode.
558 "Exit from hg-view-mode.
559 We delete the current window if entering hg-view-mode split the
559 We delete the current window if entering hg-view-mode split the
560 current frame."
560 current frame."
561 (when (and (eq buf (current-buffer))
561 (when (and (eq buf (current-buffer))
562 (> (length (window-list)) 1))
562 (> (length (window-list)) 1))
563 (delete-window))
563 (delete-window))
564 (when (buffer-live-p buf)
564 (when (buffer-live-p buf)
565 (kill-buffer buf)))
565 (kill-buffer buf)))
566
566
567 (defun hg-view-mode (prev-buffer &optional file-name)
567 (defun hg-view-mode (prev-buffer &optional file-name)
568 (goto-char (point-min))
568 (goto-char (point-min))
569 (set-buffer-modified-p nil)
569 (set-buffer-modified-p nil)
570 (toggle-read-only t)
570 (toggle-read-only t)
571 (view-minor-mode prev-buffer 'hg-exit-view-mode)
571 (view-minor-mode prev-buffer 'hg-exit-view-mode)
572 (use-local-map hg-view-mode-map)
572 (use-local-map hg-view-mode-map)
573 (setq truncate-lines t)
573 (setq truncate-lines t)
574 (when file-name
574 (when file-name
575 (set (make-local-variable 'hg-view-file-name)
575 (set (make-local-variable 'hg-view-file-name)
576 (hg-abbrev-file-name file-name))))
576 (hg-abbrev-file-name file-name))))
577
577
578 (defun hg-file-status (file)
578 (defun hg-file-status (file)
579 "Return status of FILE, or nil if FILE does not exist or is unmanaged."
579 "Return status of FILE, or nil if FILE does not exist or is unmanaged."
580 (let* ((s (hg-run "status" file))
580 (let* ((s (hg-run "status" file))
581 (exit (car s))
581 (exit (car s))
582 (output (cdr s)))
582 (output (cdr s)))
583 (if (= exit 0)
583 (if (= exit 0)
584 (let ((state (assoc (substring output 0 (min (length output) 2))
584 (let ((state (assoc (substring output 0 (min (length output) 2))
585 '(("M " . modified)
585 '(("M " . modified)
586 ("A " . added)
586 ("A " . added)
587 ("R " . removed)
587 ("R " . removed)
588 ("! " . deleted)
588 ("! " . deleted)
589 ("? " . nil)))))
589 ("? " . nil)))))
590 (if state
590 (if state
591 (cdr state)
591 (cdr state)
592 'normal)))))
592 'normal)))))
593
593
594 (defun hg-path-status (root paths)
594 (defun hg-path-status (root paths)
595 "Return status of PATHS in repo ROOT as an alist.
595 "Return status of PATHS in repo ROOT as an alist.
596 Each entry is a pair (FILE-NAME . STATUS)."
596 Each entry is a pair (FILE-NAME . STATUS)."
597 (let ((s (apply 'hg-run "--cwd" root "status" "-marduc" paths))
597 (let ((s (apply 'hg-run "--cwd" root "status" "-marduc" paths))
598 result)
598 result)
599 (dolist (entry (split-string (hg-chomp (cdr s)) "\n") (nreverse result))
599 (dolist (entry (split-string (hg-chomp (cdr s)) "\n") (nreverse result))
600 (let ((state (cdr (assoc (substring entry 0 2)
600 (let ((state (cdr (assoc (substring entry 0 2)
601 '(("M " . modified)
601 '(("M " . modified)
602 ("A " . added)
602 ("A " . added)
603 ("R " . removed)
603 ("R " . removed)
604 ("! " . deleted)
604 ("! " . deleted)
605 ("C " . normal)
605 ("C " . normal)
606 ("I " . ignored)
606 ("I " . ignored)
607 ("? " . nil)))))
607 ("? " . nil)))))
608 (name (substring entry 2)))
608 (name (substring entry 2)))
609 (setq result (cons (cons name state) result))))))
609 (setq result (cons (cons name state) result))))))
610
610
611 (defmacro hg-view-output (args &rest body)
611 (defmacro hg-view-output (args &rest body)
612 "Execute BODY in a clean buffer, then quickly display that buffer.
612 "Execute BODY in a clean buffer, then quickly display that buffer.
613 If the buffer contains one line, its contents are displayed in the
613 If the buffer contains one line, its contents are displayed in the
614 minibuffer. Otherwise, the buffer is displayed in view-mode.
614 minibuffer. Otherwise, the buffer is displayed in view-mode.
615 ARGS is of the form (BUFFER-NAME &optional FILE), where BUFFER-NAME is
615 ARGS is of the form (BUFFER-NAME &optional FILE), where BUFFER-NAME is
616 the name of the buffer to create, and FILE is the name of the file
616 the name of the buffer to create, and FILE is the name of the file
617 being viewed."
617 being viewed."
618 (let ((prev-buf (gensym "prev-buf-"))
618 (let ((prev-buf (gensym "prev-buf-"))
619 (v-b-name (car args))
619 (v-b-name (car args))
620 (v-m-rest (cdr args)))
620 (v-m-rest (cdr args)))
621 `(let ((view-buf-name ,v-b-name)
621 `(let ((view-buf-name ,v-b-name)
622 (,prev-buf (current-buffer)))
622 (,prev-buf (current-buffer)))
623 (get-buffer-create view-buf-name)
623 (get-buffer-create view-buf-name)
624 (kill-buffer view-buf-name)
624 (kill-buffer view-buf-name)
625 (get-buffer-create view-buf-name)
625 (get-buffer-create view-buf-name)
626 (set-buffer view-buf-name)
626 (set-buffer view-buf-name)
627 (save-excursion
627 (save-excursion
628 ,@body)
628 ,@body)
629 (case (count-lines (point-min) (point-max))
629 (case (count-lines (point-min) (point-max))
630 ((0)
630 ((0)
631 (kill-buffer view-buf-name)
631 (kill-buffer view-buf-name)
632 (message "(No output)"))
632 (message "(No output)"))
633 ((1)
633 ((1)
634 (let ((msg (hg-chomp (buffer-substring (point-min) (point-max)))))
634 (let ((msg (hg-chomp (buffer-substring (point-min) (point-max)))))
635 (kill-buffer view-buf-name)
635 (kill-buffer view-buf-name)
636 (message "%s" msg)))
636 (message "%s" msg)))
637 (t
637 (t
638 (pop-to-buffer view-buf-name)
638 (pop-to-buffer view-buf-name)
639 (setq hg-prev-buffer ,prev-buf)
639 (setq hg-prev-buffer ,prev-buf)
640 (hg-view-mode ,prev-buf ,@v-m-rest))))))
640 (hg-view-mode ,prev-buf ,@v-m-rest))))))
641
641
642 (put 'hg-view-output 'lisp-indent-function 1)
642 (put 'hg-view-output 'lisp-indent-function 1)
643
643
644 ;;; Context save and restore across revert.
644 ;;; Context save and restore across revert.
645
645
646 (defun hg-position-context (pos)
646 (defun hg-position-context (pos)
647 "Return information to help find the given position again."
647 "Return information to help find the given position again."
648 (let* ((end (min (point-max) (+ pos 98))))
648 (let* ((end (min (point-max) (+ pos 98))))
649 (list pos
649 (list pos
650 (buffer-substring (max (point-min) (- pos 2)) end)
650 (buffer-substring (max (point-min) (- pos 2)) end)
651 (- end pos))))
651 (- end pos))))
652
652
653 (defun hg-buffer-context ()
653 (defun hg-buffer-context ()
654 "Return information to help restore a user's editing context.
654 "Return information to help restore a user's editing context.
655 This is useful across reverts and merges, where a context is likely
655 This is useful across reverts and merges, where a context is likely
656 to have moved a little, but not really changed."
656 to have moved a little, but not really changed."
657 (let ((point-context (hg-position-context (point)))
657 (let ((point-context (hg-position-context (point)))
658 (mark-context (let ((mark (mark-marker)))
658 (mark-context (let ((mark (mark-marker)))
659 (and mark (hg-position-context mark)))))
659 (and mark (hg-position-context mark)))))
660 (list point-context mark-context)))
660 (list point-context mark-context)))
661
661
662 (defun hg-find-context (ctx)
662 (defun hg-find-context (ctx)
663 "Attempt to find a context in the given buffer.
663 "Attempt to find a context in the given buffer.
664 Always returns a valid, hopefully sane, position."
664 Always returns a valid, hopefully sane, position."
665 (let ((pos (nth 0 ctx))
665 (let ((pos (nth 0 ctx))
666 (str (nth 1 ctx))
666 (str (nth 1 ctx))
667 (fixup (nth 2 ctx)))
667 (fixup (nth 2 ctx)))
668 (save-excursion
668 (save-excursion
669 (goto-char (max (point-min) (- pos 15000)))
669 (goto-char (max (point-min) (- pos 15000)))
670 (if (and (not (equal str ""))
670 (if (and (not (equal str ""))
671 (search-forward str nil t))
671 (search-forward str nil t))
672 (- (point) fixup)
672 (- (point) fixup)
673 (max pos (point-min))))))
673 (max pos (point-min))))))
674
674
675 (defun hg-restore-context (ctx)
675 (defun hg-restore-context (ctx)
676 "Attempt to restore the user's editing context."
676 "Attempt to restore the user's editing context."
677 (let ((point-context (nth 0 ctx))
677 (let ((point-context (nth 0 ctx))
678 (mark-context (nth 1 ctx)))
678 (mark-context (nth 1 ctx)))
679 (goto-char (hg-find-context point-context))
679 (goto-char (hg-find-context point-context))
680 (when mark-context
680 (when mark-context
681 (set-mark (hg-find-context mark-context)))))
681 (set-mark (hg-find-context mark-context)))))
682
682
683
683
684 ;;; Hooks.
684 ;;; Hooks.
685
685
686 (defun hg-mode-line-internal (status parents)
686 (defun hg-mode-line-internal (status parents)
687 (setq hg-status status
687 (setq hg-status status
688 hg-mode (and status (concat " Hg:"
688 hg-mode (and status (concat " Hg:"
689 parents
689 parents
690 (cdr (assq status
690 (cdr (assq status
691 '((normal . "")
691 '((normal . "")
692 (removed . "r")
692 (removed . "r")
693 (added . "a")
693 (added . "a")
694 (deleted . "!")
694 (deleted . "!")
695 (modified . "m"))))))))
695 (modified . "m"))))))))
696
696
697 (defun hg-mode-line (&optional force)
697 (defun hg-mode-line (&optional force)
698 "Update the modeline with the current status of a file.
698 "Update the modeline with the current status of a file.
699 An update occurs if optional argument FORCE is non-nil,
699 An update occurs if optional argument FORCE is non-nil,
700 hg-update-modeline is non-nil, or we have not yet checked the state of
700 hg-update-modeline is non-nil, or we have not yet checked the state of
701 the file."
701 the file."
702 (let ((root (hg-root)))
702 (let ((root (hg-root)))
703 (when (and root (or force hg-update-modeline (not hg-mode)))
703 (when (and root (or force hg-update-modeline (not hg-mode)))
704 (let ((status (hg-file-status buffer-file-name))
704 (let ((status (hg-file-status buffer-file-name))
705 (parents (hg-parents-for-mode-line root)))
705 (parents (hg-parents-for-mode-line root)))
706 (hg-mode-line-internal status parents)
706 (hg-mode-line-internal status parents)
707 status))))
707 status))))
708
708
709 (defun hg-mode (&optional toggle)
709 (defun hg-mode (&optional toggle)
710 "Minor mode for Mercurial distributed SCM integration.
710 "Minor mode for Mercurial distributed SCM integration.
711
711
712 The Mercurial mode user interface is based on that of VC mode, so if
712 The Mercurial mode user interface is based on that of VC mode, so if
713 you're already familiar with VC, the same keybindings and functions
713 you're already familiar with VC, the same keybindings and functions
714 will generally work.
714 will generally work.
715
715
716 Below is a list of many common SCM tasks. In the list, `G/L\'
716 Below is a list of many common SCM tasks. In the list, `G/L\'
717 indicates whether a key binding is global (G) to a repository or local
717 indicates whether a key binding is global (G) to a repository or local
718 (L) to a file. Many commands take a prefix argument.
718 (L) to a file. Many commands take a prefix argument.
719
719
720 SCM Task G/L Key Binding Command Name
720 SCM Task G/L Key Binding Command Name
721 -------- --- ----------- ------------
721 -------- --- ----------- ------------
722 Help overview (what you are reading) G C-c h h hg-help-overview
722 Help overview (what you are reading) G C-c h h hg-help-overview
723
723
724 Tell Mercurial to manage a file G C-c h a hg-add
724 Tell Mercurial to manage a file G C-c h a hg-add
725 Commit changes to current file only L C-x v n hg-commit-start
725 Commit changes to current file only L C-x v n hg-commit-start
726 Undo changes to file since commit L C-x v u hg-revert-buffer
726 Undo changes to file since commit L C-x v u hg-revert-buffer
727
727
728 Diff file vs last checkin L C-x v = hg-diff
728 Diff file vs last checkin L C-x v = hg-diff
729
729
730 View file change history L C-x v l hg-log
730 View file change history L C-x v l hg-log
731 View annotated file L C-x v a hg-annotate
731 View annotated file L C-x v a hg-annotate
732
732
733 Diff repo vs last checkin G C-c h = hg-diff-repo
733 Diff repo vs last checkin G C-c h = hg-diff-repo
734 View status of files in repo G C-c h s hg-status
734 View status of files in repo G C-c h s hg-status
735 Commit all changes G C-c h c hg-commit-start
735 Commit all changes G C-c h c hg-commit-start
736
736
737 Undo all changes since last commit G C-c h U hg-revert
737 Undo all changes since last commit G C-c h U hg-revert
738 View repo change history G C-c h l hg-log-repo
738 View repo change history G C-c h l hg-log-repo
739
739
740 See changes that can be pulled G C-c h , hg-incoming
740 See changes that can be pulled G C-c h , hg-incoming
741 Pull changes G C-c h < hg-pull
741 Pull changes G C-c h < hg-pull
742 Update working directory after pull G C-c h u hg-update
742 Update working directory after pull G C-c h u hg-update
743 See changes that can be pushed G C-c h . hg-outgoing
743 See changes that can be pushed G C-c h . hg-outgoing
744 Push changes G C-c h > hg-push"
744 Push changes G C-c h > hg-push"
745 (unless vc-make-backup-files
745 (unless vc-make-backup-files
746 (set (make-local-variable 'backup-inhibited) t))
746 (set (make-local-variable 'backup-inhibited) t))
747 (run-hooks 'hg-mode-hook))
747 (run-hooks 'hg-mode-hook))
748
748
749 (defun hg-find-file-hook ()
749 (defun hg-find-file-hook ()
750 (when (hg-mode-line)
750 (when (hg-mode-line)
751 (hg-mode)))
751 (hg-mode)))
752
752
753 (add-hook 'find-file-hooks 'hg-find-file-hook)
753 (add-hook 'find-file-hooks 'hg-find-file-hook)
754
754
755 (defun hg-after-save-hook ()
755 (defun hg-after-save-hook ()
756 (let ((old-status hg-status))
756 (let ((old-status hg-status))
757 (hg-mode-line)
757 (hg-mode-line)
758 (if (and (not old-status) hg-status)
758 (if (and (not old-status) hg-status)
759 (hg-mode))))
759 (hg-mode))))
760
760
761 (add-hook 'after-save-hook 'hg-after-save-hook)
761 (add-hook 'after-save-hook 'hg-after-save-hook)
762
762
763
763
764 ;;; User interface functions.
764 ;;; User interface functions.
765
765
766 (defun hg-help-overview ()
766 (defun hg-help-overview ()
767 "This is an overview of the Mercurial SCM mode for Emacs.
767 "This is an overview of the Mercurial SCM mode for Emacs.
768
768
769 You can find the source code, license (GPL v2), and credits for this
769 You can find the source code, license (GPL v2), and credits for this
770 code by typing `M-x find-library mercurial RET'."
770 code by typing `M-x find-library mercurial RET'."
771 (interactive)
771 (interactive)
772 (hg-view-output ("Mercurial Help Overview")
772 (hg-view-output ("Mercurial Help Overview")
773 (insert (documentation 'hg-help-overview))
773 (insert (documentation 'hg-help-overview))
774 (let ((pos (point)))
774 (let ((pos (point)))
775 (insert (documentation 'hg-mode))
775 (insert (documentation 'hg-mode))
776 (goto-char pos)
776 (goto-char pos)
777 (end-of-line 1)
777 (end-of-line 1)
778 (delete-region pos (point)))
778 (delete-region pos (point)))
779 (let ((hg-root-dir (hg-root)))
779 (let ((hg-root-dir (hg-root)))
780 (if (not hg-root-dir)
780 (if (not hg-root-dir)
781 (error "error: %s: directory is not part of a Mercurial repository."
781 (error "error: %s: directory is not part of a Mercurial repository."
782 default-directory)
782 default-directory)
783 (cd hg-root-dir)))))
783 (cd hg-root-dir)))))
784
784
785 (defun hg-fix-paths ()
785 (defun hg-fix-paths ()
786 "Fix paths reported by some Mercurial commands."
786 "Fix paths reported by some Mercurial commands."
787 (save-excursion
787 (save-excursion
788 (goto-char (point-min))
788 (goto-char (point-min))
789 (while (re-search-forward " \\.\\.." nil t)
789 (while (re-search-forward " \\.\\.." nil t)
790 (replace-match " " nil nil))))
790 (replace-match " " nil nil))))
791
791
792 (defun hg-add (path)
792 (defun hg-add (path)
793 "Add PATH to the Mercurial repository on the next commit.
793 "Add PATH to the Mercurial repository on the next commit.
794 With a prefix argument, prompt for the path to add."
794 With a prefix argument, prompt for the path to add."
795 (interactive (list (hg-read-file-name " to add")))
795 (interactive (list (hg-read-file-name " to add")))
796 (let ((buf (current-buffer))
796 (let ((buf (current-buffer))
797 (update (equal buffer-file-name path)))
797 (update (equal buffer-file-name path)))
798 (hg-view-output (hg-output-buffer-name)
798 (hg-view-output (hg-output-buffer-name)
799 (apply 'call-process (hg-binary) nil t nil (list "add" path))
799 (apply 'call-process (hg-binary) nil t nil (list "add" path))
800 (hg-fix-paths)
800 (hg-fix-paths)
801 (goto-char (point-min))
801 (goto-char (point-min))
802 (cd (hg-root path)))
802 (cd (hg-root path)))
803 (when update
803 (when update
804 (unless vc-make-backup-files
804 (unless vc-make-backup-files
805 (set (make-local-variable 'backup-inhibited) t))
805 (set (make-local-variable 'backup-inhibited) t))
806 (with-current-buffer buf
806 (with-current-buffer buf
807 (hg-mode-line)))))
807 (hg-mode-line)))))
808
808
809 (defun hg-addremove ()
809 (defun hg-addremove ()
810 (interactive)
810 (interactive)
811 (error "not implemented"))
811 (error "not implemented"))
812
812
813 (defun hg-annotate ()
813 (defun hg-annotate ()
814 (interactive)
814 (interactive)
815 (error "not implemented"))
815 (error "not implemented"))
816
816
817 (defun hg-commit-toggle-file (pos)
817 (defun hg-commit-toggle-file (pos)
818 "Toggle whether or not the file at POS will be committed."
818 "Toggle whether or not the file at POS will be committed."
819 (interactive "d")
819 (interactive "d")
820 (save-excursion
820 (save-excursion
821 (goto-char pos)
821 (goto-char pos)
822 (let ((face (get-text-property pos 'face))
822 (let ((face (get-text-property pos 'face))
823 (inhibit-read-only t)
823 (inhibit-read-only t)
824 bol)
824 bol)
825 (beginning-of-line)
825 (beginning-of-line)
826 (setq bol (+ (point) 4))
826 (setq bol (+ (point) 4))
827 (end-of-line)
827 (end-of-line)
828 (if (eq face 'bold)
828 (if (eq face 'bold)
829 (progn
829 (progn
830 (remove-text-properties bol (point) '(face nil))
830 (remove-text-properties bol (point) '(face nil))
831 (message "%s will not be committed"
831 (message "%s will not be committed"
832 (buffer-substring bol (point))))
832 (buffer-substring bol (point))))
833 (add-text-properties bol (point) '(face bold))
833 (add-text-properties bol (point) '(face bold))
834 (message "%s will be committed"
834 (message "%s will be committed"
835 (buffer-substring bol (point)))))))
835 (buffer-substring bol (point)))))))
836
836
837 (defun hg-commit-mouse-clicked (event)
837 (defun hg-commit-mouse-clicked (event)
838 "Toggle whether or not the file at POS will be committed."
838 "Toggle whether or not the file at POS will be committed."
839 (interactive "@e")
839 (interactive "@e")
840 (hg-commit-toggle-file (hg-event-point event)))
840 (hg-commit-toggle-file (hg-event-point event)))
841
841
842 (defun hg-commit-kill ()
842 (defun hg-commit-kill ()
843 "Kill the commit currently being prepared."
843 "Kill the commit currently being prepared."
844 (interactive)
844 (interactive)
845 (when (or (not (buffer-modified-p)) (y-or-n-p "Really kill this commit? "))
845 (when (or (not (buffer-modified-p)) (y-or-n-p "Really kill this commit? "))
846 (let ((buf hg-prev-buffer))
846 (let ((buf hg-prev-buffer))
847 (kill-buffer nil)
847 (kill-buffer nil)
848 (switch-to-buffer buf))))
848 (switch-to-buffer buf))))
849
849
850 (defun hg-commit-finish ()
850 (defun hg-commit-finish ()
851 "Finish preparing a commit, and perform the actual commit.
851 "Finish preparing a commit, and perform the actual commit.
852 The hook hg-pre-commit-hook is run before anything else is done. If
852 The hook hg-pre-commit-hook is run before anything else is done. If
853 the commit message is empty and hg-commit-allow-empty-message is nil,
853 the commit message is empty and hg-commit-allow-empty-message is nil,
854 an error is raised. If the list of files to commit is empty and
854 an error is raised. If the list of files to commit is empty and
855 hg-commit-allow-empty-file-list is nil, an error is raised."
855 hg-commit-allow-empty-file-list is nil, an error is raised."
856 (interactive)
856 (interactive)
857 (let ((root hg-root))
857 (let ((root hg-root))
858 (save-excursion
858 (save-excursion
859 (run-hooks 'hg-pre-commit-hook)
859 (run-hooks 'hg-pre-commit-hook)
860 (goto-char (point-min))
860 (goto-char (point-min))
861 (search-forward hg-commit-message-start)
861 (search-forward hg-commit-message-start)
862 (let (message files)
862 (let (message files)
863 (let ((start (point)))
863 (let ((start (point)))
864 (goto-char (point-max))
864 (goto-char (point-max))
865 (search-backward hg-commit-message-end)
865 (search-backward hg-commit-message-end)
866 (setq message (hg-strip (buffer-substring start (point)))))
866 (setq message (hg-strip (buffer-substring start (point)))))
867 (when (and (= (length message) 0)
867 (when (and (= (length message) 0)
868 (not hg-commit-allow-empty-message))
868 (not hg-commit-allow-empty-message))
869 (error "Cannot proceed - commit message is empty"))
869 (error "Cannot proceed - commit message is empty"))
870 (forward-line 1)
870 (forward-line 1)
871 (beginning-of-line)
871 (beginning-of-line)
872 (while (< (point) (point-max))
872 (while (< (point) (point-max))
873 (let ((pos (+ (point) 4)))
873 (let ((pos (+ (point) 4)))
874 (end-of-line)
874 (end-of-line)
875 (when (eq (get-text-property pos 'face) 'bold)
875 (when (eq (get-text-property pos 'face) 'bold)
876 (end-of-line)
876 (end-of-line)
877 (setq files (cons (buffer-substring pos (point)) files))))
877 (setq files (cons (buffer-substring pos (point)) files))))
878 (forward-line 1))
878 (forward-line 1))
879 (when (and (= (length files) 0)
879 (when (and (= (length files) 0)
880 (not hg-commit-allow-empty-file-list))
880 (not hg-commit-allow-empty-file-list))
881 (error "Cannot proceed - no files to commit"))
881 (error "Cannot proceed - no files to commit"))
882 (setq message (concat message "\n"))
882 (setq message (concat message "\n"))
883 (apply 'hg-run0 "--cwd" hg-root "commit" "-m" message files))
883 (apply 'hg-run0 "--cwd" hg-root "commit" "-m" message files))
884 (let ((buf hg-prev-buffer))
884 (let ((buf hg-prev-buffer))
885 (kill-buffer nil)
885 (kill-buffer nil)
886 (switch-to-buffer buf))
886 (switch-to-buffer buf))
887 (hg-update-mode-lines root))))
887 (hg-update-mode-lines root))))
888
888
889 (defun hg-commit-mode ()
889 (defun hg-commit-mode ()
890 "Mode for describing a commit of changes to a Mercurial repository.
890 "Mode for describing a commit of changes to a Mercurial repository.
891 This involves two actions: describing the changes with a commit
891 This involves two actions: describing the changes with a commit
892 message, and choosing the files to commit.
892 message, and choosing the files to commit.
893
893
894 To describe the commit, simply type some text in the designated area.
894 To describe the commit, simply type some text in the designated area.
895
895
896 By default, all modified, added and removed files are selected for
896 By default, all modified, added and removed files are selected for
897 committing. Files that will be committed are displayed in bold face\;
897 committing. Files that will be committed are displayed in bold face\;
898 those that will not are displayed in normal face.
898 those that will not are displayed in normal face.
899
899
900 To toggle whether a file will be committed, move the cursor over a
900 To toggle whether a file will be committed, move the cursor over a
901 particular file and hit space or return. Alternatively, middle click
901 particular file and hit space or return. Alternatively, middle click
902 on the file.
902 on the file.
903
903
904 Key bindings
904 Key bindings
905 ------------
905 ------------
906 \\[hg-commit-finish] proceed with commit
906 \\[hg-commit-finish] proceed with commit
907 \\[hg-commit-kill] kill commit
907 \\[hg-commit-kill] kill commit
908
908
909 \\[hg-diff-repo] view diff of pending changes"
909 \\[hg-diff-repo] view diff of pending changes"
910 (interactive)
910 (interactive)
911 (use-local-map hg-commit-mode-map)
911 (use-local-map hg-commit-mode-map)
912 (set-syntax-table text-mode-syntax-table)
912 (set-syntax-table text-mode-syntax-table)
913 (setq local-abbrev-table text-mode-abbrev-table
913 (setq local-abbrev-table text-mode-abbrev-table
914 major-mode 'hg-commit-mode
914 major-mode 'hg-commit-mode
915 mode-name "Hg-Commit")
915 mode-name "Hg-Commit")
916 (set-buffer-modified-p nil)
916 (set-buffer-modified-p nil)
917 (setq buffer-undo-list nil)
917 (setq buffer-undo-list nil)
918 (run-hooks 'text-mode-hook 'hg-commit-mode-hook))
918 (run-hooks 'text-mode-hook 'hg-commit-mode-hook))
919
919
920 (defun hg-commit-start ()
920 (defun hg-commit-start ()
921 "Prepare a commit of changes to the repository containing the current file."
921 "Prepare a commit of changes to the repository containing the current file."
922 (interactive)
922 (interactive)
923 (while hg-prev-buffer
923 (while hg-prev-buffer
924 (set-buffer hg-prev-buffer))
924 (set-buffer hg-prev-buffer))
925 (let ((root (hg-root))
925 (let ((root (hg-root))
926 (prev-buffer (current-buffer))
926 (prev-buffer (current-buffer))
927 modified-files)
927 modified-files)
928 (unless root
928 (unless root
929 (error "Cannot commit outside a repository!"))
929 (error "Cannot commit outside a repository!"))
930 (hg-sync-buffers root)
930 (hg-sync-buffers root)
931 (setq modified-files (hg-chomp (hg-run0 "--cwd" root "status" "-arm")))
931 (setq modified-files (hg-chomp (hg-run0 "--cwd" root "status" "-arm")))
932 (when (and (= (length modified-files) 0)
932 (when (and (= (length modified-files) 0)
933 (not hg-commit-allow-empty-file-list))
933 (not hg-commit-allow-empty-file-list))
934 (error "No pending changes to commit"))
934 (error "No pending changes to commit"))
935 (let* ((buf-name (format "*Mercurial: Commit %s*" root)))
935 (let* ((buf-name (format "*Mercurial: Commit %s*" root)))
936 (pop-to-buffer (get-buffer-create buf-name))
936 (pop-to-buffer (get-buffer-create buf-name))
937 (when (= (point-min) (point-max))
937 (when (= (point-min) (point-max))
938 (set (make-local-variable 'hg-root) root)
938 (set (make-local-variable 'hg-root) root)
939 (setq hg-prev-buffer prev-buffer)
939 (setq hg-prev-buffer prev-buffer)
940 (insert "\n")
940 (insert "\n")
941 (let ((bol (point)))
941 (let ((bol (point)))
942 (insert hg-commit-message-end)
942 (insert hg-commit-message-end)
943 (add-text-properties bol (point) '(face bold-italic)))
943 (add-text-properties bol (point) '(face bold-italic)))
944 (let ((file-area (point)))
944 (let ((file-area (point)))
945 (insert modified-files)
945 (insert modified-files)
946 (goto-char file-area)
946 (goto-char file-area)
947 (while (< (point) (point-max))
947 (while (< (point) (point-max))
948 (let ((bol (point)))
948 (let ((bol (point)))
949 (forward-char 1)
949 (forward-char 1)
950 (insert " ")
950 (insert " ")
951 (end-of-line)
951 (end-of-line)
952 (add-text-properties (+ bol 4) (point)
952 (add-text-properties (+ bol 4) (point)
953 '(face bold mouse-face highlight)))
953 '(face bold mouse-face highlight)))
954 (forward-line 1))
954 (forward-line 1))
955 (goto-char file-area)
955 (goto-char file-area)
956 (add-text-properties (point) (point-max)
956 (add-text-properties (point) (point-max)
957 `(keymap ,hg-commit-mode-file-map))
957 `(keymap ,hg-commit-mode-file-map))
958 (goto-char (point-min))
958 (goto-char (point-min))
959 (insert hg-commit-message-start)
959 (insert hg-commit-message-start)
960 (add-text-properties (point-min) (point) '(face bold-italic))
960 (add-text-properties (point-min) (point) '(face bold-italic))
961 (insert "\n\n")
961 (insert "\n\n")
962 (forward-line -1)
962 (forward-line -1)
963 (save-excursion
963 (save-excursion
964 (goto-char (point-max))
964 (goto-char (point-max))
965 (search-backward hg-commit-message-end)
965 (search-backward hg-commit-message-end)
966 (add-text-properties (match-beginning 0) (point-max)
966 (add-text-properties (match-beginning 0) (point-max)
967 '(read-only t))
967 '(read-only t))
968 (goto-char (point-min))
968 (goto-char (point-min))
969 (search-forward hg-commit-message-start)
969 (search-forward hg-commit-message-start)
970 (add-text-properties (match-beginning 0) (match-end 0)
970 (add-text-properties (match-beginning 0) (match-end 0)
971 '(read-only t)))
971 '(read-only t)))
972 (hg-commit-mode)
972 (hg-commit-mode)
973 (cd root))))))
973 (cd root))))))
974
974
975 (defun hg-diff (path &optional rev1 rev2)
975 (defun hg-diff (path &optional rev1 rev2)
976 "Show the differences between REV1 and REV2 of PATH.
976 "Show the differences between REV1 and REV2 of PATH.
977 When called interactively, the default behaviour is to treat REV1 as
977 When called interactively, the default behaviour is to treat REV1 as
978 the \"parent\" revision, REV2 as the current edited version of the file, and
978 the \"parent\" revision, REV2 as the current edited version of the file, and
979 PATH as the file edited in the current buffer.
979 PATH as the file edited in the current buffer.
980 With a prefix argument, prompt for all of these."
980 With a prefix argument, prompt for all of these."
981 (interactive (list (hg-read-file-name " to diff")
981 (interactive (list (hg-read-file-name " to diff")
982 (let ((rev1 (hg-read-rev " to start with" 'parent)))
982 (let ((rev1 (hg-read-rev " to start with" 'parent)))
983 (and (not (eq rev1 'parent)) rev1))
983 (and (not (eq rev1 'parent)) rev1))
984 (let ((rev2 (hg-read-rev " to end with" 'working-dir)))
984 (let ((rev2 (hg-read-rev " to end with" 'working-dir)))
985 (and (not (eq rev2 'working-dir)) rev2))))
985 (and (not (eq rev2 'working-dir)) rev2))))
986 (hg-sync-buffers path)
986 (hg-sync-buffers path)
987 (let ((a-path (hg-abbrev-file-name path))
987 (let ((a-path (hg-abbrev-file-name path))
988 ;; none revision is specified explicitly
988 ;; none revision is specified explicitly
989 (none (and (not rev1) (not rev2)))
989 (none (and (not rev1) (not rev2)))
990 ;; only one revision is specified explicitly
990 ;; only one revision is specified explicitly
991 (one (or (and (or (equal rev1 rev2) (not rev2)) rev1)
991 (one (or (and (or (equal rev1 rev2) (not rev2)) rev1)
992 (and (not rev1) rev2)))
992 (and (not rev1) rev2)))
993 diff)
993 diff)
994 (hg-view-output ((cond
994 (hg-view-output ((cond
995 (none
995 (none
996 (format "Mercurial: Diff against parent of %s" a-path))
996 (format "Mercurial: Diff against parent of %s" a-path))
997 (one
997 (one
998 (format "Mercurial: Diff of rev %s of %s" one a-path))
998 (format "Mercurial: Diff of rev %s of %s" one a-path))
999 (t
999 (t
1000 (format "Mercurial: Diff from rev %s to %s of %s"
1000 (format "Mercurial: Diff from rev %s to %s of %s"
1001 rev1 rev2 a-path))))
1001 rev1 rev2 a-path))))
1002 (cond
1002 (cond
1003 (none
1003 (none
1004 (call-process (hg-binary) nil t nil "diff" path))
1004 (call-process (hg-binary) nil t nil "diff" path))
1005 (one
1005 (one
1006 (call-process (hg-binary) nil t nil "diff" "-r" one path))
1006 (call-process (hg-binary) nil t nil "diff" "-r" one path))
1007 (t
1007 (t
1008 (call-process (hg-binary) nil t nil "diff" "-r" rev1 "-r" rev2 path)))
1008 (call-process (hg-binary) nil t nil "diff" "-r" rev1 "-r" rev2 path)))
1009 (diff-mode)
1009 (diff-mode)
1010 (setq diff (not (= (point-min) (point-max))))
1010 (setq diff (not (= (point-min) (point-max))))
1011 (font-lock-fontify-buffer)
1011 (font-lock-fontify-buffer)
1012 (cd (hg-root path)))
1012 (cd (hg-root path)))
1013 diff))
1013 diff))
1014
1014
1015 (defun hg-diff-repo (path &optional rev1 rev2)
1015 (defun hg-diff-repo (path &optional rev1 rev2)
1016 "Show the differences between REV1 and REV2 of repository containing PATH.
1016 "Show the differences between REV1 and REV2 of repository containing PATH.
1017 When called interactively, the default behaviour is to treat REV1 as
1017 When called interactively, the default behaviour is to treat REV1 as
1018 the \"parent\" revision, REV2 as the current edited version of the file, and
1018 the \"parent\" revision, REV2 as the current edited version of the file, and
1019 PATH as the `hg-root' of the current buffer.
1019 PATH as the `hg-root' of the current buffer.
1020 With a prefix argument, prompt for all of these."
1020 With a prefix argument, prompt for all of these."
1021 (interactive (list (hg-read-file-name " to diff")
1021 (interactive (list (hg-read-file-name " to diff")
1022 (let ((rev1 (hg-read-rev " to start with" 'parent)))
1022 (let ((rev1 (hg-read-rev " to start with" 'parent)))
1023 (and (not (eq rev1 'parent)) rev1))
1023 (and (not (eq rev1 'parent)) rev1))
1024 (let ((rev2 (hg-read-rev " to end with" 'working-dir)))
1024 (let ((rev2 (hg-read-rev " to end with" 'working-dir)))
1025 (and (not (eq rev2 'working-dir)) rev2))))
1025 (and (not (eq rev2 'working-dir)) rev2))))
1026 (hg-diff (hg-root path) rev1 rev2))
1026 (hg-diff (hg-root path) rev1 rev2))
1027
1027
1028 (defun hg-forget (path)
1028 (defun hg-forget (path)
1029 "Lose track of PATH, which has been added, but not yet committed.
1029 "Lose track of PATH, which has been added, but not yet committed.
1030 This will prevent the file from being incorporated into the Mercurial
1030 This will prevent the file from being incorporated into the Mercurial
1031 repository on the next commit.
1031 repository on the next commit.
1032 With a prefix argument, prompt for the path to forget."
1032 With a prefix argument, prompt for the path to forget."
1033 (interactive (list (hg-read-file-name " to forget")))
1033 (interactive (list (hg-read-file-name " to forget")))
1034 (let ((buf (current-buffer))
1034 (let ((buf (current-buffer))
1035 (update (equal buffer-file-name path)))
1035 (update (equal buffer-file-name path)))
1036 (hg-view-output (hg-output-buffer-name)
1036 (hg-view-output (hg-output-buffer-name)
1037 (apply 'call-process (hg-binary) nil t nil (list "forget" path))
1037 (apply 'call-process (hg-binary) nil t nil (list "forget" path))
1038 ;; "hg forget" shows pathes relative NOT TO ROOT BUT TO REPOSITORY
1038 ;; "hg forget" shows pathes relative NOT TO ROOT BUT TO REPOSITORY
1039 (hg-fix-paths)
1039 (hg-fix-paths)
1040 (goto-char (point-min))
1040 (goto-char (point-min))
1041 (cd (hg-root path)))
1041 (cd (hg-root path)))
1042 (when update
1042 (when update
1043 (with-current-buffer buf
1043 (with-current-buffer buf
1044 (when (local-variable-p 'backup-inhibited)
1044 (when (local-variable-p 'backup-inhibited)
1045 (kill-local-variable 'backup-inhibited))
1045 (kill-local-variable 'backup-inhibited))
1046 (hg-mode-line)))))
1046 (hg-mode-line)))))
1047
1047
1048 (defun hg-incoming (&optional repo)
1048 (defun hg-incoming (&optional repo)
1049 "Display changesets present in REPO that are not present locally."
1049 "Display changesets present in REPO that are not present locally."
1050 (interactive (list (hg-read-repo-name " where changes would come from")))
1050 (interactive (list (hg-read-repo-name " where changes would come from")))
1051 (hg-view-output ((format "Mercurial: Incoming from %s to %s"
1051 (hg-view-output ((format "Mercurial: Incoming from %s to %s"
1052 (hg-abbrev-file-name (hg-root))
1052 (hg-abbrev-file-name (hg-root))
1053 (hg-abbrev-file-name
1053 (hg-abbrev-file-name
1054 (or repo hg-incoming-repository))))
1054 (or repo hg-incoming-repository))))
1055 (call-process (hg-binary) nil t nil "incoming"
1055 (call-process (hg-binary) nil t nil "incoming"
1056 (or repo hg-incoming-repository))
1056 (or repo hg-incoming-repository))
1057 (hg-log-mode)
1057 (hg-log-mode)
1058 (cd (hg-root))))
1058 (cd (hg-root))))
1059
1059
1060 (defun hg-init ()
1060 (defun hg-init ()
1061 (interactive)
1061 (interactive)
1062 (error "not implemented"))
1062 (error "not implemented"))
1063
1063
1064 (defun hg-log-mode ()
1064 (defun hg-log-mode ()
1065 "Mode for viewing a Mercurial change log."
1065 "Mode for viewing a Mercurial change log."
1066 (goto-char (point-min))
1066 (goto-char (point-min))
1067 (when (looking-at "^searching for changes.*$")
1067 (when (looking-at "^searching for changes.*$")
1068 (delete-region (match-beginning 0) (match-end 0)))
1068 (delete-region (match-beginning 0) (match-end 0)))
1069 (run-hooks 'hg-log-mode-hook))
1069 (run-hooks 'hg-log-mode-hook))
1070
1070
1071 (defun hg-log (path &optional rev1 rev2 log-limit)
1071 (defun hg-log (path &optional rev1 rev2 log-limit)
1072 "Display the revision history of PATH.
1072 "Display the revision history of PATH.
1073 History is displayed between REV1 and REV2.
1073 History is displayed between REV1 and REV2.
1074 Number of displayed changesets is limited to LOG-LIMIT.
1074 Number of displayed changesets is limited to LOG-LIMIT.
1075 REV1 defaults to the tip, while
1075 REV1 defaults to the tip, while
1076 REV2 defaults to `hg-rev-completion-limit' changes from the tip revision.
1076 REV2 defaults to `hg-rev-completion-limit' changes from the tip revision.
1077 LOG-LIMIT defaults to `hg-log-limit'.
1077 LOG-LIMIT defaults to `hg-log-limit'.
1078 With a prefix argument, prompt for each parameter."
1078 With a prefix argument, prompt for each parameter."
1079 (interactive (list (hg-read-file-name " to log")
1079 (interactive (list (hg-read-file-name " to log")
1080 (hg-read-rev " to start with"
1080 (hg-read-rev " to start with"
1081 "tip")
1081 "tip")
1082 (hg-read-rev " to end with"
1082 (hg-read-rev " to end with"
1083 (format "%d" (- hg-rev-completion-limit)))
1083 (format "%d" (- hg-rev-completion-limit)))
1084 (hg-read-number "Output limited to: "
1084 (hg-read-number "Output limited to: "
1085 hg-log-limit)))
1085 hg-log-limit)))
1086 (let ((a-path (hg-abbrev-file-name path))
1086 (let ((a-path (hg-abbrev-file-name path))
1087 (r1 (or rev1 (format "-%d" hg-rev-completion-limit)))
1087 (r1 (or rev1 (format "-%d" hg-rev-completion-limit)))
1088 (r2 (or rev2 rev1 "tip"))
1088 (r2 (or rev2 rev1 "tip"))
1089 (limit (format "%d" (or log-limit hg-log-limit))))
1089 (limit (format "%d" (or log-limit hg-log-limit))))
1090 (hg-view-output ((if (equal r1 r2)
1090 (hg-view-output ((if (equal r1 r2)
1091 (format "Mercurial: Log of rev %s of %s" rev1 a-path)
1091 (format "Mercurial: Log of rev %s of %s" rev1 a-path)
1092 (format
1092 (format
1093 "Mercurial: at most %s log(s) from rev %s to %s of %s"
1093 "Mercurial: at most %s log(s) from rev %s to %s of %s"
1094 limit r1 r2 a-path)))
1094 limit r1 r2 a-path)))
1095 (eval (list* 'call-process (hg-binary) nil t nil
1095 (eval (list* 'call-process (hg-binary) nil t nil
1096 "log"
1096 "log"
1097 "-r" (format "%s:%s" r1 r2)
1097 "-r" (format "%s:%s" r1 r2)
1098 "-l" limit
1098 "-l" limit
1099 (if (> (length path) (length (hg-root path)))
1099 (if (> (length path) (length (hg-root path)))
1100 (cons path nil)
1100 (cons path nil)
1101 nil)))
1101 nil)))
1102 (hg-log-mode)
1102 (hg-log-mode)
1103 (cd (hg-root path)))))
1103 (cd (hg-root path)))))
1104
1104
1105 (defun hg-log-repo (path &optional rev1 rev2 log-limit)
1105 (defun hg-log-repo (path &optional rev1 rev2 log-limit)
1106 "Display the revision history of the repository containing PATH.
1106 "Display the revision history of the repository containing PATH.
1107 History is displayed between REV1 and REV2.
1107 History is displayed between REV1 and REV2.
1108 Number of displayed changesets is limited to LOG-LIMIT,
1108 Number of displayed changesets is limited to LOG-LIMIT,
1109 REV1 defaults to the tip, while
1109 REV1 defaults to the tip, while
1110 REV2 defaults to `hg-rev-completion-limit' changes from the tip revision.
1110 REV2 defaults to `hg-rev-completion-limit' changes from the tip revision.
1111 LOG-LIMIT defaults to `hg-log-limit'.
1111 LOG-LIMIT defaults to `hg-log-limit'.
1112 With a prefix argument, prompt for each parameter."
1112 With a prefix argument, prompt for each parameter."
1113 (interactive (list (hg-read-file-name " to log")
1113 (interactive (list (hg-read-file-name " to log")
1114 (hg-read-rev " to start with"
1114 (hg-read-rev " to start with"
1115 "tip")
1115 "tip")
1116 (hg-read-rev " to end with"
1116 (hg-read-rev " to end with"
1117 (format "%d" (- hg-rev-completion-limit)))
1117 (format "%d" (- hg-rev-completion-limit)))
1118 (hg-read-number "Output limited to: "
1118 (hg-read-number "Output limited to: "
1119 hg-log-limit)))
1119 hg-log-limit)))
1120 (hg-log (hg-root path) rev1 rev2 log-limit))
1120 (hg-log (hg-root path) rev1 rev2 log-limit))
1121
1121
1122 (defun hg-outgoing (&optional repo)
1122 (defun hg-outgoing (&optional repo)
1123 "Display changesets present locally that are not present in REPO."
1123 "Display changesets present locally that are not present in REPO."
1124 (interactive (list (hg-read-repo-name " where changes would go to" nil
1124 (interactive (list (hg-read-repo-name " where changes would go to" nil
1125 hg-outgoing-repository)))
1125 hg-outgoing-repository)))
1126 (hg-view-output ((format "Mercurial: Outgoing from %s to %s"
1126 (hg-view-output ((format "Mercurial: Outgoing from %s to %s"
1127 (hg-abbrev-file-name (hg-root))
1127 (hg-abbrev-file-name (hg-root))
1128 (hg-abbrev-file-name
1128 (hg-abbrev-file-name
1129 (or repo hg-outgoing-repository))))
1129 (or repo hg-outgoing-repository))))
1130 (call-process (hg-binary) nil t nil "outgoing"
1130 (call-process (hg-binary) nil t nil "outgoing"
1131 (or repo hg-outgoing-repository))
1131 (or repo hg-outgoing-repository))
1132 (hg-log-mode)
1132 (hg-log-mode)
1133 (cd (hg-root))))
1133 (cd (hg-root))))
1134
1134
1135 (defun hg-pull (&optional repo)
1135 (defun hg-pull (&optional repo)
1136 "Pull changes from repository REPO.
1136 "Pull changes from repository REPO.
1137 This does not update the working directory."
1137 This does not update the working directory."
1138 (interactive (list (hg-read-repo-name " to pull from")))
1138 (interactive (list (hg-read-repo-name " to pull from")))
1139 (hg-view-output ((format "Mercurial: Pull to %s from %s"
1139 (hg-view-output ((format "Mercurial: Pull to %s from %s"
1140 (hg-abbrev-file-name (hg-root))
1140 (hg-abbrev-file-name (hg-root))
1141 (hg-abbrev-file-name
1141 (hg-abbrev-file-name
1142 (or repo hg-incoming-repository))))
1142 (or repo hg-incoming-repository))))
1143 (call-process (hg-binary) nil t nil "pull"
1143 (call-process (hg-binary) nil t nil "pull"
1144 (or repo hg-incoming-repository))
1144 (or repo hg-incoming-repository))
1145 (cd (hg-root))))
1145 (cd (hg-root))))
1146
1146
1147 (defun hg-push (&optional repo)
1147 (defun hg-push (&optional repo)
1148 "Push changes to repository REPO."
1148 "Push changes to repository REPO."
1149 (interactive (list (hg-read-repo-name " to push to")))
1149 (interactive (list (hg-read-repo-name " to push to")))
1150 (hg-view-output ((format "Mercurial: Push from %s to %s"
1150 (hg-view-output ((format "Mercurial: Push from %s to %s"
1151 (hg-abbrev-file-name (hg-root))
1151 (hg-abbrev-file-name (hg-root))
1152 (hg-abbrev-file-name
1152 (hg-abbrev-file-name
1153 (or repo hg-outgoing-repository))))
1153 (or repo hg-outgoing-repository))))
1154 (call-process (hg-binary) nil t nil "push"
1154 (call-process (hg-binary) nil t nil "push"
1155 (or repo hg-outgoing-repository))
1155 (or repo hg-outgoing-repository))
1156 (cd (hg-root))))
1156 (cd (hg-root))))
1157
1157
1158 (defun hg-revert-buffer-internal ()
1158 (defun hg-revert-buffer-internal ()
1159 (let ((ctx (hg-buffer-context)))
1159 (let ((ctx (hg-buffer-context)))
1160 (message "Reverting %s..." buffer-file-name)
1160 (message "Reverting %s..." buffer-file-name)
1161 (hg-run0 "revert" buffer-file-name)
1161 (hg-run0 "revert" buffer-file-name)
1162 (revert-buffer t t t)
1162 (revert-buffer t t t)
1163 (hg-restore-context ctx)
1163 (hg-restore-context ctx)
1164 (hg-mode-line)
1164 (hg-mode-line)
1165 (message "Reverting %s...done" buffer-file-name)))
1165 (message "Reverting %s...done" buffer-file-name)))
1166
1166
1167 (defun hg-revert-buffer ()
1167 (defun hg-revert-buffer ()
1168 "Revert current buffer's file back to the latest committed version.
1168 "Revert current buffer's file back to the latest committed version.
1169 If the file has not changed, nothing happens. Otherwise, this
1169 If the file has not changed, nothing happens. Otherwise, this
1170 displays a diff and asks for confirmation before reverting."
1170 displays a diff and asks for confirmation before reverting."
1171 (interactive)
1171 (interactive)
1172 (let ((vc-suppress-confirm nil)
1172 (let ((vc-suppress-confirm nil)
1173 (obuf (current-buffer))
1173 (obuf (current-buffer))
1174 diff)
1174 diff)
1175 (vc-buffer-sync)
1175 (vc-buffer-sync)
1176 (unwind-protect
1176 (unwind-protect
1177 (setq diff (hg-diff buffer-file-name))
1177 (setq diff (hg-diff buffer-file-name))
1178 (when diff
1178 (when diff
1179 (unless (yes-or-no-p "Discard changes? ")
1179 (unless (yes-or-no-p "Discard changes? ")
1180 (error "Revert cancelled")))
1180 (error "Revert cancelled")))
1181 (when diff
1181 (when diff
1182 (let ((buf (current-buffer)))
1182 (let ((buf (current-buffer)))
1183 (delete-window (selected-window))
1183 (delete-window (selected-window))
1184 (kill-buffer buf))))
1184 (kill-buffer buf))))
1185 (set-buffer obuf)
1185 (set-buffer obuf)
1186 (when diff
1186 (when diff
1187 (hg-revert-buffer-internal))))
1187 (hg-revert-buffer-internal))))
1188
1188
1189 (defun hg-root (&optional path)
1189 (defun hg-root (&optional path)
1190 "Return the root of the repository that contains the given path.
1190 "Return the root of the repository that contains the given path.
1191 If the path is outside a repository, return nil.
1191 If the path is outside a repository, return nil.
1192 When called interactively, the root is printed. A prefix argument
1192 When called interactively, the root is printed. A prefix argument
1193 prompts for a path to check."
1193 prompts for a path to check."
1194 (interactive (list (hg-read-file-name)))
1194 (interactive (list (hg-read-file-name)))
1195 (if (or path (not hg-root))
1195 (if (or path (not hg-root))
1196 (let ((root (do ((prev nil dir)
1196 (let ((root (do ((prev nil dir)
1197 (dir (file-name-directory
1197 (dir (file-name-directory
1198 (or
1198 (or
1199 path
1199 path
1200 buffer-file-name
1200 buffer-file-name
1201 (expand-file-name default-directory)))
1201 (expand-file-name default-directory)))
1202 (file-name-directory (directory-file-name dir))))
1202 (file-name-directory (directory-file-name dir))))
1203 ((equal prev dir))
1203 ((equal prev dir))
1204 (when (file-directory-p (concat dir ".hg"))
1204 (when (file-directory-p (concat dir ".hg"))
1205 (return dir)))))
1205 (return dir)))))
1206 (when (interactive-p)
1206 (when (interactive-p)
1207 (if root
1207 (if root
1208 (message "The root of this repository is `%s'." root)
1208 (message "The root of this repository is `%s'." root)
1209 (message "The path `%s' is not in a Mercurial repository."
1209 (message "The path `%s' is not in a Mercurial repository."
1210 (hg-abbrev-file-name path))))
1210 (hg-abbrev-file-name path))))
1211 root)
1211 root)
1212 hg-root))
1212 hg-root))
1213
1213
1214 (defun hg-cwd (&optional path)
1215 "Return the current directory of PATH within the repository."
1216 (do ((stack nil (cons (file-name-nondirectory
1217 (directory-file-name dir))
1218 stack))
1219 (prev nil dir)
1220 (dir (file-name-directory (or path buffer-file-name
1221 (expand-file-name default-directory)))
1222 (file-name-directory (directory-file-name dir))))
1223 ((equal prev dir))
1224 (when (file-directory-p (concat dir ".hg"))
1225 (let ((cwd (mapconcat 'identity stack "/")))
1226 (unless (equal cwd "")
1227 (return (file-name-as-directory cwd)))))))
1228
1214 (defun hg-status (path)
1229 (defun hg-status (path)
1215 "Print revision control status of a file or directory.
1230 "Print revision control status of a file or directory.
1216 With prefix argument, prompt for the path to give status for.
1231 With prefix argument, prompt for the path to give status for.
1217 Names are displayed relative to the repository root."
1232 Names are displayed relative to the repository root."
1218 (interactive (list (hg-read-file-name " for status" (hg-root))))
1233 (interactive (list (hg-read-file-name " for status" (hg-root))))
1219 (let ((root (hg-root)))
1234 (let ((root (hg-root)))
1220 (hg-view-output ((format "Mercurial: Status of %s in %s"
1235 (hg-view-output ((format "Mercurial: Status of %s in %s"
1221 (let ((name (substring (expand-file-name path)
1236 (let ((name (substring (expand-file-name path)
1222 (length root))))
1237 (length root))))
1223 (if (> (length name) 0)
1238 (if (> (length name) 0)
1224 name
1239 name
1225 "*"))
1240 "*"))
1226 (hg-abbrev-file-name root)))
1241 (hg-abbrev-file-name root)))
1227 (apply 'call-process (hg-binary) nil t nil
1242 (apply 'call-process (hg-binary) nil t nil
1228 (list "--cwd" root "status" path))
1243 (list "--cwd" root "status" path))
1229 (cd (hg-root path)))))
1244 (cd (hg-root path)))))
1230
1245
1231 (defun hg-undo ()
1246 (defun hg-undo ()
1232 (interactive)
1247 (interactive)
1233 (error "not implemented"))
1248 (error "not implemented"))
1234
1249
1235 (defun hg-update ()
1250 (defun hg-update ()
1236 (interactive)
1251 (interactive)
1237 (error "not implemented"))
1252 (error "not implemented"))
1238
1253
1239 (defun hg-version-other-window ()
1254 (defun hg-version-other-window ()
1240 (interactive)
1255 (interactive)
1241 (error "not implemented"))
1256 (error "not implemented"))
1242
1257
1243
1258
1244 (provide 'mercurial)
1259 (provide 'mercurial)
1245
1260
1246
1261
1247 ;;; Local Variables:
1262 ;;; Local Variables:
1248 ;;; prompt-to-byte-compile: nil
1263 ;;; prompt-to-byte-compile: nil
1249 ;;; end:
1264 ;;; end:
General Comments 0
You need to be logged in to leave comments. Login now