945 lines
36 KiB
EmacsLisp
945 lines
36 KiB
EmacsLisp
|
;;; magit-blame.el --- blame support for Magit -*- lexical-binding: t -*-
|
||
|
|
||
|
;; Copyright (C) 2012-2020 The Magit Project Contributors
|
||
|
;;
|
||
|
;; You should have received a copy of the AUTHORS.md file which
|
||
|
;; lists all contributors. If not, see http://magit.vc/authors.
|
||
|
|
||
|
;; Author: Jonas Bernoulli <jonas@bernoul.li>
|
||
|
;; Maintainer: Jonas Bernoulli <jonas@bernoul.li>
|
||
|
|
||
|
;; Magit is free software; you can redistribute it and/or modify it
|
||
|
;; under the terms of the GNU General Public License as published by
|
||
|
;; the Free Software Foundation; either version 3, or (at your option)
|
||
|
;; any later version.
|
||
|
;;
|
||
|
;; Magit is distributed in the hope that it will be useful, but WITHOUT
|
||
|
;; ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||
|
;; or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
|
||
|
;; License for more details.
|
||
|
;;
|
||
|
;; You should have received a copy of the GNU General Public License
|
||
|
;; along with Magit. If not, see http://www.gnu.org/licenses.
|
||
|
|
||
|
;;; Commentary:
|
||
|
|
||
|
;; Annotates each line in file-visiting buffer with information from
|
||
|
;; the revision which last modified the line.
|
||
|
|
||
|
;;; Code:
|
||
|
|
||
|
(eval-when-compile
|
||
|
(require 'subr-x))
|
||
|
|
||
|
(require 'magit)
|
||
|
|
||
|
;;; Options
|
||
|
|
||
|
(defgroup magit-blame nil
|
||
|
"Blame support for Magit."
|
||
|
:link '(info-link "(magit)Blaming")
|
||
|
:group 'magit-modes)
|
||
|
|
||
|
(defcustom magit-blame-styles
|
||
|
'((headings
|
||
|
(heading-format . "%-20a %C %s\n"))
|
||
|
(margin
|
||
|
(margin-format . (" %s%f" " %C %a" " %H"))
|
||
|
(margin-width . 42)
|
||
|
(margin-face . magit-blame-margin)
|
||
|
(margin-body-face . (magit-blame-dimmed)))
|
||
|
(highlight
|
||
|
(highlight-face . magit-blame-highlight))
|
||
|
(lines
|
||
|
(show-lines . t)
|
||
|
(show-message . t)))
|
||
|
"List of styles used to visualize blame information.
|
||
|
|
||
|
Each entry has the form (IDENT (KEY . VALUE)...). IDENT has
|
||
|
to be a symbol uniquely identifying the style. The following
|
||
|
KEYs are recognized:
|
||
|
|
||
|
`show-lines'
|
||
|
Whether to prefix each chunk of lines with a thin line.
|
||
|
This has no effect if `heading-format' is non-nil.
|
||
|
`show-message'
|
||
|
Whether to display a commit's summary line in the echo area
|
||
|
when crossing chunks.
|
||
|
`highlight-face'
|
||
|
Face used to highlight the first line of each chunk.
|
||
|
If this is nil, then those lines are not highlighted.
|
||
|
`heading-format'
|
||
|
String specifying the information to be shown above each
|
||
|
chunk of lines. It must end with a newline character.
|
||
|
`margin-format'
|
||
|
String specifying the information to be shown in the left
|
||
|
buffer margin. It must NOT end with a newline character.
|
||
|
This can also be a list of formats used for the lines at
|
||
|
the same positions within the chunk. If the chunk has
|
||
|
more lines than formats are specified, then the last is
|
||
|
repeated.
|
||
|
`margin-width'
|
||
|
Width of the margin, provided `margin-format' is non-nil.
|
||
|
`margin-face'
|
||
|
Face used in the margin, provided `margin-format' is
|
||
|
non-nil. This face is used in combination with the faces
|
||
|
that are specific to the used %-specs. If this is nil,
|
||
|
then `magit-blame-margin' is used.
|
||
|
`margin-body-face'
|
||
|
Face used in the margin for all but first line of a chunk.
|
||
|
This face is used in combination with the faces that are
|
||
|
specific to the used %-specs. This can also be a list of
|
||
|
faces (usually one face), in which case only these faces
|
||
|
are used and the %-spec faces are ignored. A good value
|
||
|
might be `(magit-blame-dimmed)'. If this is nil, then
|
||
|
the same face as for the first line is used.
|
||
|
|
||
|
The following %-specs can be used in `heading-format' and
|
||
|
`margin-format':
|
||
|
|
||
|
%H hash using face `magit-blame-hash'
|
||
|
%s summary using face `magit-blame-summary'
|
||
|
%a author using face `magit-blame-name'
|
||
|
%A author time using face `magit-blame-date'
|
||
|
%c committer using face `magit-blame-name'
|
||
|
%C committer time using face `magit-blame-date'
|
||
|
|
||
|
Additionally if `margin-format' ends with %f, then the string
|
||
|
that is displayed in the margin is made at least `margin-width'
|
||
|
characters wide, which may be desirable if the used face sets
|
||
|
the background color.
|
||
|
|
||
|
The style used in the current buffer can be cycled from the blame
|
||
|
popup. Blame commands (except `magit-blame-echo') use the first
|
||
|
style as the initial style when beginning to blame in a buffer."
|
||
|
:package-version '(magit . "2.13.0")
|
||
|
:group 'magit-blame
|
||
|
:type 'string)
|
||
|
|
||
|
(defcustom magit-blame-echo-style 'lines
|
||
|
"The blame visualization style used by `magit-blame-echo'.
|
||
|
A symbol that has to be used as the identifier for one of the
|
||
|
styles defined in `magit-blame-styles'."
|
||
|
:package-version '(magit . "2.13.0")
|
||
|
:group 'magit-blame
|
||
|
:type 'symbol)
|
||
|
|
||
|
(defcustom magit-blame-time-format "%F %H:%M"
|
||
|
"Format for time strings in blame headings."
|
||
|
:group 'magit-blame
|
||
|
:type 'string)
|
||
|
|
||
|
(defcustom magit-blame-read-only t
|
||
|
"Whether to initially make the blamed buffer read-only."
|
||
|
:package-version '(magit . "2.13.0")
|
||
|
:group 'magit-blame
|
||
|
:type 'boolean)
|
||
|
|
||
|
(defcustom magit-blame-disable-modes '(fci-mode yascroll-bar-mode)
|
||
|
"List of modes not compatible with Magit-Blame mode.
|
||
|
This modes are turned off when Magit-Blame mode is turned on,
|
||
|
and then turned on again when turning off the latter."
|
||
|
:group 'magit-blame
|
||
|
:type '(repeat (symbol :tag "Mode")))
|
||
|
|
||
|
(defcustom magit-blame-mode-lighter " Blame"
|
||
|
"The mode-line lighter of the Magit-Blame mode."
|
||
|
:group 'magit-blame
|
||
|
:type '(choice (const :tag "No lighter" "") string))
|
||
|
|
||
|
(defcustom magit-blame-goto-chunk-hook
|
||
|
'(magit-blame-maybe-update-revision-buffer
|
||
|
magit-blame-maybe-show-message)
|
||
|
"Hook run after point entered another chunk."
|
||
|
:package-version '(magit . "2.13.0")
|
||
|
:group 'magit-blame
|
||
|
:type 'hook
|
||
|
:get 'magit-hook-custom-get
|
||
|
:options '(magit-blame-maybe-update-revision-buffer
|
||
|
magit-blame-maybe-show-message))
|
||
|
|
||
|
;;; Faces
|
||
|
|
||
|
(defface magit-blame-highlight
|
||
|
`((((class color) (background light))
|
||
|
,@(and (>= emacs-major-version 27) '(:extend t))
|
||
|
:background "grey80"
|
||
|
:foreground "black")
|
||
|
(((class color) (background dark))
|
||
|
,@(and (>= emacs-major-version 27) '(:extend t))
|
||
|
:background "grey25"
|
||
|
:foreground "white"))
|
||
|
"Face used for highlighting when blaming.
|
||
|
Also see option `magit-blame-styles'."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-margin
|
||
|
'((t :inherit magit-blame-highlight
|
||
|
:weight normal
|
||
|
:slant normal))
|
||
|
"Face used for the blame margin by default when blaming.
|
||
|
Also see option `magit-blame-styles'."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-dimmed
|
||
|
'((t :inherit magit-dimmed
|
||
|
:weight normal
|
||
|
:slant normal))
|
||
|
"Face used for the blame margin in some cases when blaming.
|
||
|
Also see option `magit-blame-styles'."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-heading
|
||
|
`((t ,@(and (>= emacs-major-version 27) '(:extend t))
|
||
|
:inherit magit-blame-highlight
|
||
|
:weight normal
|
||
|
:slant normal))
|
||
|
"Face used for blame headings by default when blaming.
|
||
|
Also see option `magit-blame-styles'."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-summary nil
|
||
|
"Face used for commit summaries when blaming."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-hash nil
|
||
|
"Face used for commit hashes when blaming."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-name nil
|
||
|
"Face used for author and committer names when blaming."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
(defface magit-blame-date nil
|
||
|
"Face used for dates when blaming."
|
||
|
:group 'magit-faces)
|
||
|
|
||
|
;;; Chunks
|
||
|
|
||
|
(defclass magit-blame-chunk ()
|
||
|
(;; <orig-rev> <orig-line> <final-line> <num-lines>
|
||
|
(orig-rev :initarg :orig-rev)
|
||
|
(orig-line :initarg :orig-line)
|
||
|
(final-line :initarg :final-line)
|
||
|
(num-lines :initarg :num-lines)
|
||
|
;; previous <prev-rev> <prev-file>
|
||
|
(prev-rev :initform nil)
|
||
|
(prev-file :initform nil)
|
||
|
;; filename <orig-file>
|
||
|
(orig-file)))
|
||
|
|
||
|
(defun magit-current-blame-chunk (&optional type)
|
||
|
(or (and (not (and type (not (eq type magit-blame-type))))
|
||
|
(magit-blame-chunk-at (point)))
|
||
|
(and type
|
||
|
(let ((rev (or magit-buffer-refname magit-buffer-revision))
|
||
|
(file (magit-file-relative-name nil (not magit-buffer-file-name)))
|
||
|
(line (format "%i,+1" (line-number-at-pos))))
|
||
|
(unless file
|
||
|
(error "Buffer does not visit a tracked file"))
|
||
|
(with-temp-buffer
|
||
|
(magit-with-toplevel
|
||
|
(magit-git-insert
|
||
|
"blame" "--porcelain"
|
||
|
(if (memq magit-blame-type '(final removal))
|
||
|
(cons "--reverse" (magit-blame-arguments))
|
||
|
(magit-blame-arguments))
|
||
|
"-L" line rev "--" file)
|
||
|
(goto-char (point-min))
|
||
|
(car (magit-blame--parse-chunk type))))))))
|
||
|
|
||
|
(defun magit-blame-chunk-at (pos)
|
||
|
(--some (overlay-get it 'magit-blame-chunk)
|
||
|
(overlays-at pos)))
|
||
|
|
||
|
(defun magit-blame--overlay-at (&optional pos key)
|
||
|
(unless pos
|
||
|
(setq pos (point)))
|
||
|
(--first (overlay-get it (or key 'magit-blame-chunk))
|
||
|
(nconc (overlays-at pos)
|
||
|
(overlays-in pos pos))))
|
||
|
|
||
|
;;; Keymaps
|
||
|
|
||
|
(defvar magit-blame-mode-map
|
||
|
(let ((map (make-sparse-keymap)))
|
||
|
(define-key map (kbd "C-c C-q") 'magit-blame-quit)
|
||
|
map)
|
||
|
"Keymap for `magit-blame-mode'.
|
||
|
Note that most blaming key bindings are defined
|
||
|
in `magit-blame-read-only-mode-map' instead.")
|
||
|
|
||
|
(defvar magit-blame-read-only-mode-map
|
||
|
(let ((map (make-sparse-keymap)))
|
||
|
(define-key map (kbd "C-m") 'magit-show-commit)
|
||
|
(define-key map (kbd "p") 'magit-blame-previous-chunk)
|
||
|
(define-key map (kbd "P") 'magit-blame-previous-chunk-same-commit)
|
||
|
(define-key map (kbd "n") 'magit-blame-next-chunk)
|
||
|
(define-key map (kbd "N") 'magit-blame-next-chunk-same-commit)
|
||
|
(define-key map (kbd "b") 'magit-blame-addition)
|
||
|
(define-key map (kbd "r") 'magit-blame-removal)
|
||
|
(define-key map (kbd "f") 'magit-blame-reverse)
|
||
|
(define-key map (kbd "B") 'magit-blame)
|
||
|
(define-key map (kbd "c") 'magit-blame-cycle-style)
|
||
|
(define-key map (kbd "q") 'magit-blame-quit)
|
||
|
(define-key map (kbd "M-w") 'magit-blame-copy-hash)
|
||
|
(define-key map (kbd "SPC") 'magit-diff-show-or-scroll-up)
|
||
|
(define-key map (kbd "S-SPC") 'magit-diff-show-or-scroll-down)
|
||
|
(define-key map (kbd "DEL") 'magit-diff-show-or-scroll-down)
|
||
|
map)
|
||
|
"Keymap for `magit-blame-read-only-mode'.")
|
||
|
|
||
|
;;; Modes
|
||
|
;;;; Variables
|
||
|
|
||
|
(defvar-local magit-blame-buffer-read-only nil)
|
||
|
(defvar-local magit-blame-cache nil)
|
||
|
(defvar-local magit-blame-disabled-modes nil)
|
||
|
(defvar-local magit-blame-process nil)
|
||
|
(defvar-local magit-blame-recursive-p nil)
|
||
|
(defvar-local magit-blame-type nil)
|
||
|
(defvar-local magit-blame-separator nil)
|
||
|
(defvar-local magit-blame-previous-chunk nil)
|
||
|
|
||
|
(defvar-local magit-blame--style nil)
|
||
|
|
||
|
(defsubst magit-blame--style-get (key)
|
||
|
(cdr (assoc key (cdr magit-blame--style))))
|
||
|
|
||
|
;;;; Base Mode
|
||
|
|
||
|
(define-minor-mode magit-blame-mode
|
||
|
"Display blame information inline."
|
||
|
:lighter magit-blame-mode-lighter
|
||
|
(cond (magit-blame-mode
|
||
|
(when (called-interactively-p 'any)
|
||
|
(setq magit-blame-mode nil)
|
||
|
(user-error
|
||
|
(concat "Don't call `magit-blame-mode' directly; "
|
||
|
"instead use `magit-blame'")))
|
||
|
(add-hook 'after-save-hook 'magit-blame--refresh t t)
|
||
|
(add-hook 'post-command-hook 'magit-blame-goto-chunk-hook t t)
|
||
|
(add-hook 'before-revert-hook 'magit-blame--remove-overlays t t)
|
||
|
(add-hook 'after-revert-hook 'magit-blame--refresh t t)
|
||
|
(add-hook 'read-only-mode-hook 'magit-blame-toggle-read-only t t)
|
||
|
(setq magit-blame-buffer-read-only buffer-read-only)
|
||
|
(when (or magit-blame-read-only magit-buffer-file-name)
|
||
|
(read-only-mode 1))
|
||
|
(dolist (mode magit-blame-disable-modes)
|
||
|
(when (and (boundp mode) (symbol-value mode))
|
||
|
(funcall mode -1)
|
||
|
(push mode magit-blame-disabled-modes)))
|
||
|
(setq magit-blame-separator (magit-blame--format-separator))
|
||
|
(unless magit-blame--style
|
||
|
(setq magit-blame--style (car magit-blame-styles)))
|
||
|
(magit-blame--update-margin))
|
||
|
(t
|
||
|
(when (process-live-p magit-blame-process)
|
||
|
(kill-process magit-blame-process)
|
||
|
(while magit-blame-process
|
||
|
(sit-for 0.01))) ; avoid racing the sentinel
|
||
|
(remove-hook 'after-save-hook 'magit-blame--refresh t)
|
||
|
(remove-hook 'post-command-hook 'magit-blame-goto-chunk-hook t)
|
||
|
(remove-hook 'before-revert-hook 'magit-blame--remove-overlays t)
|
||
|
(remove-hook 'after-revert-hook 'magit-blame--refresh t)
|
||
|
(remove-hook 'read-only-mode-hook 'magit-blame-toggle-read-only t)
|
||
|
(unless magit-blame-buffer-read-only
|
||
|
(read-only-mode -1))
|
||
|
(magit-blame-read-only-mode -1)
|
||
|
(dolist (mode magit-blame-disabled-modes)
|
||
|
(funcall mode 1))
|
||
|
(kill-local-variable 'magit-blame-disabled-modes)
|
||
|
(kill-local-variable 'magit-blame-type)
|
||
|
(kill-local-variable 'magit-blame--style)
|
||
|
(magit-blame--update-margin)
|
||
|
(magit-blame--remove-overlays))))
|
||
|
|
||
|
(defun magit-blame--refresh ()
|
||
|
(magit-blame--run (magit-blame-arguments)))
|
||
|
|
||
|
(defun magit-blame-goto-chunk-hook ()
|
||
|
(let ((chunk (magit-blame-chunk-at (point))))
|
||
|
(when (cl-typep chunk 'magit-blame-chunk)
|
||
|
(unless (eq chunk magit-blame-previous-chunk)
|
||
|
(run-hooks 'magit-blame-goto-chunk-hook))
|
||
|
(setq magit-blame-previous-chunk chunk))))
|
||
|
|
||
|
(defun magit-blame-toggle-read-only ()
|
||
|
(magit-blame-read-only-mode (if buffer-read-only 1 -1)))
|
||
|
|
||
|
;;;; Read-Only Mode
|
||
|
|
||
|
(define-minor-mode magit-blame-read-only-mode
|
||
|
"Provide keybindings for Magit-Blame mode.
|
||
|
|
||
|
This minor-mode provides the key bindings for Magit-Blame mode,
|
||
|
but only when Read-Only mode is also enabled because these key
|
||
|
bindings would otherwise conflict badly with regular bindings.
|
||
|
|
||
|
When both Magit-Blame mode and Read-Only mode are enabled, then
|
||
|
this mode gets automatically enabled too and when one of these
|
||
|
modes is toggled, then this mode also gets toggled automatically.
|
||
|
|
||
|
\\{magit-blame-read-only-mode-map}")
|
||
|
|
||
|
;;;; Kludges
|
||
|
|
||
|
(defun magit-blame-put-keymap-before-view-mode ()
|
||
|
"Put `magit-blame-read-only-mode' ahead of `view-mode' in `minor-mode-map-alist'."
|
||
|
(--when-let (assq 'magit-blame-read-only-mode
|
||
|
(cl-member 'view-mode minor-mode-map-alist :key #'car))
|
||
|
(setq minor-mode-map-alist
|
||
|
(cons it (delq it minor-mode-map-alist))))
|
||
|
(remove-hook 'view-mode-hook #'magit-blame-put-keymap-before-view-mode))
|
||
|
|
||
|
(add-hook 'view-mode-hook #'magit-blame-put-keymap-before-view-mode)
|
||
|
|
||
|
;;; Process
|
||
|
|
||
|
(defun magit-blame--run (args)
|
||
|
(magit-with-toplevel
|
||
|
(unless magit-blame-mode
|
||
|
(magit-blame-mode 1))
|
||
|
(message "Blaming...")
|
||
|
(magit-blame-run-process
|
||
|
(or magit-buffer-refname magit-buffer-revision)
|
||
|
(magit-file-relative-name nil (not magit-buffer-file-name))
|
||
|
(if (memq magit-blame-type '(final removal))
|
||
|
(cons "--reverse" args)
|
||
|
args)
|
||
|
(list (line-number-at-pos (window-start))
|
||
|
(line-number-at-pos (1- (window-end nil t)))))
|
||
|
(set-process-sentinel magit-this-process
|
||
|
'magit-blame-process-quickstart-sentinel)))
|
||
|
|
||
|
(defun magit-blame-run-process (revision file args &optional lines)
|
||
|
(let ((process (magit-parse-git-async
|
||
|
"blame" "--incremental" args
|
||
|
(and lines (list "-L" (apply #'format "%s,%s" lines)))
|
||
|
revision "--" file)))
|
||
|
(set-process-filter process 'magit-blame-process-filter)
|
||
|
(set-process-sentinel process 'magit-blame-process-sentinel)
|
||
|
(process-put process 'arguments (list revision file args))
|
||
|
(setq magit-blame-cache (make-hash-table :test 'equal))
|
||
|
(setq magit-blame-process process)))
|
||
|
|
||
|
(defun magit-blame-process-quickstart-sentinel (process event)
|
||
|
(when (memq (process-status process) '(exit signal))
|
||
|
(magit-blame-process-sentinel process event t)
|
||
|
(magit-blame-assert-buffer process)
|
||
|
(with-current-buffer (process-get process 'command-buf)
|
||
|
(when magit-blame-mode
|
||
|
(let ((default-directory (magit-toplevel)))
|
||
|
(apply #'magit-blame-run-process
|
||
|
(process-get process 'arguments)))))))
|
||
|
|
||
|
(defun magit-blame-process-sentinel (process _event &optional quiet)
|
||
|
(let ((status (process-status process)))
|
||
|
(when (memq status '(exit signal))
|
||
|
(kill-buffer (process-buffer process))
|
||
|
(if (and (eq status 'exit)
|
||
|
(zerop (process-exit-status process)))
|
||
|
(unless quiet
|
||
|
(message "Blaming...done"))
|
||
|
(magit-blame-assert-buffer process)
|
||
|
(with-current-buffer (process-get process 'command-buf)
|
||
|
(if magit-blame-mode
|
||
|
(progn (magit-blame-mode -1)
|
||
|
(message "Blaming...failed"))
|
||
|
(message "Blaming...aborted"))))
|
||
|
(kill-local-variable 'magit-blame-process))))
|
||
|
|
||
|
(defun magit-blame-process-filter (process string)
|
||
|
(internal-default-process-filter process string)
|
||
|
(let ((buf (process-get process 'command-buf))
|
||
|
(pos (process-get process 'parsed))
|
||
|
(mark (process-mark process))
|
||
|
type cache)
|
||
|
(with-current-buffer buf
|
||
|
(setq type magit-blame-type)
|
||
|
(setq cache magit-blame-cache))
|
||
|
(with-current-buffer (process-buffer process)
|
||
|
(goto-char pos)
|
||
|
(while (and (< (point) mark)
|
||
|
(save-excursion (re-search-forward "^filename .+\n" nil t)))
|
||
|
(pcase-let* ((`(,chunk ,revinfo)
|
||
|
(magit-blame--parse-chunk type))
|
||
|
(rev (oref chunk orig-rev)))
|
||
|
(if revinfo
|
||
|
(puthash rev revinfo cache)
|
||
|
(setq revinfo
|
||
|
(or (gethash rev cache)
|
||
|
(puthash rev (magit-blame--commit-alist rev) cache))))
|
||
|
(magit-blame--make-overlays buf chunk revinfo))
|
||
|
(process-put process 'parsed (point))))))
|
||
|
|
||
|
(defun magit-blame--parse-chunk (type)
|
||
|
(let (chunk revinfo)
|
||
|
(looking-at "^\\(.\\{40\\}\\) \\([0-9]+\\) \\([0-9]+\\) \\([0-9]+\\)")
|
||
|
(with-slots (orig-rev orig-file prev-rev prev-file)
|
||
|
(setq chunk (magit-blame-chunk
|
||
|
:orig-rev (match-string 1)
|
||
|
:orig-line (string-to-number (match-string 2))
|
||
|
:final-line (string-to-number (match-string 3))
|
||
|
:num-lines (string-to-number (match-string 4))))
|
||
|
(forward-line)
|
||
|
(let (done)
|
||
|
(while (not done)
|
||
|
(cond ((looking-at "^filename \\(.+\\)")
|
||
|
(setq done t)
|
||
|
(setf orig-file (match-string 1)))
|
||
|
((looking-at "^previous \\(.\\{40\\}\\) \\(.+\\)")
|
||
|
(setf prev-rev (match-string 1))
|
||
|
(setf prev-file (match-string 2)))
|
||
|
((looking-at "^\\([^ ]+\\) \\(.+\\)")
|
||
|
(push (cons (match-string 1)
|
||
|
(match-string 2)) revinfo)))
|
||
|
(forward-line)))
|
||
|
(when (and (eq type 'removal) prev-rev)
|
||
|
(cl-rotatef orig-rev prev-rev)
|
||
|
(cl-rotatef orig-file prev-file)
|
||
|
(setq revinfo nil)))
|
||
|
(list chunk revinfo)))
|
||
|
|
||
|
(defun magit-blame--commit-alist (rev)
|
||
|
(cl-mapcar 'cons
|
||
|
'("summary"
|
||
|
"author" "author-time" "author-tz"
|
||
|
"committer" "committer-time" "committer-tz")
|
||
|
(split-string (magit-rev-format "%s\v%an\v%ad\v%cn\v%cd" rev
|
||
|
"--date=format:%s\v%z")
|
||
|
"\v")))
|
||
|
|
||
|
(defun magit-blame-assert-buffer (process)
|
||
|
(unless (buffer-live-p (process-get process 'command-buf))
|
||
|
(kill-process process)
|
||
|
(user-error "Buffer being blamed has been killed")))
|
||
|
|
||
|
;;; Display
|
||
|
|
||
|
(defun magit-blame--make-overlays (buf chunk revinfo)
|
||
|
(with-current-buffer buf
|
||
|
(save-excursion
|
||
|
(save-restriction
|
||
|
(widen)
|
||
|
(goto-char (point-min))
|
||
|
(forward-line (1- (oref chunk final-line)))
|
||
|
(let ((beg (point))
|
||
|
(end (save-excursion
|
||
|
(forward-line (oref chunk num-lines))
|
||
|
(point))))
|
||
|
(magit-blame--remove-overlays beg end)
|
||
|
(magit-blame--make-margin-overlays chunk revinfo beg end)
|
||
|
(magit-blame--make-heading-overlay chunk revinfo beg end)
|
||
|
(magit-blame--make-highlight-overlay chunk beg))))))
|
||
|
|
||
|
(defun magit-blame--make-margin-overlays (chunk revinfo _beg end)
|
||
|
(save-excursion
|
||
|
(let ((line 0))
|
||
|
(while (< (point) end)
|
||
|
(magit-blame--make-margin-overlay chunk revinfo line)
|
||
|
(forward-line)
|
||
|
(cl-incf line)))))
|
||
|
|
||
|
(defun magit-blame--make-margin-overlay (chunk revinfo line)
|
||
|
(let* ((end (line-end-position))
|
||
|
;; If possible avoid putting this on the first character
|
||
|
;; of the line to avoid a conflict with the line overlay.
|
||
|
(beg (min (1+ (line-beginning-position)) end))
|
||
|
(ov (make-overlay beg end)))
|
||
|
(overlay-put ov 'magit-blame-chunk chunk)
|
||
|
(overlay-put ov 'magit-blame-revinfo revinfo)
|
||
|
(overlay-put ov 'magit-blame-margin line)
|
||
|
(magit-blame--update-margin-overlay ov)))
|
||
|
|
||
|
(defun magit-blame--make-heading-overlay (chunk revinfo beg end)
|
||
|
(let ((ov (make-overlay beg end)))
|
||
|
(overlay-put ov 'magit-blame-chunk chunk)
|
||
|
(overlay-put ov 'magit-blame-revinfo revinfo)
|
||
|
(overlay-put ov 'magit-blame-heading t)
|
||
|
(magit-blame--update-heading-overlay ov)))
|
||
|
|
||
|
(defun magit-blame--make-highlight-overlay (chunk beg)
|
||
|
(let ((ov (make-overlay beg (1+ (line-end-position)))))
|
||
|
(overlay-put ov 'magit-blame-chunk chunk)
|
||
|
(overlay-put ov 'magit-blame-highlight t)
|
||
|
(magit-blame--update-highlight-overlay ov)))
|
||
|
|
||
|
(defun magit-blame--update-margin ()
|
||
|
(setq left-margin-width (or (magit-blame--style-get 'margin-width) 0))
|
||
|
(set-window-buffer (selected-window) (current-buffer)))
|
||
|
|
||
|
(defun magit-blame--update-overlays ()
|
||
|
(save-restriction
|
||
|
(widen)
|
||
|
(dolist (ov (overlays-in (point-min) (point-max)))
|
||
|
(cond ((overlay-get ov 'magit-blame-heading)
|
||
|
(magit-blame--update-heading-overlay ov))
|
||
|
((overlay-get ov 'magit-blame-margin)
|
||
|
(magit-blame--update-margin-overlay ov))
|
||
|
((overlay-get ov 'magit-blame-highlight)
|
||
|
(magit-blame--update-highlight-overlay ov))))))
|
||
|
|
||
|
(defun magit-blame--update-margin-overlay (ov)
|
||
|
(overlay-put
|
||
|
ov 'before-string
|
||
|
(and (magit-blame--style-get 'margin-width)
|
||
|
(propertize
|
||
|
"o" 'display
|
||
|
(list (list 'margin 'left-margin)
|
||
|
(let ((line (overlay-get ov 'magit-blame-margin))
|
||
|
(format (magit-blame--style-get 'margin-format))
|
||
|
(face (magit-blame--style-get 'margin-face)))
|
||
|
(magit-blame--format-string
|
||
|
ov
|
||
|
(or (and (atom format)
|
||
|
format)
|
||
|
(nth line format)
|
||
|
(car (last format)))
|
||
|
(or (and (not (zerop line))
|
||
|
(magit-blame--style-get 'margin-body-face))
|
||
|
face
|
||
|
'magit-blame-margin))))))))
|
||
|
|
||
|
(defun magit-blame--update-heading-overlay (ov)
|
||
|
(overlay-put
|
||
|
ov 'before-string
|
||
|
(--if-let (magit-blame--style-get 'heading-format)
|
||
|
(magit-blame--format-string ov it 'magit-blame-heading)
|
||
|
(and (magit-blame--style-get 'show-lines)
|
||
|
(or (not (magit-blame--style-get 'margin-format))
|
||
|
(save-excursion
|
||
|
(goto-char (overlay-start ov))
|
||
|
;; Special case of the special case described in
|
||
|
;; `magit-blame--make-margin-overlay'. For empty
|
||
|
;; lines it is not possible to show both overlays
|
||
|
;; without the line being to high.
|
||
|
(not (= (point) (line-end-position)))))
|
||
|
magit-blame-separator))))
|
||
|
|
||
|
(defun magit-blame--update-highlight-overlay (ov)
|
||
|
(overlay-put ov 'font-lock-face (magit-blame--style-get 'highlight-face)))
|
||
|
|
||
|
(defun magit-blame--format-string (ov format face)
|
||
|
(let* ((chunk (overlay-get ov 'magit-blame-chunk))
|
||
|
(revinfo (overlay-get ov 'magit-blame-revinfo))
|
||
|
(key (list format face))
|
||
|
(string (cdr (assoc key revinfo))))
|
||
|
(unless string
|
||
|
(setq string
|
||
|
(and format
|
||
|
(magit-blame--format-string-1 (oref chunk orig-rev)
|
||
|
revinfo format face)))
|
||
|
(nconc revinfo (list (cons key string))))
|
||
|
string))
|
||
|
|
||
|
(defun magit-blame--format-string-1 (rev revinfo format face)
|
||
|
(let ((str
|
||
|
(if (equal rev "0000000000000000000000000000000000000000")
|
||
|
(propertize (concat (if (string-prefix-p "\s" format) "\s" "")
|
||
|
"Not Yet Committed"
|
||
|
(if (string-suffix-p "\n" format) "\n" ""))
|
||
|
'font-lock-face face)
|
||
|
(magit--format-spec
|
||
|
(propertize format 'font-lock-face face)
|
||
|
(cl-flet* ((p0 (s f)
|
||
|
(propertize s 'font-lock-face
|
||
|
(if face
|
||
|
(if (listp face)
|
||
|
face
|
||
|
(list f face))
|
||
|
f)))
|
||
|
(p1 (k f)
|
||
|
(p0 (cdr (assoc k revinfo)) f))
|
||
|
(p2 (k1 k2 f)
|
||
|
(p0 (magit-blame--format-time-string
|
||
|
(cdr (assoc k1 revinfo))
|
||
|
(cdr (assoc k2 revinfo)))
|
||
|
f)))
|
||
|
`((?H . ,(p0 rev 'magit-blame-hash))
|
||
|
(?s . ,(p1 "summary" 'magit-blame-summary))
|
||
|
(?a . ,(p1 "author" 'magit-blame-name))
|
||
|
(?c . ,(p1 "committer" 'magit-blame-name))
|
||
|
(?A . ,(p2 "author-time" "author-tz" 'magit-blame-date))
|
||
|
(?C . ,(p2 "committer-time" "committer-tz" 'magit-blame-date))
|
||
|
(?f . "")))))))
|
||
|
(if-let ((width (and (string-suffix-p "%f" format)
|
||
|
(magit-blame--style-get 'margin-width))))
|
||
|
(concat str
|
||
|
(propertize (make-string (max 0 (- width (length str))) ?\s)
|
||
|
'font-lock-face face))
|
||
|
str)))
|
||
|
|
||
|
(defun magit-blame--format-separator ()
|
||
|
(propertize
|
||
|
(concat (propertize "\s" 'display '(space :height (2)))
|
||
|
(propertize "\n" 'line-height t))
|
||
|
'font-lock-face (list :background
|
||
|
(face-attribute 'magit-blame-heading
|
||
|
:background nil t))))
|
||
|
|
||
|
(defun magit-blame--format-time-string (time tz)
|
||
|
(let* ((time-format (or (magit-blame--style-get 'time-format)
|
||
|
magit-blame-time-format))
|
||
|
(tz-in-second (and (string-match "%z" time-format)
|
||
|
(car (last (parse-time-string tz))))))
|
||
|
(format-time-string time-format
|
||
|
(seconds-to-time (string-to-number time))
|
||
|
tz-in-second)))
|
||
|
|
||
|
(defun magit-blame--remove-overlays (&optional beg end)
|
||
|
(save-restriction
|
||
|
(widen)
|
||
|
(dolist (ov (overlays-in (or beg (point-min))
|
||
|
(or end (point-max))))
|
||
|
(when (overlay-get ov 'magit-blame-chunk)
|
||
|
(delete-overlay ov)))))
|
||
|
|
||
|
(defun magit-blame-maybe-show-message ()
|
||
|
(when (magit-blame--style-get 'show-message)
|
||
|
(let ((message-log-max 0))
|
||
|
(if-let ((msg (cdr (assoc "summary"
|
||
|
(gethash (oref (magit-current-blame-chunk)
|
||
|
orig-rev)
|
||
|
magit-blame-cache)))))
|
||
|
(progn (set-text-properties 0 (length msg) nil msg)
|
||
|
(message msg))
|
||
|
(message "Commit data not available yet. Still blaming.")))))
|
||
|
|
||
|
;;; Commands
|
||
|
|
||
|
;;;###autoload (autoload 'magit-blame-echo "magit-blame" nil t)
|
||
|
(transient-define-suffix magit-blame-echo (args)
|
||
|
"For each line show the revision in which it was added.
|
||
|
Show the information about the chunk at point in the echo area
|
||
|
when moving between chunks. Unlike other blaming commands, do
|
||
|
not turn on `read-only-mode'."
|
||
|
:if (lambda ()
|
||
|
(and buffer-file-name
|
||
|
(or (not magit-blame-mode)
|
||
|
buffer-read-only)))
|
||
|
(interactive (list (magit-blame-arguments)))
|
||
|
(when magit-buffer-file-name
|
||
|
(user-error "Blob buffers aren't supported"))
|
||
|
(setq-local magit-blame--style
|
||
|
(assq magit-blame-echo-style magit-blame-styles))
|
||
|
(setq-local magit-blame-disable-modes
|
||
|
(cons 'eldoc-mode magit-blame-disable-modes))
|
||
|
(if (not magit-blame-mode)
|
||
|
(let ((magit-blame-read-only nil))
|
||
|
(magit-blame--pre-blame-assert 'addition)
|
||
|
(magit-blame--pre-blame-setup 'addition)
|
||
|
(magit-blame--run args))
|
||
|
(read-only-mode -1)
|
||
|
(magit-blame--update-overlays)))
|
||
|
|
||
|
;;;###autoload (autoload 'magit-blame-addition "magit-blame" nil t)
|
||
|
(transient-define-suffix magit-blame-addition (args)
|
||
|
"For each line show the revision in which it was added."
|
||
|
(interactive (list (magit-blame-arguments)))
|
||
|
(magit-blame--pre-blame-assert 'addition)
|
||
|
(magit-blame--pre-blame-setup 'addition)
|
||
|
(magit-blame--run args))
|
||
|
|
||
|
;;;###autoload (autoload 'magit-blame-removal "magit-blame" nil t)
|
||
|
(transient-define-suffix magit-blame-removal (args)
|
||
|
"For each line show the revision in which it was removed."
|
||
|
:if-nil 'buffer-file-name
|
||
|
(interactive (list (magit-blame-arguments)))
|
||
|
(unless magit-buffer-file-name
|
||
|
(user-error "Only blob buffers can be blamed in reverse"))
|
||
|
(magit-blame--pre-blame-assert 'removal)
|
||
|
(magit-blame--pre-blame-setup 'removal)
|
||
|
(magit-blame--run args))
|
||
|
|
||
|
;;;###autoload (autoload 'magit-blame-reverse "magit-blame" nil t)
|
||
|
(transient-define-suffix magit-blame-reverse (args)
|
||
|
"For each line show the last revision in which it still exists."
|
||
|
:if-nil 'buffer-file-name
|
||
|
(interactive (list (magit-blame-arguments)))
|
||
|
(unless magit-buffer-file-name
|
||
|
(user-error "Only blob buffers can be blamed in reverse"))
|
||
|
(magit-blame--pre-blame-assert 'final)
|
||
|
(magit-blame--pre-blame-setup 'final)
|
||
|
(magit-blame--run args))
|
||
|
|
||
|
(defun magit-blame--pre-blame-assert (type)
|
||
|
(unless (magit-toplevel)
|
||
|
(magit--not-inside-repository-error))
|
||
|
(if (and magit-blame-mode
|
||
|
(eq type magit-blame-type))
|
||
|
(if-let ((chunk (magit-current-blame-chunk)))
|
||
|
(unless (oref chunk prev-rev)
|
||
|
(user-error "Chunk has no further history"))
|
||
|
(user-error "Commit data not available yet. Still blaming."))
|
||
|
(unless (magit-file-relative-name nil (not magit-buffer-file-name))
|
||
|
(if buffer-file-name
|
||
|
(user-error "Buffer isn't visiting a tracked file")
|
||
|
(user-error "Buffer isn't visiting a file")))))
|
||
|
|
||
|
(defun magit-blame--pre-blame-setup (type)
|
||
|
(when magit-blame-mode
|
||
|
(if (eq type magit-blame-type)
|
||
|
(let ((style magit-blame--style))
|
||
|
(magit-blame-visit-other-file)
|
||
|
(setq-local magit-blame--style style)
|
||
|
(setq-local magit-blame-recursive-p t)
|
||
|
;; Set window-start for the benefit of quickstart.
|
||
|
(redisplay))
|
||
|
(magit-blame--remove-overlays)))
|
||
|
(setq magit-blame-type type))
|
||
|
|
||
|
(defun magit-blame-visit-other-file ()
|
||
|
"Visit another blob related to the current chunk."
|
||
|
(interactive)
|
||
|
(with-slots (prev-rev prev-file orig-line)
|
||
|
(magit-current-blame-chunk)
|
||
|
(unless prev-rev
|
||
|
(user-error "Chunk has no further history"))
|
||
|
(magit-with-toplevel
|
||
|
(magit-find-file prev-rev prev-file))
|
||
|
;; TODO Adjust line like magit-diff-visit-file.
|
||
|
(goto-char (point-min))
|
||
|
(forward-line (1- orig-line))))
|
||
|
|
||
|
(defun magit-blame-visit-file ()
|
||
|
"Visit the blob related to the current chunk."
|
||
|
(interactive)
|
||
|
(with-slots (orig-rev orig-file orig-line)
|
||
|
(magit-current-blame-chunk)
|
||
|
(magit-with-toplevel
|
||
|
(magit-find-file orig-rev orig-file))
|
||
|
(goto-char (point-min))
|
||
|
(forward-line (1- orig-line))))
|
||
|
|
||
|
(transient-define-suffix magit-blame-quit ()
|
||
|
"Turn off Magit-Blame mode.
|
||
|
If the buffer was created during a recursive blame,
|
||
|
then also kill the buffer."
|
||
|
:if-non-nil 'magit-blame-mode
|
||
|
(interactive)
|
||
|
(magit-blame-mode -1)
|
||
|
(when magit-blame-recursive-p
|
||
|
(kill-buffer)))
|
||
|
|
||
|
(defun magit-blame-next-chunk ()
|
||
|
"Move to the next chunk."
|
||
|
(interactive)
|
||
|
(--if-let (next-single-char-property-change (point) 'magit-blame-chunk)
|
||
|
(goto-char it)
|
||
|
(user-error "No more chunks")))
|
||
|
|
||
|
(defun magit-blame-previous-chunk ()
|
||
|
"Move to the previous chunk."
|
||
|
(interactive)
|
||
|
(--if-let (previous-single-char-property-change (point) 'magit-blame-chunk)
|
||
|
(goto-char it)
|
||
|
(user-error "No more chunks")))
|
||
|
|
||
|
(defun magit-blame-next-chunk-same-commit (&optional previous)
|
||
|
"Move to the next chunk from the same commit.\n\n(fn)"
|
||
|
(interactive)
|
||
|
(if-let ((rev (oref (magit-current-blame-chunk) orig-rev)))
|
||
|
(let ((pos (point)) ov)
|
||
|
(save-excursion
|
||
|
(while (and (not ov)
|
||
|
(not (= pos (if previous (point-min) (point-max))))
|
||
|
(setq pos (funcall
|
||
|
(if previous
|
||
|
'previous-single-char-property-change
|
||
|
'next-single-char-property-change)
|
||
|
pos 'magit-blame-chunk)))
|
||
|
(--when-let (magit-blame--overlay-at pos)
|
||
|
(when (equal (oref (magit-blame-chunk-at pos) orig-rev) rev)
|
||
|
(setq ov it)))))
|
||
|
(if ov
|
||
|
(goto-char (overlay-start ov))
|
||
|
(user-error "No more chunks from same commit")))
|
||
|
(user-error "This chunk hasn't been blamed yet")))
|
||
|
|
||
|
(defun magit-blame-previous-chunk-same-commit ()
|
||
|
"Move to the previous chunk from the same commit."
|
||
|
(interactive)
|
||
|
(magit-blame-next-chunk-same-commit 'previous-single-char-property-change))
|
||
|
|
||
|
(defun magit-blame-cycle-style ()
|
||
|
"Change how blame information is visualized.
|
||
|
Cycle through the elements of option `magit-blame-styles'."
|
||
|
(interactive)
|
||
|
(setq magit-blame--style
|
||
|
(or (cadr (cl-member (car magit-blame--style)
|
||
|
magit-blame-styles :key #'car))
|
||
|
(car magit-blame-styles)))
|
||
|
(magit-blame--update-margin)
|
||
|
(magit-blame--update-overlays))
|
||
|
|
||
|
(defun magit-blame-copy-hash ()
|
||
|
"Save hash of the current chunk's commit to the kill ring.
|
||
|
|
||
|
When the region is active, then save the region's content
|
||
|
instead of the hash, like `kill-ring-save' would."
|
||
|
(interactive)
|
||
|
(if (use-region-p)
|
||
|
(call-interactively #'copy-region-as-kill)
|
||
|
(kill-new (message "%s" (oref (magit-current-blame-chunk) orig-rev)))))
|
||
|
|
||
|
;;; Popup
|
||
|
|
||
|
;;;###autoload (autoload 'magit-blame "magit-blame" nil t)
|
||
|
(transient-define-prefix magit-blame ()
|
||
|
"Show the commits that added or removed lines in the visited file."
|
||
|
:man-page "git-blame"
|
||
|
:value '("-w")
|
||
|
["Arguments"
|
||
|
("-w" "Ignore whitespace" "-w")
|
||
|
("-r" "Do not treat root commits as boundaries" "--root")
|
||
|
(magit-blame:-M)
|
||
|
(magit-blame:-C)]
|
||
|
["Actions"
|
||
|
("b" "Show commits adding lines" magit-blame-addition)
|
||
|
("r" "Show commits removing lines" magit-blame-removal)
|
||
|
("f" "Show last commits that still have lines" magit-blame-reverse)
|
||
|
("m" "Blame echo" magit-blame-echo)
|
||
|
("q" "Quit blaming" magit-blame-quit)]
|
||
|
["Refresh"
|
||
|
:if-non-nil magit-blame-mode
|
||
|
("c" "Cycle style" magit-blame-cycle-style)])
|
||
|
|
||
|
(defun magit-blame-arguments ()
|
||
|
(transient-args 'magit-blame))
|
||
|
|
||
|
(transient-define-argument magit-blame:-M ()
|
||
|
:description "Detect lines moved or copied within a file"
|
||
|
:class 'transient-option
|
||
|
:argument "-M"
|
||
|
:reader 'transient-read-number-N+)
|
||
|
|
||
|
(transient-define-argument magit-blame:-C ()
|
||
|
:description "Detect lines moved or copied between files"
|
||
|
:class 'transient-option
|
||
|
:argument "-C"
|
||
|
:reader 'transient-read-number-N+)
|
||
|
|
||
|
;;; Utilities
|
||
|
|
||
|
(defun magit-blame-maybe-update-revision-buffer ()
|
||
|
(when-let ((chunk (magit-current-blame-chunk))
|
||
|
(commit (oref chunk orig-rev))
|
||
|
(buffer (magit-get-mode-buffer 'magit-revision-mode nil t)))
|
||
|
(if magit--update-revision-buffer
|
||
|
(setq magit--update-revision-buffer (list commit buffer))
|
||
|
(setq magit--update-revision-buffer (list commit buffer))
|
||
|
(run-with-idle-timer
|
||
|
magit-update-other-window-delay nil
|
||
|
(lambda ()
|
||
|
(pcase-let ((`(,rev ,buf) magit--update-revision-buffer))
|
||
|
(setq magit--update-revision-buffer nil)
|
||
|
(when (buffer-live-p buf)
|
||
|
(let ((magit-display-buffer-noselect t))
|
||
|
(apply #'magit-show-commit rev
|
||
|
(magit-diff-arguments 'magit-revision-mode))))))))))
|
||
|
|
||
|
;;; _
|
||
|
(provide 'magit-blame)
|
||
|
;;; magit-blame.el ends here
|