Files
emacs/lisp/treemacs-magit.el
Daniel Weschke 82f05baffe pkg update and first config fix
org-brain not working, add org-roam
2022-12-19 23:02:34 +01:00

211 lines
9.9 KiB
EmacsLisp

;;; treemacs-magit.el --- Magit integration for treemacs -*- lexical-binding: t -*-
;; Copyright (C) 2022 Alexander Miller
;; Author: Alexander Miller <alexanderm@web.de>
;; Package-Requires: ((emacs "26.1") (treemacs "0.0") (pfuture "1.3" ) (magit "2.90.0"))
;; Package-Version: 20220917.1026
;; Package-Commit: 13a8a060b784021f3d6bd7c27f2a0bcf6ea0d087
;; Version: 0
;; Homepage: https://github.com/Alexander-Miller/treemacs
;; This program 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 of the License, or
;; (at your option) any later version.
;; This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
;;; Commentary:
;;; Closing the gaps for filewatch- and git-modes in conjunction with magit.
;;; Specifically this package will hook into magit so as to artificially
;;; produce filewatch events for changes that treemacs would otherwise
;;; not catch, nameley the committing and (un)staging of files.
;;; Code:
(require 'treemacs)
(require 'magit)
(require 'pfuture)
(require 'seq)
;; no need for dash for a single when-let
(eval-when-compile
(when (version< emacs-version "26")
(defalias 'if-let* #'if-let)
(defalias 'when-let* #'when-let)))
;;;; Filewatch
(defvar treemacs-magit--timers nil
"Cached list of roots an update is scheduled for.")
(defun treemacs-magit--schedule-update ()
"Schedule an update to potentially run after 3 seconds of idle time.
In order for the update to fully run several conditions must be met:
* A timer for an update for the given directory must not already exist
(see `treemacs-magit--timers')
* The directory must be part of a treemacs workspace, and
* The project must not be set for refresh already."
(when treemacs-git-mode
(let ((magit-root (treemacs-canonical-path (magit-toplevel))))
(unless (member magit-root treemacs-magit--timers)
(push magit-root treemacs-magit--timers)
(run-with-idle-timer
3 nil
(lambda ()
(unwind-protect
(pcase treemacs--git-mode
('simple
(treemacs-magit--simple-git-mode-update magit-root))
((or 'extended 'deferred)
(treemacs-magit--extended-git-mode-update magit-root)))
(setf treemacs-magit--timers (delete magit-root treemacs-magit--timers)))))))))
(defun treemacs-magit--simple-git-mode-update (magit-root)
"Update the project at the given MAGIT-ROOT.
Without the parsing ability of extended git-mode this update uses
filewatch-mode's mechanics to update the entire project."
(treemacs-run-in-every-buffer
(when-let* ((project (treemacs--find-project-for-path magit-root))
(dom-node (treemacs-find-in-dom (treemacs-project->path project))))
(push (cons (treemacs-project->path project) 'force-refresh)
(treemacs-dom-node->refresh-flag dom-node))
(treemacs--start-filewatch-timer))))
(defun treemacs-magit--extended-git-mode-update (magit-root)
"Update the project at the given MAGIT-ROOT.
This runs due to a commit or stash action, so we know that no files have
actually been added or deleted. This allows us to forego rebuilding the entire
project structure just to be sure we caught everything. Instead we grab the
current git status and just go through the lines as they are right now."
;; we run a single git process to update every buffer, so we need to gather
;; the visible dirs in every buffer
;; this collection may contain duplicates, but they are removed in python
(-let [visible-dirs nil]
(treemacs-run-in-every-buffer
(dolist (dir (-some->> magit-root
(treemacs-find-in-dom)
(treemacs-dom-node->children)
(-map #'treemacs-dom-node->key)))
(when (stringp dir)
(push dir visible-dirs))))
(pfuture-callback `(,treemacs-python-executable
"-O" "-S"
,treemacs--git-status.py
,magit-root
,(number-to-string treemacs-max-git-entries)
,treemacs-git-command-pipe
,@visible-dirs)
:directory magit-root
:on-success
(progn
(ignore status)
(treemacs-magit--update-callback magit-root pfuture-buffer)))))
(defun treemacs-magit--update-callback (magit-root pfuture-buffer)
"Run the update as a pfuture callback.
Will update nodes under MAGIT-ROOT with output in PFUTURE-BUFFER."
(let ((ht (read (pfuture-output-from-buffer pfuture-buffer))))
(treemacs-run-in-every-buffer
(let ((dom-node (or (treemacs-find-in-dom magit-root)
(when-let* ((project
(seq-find
(lambda (pr) (treemacs-is-path (treemacs-project->path pr) :in magit-root))
(treemacs-workspace->projects (treemacs-current-workspace)))))
(treemacs-find-in-dom (treemacs-project->path project))))))
(when (and dom-node
(treemacs-dom-node->position dom-node)
(treemacs-is-node-expanded? (treemacs-dom-node->position dom-node))
(null (treemacs-dom-node->refresh-flag dom-node)))
(save-excursion
(goto-char (treemacs-dom-node->position dom-node))
(forward-line 1)
(let* ((node (treemacs-node-at-point))
(start-depth (-some-> node (treemacs-button-get :depth)))
(curr-depth start-depth)
(path (-some-> node (treemacs-button-get :key))))
(treemacs-with-writable-buffer
(while (and node
(>= curr-depth start-depth))
(when (and (stringp path)
(file-exists-p path))
(treemacs--git-face-quick-change
(treemacs-button-get node :key)
(or (ht-get ht path)
(if (memq (treemacs-button-get node :state)
'(file-node-open file-node-closed))
'treemacs-git-unmodified-face
'treemacs-directory-face)))
(put-text-property (treemacs-button-start node) (treemacs-button-end node) 'face
(or (ht-get ht path)
(if (memq (treemacs-button-get node :state)
'(file-node-open file-node-closed))
'treemacs-git-unmodified-face
'treemacs-directory-face))))
(forward-line 1)
(if (eobp)
(setf node nil)
(setf node (treemacs-node-at-point)
path (-some-> node (treemacs-button-get :path))
curr-depth (-some-> node (treemacs-button-get :depth)))))))))))))
(unless (featurep 'treemacs-magit)
(add-hook 'magit-post-commit-hook #'treemacs-magit--schedule-update)
(add-hook 'git-commit-post-finish-hook #'treemacs-magit--schedule-update)
(add-hook 'magit-post-stage-hook #'treemacs-magit--schedule-update)
(add-hook 'magit-post-unstage-hook #'treemacs-magit--schedule-update))
;;;; Git Commit Diff
(defvar treemacs--git-commit-diff.py)
(defvar treemacs--commit-diff-ann-source)
(defconst treemacs--commit-diff-update-commands
(list "pull" "push" "commit" "merge" "rebase" "cherry-pick" "fetch" "checkout")
"List of git commands that change local/remote commit status info.
Relevant for integrating with `treemacs-git-commit-diff-mode'.")
(defun treemacs--update-commit-diff-after-magit-process (process &rest _)
"Update commit diffs after completion of a magit git PROCESS."
(when (memq (process-status process) '(exit signal))
(let* ((args (process-command process))
(command (car (nthcdr (1+ (length magit-git-global-arguments)) args))))
(when (member command treemacs--commit-diff-update-commands)
(-let [path (process-get process 'default-dir)]
(pfuture-callback `(,treemacs-python-executable "-O" ,treemacs--git-commit-diff.py ,path)
:directory path
:on-success
(-let [out (-> (pfuture-callback-output)
(string-trim-right)
(read))]
(treemacs-run-in-every-buffer
(-when-let* ((project (treemacs--find-project-for-path path))
(project-path (treemacs-project->path project)))
(if out
(treemacs-set-annotation-suffix
project-path out treemacs--commit-diff-ann-source)
(treemacs-remove-annotation-suffix project-path treemacs--commit-diff-ann-source))
(treemacs-apply-single-annotation project-path))))))))))
(defun treemacs--magit-commit-diff-setup ()
"Enable or disable magit advice for `treemacs-git-commit-diff-mode'."
(if (bound-and-true-p treemacs-git-commit-diff-mode)
(advice-add #'magit-process-sentinel :after #'treemacs--update-commit-diff-after-magit-process)
(advice-remove #'magit-process-sentinel #'treemacs--update-commit-diff-after-magit-process)))
(unless (featurep 'treemacs-magit)
(add-hook 'treemacs-git-commit-diff-mode-hook #'treemacs--magit-commit-diff-setup)
(when (bound-and-true-p treemacs-git-commit-diff-mode)
(advice-add #'magit-process-sentinel :after #'treemacs--update-commit-diff-after-magit-process)))
(provide 'treemacs-magit)
;;; treemacs-magit.el ends here