]> git.eshelyaron.com Git - emacs.git/commitdiff
(vc-process-sentinel): New function.
authorStefan Monnier <monnier@iro.umontreal.ca>
Sat, 15 Sep 2007 22:25:55 +0000 (22:25 +0000)
committerStefan Monnier <monnier@iro.umontreal.ca>
Sat, 15 Sep 2007 22:25:55 +0000 (22:25 +0000)
(vc-exec-after): Use it instead of using ugly hackish analysis and
construction of Elisp code.
(vc-sentinel-movepoint): New dynamically scoped var.
(vc-print-log, vc-annotate): Set it to move the user's point.

lisp/ChangeLog
lisp/vc.el

index 9f195dfe7b61c18d0a95104667ccab45628b8250..6dcd86c9f55a6c80458822d09a56ccb3cf8973fd 100644 (file)
@@ -1,5 +1,11 @@
 2007-09-15  Stefan Monnier  <monnier@iro.umontreal.ca>
 
+       * vc.el (vc-process-sentinel): New function.
+       (vc-exec-after): Use it instead of using ugly hackish analysis and
+       construction of Elisp code.
+       (vc-sentinel-movepoint): New dynamically scoped var.
+       (vc-print-log, vc-annotate): Set it to move the user's point.
+
        * vc-cvs.el (vc-cvs-annotate-time): Use inhibit-read-only and
        inhibit-modification-hooks.
 
index 222825b054f1b1f12fba7f0793639db5e3e4af5b..6c06f9a9032a1f2e36ef6903bcd0d2e81fa7606c 100644 (file)
@@ -975,6 +975,33 @@ BUF defaults to \"*vc*\", can be a string and will be created if necessary."
           (inhibit-read-only t))
       (erase-buffer))))
 
+(defvar vc-sentinel-movepoint)          ;Dynamically scoped.
+
+(defun vc-process-sentinel (p s)
+  (let ((previous (process-get p 'vc-previous-sentinel)))
+    (if previous (funcall previous p s))
+    (with-current-buffer (process-buffer p)
+      (let (vc-sentinel-movepoint)
+        ;; Normally, we want async code such as sentinels to not move point.
+        (save-excursion
+          (goto-char (process-mark p))
+          (let ((cmds (process-get p 'vc-sentinel-commands)))
+            (process-put p 'vc-postprocess nil)
+            (dolist (cmd cmds)
+              ;; Each sentinel may move point and the next one should be run
+              ;; at that new point.  We could get the same result by having
+              ;; each sentinel read&set process-mark, but since `cmd' needs
+              ;; to work both for async and sync processes, this would be
+              ;; difficult to achieve.
+              (vc-exec-after cmd))))
+        ;; But sometimes the sentinels really want to move point.
+        (if vc-sentinel-movepoint
+            (let ((win (get-buffer-window (current-buffer) 0)))
+              (if (not win)
+                  (goto-char vc-sentinel-movepoint)
+                (with-selected-window win
+                  (goto-char vc-sentinel-movepoint)))))))))
+
 (defun vc-exec-after (code)
   "Eval CODE when the current buffer's process is done.
 If the current buffer has no process, just evaluate CODE.
@@ -992,17 +1019,12 @@ Else, add CODE to the process' sentinel."
       (eval code))
      ;; If a process is running, add CODE to the sentinel
      ((eq (process-status proc) 'run)
-      (let ((sentinel (process-sentinel proc)))
-       (set-process-sentinel proc
-         `(lambda (p s)
-            (with-current-buffer ',(current-buffer)
-               (save-excursion
-                (goto-char (process-mark p))
-               ,@(append (cdr (cdr (car   ;Strip off (save-exc (goto-char...)
-                           (cdr (cdr       ;Strip off (with-current-buffer buf
-                           (car (cdr (cdr ;Strip off (lambda (p s)
-                             sentinel))))))))
-                         (list `(vc-exec-after ',code)))))))))
+      (let ((previous (process-sentinel proc)))
+        (unless (eq previous 'vc-process-sentinel)
+          (process-put proc 'vc-previous-sentinel previous))
+        (set-process-sentinel proc 'vc-process-sentinel))
+      (process-put proc 'vc-sentinel-commands
+                   (cons code (process-get proc 'vc-sentinel-commands))))
      (t (error "Unexpected process state"))))
   nil)
 
@@ -1087,7 +1109,8 @@ that is inserted into the command line before the filename."
          (if vc-command-messages
              (message "Running %s...OK" full-command)))
        (vc-exec-after
-        `(run-hook-with-args 'vc-post-command-functions ',command ',file-or-list ',flags))
+        `(run-hook-with-args 'vc-post-command-functions
+                              ',command ',file-or-list ',flags))
        status))))
 
 (defun vc-position-context (posn)
@@ -2557,6 +2580,7 @@ If FOCUS-REV is non-nil, leave the point at that revision."
        (vc-call-backend ',(vc-backend file)
                         'show-log-entry
                         ',focus-rev)
+        (setq vc-sentinel-movepoint (point))
         (set-buffer-modified-p nil)))))
 
 (defun vc-default-log-view-mode (backend) (log-view-mode))
@@ -3279,10 +3303,8 @@ colors. `vc-annotate-background' specifies the background color."
           ;; moved it elsewhere, but really point here is not the position
           ;; of the user's cursor :-(
           (when ,current-line           ;(and (bobp))
-            (let ((win (get-buffer-window (current-buffer) 0)))
-              (when win
-                (with-selected-window win
-                  (goto-line ,current-line)))))
+            (goto-line ,current-line)
+            (setq vc-sentinel-movepoint))
           (unless (active-minibuffer-window)
             (message "Annotating... done")))))))