my-emacs-d/elpa/magit-20161022.1845/magit-log.el

1522 lines
61 KiB
EmacsLisp
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

;;; magit-log.el --- inspect Git history -*- lexical-binding: t -*-
;; Copyright (C) 2010-2016 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:
;; This library implements support for looking at Git logs, including
;; special logs like reflogs and cherry-logs, as well as for selecting
;; a commit from a log.
;;; Code:
(require 'magit-core)
(require 'magit-diff)
(declare-function magit-blob-visit 'magit)
(declare-function magit-insert-head-branch-header 'magit)
(declare-function magit-insert-upstream-branch-header 'magit)
(declare-function magit-read-file-from-rev 'magit)
(declare-function magit-show-commit 'magit)
(defvar magit-refs-indent-cherry-lines)
(defvar magit-refs-show-commit-count)
(defvar magit-status-sections-hook)
(require 'ansi-color)
(require 'crm)
;;; Options
;;;; Log Mode
(defgroup magit-log nil
"Inspect and manipulate Git history."
:group 'magit-modes)
(defcustom magit-log-mode-hook nil
"Hook run after entering Magit-Log mode."
:group 'magit-log
:type 'hook)
(defcustom magit-log-arguments '("-n256" "--graph" "--decorate")
"The log arguments used in `magit-log-mode' buffers."
:package-version '(magit . "2.3.0")
:group 'magit-log
:group 'magit-commands
:type '(repeat (string :tag "Argument")))
(defcustom magit-log-remove-graph-args '("--follow" "--grep" "-G" "-S" "-L")
"The log arguments that cause the `--graph' argument to be dropped."
:package-version '(magit . "2.3.0")
:group 'magit-log
:type '(repeat (string :tag "Argument"))
:options '("--follow" "--grep" "-G" "-S" "-L"))
(defcustom magit-log-revision-headers-format "\
%+b
Author: %aN <%aE>
Committer: %cN <%cE>"
"Additional format string used with the `++header' argument."
:package-version '(magit . "2.3.0")
:group 'magit-log
:type 'string)
(defcustom magit-log-auto-more nil
"Insert more log entries automatically when moving past the last entry.
Only considered when moving past the last entry with
`magit-goto-*-section' commands."
:group 'magit-log
:type 'boolean)
(defcustom magit-log-show-margin t
"Whether to initially show the margin in log buffers.
When non-nil the author name and date are initially displayed in
the margin of log buffers. The margin can be shown or hidden in
the current buffer using the command `magit-toggle-margin'. In
status buffers this option is ignored but it is possible to show
the margin using the mentioned command."
:package-version '(magit . "2.1.0")
:group 'magit-log
:type 'boolean)
(defcustom magit-duration-spec
`((?Y "year" "years" ,(round (* 60 60 24 365.2425)))
(?M "month" "months" ,(round (* 60 60 24 30.436875)))
(?w "week" "weeks" ,(* 60 60 24 7))
(?d "day" "days" ,(* 60 60 24))
(?h "hour" "hours" ,(* 60 60))
(?m "minute" "minutes" 60)
(?s "second" "seconds" 1))
"Units used to display durations in a human format.
The value is a list of time units, beginning with the longest.
Each element has the form (CHAR UNIT UNITS SECONDS). UNIT is the
time unit, UNITS is the plural of that unit. CHAR is a character
abbreviation. And SECONDS is the number of seconds in one UNIT.
Also see option `magit-log-margin-spec'."
:package-version '(magit . "2.1.0")
:group 'magit-log
:type '(repeat (list (character :tag "Unit character")
(string :tag "Unit singular string")
(string :tag "Unit plural string")
(integer :tag "Seconds in unit"))))
(defcustom magit-log-margin-spec '(28 7 magit-duration-spec)
"How to format the log margin.
The log margin is used to display each commit's author followed
by the commit's age. This option controls the total width of the
margin and how time units are formatted, the value has the form:
(WIDTH UNIT-WIDTH DURATION-SPEC)
WIDTH specifies the total width of the log margin. UNIT-WIDTH is
either the integer 1, in which case time units are displayed as a
single characters, leaving more room for author names; or it has
to be the width of the longest time unit string in DURATION-SPEC.
DURATION-SPEC has to be a variable, its value controls which time
units, in what language, are being used."
:package-version '(magit . "2.1.0")
:group 'magit-log
:set-after '(magit-duration-spec)
:type '(list (integer :tag "Margin width")
(choice :tag "Time unit style"
(const :format "%t\n"
:tag "abbreviate to single character" 1)
(integer :format "%t\n"
:tag "show full name" 7))
(variable :tag "Duration spec variable")))
(defcustom magit-log-show-refname-after-summary nil
"Whether to show refnames after commit summaries.
This is useful if you use really long branch names."
:package-version '(magit . "2.2.0")
:group 'magit-log
:type 'boolean)
(defface magit-log-graph
'((((class color) (background light)) :foreground "grey30")
(((class color) (background dark)) :foreground "grey80"))
"Face for the graph part of the log output."
:group 'magit-faces)
(defface magit-log-author
'((((class color) (background light)) :foreground "firebrick")
(((class color) (background dark)) :foreground "tomato"))
"Face for the author part of the log output."
:group 'magit-faces)
(defface magit-log-date
'((((class color) (background light)) :foreground "grey30")
(((class color) (background dark)) :foreground "grey80"))
"Face for the date part of the log output."
:group 'magit-faces)
;;;; File Log
(defcustom magit-log-buffer-file-locked t
"Whether `magit-log-buffer-file' uses a decicated buffer."
:package-version '(magit . "2.7.0")
:group 'magit-commands
:group 'magit-log
:type 'boolean)
;;;; Select Mode
(defcustom magit-log-select-arguments '("-n256" "--decorate")
"The log arguments used in `magit-log-select-mode' buffers."
:package-version '(magit . "2.3.0")
:group 'magit-log
:type '(repeat (string :tag "Argument")))
(defcustom magit-log-select-show-usage 'both
"Whether to show usage information when selecting a commit from a log.
The message can be shown in the `echo-area' or the `header-line', or in
`both' places. If the value isn't one of these symbols, then it should
be nil, in which case no usage information is shown."
:package-version '(magit . "2.1.0")
:group 'magit-log
:type '(choice (const :tag "in echo-area" echo-area)
(const :tag "in header-line" header-line)
(const :tag "in both places" both)
(const :tag "nowhere")))
;;;; Cherry Mode
(defcustom magit-cherry-sections-hook
'(magit-insert-cherry-headers
magit-insert-cherry-commits)
"Hook run to insert sections into the cherry buffer."
:package-version '(magit . "2.1.0")
:group 'magit-log
:type 'hook)
;;;; Reflog Mode
(defcustom magit-reflog-arguments '("-n256")
"The log arguments used in `magit-reflog-mode' buffers."
:package-version '(magit . "2.3.0")
:group 'magit-log
:group 'magit-commands
:type '(repeat (string :tag "Argument")))
(defcustom magit-reflog-show-margin t
"Whether to initially show the margin in reflog buffers.
When non-nil the author name and date are initially displayed in
the margin of reflog buffers. The margin can be shown or hidden
in the current buffer using the command `magit-toggle-margin'."
:package-version '(magit . "2.1.0")
:group 'magit-log
:type 'boolean)
(defface magit-reflog-commit '((t :foreground "green"))
"Face for commit commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-amend '((t :foreground "magenta"))
"Face for amend commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-merge '((t :foreground "green"))
"Face for merge, checkout and branch commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-checkout '((t :foreground "blue"))
"Face for checkout commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-reset '((t :foreground "red"))
"Face for reset commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-rebase '((t :foreground "magenta"))
"Face for rebase commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-cherry-pick '((t :foreground "green"))
"Face for cherry-pick commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-remote '((t :foreground "cyan"))
"Face for pull and clone commands in reflogs."
:group 'magit-faces)
(defface magit-reflog-other '((t :foreground "cyan"))
"Face for other commands in reflogs."
:group 'magit-faces)
;;;; Log Sections
(defcustom magit-log-section-commit-count 10
"How many recent commits to show in certain log sections.
How many recent commits `magit-insert-recent-commits' and
`magit-insert-unpulled-from-upstream-or-recent' (provided
the upstream isn't ahead of the current branch) show."
:package-version '(magit . "2.1.0")
:group 'magit-status
:type 'number)
(defcustom magit-log-section-arguments '("-n256" "--decorate")
"The log arguments used in buffers that show other things besides logs."
:package-version '(magit . "2.4.0")
:group 'magit-log
:group 'magit-status
:type '(repeat (string :tag "Argument")))
;;; Commands
(defvar magit-log-popup
'(:variable magit-log-arguments
:man-page "git-log"
:switches ((?g "Show graph" "--graph")
(?c "Show graph in color" "--color")
(?d "Show refnames" "--decorate")
(?S "Show signatures" "--show-signature")
(?u "Show diffs" "--patch")
(?s "Show diffstats" "--stat")
(?h "Show header" "++header")
(?D "Simplify by decoration" "--simplify-by-decoration")
(?f "Follow renames when showing single-file log" "--follow"))
:options ((?n "Limit number of commits" "-n")
(?f "Limit to files" "-- " magit-read-files)
(?a "Limit to author" "--author=")
(?o "Order commits by" "++order=" magit-log-select-order)
(?g "Search messages" "--grep=")
(?G "Search changes" "-G")
(?S "Search occurences" "-S")
(?L "Trace line evolution" "-L" magit-read-file-trace))
:actions ((?l "Log current" magit-log-current)
(?L "Log local branches" magit-log-branches)
(?r "Reflog current" magit-reflog-current)
(?o "Log other" magit-log)
(?b "Log all branches" magit-log-all-branches)
(?O "Reflog other" magit-reflog)
(?h "Log HEAD" magit-log-head)
(?a "Log all references" magit-log-all)
(?H "Reflog HEAD" magit-reflog-head))
:default-action magit-log-current
:max-action-columns 3))
(defvar magit-log-mode-refresh-popup
'(:variable magit-log-arguments
:man-page "git-log"
:switches ((?g "Show graph" "--graph")
(?c "Show graph in color" "--color")
(?d "Show refnames" "--decorate")
(?S "Show signatures" "--show-signature")
(?u "Show diffs" "--patch")
(?s "Show diffstats" "--stat")
(?D "Simplify by decoration" "--simplify-by-decoration")
(?f "Follow renames when showing single-file log" "--follow"))
:options ((?n "Limit number of commits" "-n")
(?f "Limit to files" "-- " magit-read-files)
(?a "Limit to author" "--author=")
(?o "Order commits by" "++order=" magit-log-select-order)
(?g "Search messages" "--grep=")
(?G "Search changes" "-G")
(?S "Search occurences" "-S")
(?L "Trace line evolution" "-L" magit-read-file-trace))
:actions ((?g "Refresh" magit-log-refresh)
(?t "Toggle margin" magit-toggle-margin)
(?s "Set defaults" magit-log-set-default-arguments) nil
(?w "Save defaults" magit-log-save-default-arguments))
:max-action-columns 2))
(defvar magit-reflog-mode-refresh-popup
'(:variable magit-reflog-arguments
:man-page "git-reflog"
:options ((?n "Limit number of commits" "-n"))))
(defvar magit-log-refresh-popup
'(:variable magit-log-arguments
:man-page "git-log"
:switches ((?g "Show graph" "--graph")
(?c "Show graph in color" "--color")
(?d "Show refnames" "--decorate"))
:options ((?n "Limit number of commits" "-n")
(?o "Order commits by" "++order=" magit-log-select-order))
:actions ((?g "Refresh" magit-log-refresh)
(?t "Toggle margin" magit-toggle-margin)
(?s "Set defaults" magit-log-set-default-arguments) nil
(?w "Save defaults" magit-log-save-default-arguments))
:max-action-columns 2))
(magit-define-popup-keys-deferred 'magit-log-popup)
(magit-define-popup-keys-deferred 'magit-log-mode-refresh-popup)
(magit-define-popup-keys-deferred 'magit-log-refresh-popup)
(defun magit-read-file-trace (&rest _ignored)
(let ((file (magit-read-file-from-rev "HEAD" "File"))
(trace (magit-read-string "Trace")))
(if (string-match
"^\\(/.+/\\|:[^:]+\\|[0-9]+,[-+]?[0-9]+\\)\\(:\\)?$" trace)
(concat trace (or (match-string 2 trace) ":") file)
(user-error "Trace is invalid, see man git-log"))))
(defun magit-log-select-order (&rest _ignored)
(magit-read-char-case "Order commits by " t
(?t "[t]opography" "topo")
(?a "[a]uthor date" "author-date")
(?c "[c]ommitter date" "date")))
(defun magit-log-arguments (&optional refresh)
(cond ((memq magit-current-popup
'(magit-log-popup magit-log-refresh-popup))
(magit-popup-export-file-args magit-current-popup-args))
((derived-mode-p 'magit-log-mode)
(list (nth 1 magit-refresh-args)
(nth 2 magit-refresh-args)))
(refresh
(list magit-log-section-arguments nil))
(t
(-if-let (buffer (magit-mode-get-buffer 'magit-log-mode))
(with-current-buffer buffer
(list (nth 1 magit-refresh-args)
(nth 2 magit-refresh-args)))
(list (default-value 'magit-log-arguments) nil)))))
(defun magit-log-popup (arg)
"Popup console for log commands."
(interactive "P")
(let ((magit-log-refresh-popup
(pcase major-mode
(`magit-log-mode magit-log-mode-refresh-popup)
(_ magit-log-refresh-popup)))
(magit-log-arguments
(-if-let (buffer (magit-mode-get-buffer 'magit-log-mode))
(with-current-buffer buffer
(magit-popup-import-file-args (nth 1 magit-refresh-args)
(nth 2 magit-refresh-args)))
(default-value 'magit-log-arguments))))
(magit-invoke-popup 'magit-log-popup nil arg)))
;;;###autoload
(defun magit-log-buffer-file-popup ()
"Popup console for log commands.
This is a variant of `magit-log-popup' which shows the same popup
but which limits the log to the file being visited in the current
buffer."
(interactive)
(-if-let (file (magit-file-relative-name))
(let ((magit-log-arguments
(magit-popup-import-file-args
(-if-let (buffer (magit-mode-get-buffer 'magit-log-mode))
(with-current-buffer buffer
(nth 2 magit-refresh-args))
(default-value 'magit-log-arguments))
(list file))))
(magit-invoke-popup 'magit-log-popup nil nil))
(user-error "Buffer isn't visiting a file")))
(defun magit-log-refresh-popup (arg)
"Popup console for changing log arguments in the current buffer."
(interactive "P")
(magit-log-refresh-assert)
(let ((magit-log-refresh-popup
(cond ((derived-mode-p 'magit-log-select-mode)
magit-log-refresh-popup)
((derived-mode-p 'magit-log-mode)
(let ((def (copy-sequence magit-log-refresh-popup)))
(plist-put def :switches (plist-get magit-log-popup :switches))
(plist-put def :options (plist-get magit-log-popup :options))
def))
(t
magit-log-refresh-popup)))
(magit-log-arguments
(cond ((derived-mode-p 'magit-log-select-mode)
(cadr magit-refresh-args))
((derived-mode-p 'magit-log-mode)
(magit-popup-import-file-args (nth 1 magit-refresh-args)
(nth 2 magit-refresh-args)))
(t
magit-log-section-arguments))))
(magit-invoke-popup 'magit-log-refresh-popup nil arg)))
(defun magit-log-refresh (args files)
"Set the local log arguments for the current buffer."
(interactive (magit-log-arguments t))
(magit-log-refresh-assert)
(cond ((derived-mode-p 'magit-log-select-mode)
(setcar (cdr magit-refresh-args) args))
((derived-mode-p 'magit-log-mode)
(setcdr magit-refresh-args (list args files)))
(t
(setq-local magit-log-section-arguments args)))
(magit-refresh))
(defun magit-log-set-default-arguments (args files)
"Set the global log arguments for the current buffer."
(interactive (magit-log-arguments t))
(magit-log-refresh-assert)
(cond ((derived-mode-p 'magit-log-select-mode)
(customize-set-variable 'magit-log-select-arguments args)
(setcar (cdr magit-refresh-args) args))
((derived-mode-p 'magit-log-mode)
(customize-set-variable 'magit-log-arguments args)
(setcdr magit-refresh-args (list args files)))
(t
(customize-set-variable 'magit-log-section-arguments args)
(kill-local-variable 'magit-log-section-arguments)))
(magit-refresh))
(defun magit-log-save-default-arguments (args files)
"Set and save the global log arguments for the current buffer."
(interactive (magit-log-arguments t))
(magit-log-refresh-assert)
(cond ((derived-mode-p 'magit-log-select-mode)
(customize-save-variable 'magit-log-select-arguments args)
(setcar (cdr magit-refresh-args) args))
((derived-mode-p 'magit-log-mode)
(customize-save-variable 'magit-log-arguments args)
(setcdr magit-refresh-args (list args files)))
(t
(customize-save-variable 'magit-log-section-arguments args)
(kill-local-variable 'magit-log-section-arguments)))
(magit-refresh))
(defun magit-log-refresh-assert ()
(cond ((derived-mode-p 'magit-reflog-mode)
(user-error "Cannot change log arguments in reflog buffers"))
((derived-mode-p 'magit-cherry-mode)
(user-error "Cannot change log arguments in cherry buffers"))))
(defvar magit-log-read-revs-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map crm-local-completion-map)
(define-key map "\s" 'self-insert-command)
map))
(defun magit-log-read-revs (&optional use-current)
(or (and use-current (--when-let (magit-get-current-branch) (list it)))
(let* ((choose-completion-string-functions
'(crm--choose-completion-string))
(minibuffer-completion-table #'crm--collection-fn)
(minibuffer-completion-confirm t)
(crm-completion-table
`(,@(and (file-exists-p (magit-git-dir "FETCH_HEAD"))
(list "FETCH_HEAD"))
,@(magit-list-branch-names)))
(crm-separator "\\(\\.\\.\\.?\\|[, ]\\)")
(default (or (magit-branch-or-commit-at-point)
(unless use-current
(magit-get-previous-branch))))
(input (read-from-minibuffer
(format "Log rev,s%s: "
(if default (format " (%s)" default) ""))
nil magit-log-read-revs-map
nil 'magit-revision-history default)))
(when (string-equal input "")
(or (setq input default)
(user-error "Nothing selected")))
(split-string input "[, ]" t))))
;;;###autoload
(defun magit-log-current (revs &optional args files)
"Show log for the current branch.
When `HEAD' is detached or with a prefix argument show log for
one or more revs read from the minibuffer."
(interactive (cons (magit-log-read-revs t)
(magit-log-arguments)))
(magit-log revs args files))
;;;###autoload
(defun magit-log (revs &optional args files)
"Show log for one or more revs read from the minibuffer.
The user can input any revision or revisions separated by a
space, or even ranges, but only branches and tags, and a
representation of the commit at point, are available as
completion candidates."
(interactive (cons (magit-log-read-revs)
(magit-log-arguments)))
(require 'magit)
(magit-mode-setup #'magit-log-mode revs args files)
(magit-log-goto-same-commit))
;;;###autoload
(defun magit-log-head (&optional args files)
"Show log for `HEAD'."
(interactive (magit-log-arguments))
(magit-log (list "HEAD") args files))
;;;###autoload
(defun magit-log-branches (&optional args files)
"Show log for all local branches and `HEAD'."
(interactive (magit-log-arguments))
(magit-log (if (magit-get-current-branch)
(list "--branches")
(list "HEAD" "--branches"))
args files))
;;;###autoload
(defun magit-log-all-branches (&optional args files)
"Show log for all local and remote branches and `HEAD'."
(interactive (magit-log-arguments))
(magit-log (if (magit-get-current-branch)
(list "--branches" "--remotes")
(list "HEAD" "--branches" "--remotes"))
args files))
;;;###autoload
(defun magit-log-all (&optional args files)
"Show log for all references and `HEAD'."
(interactive (magit-log-arguments))
(magit-log (if (magit-get-current-branch)
(list "--all")
(list "HEAD" "--all"))
args files))
;;;###autoload
(defun magit-log-buffer-file (&optional follow beg end)
"Show log for the blob or file visited in the current buffer.
With a prefix argument or when `--follow' is part of
`magit-log-arguments', then follow renames."
(interactive (if (region-active-p)
(list current-prefix-arg
(1- (line-number-at-pos (region-beginning)))
(1- (line-number-at-pos (region-end))))
(list current-prefix-arg)))
(require 'magit)
(-if-let (file (magit-file-relative-name))
(magit-mode-setup-internal
#'magit-log-mode
(list (list (or magit-buffer-refname
(magit-get-current-branch)
"HEAD"))
(let ((args (car (magit-log-arguments))))
(when (and follow (not (member "--follow" args)))
(push "--follow" args))
(when (and beg end)
(setq args (cons (format "-L%s,%s:%s" beg end file)
(cl-delete "-L" args :test
'string-prefix-p)))
(setq file nil))
args)
(and file (list file)))
magit-log-buffer-file-locked)
(user-error "Buffer isn't visiting a file"))
(magit-log-goto-same-commit))
;;;###autoload
(defun magit-reflog-current ()
"Display the reflog of the current branch."
(interactive)
(magit-reflog (magit-get-current-branch)))
;;;###autoload
(defun magit-reflog (ref)
"Display the reflog of a branch."
(interactive (list (magit-read-local-branch-or-ref "Show reflog for")))
(require 'magit)
(magit-mode-setup #'magit-reflog-mode ref magit-reflog-arguments))
;;;###autoload
(defun magit-reflog-head ()
"Display the `HEAD' reflog."
(interactive)
(magit-reflog "HEAD"))
(defun magit-log-toggle-commit-limit ()
"Toggle the number of commits the current log buffer is limited to.
If the number of commits is currently limited, then remove that
limit. Otherwise set it to 256."
(interactive)
(magit-log-set-commit-limit (lambda (&rest _) nil)))
(defun magit-log-double-commit-limit ()
"Double the number of commits the current log buffer is limited to."
(interactive)
(magit-log-set-commit-limit '*))
(defun magit-log-half-commit-limit ()
"Half the number of commits the current log buffer is limited to."
(interactive)
(magit-log-set-commit-limit '/))
(defun magit-log-set-commit-limit (fn)
(let* ((val (car (magit-log-arguments t)))
(arg (--first (string-match "^-n\\([0-9]+\\)?$" it) val))
(num (and arg (string-to-number (match-string 1 arg))))
(num (if num (funcall fn num 2) 256)))
(setq val (delete arg val))
(setcar (cdr magit-refresh-args)
(if (and num (> num 0))
(cons (format "-n%i" num) val)
val)))
(magit-refresh))
(defun magit-log-get-commit-limit ()
(--when-let (--first (string-match "^-n\\([0-9]+\\)?$" it)
(car (magit-log-arguments t)))
(string-to-number (match-string 1 it))))
(defun magit-log-bury-buffer (&optional arg)
"Bury the current buffer or the revision buffer in the same frame.
Like `magit-mode-bury-buffer' (which see) but with a negative
prefix argument instead bury the revision buffer, provided it
is displayed in the current frame."
(interactive "p")
(if (< arg 0)
(let* ((buf (magit-mode-get-buffer 'magit-revision-mode))
(win (and buf (get-buffer-window buf (selected-frame)))))
(if win
(with-selected-window win
(with-current-buffer buf
(magit-mode-bury-buffer (> (abs arg) 1))))
(user-error "No revision buffer in this frame")))
(magit-mode-bury-buffer (> arg 1))))
;;;###autoload
(defun magit-log-move-to-parent (&optional n)
"Move to the Nth parent of the current commit."
(interactive "p")
(when (derived-mode-p 'magit-log-mode)
(magit-section-when commit
(let ((parent-rev (format "%s^%s" (magit-section-value it) (or n 1))))
(-if-let (parent-hash (magit-rev-parse "--short" parent-rev))
(-if-let (section (--first (equal (magit-section-value it)
parent-hash)
(magit-section-siblings it 'next)))
(magit-section-goto section)
(user-error
(substitute-command-keys
(concat "Parent " parent-hash " not found. Try typing "
"\\[magit-log-double-commit-limit] first"))))
(user-error "Parent %s does not exist" parent-rev))))))
;;; Log Mode
(defvar magit-log-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map magit-mode-map)
(define-key map "\C-c\C-b" 'magit-go-backward)
(define-key map "\C-c\C-f" 'magit-go-forward)
(define-key map "\C-c\C-n" 'magit-log-move-to-parent)
(define-key map "=" 'magit-log-toggle-commit-limit)
(define-key map "+" 'magit-log-double-commit-limit)
(define-key map "-" 'magit-log-half-commit-limit)
(define-key map "q" 'magit-log-bury-buffer)
map)
"Keymap for `magit-log-mode'.")
(define-derived-mode magit-log-mode magit-mode "Magit Log"
"Mode for looking at Git log.
This mode is documented in info node `(magit)Log Buffer'.
\\<magit-mode-map>\
Type \\[magit-refresh] to refresh the current buffer.
Type \\[magit-visit-thing] or \\[magit-diff-show-or-scroll-up] \
to visit the commit at point.
Type \\[magit-branch-popup] to see available branch commands.
Type \\[magit-merge-popup] to merge the branch or commit at point.
Type \\[magit-cherry-pick-popup] to apply the commit at point.
Type \\[magit-reset] to reset HEAD to the commit at point.
\\{magit-log-mode-map}"
:group 'magit-log
(hack-dir-local-variables-non-file-buffer))
(defvar magit-log-disable-graph-hack-args
'("-G" "--grep" "--author")
"Arguments which disable the graph speedup hack.")
(defun magit-log-refresh-buffer (revs args files)
(setq header-line-format
(propertize
(concat " Commits in " (mapconcat 'identity revs " ")
(and files (concat " touching "
(mapconcat 'identity files " "))))
'face 'magit-header-line))
(unless (= (length files) 1)
(setq args (remove "--follow" args)))
(when (--any-p (string-match-p
(concat "^" (regexp-opt magit-log-remove-graph-args)) it)
args)
(setq args (remove "--graph" args)))
(unless (member "--graph" args)
(setq args (remove "--color" args)))
(-when-let* ((limit (magit-log-get-commit-limit))
(limit (* 2 limit)) ; increase odds for complete graph
(count (and (= (length revs) 1)
(> limit 1024) ; otherwise it's fast enough
(setq revs (car revs))
(not (string-match-p "\\.\\." revs))
(not (member revs '("--all" "--branches")))
(-none-p (lambda (arg)
(--any-p (string-prefix-p it arg)
magit-log-disable-graph-hack-args))
args)
(magit-git-string "rev-list" "--count"
"--first-parent" args revs))))
(setq revs (if (< (string-to-number count) limit)
revs
(format "%s~%s..%s" revs limit revs))))
(magit-insert-section (logbuf)
(magit-insert-log revs args files)))
(defun magit-insert-log (revs &optional args files)
"Insert a log section.
Do not add this to a hook variable."
(let ((magit-git-global-arguments
(remove "--literal-pathspecs" magit-git-global-arguments)))
(magit-git-wash (apply-partially #'magit-log-wash-log 'log)
"log"
(format "--format=%%h%s %s[%%aN][%%at]%%s%s"
(if (member "--decorate" args) "%d" "")
(if (member "--show-signature" args)
(progn (setq args (remove "--show-signature" args)) "%G?")
"")
(if (member "++header" args)
(if (member "--graph" (setq args (remove "++header" args)))
(concat "\n" magit-log-revision-headers-format "\n")
(concat "\n" magit-log-revision-headers-format "\n"))
""))
(progn
(--when-let (--first (string-match "^\\+\\+order=\\(.+\\)$" it) args)
(setq args (cons (format "--%s-order" (match-string 1 it))
(remove it args))))
(if (member "--decorate" args)
(cons "--decorate=full" (remove "--decorate" args))
args))
"--use-mailmap" "--no-prefix" revs "--" files)))
(defvar magit-commit-section-map
(let ((map (make-sparse-keymap)))
(define-key map [remap magit-visit-thing] 'magit-show-commit)
(define-key map "a" 'magit-cherry-apply)
map)
"Keymap for `commit' sections.")
(defvar magit-module-commit-section-map
(let ((map (make-sparse-keymap)))
(define-key map [remap magit-visit-thing] 'magit-show-commit)
map)
"Keymap for `module-commit' sections.")
(defconst magit-log-heading-re
(concat "^"
"\\(?4:[-_/|\\*o. ]*\\)" ; graph
"\\(?1:[0-9a-fA-F]+\\) " ; sha1
"\\(?:\\(?3:([^()]+)\\) \\)?" ; refs
"\\(?7:[BGUN]\\)?" ; gpg
"\\[\\(?5:[^]]*\\)\\]" ; author
"\\[\\(?6:[^]]*\\)\\]" ; date
"\\(?2:.*\\)$")) ; msg
(defconst magit-log-cherry-re
(concat "^"
"\\(?8:[-+]\\) " ; cherry
"\\(?1:[0-9a-fA-F]+\\) " ; sha1
"\\(?2:.*\\)$")) ; msg
(defconst magit-log-module-re
(concat "^"
"\\(?:\\(?11:[<>]\\) \\)?" ; side
"\\(?1:[0-9a-fA-F]+\\) " ; sha1
"\\(?2:.*\\)$")) ; msg
(defconst magit-log-bisect-vis-re
(concat "^"
"\\(?4:[-_/|\\*o. ]*\\)" ; graph
"\\(?1:[0-9a-fA-F]+\\) " ; sha1
"\\(?:\\(?3:([^()]+)\\) \\)?" ; refs
"\\(?2:.*\\)$")) ; msg
(defconst magit-log-bisect-log-re
(concat "^# "
"\\(?3:bad:\\|skip:\\|good:\\) " ; "refs"
"\\[\\(?1:[^]]+\\)\\] " ; sha1
"\\(?2:.*\\)$")) ; msg
(defconst magit-log-reflog-re
(concat "^"
"\\(?1:[^ ]+\\) " ; sha1
"\\(?:\\(?:[^@]+@{\\(?6:[^}]+\\)} " ; date
"\\(?10:merge \\|autosave \\|restart \\|[^:]+: \\)?" ; refsub
"\\(?2:.*\\)?\\)\\| \\)$")) ; msg
(defconst magit-reflog-subject-re
(concat "\\(?1:[^ ]+\\) ?" ; command
"\\(?2:\\(?: ?-[^ ]+\\)+\\)?" ; option
"\\(?: ?(\\(?3:[^)]+\\))\\)?")) ; type
(defconst magit-log-stash-re
(concat "^"
"\\(?1:[^ ]+\\)" ; "sha1"
"\\(?5: \\)" ; "author"
"\\(?6:[^ ]+\\) " ; date
"\\(?2:.*\\)$")) ; msg
(defvar magit-log-count nil)
(defun magit-log-wash-log (style args)
(setq args (-flatten args))
(when (and (member "--graph" args)
(member "--color" args))
(let ((ansi-color-apply-face-function
(lambda (beg end face)
(put-text-property beg end 'font-lock-face
(or face 'magit-log-graph)))))
(ansi-color-apply-on-region (point-min) (point-max))))
(when (eq style 'cherry)
(reverse-region (point-min) (point-max)))
(let ((magit-log-count 0)
(abbrev (magit-abbrev-length)))
(magit-wash-sequence (apply-partially 'magit-log-wash-rev style abbrev))
(if (derived-mode-p 'magit-log-mode)
(when (eq magit-log-count (magit-log-get-commit-limit))
(magit-insert-section (longer)
(insert-text-button
(substitute-command-keys
(format "Type \\<%s>\\[%s] to show more history"
'magit-log-mode-map
'magit-log-double-commit-limit))
'action (lambda (_button)
(magit-log-double-commit-limit))
'follow-link t
'mouse-face 'magit-section-highlight)))
(unless (equal (car args) "cherry")
(insert ?\n)))))
(defun magit-log-wash-rev (style abbrev)
(when (derived-mode-p 'magit-log-mode)
(cl-incf magit-log-count))
(looking-at (pcase style
(`log magit-log-heading-re)
(`cherry magit-log-cherry-re)
(`module magit-log-module-re)
(`reflog magit-log-reflog-re)
(`stash magit-log-stash-re)
(`bisect-vis magit-log-bisect-vis-re)
(`bisect-log magit-log-bisect-log-re)))
(magit-bind-match-strings
(hash msg refs graph author date gpg cherry _ refsub side) nil
(let ((align (not (member "--stat" (cadr magit-refresh-args))))
(non-graph-re (if (eq style 'bisect-vis)
magit-log-bisect-vis-re
magit-log-heading-re)))
(magit-delete-line)
(magit-insert-section section (commit hash)
(pcase style
(`stash (setf (magit-section-type section) 'stash))
(`module (setf (magit-section-type section) 'module-commit))
(`bisect-log (setq hash (magit-rev-parse "--short" hash))))
(when cherry
(when (and (derived-mode-p 'magit-refs-mode)
magit-refs-show-commit-count)
(insert (make-string magit-refs-indent-cherry-lines ?\s)))
(insert (propertize cherry 'face (if (string= cherry "-")
'magit-cherry-equivalent
'magit-cherry-unmatched)))
(insert ?\s))
(when side
(insert (propertize side 'face (if (string= side "<")
'magit-cherry-equivalent
'magit-cherry-unmatched)))
(insert ?\s))
(when align
(insert (propertize hash 'face 'magit-hash) ?\s))
(when graph
(insert graph))
(unless align
(insert (propertize hash 'face 'magit-hash) ?\s))
(when (and refs (not magit-log-show-refname-after-summary))
(insert (magit-format-ref-labels refs) ?\s))
(when refsub
(insert (format "%-2s " (1- magit-log-count)))
(insert (magit-reflog-format-subject
(substring refsub 0 (if (string-match-p ":" refsub) -2 -1)))))
(when msg
(insert (propertize msg 'face
(pcase (and gpg (aref gpg 0))
(?G 'magit-signature-good)
(?B 'magit-signature-bad)
(?U 'magit-signature-untrusted)))))
(when (and refs magit-log-show-refname-after-summary)
(insert ?\s)
(insert (magit-format-ref-labels refs)))
(insert ?\n)
(when (memq style '(log reflog stash))
(goto-char (line-beginning-position))
(when (and refsub
(string-match "\\`\\([^ ]\\) \\+\\(..\\)\\(..\\)" date))
(setq date (+ (string-to-number (match-string 1 date))
(* (string-to-number (match-string 2 date)) 60 60)
(* (string-to-number (match-string 3 date)) 60))))
(save-excursion
(backward-char)
(magit-format-log-margin author date)))
(when (and graph
(not (eobp))
(not (looking-at non-graph-re)))
(when (looking-at "")
(magit-insert-heading)
(delete-char 1)
(magit-insert-section (commit-header)
(forward-line)
(magit-insert-heading)
(re-search-forward "")
(backward-delete-char 1)
(forward-char)
(insert ?\n))
(delete-char 1))
(if (looking-at "^\\(---\\|\n\s\\|\ndiff\\)")
(let ((limit (save-excursion
(and (re-search-forward non-graph-re nil t)
(match-beginning 0)))))
(unless (magit-section-content magit-insert-section--current)
(magit-insert-heading))
(delete-char (if (looking-at "\n") 1 4))
(magit-diff-wash-diffs (list "--stat") limit))
(when align
(setq align (make-string (1+ abbrev) ? )))
(when (and (not (eobp)) (not (looking-at non-graph-re)))
(when align
(setq align (make-string (1+ abbrev) ? )))
(while (and (not (eobp)) (not (looking-at non-graph-re)))
(when align
(save-excursion (insert align)))
(magit-format-log-margin)
(forward-line))
;; When `--format' is used and its value isn't one of the
;; predefined formats, then `git-log' does not insert a
;; separator line.
(save-excursion
(forward-line -1)
(looking-at "[-_/|\\*o. ]*"))
(setq graph (match-string 0))
(unless (string-match-p "[/\\]" graph)
(insert graph ?\n))))))))
t)
(defun magit-format-log-margin (&optional author date)
(-let [(width unit-width duration-spec) magit-log-margin-spec]
(when (and date (not author))
(setq width (+ (if (= unit-width 1) 1 (1+ unit-width))
(if (derived-mode-p 'magit-log-mode) 1 0))))
(if date
(magit-make-margin-overlay
(and author
(concat (propertize (truncate-string-to-width
(or author "")
(- width 1 3 ; gap, digits
(if (= unit-width 1) 1 (1+ unit-width))
(if (derived-mode-p 'magit-log-mode) 1 0))
nil ?\s (make-string 1 magit-ellipsis))
'face 'magit-log-author)
" "))
(propertize (magit-format-duration
(abs (- (float-time)
(string-to-number date)))
(symbol-value duration-spec)
unit-width)
'face 'magit-log-date)
(and (derived-mode-p 'magit-log-mode)
(propertize " " 'face 'fringe)))
(magit-make-margin-overlay
(propertize (make-string (1- width) ?\s) 'face 'default)
(propertize " " 'face 'fringe)))))
(defun magit-format-duration (duration spec &optional width)
(-let [(char unit units weight) (car spec)]
(let ((cnt (round (/ duration weight 1.0))))
(if (or (not (cdr spec))
(>= (/ duration weight) 1))
(if (eq width 1)
(format "%3i%c" cnt char)
(format (if width (format "%%3i %%-%is" width) "%i %s")
cnt (if (= cnt 1) unit units)))
(magit-format-duration duration (cdr spec) width)))))
(defun magit-log-maybe-show-more-commits (section)
"Automatically insert more commit sections in a log.
Only do so if `point' is on the \"show more\" section,
and `magit-log-auto-more' is non-nil."
(when (and (eq (magit-section-type section) 'longer)
magit-log-auto-more)
(magit-log-double-commit-limit)
(forward-line -1)
(magit-section-forward)))
(defvar magit--update-revision-buffer nil)
(defun magit-log-maybe-update-revision-buffer (&optional _)
"When moving in the log buffer, update the revision buffer.
If there is no revision buffer in the same frame, then do nothing."
(when (derived-mode-p 'magit-log-mode)
(magit-log-maybe-update-revision-buffer-1)))
(defun magit-log-maybe-update-revision-buffer-1 ()
(unless magit--update-revision-buffer
(-when-let* ((commit (magit-section-when 'commit))
(buffer (magit-mode-get-buffer 'magit-revision-mode nil t)))
(setq magit--update-revision-buffer (list commit buffer))
(run-with-idle-timer
magit-update-other-window-delay nil
(lambda ()
(-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)))))
(setq magit--update-revision-buffer nil))))))
(defvar magit--update-blob-buffer nil)
(defun magit-log-maybe-update-blob-buffer (&optional _)
"When moving in the log buffer, update the blob buffer.
If there is no blob buffer in the same frame, then do nothing."
(when (derived-mode-p 'magit-log-mode)
(magit-log-maybe-update-blob-buffer-1)))
(defun magit-log-maybe-update-blob-buffer-1 ()
(unless magit--update-revision-buffer
(-when-let* ((commit (magit-section-when 'commit))
(buffer (--first (with-current-buffer it magit-buffer-revision)
(-map #'window-buffer (window-list)))))
(setq magit--update-blob-buffer (list commit buffer))
(run-with-idle-timer
magit-update-other-window-delay nil
(lambda ()
(-let [(rev buf) magit--update-blob-buffer]
(setq magit--update-blob-buffer nil)
(when (buffer-live-p buf)
(save-excursion
(with-selected-window (get-buffer-window buf)
(with-current-buffer buf
(magit-blob-visit (list (magit-rev-parse rev)
(magit-file-relative-name
magit-buffer-file-name))
(line-number-at-pos))))))))))))
(defun magit-log-goto-same-commit ()
(-when-let* ((prev magit-previous-section)
(rev (cond ((magit-section-match 'commit prev)
(magit-section-value prev))
((magit-section-match 'branch prev)
(magit-rev-format
"%h" (magit-section-value prev)))))
(same (--first (equal (magit-section-value it) rev)
(magit-section-children magit-root-section))))
(goto-char (magit-section-start same))))
;;; Select Mode
(defvar magit-log-select-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map magit-log-mode-map)
(define-key map "\C-c\C-b" 'undefined)
(define-key map "\C-c\C-f" 'undefined)
(define-key map "." 'magit-log-select-pick)
(define-key map "e" 'magit-log-select-pick)
(define-key map "\C-c\C-c" 'magit-log-select-pick)
(define-key map "q" 'magit-log-select-quit)
(define-key map "\C-c\C-k" 'magit-log-select-quit)
map)
"Keymap for `magit-log-select-mode'.")
(put 'magit-log-select-pick :advertised-binding [?\C-c ?\C-c])
(put 'magit-log-select-quit :advertised-binding [?\C-c ?\C-k])
(define-derived-mode magit-log-select-mode magit-log-mode "Magit Select"
"Mode for selecting a commit from history.
This mode is documented in info node `(magit)Select from log'.
\\<magit-mode-map>\
Type \\[magit-refresh] to refresh the current buffer.
Type \\[magit-visit-thing] or \\[magit-diff-show-or-scroll-up] \
to visit the commit at point.
\\<magit-log-select-mode-map>\
Type \\[magit-log-select-pick] to select the commit at point.
Type \\[magit-log-select-quit] to abort without selecting a commit."
:group 'magit-log
(hack-dir-local-variables-non-file-buffer))
(defun magit-log-select-refresh-buffer (rev args)
(magit-insert-section (logbuf)
(magit-insert-log rev args)))
(defvar-local magit-log-select-pick-function nil)
(defvar-local magit-log-select-quit-function nil)
(defun magit-log-select (pick &optional msg quit branch)
(declare (indent defun))
(magit-mode-setup #'magit-log-select-mode
(or branch (magit-get-current-branch) "HEAD")
magit-log-select-arguments)
(magit-log-goto-same-commit)
(setq magit-log-select-pick-function pick)
(setq magit-log-select-quit-function quit)
(when magit-log-select-show-usage
(setq msg (substitute-command-keys
(format-spec
(if msg
(if (string-suffix-p "," msg)
(concat msg " or %q to abort")
msg)
"Type %p to select commit at point, or %q to abort")
'((?p . "\\[magit-log-select-pick]")
(?q . "\\[magit-log-select-quit]")))))
(when (memq magit-log-select-show-usage '(both header-line))
(setq header-line-format (propertize (concat " " msg) 'face 'bold)))
(when (memq magit-log-select-show-usage '(both echo-area))
(message "%s" msg))))
(defun magit-log-select-pick ()
"Select the commit at point and act on it.
Call `magit-log-select-pick-function' with the selected
commit as argument."
(interactive)
(let ((fun magit-log-select-pick-function)
(rev (magit-commit-at-point)))
(quit-restore-window nil 'kill)
(funcall fun rev)))
(defun magit-log-select-quit ()
"Abort selecting a commit, don't act on any commit."
(interactive)
(quit-restore-window nil 'kill)
(when magit-log-select-quit-function
(funcall magit-log-select-quit-function)))
;;; Cherry Mode
(defvar magit-cherry-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map magit-mode-map)
(define-key map "q" 'magit-log-bury-buffer)
(define-key map "L" 'magit-toggle-margin)
map)
"Keymap for `magit-cherry-mode'.")
(define-derived-mode magit-cherry-mode magit-mode "Magit Cherry"
"Mode for looking at commits not merged upstream.
\\<magit-mode-map>\
Type \\[magit-refresh] to refresh the current buffer.
Type \\[magit-visit-thing] or \\[magit-diff-show-or-scroll-up] \
to visit the commit at point.
Type \\[magit-cherry-pick-popup] to apply the commit at point.
\\{magit-cherry-mode-map}"
:group 'magit-log
(hack-dir-local-variables-non-file-buffer))
;;;###autoload
(defun magit-cherry (head upstream)
"Show commits in a branch that are not merged in the upstream branch."
(interactive
(let ((head (magit-read-branch "Cherry head")))
(list head (magit-read-other-branch "Cherry upstream" head
(magit-get-upstream-branch head)))))
(require 'magit)
(magit-mode-setup #'magit-cherry-mode upstream head))
(defun magit-cherry-refresh-buffer (_upstream _head)
(magit-insert-section (cherry)
(run-hooks 'magit-cherry-sections-hook)))
(defun magit-insert-cherry-headers ()
"Insert headers appropriate for `magit-cherry-mode' buffers."
(magit-insert-head-branch-header (nth 1 magit-refresh-args))
(magit-insert-upstream-branch-header (nth 1 magit-refresh-args)
(nth 0 magit-refresh-args)
"Upstream: ")
(insert ?\n))
(defun magit-insert-cherry-commits ()
"Insert commit sections into a `magit-cherry-mode' buffer."
(magit-insert-section (cherries)
(magit-insert-heading "Cherry commits:")
(magit-git-wash (apply-partially 'magit-log-wash-log 'cherry)
"cherry" "-v" "--abbrev" magit-refresh-args)))
;;; Reflog Mode
(defvar magit-reflog-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map magit-log-mode-map)
(define-key map "L" 'magit-toggle-margin)
map)
"Keymap for `magit-reflog-mode'.")
(define-derived-mode magit-reflog-mode magit-log-mode "Magit Reflog"
"Mode for looking at Git reflog.
This mode is documented in info node `(magit)Reflog'.
\\<magit-mode-map>\
Type \\[magit-refresh] to refresh the current buffer.
Type \\[magit-visit-thing] or \\[magit-diff-show-or-scroll-up] \
to visit the commit at point.
Type \\[magit-cherry-pick-popup] to apply the commit at point.
Type \\[magit-reset] to reset HEAD to the commit at point.
\\{magit-reflog-mode-map}"
:group 'magit-log
(hack-dir-local-variables-non-file-buffer))
(defun magit-reflog-refresh-buffer (ref args)
(setq header-line-format
(propertize (concat " Reflog for " ref) 'face 'magit-header-line))
(magit-insert-section (reflogbuf)
(magit-git-wash (apply-partially 'magit-log-wash-log 'reflog)
"reflog" "show" "--format=%h %gd %gs" "--date=raw" args ref)))
(defvar magit-reflog-labels
'(("commit" . magit-reflog-commit)
("amend" . magit-reflog-amend)
("merge" . magit-reflog-merge)
("checkout" . magit-reflog-checkout)
("branch" . magit-reflog-checkout)
("reset" . magit-reflog-reset)
("rebase" . magit-reflog-rebase)
("cherry-pick" . magit-reflog-cherry-pick)
("initial" . magit-reflog-commit)
("pull" . magit-reflog-remote)
("clone" . magit-reflog-remote)
("autosave" . magit-reflog-commit)
("restart" . magit-reflog-reset)))
(defun magit-reflog-format-subject (subject)
(let* ((match (string-match magit-reflog-subject-re subject))
(command (and match (match-string 1 subject)))
(option (and match (match-string 2 subject)))
(type (and match (match-string 3 subject)))
(label (if (string= command "commit")
(or type command)
command))
(text (if (string= command "commit")
label
(mapconcat #'identity
(delq nil (list command option type))
" "))))
(format "%-16s "
(propertize text 'face
(or (cdr (assoc label magit-reflog-labels))
'magit-reflog-other)))))
;;; Log Sections
;;;; Standard Log Sections
(defvar magit-unpulled-section-map
(let ((map (make-sparse-keymap)))
(define-key map [remap magit-visit-thing] 'magit-diff-dwim)
map)
"Keymap for `unpulled' sections.")
(magit-define-section-jumper magit-jump-to-unpulled-from-upstream
"Unpulled from @{upstream}" unpulled "..@{upstream}")
(defun magit-insert-unpulled-from-upstream ()
"Insert commits that haven't been pulled from the upstream yet."
(when (magit-git-success "rev-parse" "@{upstream}")
(magit-insert-section (unpulled "..@{upstream}")
(magit-insert-heading
(format (propertize "Unpulled from %s:" 'face 'magit-section-heading)
(magit-get-upstream-branch)))
(magit-insert-log "..@{upstream}" magit-log-section-arguments))))
(magit-define-section-jumper magit-jump-to-unpulled-from-pushremote
"Unpulled from <push-remote>" unpulled
(concat ".." (magit-get-push-branch)))
(defun magit-insert-unpulled-from-pushremote ()
"Insert commits that haven't been pulled from the push-remote yet."
(--when-let (magit-get-push-branch)
(unless (and (equal (magit-rev-name it)
(magit-rev-name "@{upstream}"))
(or (memq 'magit-insert-unpulled-from-upstream
magit-status-sections-hook)
(memq 'magit-insert-unpulled-from-upstream-or-recent
magit-status-sections-hook)))
(magit-insert-section (unpulled (concat ".." it))
(magit-insert-heading
(format (propertize "Unpulled from %s:" 'face 'magit-section-heading)
(propertize it 'face 'magit-branch-remote)))
(magit-insert-log (concat ".." it) magit-log-section-arguments)))))
(defvar magit-unpushed-section-map
(let ((map (make-sparse-keymap)))
(define-key map [remap magit-visit-thing] 'magit-diff-dwim)
map)
"Keymap for `unpushed' sections.")
(magit-define-section-jumper magit-jump-to-unpushed-to-upstream
"Unpushed to @{upstream}" unpushed "@{upstream}..")
(defun magit-insert-unpushed-to-upstream ()
"Insert commits that haven't been pushed to the upstream yet."
(when (magit-git-success "rev-parse" "@{upstream}")
(magit-insert-section (unpushed "@{upstream}..")
(magit-insert-heading
(format (propertize "Unmerged into %s:" 'face 'magit-section-heading)
(magit-get-upstream-branch)))
(magit-insert-log "@{upstream}.." magit-log-section-arguments))))
(magit-define-section-jumper magit-jump-to-unpushed-to-pushremote
"Unpushed to <push-remote>" unpushed
(concat (magit-get-push-branch) ".."))
(defun magit-insert-unpushed-to-pushremote ()
"Insert commits that haven't been pushed to the push-remote yet."
(--when-let (magit-get-push-branch)
(unless (and (equal (magit-rev-name it)
(magit-rev-name "@{upstream}"))
(memq 'magit-insert-unpushed-to-upstream
magit-status-sections-hook))
(magit-insert-section (unpushed (concat it ".."))
(magit-insert-heading
(format (propertize "Unpushed to %s:" 'face 'magit-section-heading)
(propertize it 'face 'magit-branch-remote)))
(magit-insert-log (concat it "..") magit-log-section-arguments)))))
(defun magit-insert-recent-commits (&optional collapse)
"Insert section showing recent commits.
Show the last `magit-log-section-commit-count' commits."
(let* ((start (format "HEAD~%s" magit-log-section-commit-count))
(range (and (magit-rev-verify start)
(concat start "..HEAD"))))
(magit-insert-section (recent range collapse)
(magit-insert-heading "Recent commits:")
(magit-insert-log range
(cons (format "-%d" magit-log-section-commit-count)
magit-log-section-arguments)))))
(defun magit-insert-unpulled-from-upstream-or-recent ()
"Insert section showing unpulled or recent commits.
If an upstream is configured for the current branch and it is
ahead of the current branch, then show the commits that have
not yet been pulled into the current branch. If no upstream is
configured or if the upstream is not ahead of the current branch,
then show the last `magit-log-section-commit-count' commits."
(let ((upstream (magit-rev-parse "@{upstream}")))
(if (or (not upstream)
(equal upstream (magit-rev-parse "HEAD")))
(magit-insert-recent-commits t)
(magit-insert-unpulled-from-upstream))))
;;;; Auxiliary Log Sections
(defun magit-insert-unpulled-cherries ()
"Insert section showing unpulled commits.
Like `magit-insert-unpulled-to-upstream' but prefix each commit
which has not been applied yet (i.e. a commit with a patch-id
not shared with any local commit) with \"+\", and all others with
\"-\"."
(when (magit-git-success "rev-parse" "@{upstream}")
(magit-insert-section (unpulled "..@{upstream}")
(magit-insert-heading "Unpulled commits:")
(magit-git-wash (apply-partially 'magit-log-wash-log 'cherry)
"cherry" "-v" (magit-abbrev-arg)
(magit-get-current-branch) "@{upstream}"))))
(defun magit-insert-unpushed-cherries ()
"Insert section showing unpushed commits.
Like `magit-insert-unpushed-to-upstream' but prefix each commit
which has not been applied to upstream yet (i.e. a commit with
a patch-id not shared with any upstream commit) with \"+\", and
all others with \"-\"."
(when (magit-git-success "rev-parse" "@{upstream}")
(magit-insert-section (unpushed "@{upstream}..")
(magit-insert-heading "Unpushed commits:")
(magit-git-wash (apply-partially 'magit-log-wash-log 'cherry)
"cherry" "-v" (magit-abbrev-arg) "@{upstream}"))))
;;; Buffer Margins
(defvar-local magit-set-buffer-margin-refresh nil)
(defvar-local magit-show-margin nil)
(put 'magit-show-margin 'permanent-local t)
(defun magit-toggle-margin ()
"Show or hide the Magit margin."
(interactive)
(unless (derived-mode-p 'magit-log-mode 'magit-status-mode 'magit-refs-mode)
(user-error "Buffer doesn't contain any logs"))
(magit-set-buffer-margin (not (cdr (window-margins)))))
(defun magit-maybe-show-margin ()
"Maybe show the margin, depending on the major-mode and an option.
Supported modes are `magit-log-mode' and `magit-reflog-mode',
and the respective options are `magit-log-show-margin' and
`magit-reflog-show-margin'."
(if (local-variable-p 'magit-show-margin)
(magit-set-buffer-margin magit-show-margin)
(pcase major-mode
(`magit-log-mode (magit-set-buffer-margin magit-log-show-margin))
(`magit-reflog-mode (magit-set-buffer-margin magit-reflog-show-margin)))))
(defun magit-set-buffer-margin (enable)
(let ((width (cond ((not enable) nil)
((derived-mode-p 'magit-reflog-mode)
(+ (cadr magit-log-margin-spec) 5))
(t (car magit-log-margin-spec)))))
(setq magit-show-margin width)
(when (and enable magit-set-buffer-margin-refresh)
(magit-refresh))
(-when-let (window (get-buffer-window))
(with-selected-window window
(set-window-margins nil (car (window-margins)) width)
(if enable
(add-hook 'window-configuration-change-hook
'magit-set-buffer-margin-1 nil t)
(remove-hook 'window-configuration-change-hook
'magit-set-buffer-margin-1 t))))))
(defun magit-set-buffer-margin-1 ()
(-when-let (window (get-buffer-window))
(with-selected-window window
(set-window-margins nil (car (window-margins)) magit-show-margin))))
(defun magit-make-margin-overlay (&rest strings)
;; Don't put the overlay on the complete line to work around #1880.
(let ((o (make-overlay (1+ (line-beginning-position))
(line-end-position)
nil t)))
(overlay-put o 'evaporate t)
(overlay-put o 'before-string
(propertize "o" 'display
(list '(margin right-margin)
(apply #'concat strings))))))
;;; magit-log.el ends soon
(define-obsolete-variable-alias 'magit-log-section-args
'magit-log-section-arguments "Magit 2.2.0")
(define-obsolete-function-alias 'magit-insert-unpulled-or-recent-commits
'magit-insert-unpulled-from-upstream-or-recent "Magit 2.4.0")
(provide 'magit-log)
;; Local Variables:
;; coding: utf-8
;; indent-tabs-mode: nil
;; End:
;;; magit-log.el ends here