;;; lsp-treemacs.el --- LSP treemacs -*- lexical-binding: t; -*- ;; Copyright (C) 2019 Ivan Yonchovski ;; Author: Ivan Yonchovski ;; Keywords: languages ;; Package-Requires: ((emacs "26.1") (dash "2.14.1") (dash-functional "2.14.1") (f "0.20.0") (ht "2.0") (treemacs "2.5") (lsp-mode "6.0")) ;; Homepage: https://github.com/emacs-lsp/lsp-treemacs ;; Version: 0.3 ;; 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 . ;;; Commentary: ;; `lsp-mode' and `treemacs' integration. ;;; Code: (require 'treemacs) (require 's) (require 'f) (require 'treemacs-extensions) (require 'treemacs-icons) (require 'lsp-treemacs-themes) (require 'lsp-mode) (require 'lsp-lens) (defconst lsp-treemacs-deps-buffer-name "*Java Dependency List*") (defgroup lsp-treemacs nil "Language Server Protocol client." :group 'tools :tag "Language Server") (defvar lsp-treemacs-deps-position-params `((side . ,treemacs-position) (slot . 1) (window-width . ,treemacs-width)) "The params which will be used by `display-buffer-in-side-window'.") (defvar lsp-treemacs-symbols-position-params `((side . ,treemacs-position) (slot . 2) (window-width . ,treemacs-width))) (defface lsp-treemacs-project-root-error '((t :inherit font-lock-keyword-face)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defface lsp-treemacs-project-root-info '((t :inherit font-lock-keyword-face)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defface lsp-treemacs-project-root-warn '((t :inherit font-lock-keyword-face)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defface lsp-treemacs-file-error '((t :inherit error)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defface lsp-treemacs-file-info '((t :inherit success)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defface lsp-treemacs-file-warn '((t :inherit warning)) "Face used for highlighting symbols being read." :group 'lsp-faces) (defcustom lsp-treemacs-face-map '((1 . lsp-treemacs-project-root-error) (2 . lsp-treemacs-project-root-warn) (3 . lsp-treemacs-project-root-info)) "Alist diagnostics to face." :type 'alist) (defcustom lsp-treemacs-file-face-map '((1 . lsp-treemacs-file-error) (2 . lsp-treemacs-file-warn) (3 . lsp-treemacs-file-info)) "Alist diagnostics to face." :type 'alist) (defcustom lsp-treemacs-error-list-severity 3 "Severity level for `lsp-treemacs-error-list-mode'. 1 (highest) to 3 (lowest)" :type 'number) (defun lsp-treemacs--match-diagnostic-severity (diagnostic) (<= (lsp:diagnostic-severity? diagnostic) (prefix-numeric-value lsp-treemacs-error-list-severity))) (defun lsp-treemacs--diagnostics-match-selected-severity (diagnostics) (-some #'lsp-treemacs--match-diagnostic-severity diagnostics)) (defun lsp-treemacs--root-folders () "Get root folders containing errors." (let ((diagnostics (lsp-diagnostics))) (->> (lsp-session) lsp-session-folders (-filter (lambda (folder-name) (-some (-lambda ((file-name . file-diagnostics)) (and (s-starts-with? folder-name file-name) (lsp-treemacs--diagnostics-match-selected-severity file-diagnostics))) (ht->alist diagnostics)))) (-map (lambda (root-folder) (cons (format (propertize "%s %s %s" 'face 'default) (propertize (f-filename root-folder) 'face (lsp-treemacs--face root-folder diagnostics)) (lsp-treemacs--diag-statistics (lsp-treemacs--project-diagnostics root-folder diagnostics)) (propertize (f-dirname root-folder) 'face 'lsp-lens-face)) root-folder)))))) (defun lsp-treemacs-quick-fix () "Select the element under cursor." (interactive) (let ((key (button-get (treemacs-node-at-point) :data))) (if (and (consp key) (ht? (cdr key))) (-let (((file . _diagnostic) key)) (with-current-buffer (find-file-noselect file) (with-lsp-workspaces (lsp--try-project-root-workspaces nil nil) (save-excursion (goto-char (lsp--position-to-point :start)) (lsp-execute-code-action-by-kind "quickfix"))))) (user-error "Not on a diagnostic")))) (defun lsp-treemacs-cycle-severity () "Cycle through the severity levels shown in the errors list" (interactive) (setq lsp-treemacs-error-list-severity (if (= lsp-treemacs-error-list-severity 1) 3 (1- lsp-treemacs-error-list-severity))) (lsp-treemacs--after-diagnostics)) (defun lsp-treemacs--open-file-in-mru (file) (select-window (get-mru-window (selected-frame) nil :not-selected)) (find-file file)) (defun lsp-treemacs-open-file (&rest _) "Open file." (interactive) (let ((file (button-get (treemacs-node-at-point) :key))) (lsp-treemacs--open-file-in-mru file))) (defun lsp-treemacs-open-error (&rest _) "Open error." (interactive) (-let [(file . (&Diagnostic :range (&Range :start))) (button-get (treemacs-node-at-point) :data)] (find-file-other-window file) (goto-char (lsp--position-to-point start)))) (defun lsp-treemacs--face (root-folder diagnostics) "Calculate ROOT-FOLDER face based on DIAGNOSTICS." (--> diagnostics ht->alist (-keep (-lambda ((file-name . file-diagnostics)) (when (s-starts-with? root-folder file-name) (lsp:diagnostic-severity? (-min-by (-lambda ((&Diagnostic :severity? left?) (&Diagnostic :severity? right?)) (> (or left? 0) (or right? 0))) file-diagnostics)))) it) -min (assoc it lsp-treemacs-face-map) cl-rest)) (defun lsp-treemacs--project-diagnostics (root-folder diagnostics) "Calculate ROOT-FOLDER face based on DIAGNOSTICS." (->> diagnostics (ht-map (lambda (file-name file-diags) (when (and (s-starts-with? root-folder file-name) (lsp-treemacs--diagnostics-match-selected-severity file-diags)) file-diags))) (apply #'append))) (defun lsp-treemacs--diag-statistics (file-diagnostics) "Calculate FILE-DIAGNOSTICS statistics." (->> file-diagnostics (-filter #'lsp-treemacs--match-diagnostic-severity) (-group-by 'lsp:diagnostic-severity?) (-sort (-lambda ((left) (right)) (< left right))) (-map (-lambda ((severity . diagnostics)) (propertize (f-filename (number-to-string (length diagnostics))) 'face (cl-rest (assoc severity lsp-treemacs-file-face-map))))) (s-join "/"))) (defun lsp-treemacs--get-files (project-root) "Get files with errors in PROJECT-ROOT." (--> (lsp-diagnostics) ht->alist (-keep (-lambda ((file-name . file-diagnostics)) (when (and (s-starts-with? project-root file-name) (lsp-treemacs--diagnostics-match-selected-severity file-diagnostics)) (cons file-name (format (propertize "%s %s %s" 'face 'default) (propertize (f-filename file-name) 'face 'default) (lsp-treemacs--diag-statistics file-diagnostics) (propertize (f-dirname (f-relative file-name project-root)) 'face 'lsp-lens-face))))) it))) (defun lsp-treemacs--errors (file-name) "Get errors for FILE-NAME." (->> (lsp-diagnostics) (gethash file-name) (-filter #'lsp-treemacs--match-diagnostic-severity) (-sort (-lambda ((&Diagnostic :range (&Range :start (&Position :character char-a :line line-a))) (&Diagnostic :range (&Range :start (&Position :character char-b :line line-b)))) (if (= line-a line-b) (< char-a char-b) (< line-a line-b)))) (--map (cons file-name it)))) (defun lsp-treemacs--diagnostic-icon (diagnostic) "Get the icon for DIAGNOSTIC." (cl-case (lsp:diagnostic-severity? diagnostic) (1 treemacs-icon-error) (2 treemacs-icon-warning) (t treemacs-icon-info))) (treemacs-define-expandable-node lsp-error :icon-open-form (lsp-treemacs--diagnostic-icon (cl-rest (treemacs-button-get node :data))) :icon-closed-form (lsp-treemacs--diagnostic-icon (cl-rest (treemacs-button-get node :data))) :query-function (lsp-treemacs--errors (treemacs-button-get node :data)) :ret-action 'lsp-treemacs-open-error :render-action (treemacs-render-node :icon (treemacs-as-icon ". " 'face 'font-lock-string-face) :label-form (propertize (lsp:diagnostic-message item) 'face 'default) :state treemacs-lsp-error-open-state :key-form item)) (treemacs-define-expandable-node lsp-files :icon-open-form (treemacs-icon-for-file (treemacs-button-get node :key)) :icon-closed-form (treemacs-icon-for-file (treemacs-button-get node :key)) :query-function (lsp-treemacs--errors (treemacs-button-get node :key)) :ret-action 'lsp-treemacs-open-file :render-action (-let* (((&Diagnostic :source? :message :range (&Range :start (&Position :line :character))) (cl-rest item)) (label (format (propertize "%s %s %s" 'face 'default) (if source? (propertize (format "[%s]" source?) 'face 'shadow) "") message (propertize (format "(%s:%s)" line character) 'face 'lsp-lens-face)))) (treemacs-render-node :icon (lsp-treemacs--diagnostic-icon (cl-rest item)) :label-form label :state treemacs-lsp-error-open-state :key-form label :more-properties (:data item)))) (treemacs-define-expandable-node lsp-projects :icon-open (treemacs-get-icon-value 'root nil lsp-treemacs-theme) :icon-closed (treemacs-get-icon-value 'root nil lsp-treemacs-theme) :query-function (lsp-treemacs--get-files (treemacs-button-get node :key)) :ret-action 'lsp-treemacs-open-file :render-action (treemacs-render-node :icon (treemacs-icon-for-file (cl-first item)) :label-form (cl-rest item) :state treemacs-lsp-files-closed-state :key-form (cl-first item))) (treemacs-define-variadic-node lsp-error-list :query-function (lsp-treemacs--root-folders) :render-action (treemacs-render-node :icon (treemacs-get-icon-value 'root nil lsp-treemacs-theme) :label-form (cl-first item) :state treemacs-lsp-projects-closed-state :key-form (cl-rest item)) :root-key-form 'LSP-Errors) (defun lsp-treemacs--after-diagnostics () "After diagnostics handler." (save-excursion (condition-case _err (with-current-buffer (get-buffer-create "*LSP Error List*") (treemacs-update-node '(:custom LSP-Errors) t)) (error)))) (defun lsp-treemacs--kill-buffer () "Kill buffer hook." (remove-hook 'lsp-diagnostics-updated-hook #'lsp-treemacs--after-diagnostics)) (defvar lsp-treemacs-error-list-mode-map (let ((m (make-sparse-keymap))) (define-key m (kbd "x") #'lsp-treemacs-quick-fix) (define-key m (kbd "=") #'lsp-treemacs-cycle-severity) m) "Keymap for `lsp-treemacs-error-list-mode'.") (define-minor-mode lsp-treemacs-error-list-mode "" nil nil nil :keymap lsp-treemacs-error-list-mode-map :group 'lsp-treeemacs) ;;;###autoload (defun lsp-treemacs-errors-list () "Display error list." (interactive) (-if-let (buffer (get-buffer "*LSP Error List*")) (progn (select-window (display-buffer-in-side-window buffer '((side . bottom)))) (lsp-treemacs--after-diagnostics)) (let* ((buffer (get-buffer-create "*LSP Error List*")) (window (display-buffer-in-side-window buffer '((side . bottom))))) (select-window window) (set-window-dedicated-p window t) (treemacs-initialize) (lsp-treemacs--set-mode-line-format buffer " LSP Errors View ") (lsp-treemacs-error-list-mode 1) (setq-local treemacs-default-visit-action 'treemacs-RET-action) (setq-local treemacs-space-between-root-nodes nil) (treemacs-LSP-ERROR-LIST-extension) (add-hook 'lsp-diagnostics-updated-hook #'lsp-treemacs--after-diagnostics) (add-hook 'kill-buffer-hook 'lsp-treemacs--kill-buffer nil t)))) (defun lsp-treemacs-symbol-kind->icon (kind) (cl-case kind (1 'document) (2 'namespace) (3 'namespace) (4 'namespace) (5 'class) (6 'method) (7 'property) (8 'field) (9 'method) (10 'enumerator) (11 'interface) (12 'method ) (13 'localvariable) (14 'constant) (15 'string) (16 'numeric) (17 'boolean-data) (18 'boolean-data) (19 'namespace) (20 'indexer) (21 'boolean-data) (22 'enumitem) (23 'structure) (24 'event) (25 'operator) (26 'template))) (defun lsp-treemacs-get-icon (icon-name) "Get the treemacs ICON using current theme." (treemacs-get-icon-value icon-name nil lsp-treemacs-theme)) (defun lsp-treemacs-symbol-icon (kind) "Get icon for `kind'." (treemacs-get-icon-value (lsp-treemacs-symbol-kind->icon kind) nil lsp-treemacs-theme)) (defun lsp-treemacs--symbol-icon (symbol expanded) "Get the symbol for the the kind." (-let [(&DocumentSymbol :kind :children?) symbol] (concat (if (seq-empty-p children?) " " (if expanded " ▾ " " ▸ ")) (lsp-treemacs-symbol-icon kind)))) (treemacs-define-expandable-node lsp-symbol :icon-open-form (lsp-treemacs--symbol-icon (treemacs-button-get node :symbol) t) :icon-closed-form (lsp-treemacs--symbol-icon (treemacs-button-get node :symbol) nil) :query-function (append (lsp:document-symbol-children? (treemacs-button-get node :symbol)) nil) :ret-action 'lsp-treemacs-goto-symbol :render-action (treemacs-render-node :icon (lsp-treemacs--symbol-icon item nil) :label-form (propertize (lsp:document-symbol-name item) 'face 'default) :state treemacs-lsp-symbol-closed-state :key-form (lsp:document-symbol-name item) :more-properties (:symbol item))) (defvar-local lsp-treemacs--symbols nil) (defvar-local lsp-treemacs--symbols-tick nil) (defvar-local lsp-treemacs--empty nil) (defvar-local lsp-treemacs--symbols-state-string nil) (defvar-local lsp-treemacs--symbols-state-locals nil) (defvar lsp-treemacs--symbols-current-buffer nil) (defvar lsp-treemacs--symbols-last-buffer nil) (defvar lsp-treemacs--symbols-timer nil) (defun lsp-treemacs-sort-by-position (left right) (-let (((&plist :position left-position) left) ((&plist :position right-position) right)) (and left-position right-position (lsp--position-compare left-position right-position)))) (defun lsp-treemacs-sort-by-kind (left right) (-let (((&plist :kind left-kind) left) ((&plist :kind right-kind) right)) (and left-kind right-kind (> left-kind right-kind)))) (defun lsp-treemacs-sort-by-name (left right) (-let* (((&plist :label left-name) left) ((&plist :label right-name) right)) (string> (downcase right-name) (downcase left-name)))) (defcustom lsp-treemacs-symbols-sort-functions '(lsp-treemacs-sort-by-position) "Sort functions." :type '(repeat (choice (const :tag "Name" lsp-treemacs-sort-by-name) (const :tag "Kind" lsp-treemacs-sort-by-kind) (const :tag "Position" lsp-treemacs-sort-by-position)))) (defun lsp-treemacs--symbols->tree (items parent-key) "Convert ITEMS and PARENT-KEY to a treemacs tree." (-sort (lambda (left right) (-first (lambda (fn) (funcall fn left right)) lsp-treemacs-symbols-sort-functions)) (if (-some->> items lsp-seq-first lsp-symbol-information?) (-let [(current rest) (-separate (-lambda ((&SymbolInformation :container-name?)) (string= container-name? parent-key)) (append items nil))] (seq-map (-lambda ((&SymbolInformation :name :container-name? :kind :location (location &as &Location :range (&Range :start start-range)))) (when (string= parent-key container-name?) `(:label ,name :key ,name :icon ,(lsp-treemacs-symbol-kind->icon kind) ,@(when (-first (-lambda ((&SymbolInformation :container-name? parent)) (string= name parent)) rest) (list :children (lsp-treemacs--symbols->tree rest name))) :kind ,kind :location ,start-range :ret-action ,(lambda (&rest _) (pop-to-buffer lsp-treemacs--symbols-last-buffer) (->> start-range lsp--position-to-point goto-char) (run-hooks 'xref-after-jump-hook))))) current)) (seq-map (-lambda ((&DocumentSymbol :name :kind :selection-range (&Range :start start-range) :children? :deprecated?)) `(:label ,(if deprecated? (propertize name 'face 'lsp-face-semhl-deprecated) name) :key ,name :icon ,(lsp-treemacs-symbol-kind->icon kind) :kind ,kind :location start-range ,@(unless (seq-empty-p children?) (list :children (lsp-treemacs--symbols->tree children? name))) :ret-action ,(lambda (&rest _) (pop-to-buffer lsp-treemacs--symbols-last-buffer) (->> start-range lsp--position-to-point goto-char) (run-hooks 'xref-after-jump-hook)))) items)))) (defun lsp-treemacs--update-symbols () "After diagnostics handler." (setq-local header-line-format (unless lsp-treemacs--symbols (propertize "No symbol information." 'face 'shadow))) (lsp-treemacs-render (lsp-treemacs--symbols->tree lsp-treemacs--symbols nil) " LSP Symbols " (and lsp-treemacs--symbols (> 30 (length lsp-treemacs--symbols))) "*LSP Symbols List*" )) (defun lsp-treemacs--update () (unless (eq (current-buffer) (get-buffer "*scratch*")) (when (with-current-buffer "*LSP Symbols List*" (get-buffer-window)) (if (lsp--find-workspaces-for "textDocument/documentSymbol") (when (or (not lsp-treemacs--symbols-tick) (not (eq lsp-treemacs--symbols-tick (buffer-modified-tick))) (not (eq (current-buffer) lsp-treemacs--symbols-current-buffer))) (lsp-request-async "textDocument/documentSymbol" (lsp-make-document-symbol-params :text-document (lsp--text-document-identifier)) (lambda (document-symbols) (save-excursion (with-current-buffer "*LSP Symbols List*" (setq-local lsp-treemacs--symbols document-symbols) (lsp-treemacs--update-symbols)))) :mode 'alive) (setq-local lsp-treemacs--symbols-tick (buffer-modified-tick)) (setq lsp-treemacs--symbols-last-buffer (current-buffer))) (when (buffer-file-name) (with-current-buffer "*LSP Symbols List*" (setq-local lsp-treemacs--symbols nil) (lsp-treemacs--update-symbols))))) (setq lsp-treemacs--symbols-current-buffer (current-buffer)))) (defun lsp-treemacs-goto-symbol (&rest _) "Goto the symbol at point." (interactive) (if-let ((symbol-data (-some-> (treemacs-node-at-point) (button-get :symbol)))) (with-current-buffer lsp-treemacs--symbols-last-buffer (let ((p (lsp--position-to-point (or (-some->> symbol-data lsp:document-symbol-selection-range lsp:range-start) (-some->> symbol-data lsp:symbol-information-location lsp:location-range lsp:range-start) (error "Unable to go to location"))))) (pop-to-buffer lsp-treemacs--symbols-last-buffer) (goto-char p) (run-hooks 'xref-after-jump-hook))) (user-error "No symbol under point."))) (with-eval-after-load 'winum (when (boundp 'winum-ignored-buffers) (add-to-list 'winum-ignored-buffers "*LSP Symbols List*") (add-to-list 'winum-ignored-buffers "*LSP Error List*") (add-to-list 'winum-ignored-buffers lsp-treemacs-deps-buffer-name))) (defun lsp-treemacs--expand (root-key depth) (-when-let (root (treemacs-dom-node->position (treemacs-find-in-dom root-key))) (treemacs-save-position (lsp-treemacs--expand-recursively root depth)))) (defun lsp-treemacs--kill-symbols-buffer () (and lsp-treemacs--symbols-timer (cancel-timer lsp-treemacs--symbols-timer))) ;;;###autoload (defun lsp-treemacs-symbols () "Show symbols view." (interactive) (let ((original-buffer (current-buffer))) (if-let (buf (get-buffer "*LSP Symbols List*")) (select-window (display-buffer-in-side-window buf lsp-treemacs-symbols-position-params)) (let* ((buf (get-buffer-create "*LSP Symbols List*")) (window (display-buffer-in-side-window buf lsp-treemacs-symbols-position-params))) (select-window window) (set-window-dedicated-p window t) (setq-local treemacs-default-visit-action 'treemacs-RET-action) (setq-local treemacs-space-between-root-nodes nil) (setq lsp-treemacs--symbols-timer (run-at-time 0 1.0 #'lsp-treemacs--update)) (add-hook 'kill-buffer-hook 'lsp-treemacs--kill-symbols-buffer nil t))) (with-current-buffer original-buffer (lsp-treemacs--update)))) (defun lsp-treemacs--expand-recursively (root depth) (when (if (booleanp depth) depth (not (zerop depth))) (save-excursion (-map (lambda (btn) (unless (treemacs-is-node-expanded? btn) (goto-char (marker-position btn)) (funcall (alist-get (treemacs-button-get btn :state) treemacs-TAB-actions-config))) (lsp-treemacs--expand-recursively btn (if (booleanp depth) depth (1- depth)))) (treemacs-collect-child-nodes root))))) (defmacro lsp-treemacs-deps-with-jdtls (&rest body) "Helper macro for invoking BODY against WORKSPACE context." (declare (debug (form body)) (indent 0)) `(if-let (lsp--cur-workspace (lsp-find-workspace 'jdtls nil)) (progn ,@body) (user-error "Java Language Server is not started."))) (defun lsp-treemacs-deps--goto-element (&rest _args) (if-let ((dep (-some-> (treemacs-node-at-point) (button-get :dep)))) (lsp-treemacs--open-file-in-mru (or (-some-> (gethash "uri" dep) (lsp--uri-to-path)) (when (f-exists? (gethash "path" dep)) (gethash "path" dep)) (concat (f-parent (lsp--uri-to-path (gethash "projectUri" dep))) (gethash "path" dep)))) (user-error "No element under point."))) (defun lsp-treemacs-deps--icon (dep expanded) "Get the symbol for the the kind." (-let (((&hash "uri" "kind" "entryKind" entry-kind) dep)) (concat (if expanded " ▾ " " ▸ ") (if (or (= kind 8) (= kind 6)) (treemacs-icon-for-file uri) (treemacs-get-icon-value (cond ((eq entry-kind 2) 'jar) ((eq kind 5) 'package) ((eq kind 7) 'folder) ((eq kind 4) 'packagefolder) ((eq kind 2) 'java-project) ((eq entry-kind 3) 'packagefolder) ((eq entry-kind 5) 'library)) nil lsp-treemacs-theme))))) (defun lsp-treemacs-deps--get-children (dep) (lsp-treemacs-deps-with-jdtls (-let* (((&hash "projectUri" project-uri "rootPath" root-path "path" "kind" "name" "uri") dep) (project-uri (if (eq kind 2) uri project-uri))) (unless (or (= kind 6) (= kind 8)) (->> (lsp-send-execute-command "java.getPackageData" (vector (ht ("kind" kind) ("path" (unless (eq kind 2) (if (= 5 kind) name path))) ("rootPath" (unless (eq kind 2) (or root-path path))) ("projectUri" project-uri)))) (-mapcat (lambda (inner-dep) (puthash "projectUri" project-uri inner-dep) (when (= kind 4) (puthash "rootPath" path inner-dep)) (if (eq (gethash "entryKind" inner-dep) 3) (lsp-treemacs-deps--get-children inner-dep) (list inner-dep))))))))) (defun lsp-treemacs-deps--java-file? (dep) (-let [(&hash "kind" "entryKind" entry-kind) dep] (and (eq kind 6) (or (eq entry-kind 1) (eq entry-kind 2))))) (treemacs-define-expandable-node lsp-treemacs-deps :icon-open-form (lsp-treemacs-deps--icon (treemacs-button-get node :dep) t) :icon-closed-form (lsp-treemacs-deps--icon (treemacs-button-get node :dep) nil) :query-function (-let (((dep &as &hash "uri") (treemacs-button-get node :dep))) (if (lsp-treemacs-deps--java-file? dep) (lsp-treemacs-deps-with-jdtls (lsp-request "textDocument/documentSymbol" (lsp-make-document-symbol-params :text-document (lsp-make-text-document-item :uri uri)))) (lsp-treemacs-deps--get-children dep))) :ret-action 'lsp-treemacs-deps--goto-element :render-action (if (lsp-treemacs-deps--java-file? (treemacs-button-get node :dep)) (treemacs-render-node :icon (lsp-treemacs--symbol-icon item nil) :label-form (propertize (gethash "name" item) 'face 'default) :state treemacs-lsp-symbol-closed-state :key-form (list (gethash "name" item) (gethash "uri" item) (gethash "path" item)) :more-properties (:symbol item)) (treemacs-render-node :icon (lsp-treemacs-deps--icon item nil) :label-form (propertize (gethash "name" item) 'face 'default) :state treemacs-lsp-treemacs-deps-closed-state :key-form (list (gethash "name" item) (gethash "uri" item) (gethash "path" item)) :more-properties (:dep item)))) (defun lsp-treemacs-deps--root-folders () (lsp-treemacs-deps-with-jdtls (-mapcat (lambda (root-path) (let ((project-uri (lsp--path-to-uri root-path))) (->> project-uri (lsp-send-execute-command "java.project.list") (--map (--doto it (puthash "projectUri" project-uri it)))))) (lsp-session-folders (lsp-session))))) (treemacs-define-variadic-node lsp-treemacs-deps-list :query-function (lsp-treemacs-deps--root-folders) :render-action (treemacs-render-node :icon (lsp-treemacs-deps--icon item nil) :label-form (propertize (gethash "name" item) 'face 'default) :state treemacs-lsp-treemacs-deps-closed-state :key-form (list (gethash "name" item) (gethash "uri" item) (gethash "path" item)) :more-properties (:dep item)) :root-key-form 'LSP-Java-Dependency) (defun lsp-treemacs-java-deps-refresh () "Refresh dependecy list." (interactive) (condition-case _err (let ((inhibit-read-only t)) (with-current-buffer lsp-treemacs-deps-buffer-name (treemacs-update-node '(:custom LSP-Java-Dependency) t) (lsp--info "Refresh completed"))) (error))) ;;;###autoload (defun lsp-treemacs-java-deps-list () "Display error list." (interactive) (-if-let (buffer (get-buffer lsp-treemacs-deps-buffer-name)) (select-window (or (get-buffer-window lsp-treemacs-deps-buffer-name) (display-buffer-in-side-window buffer lsp-treemacs-deps-position-params))) (let* ((buffer (get-buffer-create lsp-treemacs-deps-buffer-name )) (window (display-buffer-in-side-window buffer lsp-treemacs-deps-position-params))) (select-window window) (set-window-dedicated-p window t) (treemacs-initialize) (lsp-treemacs--set-mode-line-format buffer " Java Dependencies ") (lsp-treemacs-deps-list-mode t) (setq-local treemacs-space-between-root-nodes nil) (setq-local treemacs-default-visit-action 'treemacs-RET-action) (treemacs-LSP-TREEMACS-DEPS-LIST-extension)))) (defun lsp-treemacs--deps-find-children-for-key (node key) (->> node treemacs-collect-child-nodes (-first (lambda (child) (goto-char (marker-position child)) (equal (treemacs-button-get child :key) key))))) (defvar lsp-treemacs-deps-list-mode-map (let ((m (make-sparse-keymap))) (define-key m (kbd "r") #'lsp-treemacs-java-deps-refresh) m) "Keymap for `lsp-treemacs-deps-list-mode'.") (define-minor-mode lsp-treemacs-deps-list-mode "" nil nil nil :keymap lsp-treemacs-deps-list-mode-map :group 'lsp-treeemacs) ;;;###autoload (defun lsp-treemacs-java-deps-follow () (interactive) (lsp-treemacs-deps-with-jdtls (let ((paths (lsp-send-execute-command "java.resolvePath" (lsp--buffer-uri)))) (select-window (with-current-buffer lsp-treemacs-deps-buffer-name (set-window-point (get-buffer-window) (marker-position (-reduce-from (-lambda (node (&hash "path" "name" "uri")) (unless (treemacs-is-node-expanded? node) (save-excursion (goto-char (marker-position node)) (funcall (alist-get (treemacs-button-get node :state) treemacs-TAB-actions-config)))) (or (lsp-treemacs--deps-find-children-for-key node (list name uri path)) (user-error "Unable to find %s in the dependency tree." (buffer-name)))) (treemacs-dom-node->position (treemacs-find-in-dom '(:custom LSP-Java-Dependency))) paths))) (get-buffer-window))) (recenter nil)))) ;; treemacs synchronization (defun lsp-treemacs--on-folder-remove (project) (lsp-workspace-folders-remove (treemacs-project->path project))) (defun lsp-treemacs--on-folder-added (project) (lsp-workspace-folders-add (treemacs-project->path project))) (defun lsp-treemacs--treemacs->lsp () (let ((lsp-folders (lsp-session-folders (lsp-session))) (treemacs-folders (->> (treemacs-current-workspace) (treemacs-workspace->projects) (-map #'treemacs-project->path) (-map #'lsp-canonical-file-name)))) (seq-do #'lsp-workspace-folders-remove (-difference lsp-folders treemacs-folders)) (seq-do #'lsp-workspace-folders-add (-difference treemacs-folders lsp-folders)))) (defun lsp-treemacs--sync-folders (added removed) (when-let (treemacs-workspace (treemacs-current-workspace)) (let ((treemacs-create-project-functions (remove #'lsp-treemacs--on-folder-added treemacs-create-project-functions)) (treemacs-delete-project-functions (remove #'lsp-treemacs--on-folder-remove treemacs-delete-project-functions))) (seq-do (lambda (folder) (when (->> treemacs-workspace (treemacs-workspace->projects) (-none? (lambda (project) (f-same? (treemacs-project->path project) folder)))) (treemacs-add-project-to-workspace folder))) added) (seq-do (lambda (folder) (when-let (project (->> treemacs-workspace (treemacs-workspace->projects) (-first (lambda (project) (f-same? (treemacs-project->path project) folder))))) (treemacs-do-remove-project-from-workspace project))) removed)))) ;;;###autoload (define-minor-mode lsp-treemacs-sync-mode "Global minor mode for synchronizing lsp-mode workspace folders and treemacs projects." :init-value nil :group 'lsp-treemacs :global t (cond (lsp-treemacs-sync-mode (add-hook 'treemacs-create-project-functions #'lsp-treemacs--on-folder-added) (add-hook 'treemacs-delete-project-functions #'lsp-treemacs--on-folder-remove) (add-hook 'lsp-workspace-folders-changed-functions #'lsp-treemacs--sync-folders) (add-hook 'treemacs-workspace-edit-hook #'lsp-treemacs--treemacs->lsp) (add-hook 'treemacs-switch-workspace-hook #'lsp-treemacs--treemacs->lsp)) (t (remove-hook 'treemacs-create-project-functions #'lsp-treemacs--on-folder-added) (remove-hook 'treemacs-delete-project-functions #'lsp-treemacs--on-folder-remove) (remove-hook 'lsp-workspace-folders-changed-functions #'lsp-treemacs--sync-folders) (remove-hook 'treemacs-workspace-edit-hook #'lsp-treemacs--treemacs->lsp) (remove-hook 'treemacs-switch-workspace-hook #'lsp-treemacs--treemacs->lsp)))) (defun lsp-treemacs--java-get-class-file (file) (-let (((_ _package _class jar-file) (s-match "jdt://contents/.*\/\\(.*\\)\/\\(.*\\).class\\?=.*?/\\(.*?\\)=\/" file))) (symbol-name (read (url-unhex-string jar-file ))))) (defvar-local lsp-treemacs-tree nil) (defvar-local lsp-treemacs--right-click-actions nil) (defun lsp-treemacs-perform-ret-action (&rest _) (interactive) (if-let (action (-> (treemacs-node-at-point) (button-get :item) (plist-get :ret-action))) (funcall-interactively action) (treemacs-pulse-on-failure "No ret action defined."))) (defmacro lsp-treemacs-wcb-unless-killed (buffer &rest body) "`with-current-buffer' unless buffer killed." (declare (indent 1) (debug t)) `(when (buffer-live-p (get-buffer ,buffer)) (with-current-buffer ,buffer ,@body))) (defvar lsp-treemacs-use-cache nil) (defvar-local lsp-treemacs--generic-cache nil) (defun lsp-treemacs--node-key (node) (let ((result (list (treemacs-button-get node :key))) (parent node)) (while (setq parent (treemacs-button-get parent :parent)) (setq result (cons (treemacs-button-get parent :key) result))) result)) (treemacs-define-expandable-node node :icon-open-form (lsp-treemacs--generic-icon (treemacs-button-get node :item) t) :icon-closed-form (lsp-treemacs--generic-icon (treemacs-button-get node :item) nil) :query-function (-let (((item &as &plist :children :children-async) (treemacs-button-get node :item)) (node-key (lsp-treemacs--node-key node))) (cond ((functionp children) (funcall children item)) ((and (gethash node-key lsp-treemacs--generic-cache) lsp-treemacs-use-cache) (cl-rest (gethash node-key lsp-treemacs--generic-cache))) (children-async (-let [buffer (current-buffer)] (funcall children-async item (lambda (result) (lsp-treemacs-wcb-unless-killed buffer (puthash node-key (cons t result) lsp-treemacs--generic-cache) (let ((lsp-treemacs-use-cache t)) (lsp-treemacs-generic-refresh)))))) (or (cl-rest (gethash node-key lsp-treemacs--generic-cache)) `((:label ,(propertize "Loading..." 'face 'shadow) :icon-literal " " :key "Loading...")))) (t children))) :ret-action #'lsp-treemacs-perform-ret-action :render-action (-let [(&plist :children :label :key :children-async) item] (treemacs-render-node :icon (lsp-treemacs--generic-icon item nil) :label-form label :state treemacs-node-closed-state :key-form key :more-properties (:children children :item item :children-async children-async)))) (treemacs-define-variadic-node generic :query-function lsp-treemacs-tree :render-action (treemacs-render-node :icon (lsp-treemacs--generic-icon item nil) :label-form (plist-get item :label) :state treemacs-node-closed-state :key-form (plist-get item :key) :more-properties (:item item)) :root-key-form 'LSP-Generic) (defun lsp-treemacs--generic-icon (item expanded?) "Get the symbol for the the kind." (concat (if (or (plist-get item :children) (plist-get item :children-async)) (if expanded? " ▾ " " ▸ ") " ") (or (plist-get item :icon-literal) (if-let ((icon (plist-get item :icon))) (treemacs-get-icon-value icon nil lsp-treemacs-theme) " ")))) (defun lsp-treemacs--get-xrefs-in-file (file-locs location-link) (-let (((filename . links) file-locs)) (list :key filename :label (format (propertize "%s %s" 'face 'default) (propertize (f-filename filename) 'face 'default) (propertize (format "%s references" (length links)) 'face 'lsp-lens-face)) :icon (f-ext filename) :children (lambda (_item) (condition-case err (let ((buf (find-buffer-visiting filename)) (fn (lambda () (seq-map (lambda (loc) (lsp-treemacs--make-ref-item (if location-link (or (lsp:location-link-target-selection-range loc) (lsp:location-link-target-range loc)) (lsp:location-range loc)) filename)) links)))) (if buf (with-current-buffer buf (funcall fn)) (when (file-readable-p filename) (with-temp-buffer (insert-file-contents-literally filename) (funcall fn))))) (error (ignore (lsp-warn "Failed to process xref entry for filename '%s': %s" filename (error-message-string err)))) (file-error (ignore (lsp-warn "Failed to process xref entry, file-error, '%s': %s" filename (error-message-string err)))))) :ret-action (lambda (&rest _) (interactive) (lsp-treemacs--open-file-in-mru filename))))) (defun lsp-treemacs--extract-line (point) "Return the line pointed to by POS (a Position object) in the current buffer." (let* ((inhibit-field-text-motion t)) (save-excursion (goto-char point) (buffer-substring (line-beginning-position) (line-end-position))))) (lsp-defun lsp-treemacs--make-ref-item ((&Range :start (start &as &Position :line start-line :character start-character) :end (&Position :character end-character)) filename) "Return a xref-item from a RANGE in FILENAME." (-let* ((start-point (lsp--position-to-point start)) (line (lsp-treemacs--extract-line start-point)) (len (length line))) (add-face-text-property (max (min start-character len) 0) (max (min end-character len) 0) 'highlight t line) ;; LINE is nil when FILENAME is not being current visited by any buffer. (list :label (s-trim (format "%s %s" line (propertize(format "%s line" (1+ start-line)) 'face 'lsp-lens-face))) :key line :point start-point :icon-literal " " :ret-action (lambda (&rest _) (interactive) (lsp-treemacs--open-file-in-mru filename) (goto-char start-point) (run-hooks 'xref-after-jump-hook))))) (defun lsp-treemacs-initialize () (unless (derived-mode-p 'treemacs-mode) (treemacs-initialize) (lsp-treemacs-generic-mode t) (treemacs-GENERIC-extension))) (defun lsp-treemacs-generic-refresh () (condition-case _err (let ((inhibit-read-only t)) (treemacs-update-node '(:custom LSP-Generic) t)) (error))) (defun lsp-treemacs-generic-right-click (event) (interactive "e") (let* ((ec (event-start event)) (p1 (posn-point ec)) (w1 (posn-window ec))) (select-window w1) (goto-char p1) (hl-line-highlight) (run-with-idle-timer 0.001 nil (lambda () (-when-let* ((actions (if-let (node (treemacs-node-at-point)) (plist-get (button-get node :item) :actions) lsp-treemacs--right-click-actions)) (menu (easy-menu-create-menu nil actions)) (choice (x-popup-menu event menu))) (when choice (call-interactively (lookup-key menu (apply 'vector choice)))) (hl-line-highlight)))))) (defvar lsp-treemacs-generic-map (-doto (make-sparse-keymap) (define-key [mouse-1] #'treemacs-TAB-action) (define-key [double-mouse-1] #'treemacs-RET-action) (define-key [mouse-3] #'lsp-treemacs-generic-right-click)) "Keymap for `lsp-treemacs-generic-mode'") (define-minor-mode lsp-treemacs-generic-mode "Treemacs generic mode." nil nil lsp-treemacs-generic-map) (defun lsp-treemacs--handle-references (refs) (->> refs (-group-by (-lambda ((&Location :uri)) (let ((type (url-type (url-generic-parse-url (url-unhex-string uri))))) (if (string= type "jdt") (lsp-treemacs--java-get-class-file uri) (lsp-workspace-root (lsp--uri-to-path uri)))))) (-map (-lambda ((path . rst)) (list :key path :label (format "%s %s" (f-filename path) (propertize (format "%s references" (length rst)) 'face 'lsp-lens-face)) :icon (if (f-file? path) (f-ext path) 'dir-open) :children (lambda (_item) (-map (lambda (it) (lsp-treemacs--get-xrefs-in-file it nil)) (-group-by (-lambda ((&Location :uri)) (lsp--uri-to-path uri)) rst))) :ret-action (lambda (&rest _) (interactive) (lsp-treemacs--open-file-in-mru path))))))) (defmacro lsp-treemacs-define-action (name keys &rest body) (declare (doc-string 3) (indent 2)) `(defun ,name (&rest args) ,(format "Code action %s" name) (interactive) (if-let (node (treemacs-node-at-point)) (-let [,(cons '&plist keys) (button-get node :item)] ,@body) (treemacs-pulse-on-failure "No node at point")))) (defun lsp-treemacs-render (tree title expand-depth &optional buffer-name right-click-actions) (let ((search-buffer (get-buffer-create (or buffer-name "*LSP Lookup*")))) (with-current-buffer search-buffer (lsp-treemacs-initialize) (setq-local treemacs-default-visit-action 'treemacs-RET-action) (setq-local lsp-treemacs--right-click-actions right-click-actions) (setq-local lsp-treemacs--generic-cache (or lsp-treemacs--generic-cache (ht))) (setq-local lsp-treemacs-tree tree) (setq-local face-remapping-alist '((button . default))) (lsp-treemacs--set-mode-line-format search-buffer title) (lsp-treemacs-generic-refresh) (when expand-depth (lsp-treemacs--expand 'LSP-Generic expand-depth)) (current-buffer)))) (defalias 'lsp-treemacs--show-references 'lsp-treemacs-render) (defun lsp-treemacs--set-mode-line-format (buffer title) "Set the mode line format of BUFFER to TITLE. This function sets the `mode-name' or `mode-line-format' depending on if a custom mode line is detected." (with-current-buffer buffer (cond ((or (fboundp 'spaceline-install) (memq 'moody-mode-line-buffer-identification (default-value 'mode-line-format)) (and (fboundp 'doom-modeline) (fboundp 'doom-modeline-def-modeline))) (setq mode-name title)) (t (setq mode-line-format title))))) (defun lsp-treemacs--do-search (method params title prefix-args) (let ((search-buffer (get-buffer-create "*LSP Lookup*")) (window (display-buffer (get-buffer-create "*LSP Lookup*")))) (lsp-request-async method params (lambda (refs) (lsp-treemacs--set-mode-line-format search-buffer " Rendering results... ") (lsp-with-cached-filetrue-name (let ((lsp-file-truename-cache (ht))) (lsp-treemacs-render (lsp-treemacs--handle-references refs) (format title (length refs)) (and prefix-args (not (equal prefix-args 0)))))) (lsp--info "Refresh completed!")) :mode 'detached :cancel-token :treemacs-lookup) (unless (equal prefix-args 0) (select-window window) ;; (set-window-dedicated-p window t) ) (with-current-buffer search-buffer (lsp-treemacs-initialize) (lsp-treemacs--set-mode-line-format search-buffer " Loading... ") (setq-local lsp-treemacs-tree nil) (lsp-treemacs-generic-refresh)))) ;;;###autoload (defun lsp-treemacs-references (arg) "Show the references for the symbol at point. With a prefix argument, select the new window and expand the tree of references automatically." (interactive "P") (lsp-treemacs--do-search "textDocument/references" `(:context (:includeDeclaration t) ,@(lsp--text-document-position-params)) " Found %s references " arg)) ;;;###autoload (defun lsp-treemacs-implementations (arg) "Show the implementations for the symbol at point. With a prefix argument, select the new window expand the tree of implementations automatically." (interactive "P") (lsp-treemacs--do-search "textDocument/implementation" (lsp--text-document-position-params) " Found %s implementations " arg)) ;; Call hierarchy. (defun lsp-treemacs--call-hierarchy-children (buffer method outgoing node callback) (-let [item (plist-get node :item)] (with-current-buffer buffer (lsp-request-async method (list :item item) (lambda (result) (funcall callback (seq-map (-lambda (node) (-let* (((child-item &as &CallHierarchyItem :name :kind :detail? :uri :selection-range (&Range :start)) (if outgoing (lsp:call-hierarchy-outgoing-call-to node) (lsp:call-hierarchy-incoming-call-from node))) (label (concat name (when detail? (propertize (concat " - " detail?) 'face 'lsp-lens-face))))) (list :label label :key label :icon (lsp-treemacs-symbol-kind->icon kind) :children-async (-partial #'lsp-treemacs--call-hierarchy-children buffer method outgoing) :ret-action (lambda (&rest _) (interactive) (lsp-treemacs--open-file-in-mru (lsp--uri-to-path uri)) (goto-char (lsp--position-to-point start)) (run-hooks 'xref-after-jump-hook)) :item child-item))) result))) :mode 'detached)))) ;;;###autoload (defun lsp-treemacs-call-hierarchy (outgoing) "Show the incoming call hierarchy for the symbol at point. With a prefix argument, show the outgoing call hierarchy." (interactive "P") (unless (lsp--find-workspaces-for "textDocument/prepareCallHierarchy") (user-error "Call hierarchy not supported by the current servers: %s" (-map #'lsp--workspace-print (lsp-workspaces)))) (let ((buffer (current-buffer))) (select-window (display-buffer-in-side-window (lsp-treemacs-render (seq-map (-lambda ((item &as &CallHierarchyItem :name :kind :detail?)) (list :label (concat name (when detail? (propertize (concat " - " detail?) 'face 'lsp-lens-face))) :key name :icon (lsp-treemacs-symbol-kind->icon kind) :children-async (-partial #'lsp-treemacs--call-hierarchy-children buffer (if outgoing "callHierarchy/outgoingCalls" "callHierarchy/incomingCalls") outgoing) :item item)) (lsp-request "textDocument/prepareCallHierarchy" (lsp--text-document-position-params))) (concat (if outgoing "Outgoing" "Incoming") " Call Hierarchy") nil "*Call Hierarchy*") nil)))) ;; Type hierarchy. (defconst lsp-treemacs--hierarchy-sub 0) (defconst lsp-treemacs--hierarchy-super 1) (defconst lsp-treemacs--hierarchy-both 2) (defun lsp-treemacs--type-hierarchy-render-nodes (result loaded? &optional direction) (-map (-lambda ((it &as &TypeHierarchyItem :name :kind :uri :range (&Range :start) :children? :parents?)) `(:label ,(concat name (cond ((eq lsp-treemacs--hierarchy-sub direction) (propertize " ↓" 'face 'shadow)) ((eq lsp-treemacs--hierarchy-super direction) (propertize " ↑" 'face 'shadow)))) :key ,name :icon ,(lsp-treemacs-symbol-kind->icon kind) ,@(if loaded? (list :children (append (lsp-treemacs--type-hierarchy-render-nodes children? nil lsp-treemacs--hierarchy-sub) (lsp-treemacs--type-hierarchy-render-nodes parents? nil lsp-treemacs--hierarchy-super))) (list :children-async (-partial #'lsp-treemacs--type-hierarchy-render it direction))) :ret-action ,(lambda (&rest _) (interactive) (lsp-treemacs--open-file-in-mru (lsp--uri-to-path uri)) (goto-char (lsp--position-to-point start)) (run-hooks 'xref-after-jump-hook)))) result)) (lsp-defun lsp-treemacs--type-hierarchy-render ((&TypeHierarchyItem :uri :range (&Range :start)) direction _ callback) (lsp-request-async "textDocument/typeHierarchy" (lsp-make-type-hierarchy-params :text-document (lsp-make-text-document-item :uri uri) :position start :direction direction :resolve 1) (-lambda ((&TypeHierarchyItem :children? :parents?)) (funcall callback (lsp-treemacs--type-hierarchy-render-nodes (if (eq direction lsp-treemacs--hierarchy-sub) children? parents?) nil direction))))) (defun lsp-treemacs-type-hierarchy (direction) "Show the type hierarchy for the symbol at point. With prefix 0 show sub-types. With prefix 1 show super-types. With prefix 2 show both." (interactive "P") (unless (lsp--find-workspaces-for "textDocument/typeHierarchy") (user-error "Type hierarchy not supported by the current servers: %s" (-map #'lsp--workspace-print (lsp-workspaces)))) (setq direction (or direction 0)) (let ((workspaces (lsp-workspaces)) (result (lsp-request "textDocument/typeHierarchy" (-> (lsp--text-document-position-params) (plist-put :direction direction) (plist-put :resolve 1))))) (if result (pop-to-buffer (lsp-treemacs-render (lsp-treemacs--type-hierarchy-render-nodes (vector result) t) (concat (cond ((eq lsp-treemacs--hierarchy-sub direction) "Sub") ((eq lsp-treemacs--hierarchy-super direction) "Super") ((eq lsp-treemacs--hierarchy-both direction) "Sub/Super")) " Type Hierarchy") nil "*lsp-treemacs-call-hierarchy*")) (user-error "No class under point.")) (setq lsp--buffer-workspaces workspaces))) (provide 'lsp-treemacs) ;;; lsp-treemacs.el ends here ;; Local Variables: ;; flycheck-disabled-checkers: (emacs-lisp-checkdoc) ;; End: