;;; geiser-edit.el -- scheme edit locations -*- lexical-binding: t; -*- ;; Copyright (C) 2009, 2010, 2012, 2013, 2019-2024 Jose Antonio Ortega Ruiz ;; This program is free software; you can redistribute it and/or ;; modify it under the terms of the Modified BSD License. You should ;; have received a copy of the license along with this program. If ;; not, see . ;; Start date: Wed Feb 11, 2009 21:07 ;;; Code: (require 'geiser-completion) (require 'geiser-eval) (require 'geiser-custom) (require 'geiser-base) (require 'etags) (eval-when-compile (require 'subr-x)) ;;; Customization: (defmacro geiser-edit--define-custom-visit (var group doc) `(geiser-custom--defcustom ,var nil ,doc :group ',group :type '(choice (const :tag "Other window" window) (const :tag "Other frame" frame) (const :tag "Current window" nil)))) (geiser-edit--define-custom-visit geiser-edit-symbol-method geiser-mode "How the new buffer is opened when invoking \\[geiser-edit-symbol-at-point] or following links in error buffers.") (defgroup geiser-edit nil "Customizations for scheme buffers and information about them." :group 'geiser) (geiser-custom--defface error-link 'link geiser-edit "links in error buffers") (geiser-custom--defcustom geiser-insert-actual-lambda t "Whether geiser-insert-lambda should insert \"λ\" or \"lambda\"." :type 'boolean) ;;; Auxiliary functions: (defun geiser-edit--visit-file (file method) (cond ((eq method 'window) (pop-to-buffer (find-file-noselect file t))) ((eq method 'frame) (find-file-other-frame file)) ((eq method 'noselect) (find-file-noselect file t)) (t (find-file file)))) (defsubst geiser-edit--location-name (loc) (cdr (assoc "name" loc))) (defsubst geiser-edit--location-file (loc) (when-let ((file-name (cdr (assoc "file" loc)))) (concat (or (file-remote-p default-directory) "") file-name))) (defsubst geiser-edit--to-number (x) (cond ((numberp x) x) ((stringp x) (string-to-number x)))) (defsubst geiser-edit--location-line (loc) (geiser-edit--to-number (cdr (assoc "line" loc)))) (defsubst geiser-edit--location-column (loc) (geiser-edit--to-number (cdr (assoc "column" loc)))) (defsubst geiser-edit--location-char (loc) (geiser-edit--to-number (cdr (assoc "char" loc)))) (defsubst geiser-edit--make-location (name file line column) (if (equal line "") `(("name" . ,name) ("file" . ,file) ("char" . ,column)) `(("name" . ,name) ("file" . ,file) ("line" . ,line) ("column" . ,column)))) (defconst geiser-edit--def-re (regexp-opt '("define" "defmacro" "define-macro" "define-syntax" "define-syntax-rule" "-define-syntax" "-define" "define*" "define-method" "define-class" "define-struct"))) (defconst geiser-edit--def-re* (regexp-opt '("define-syntaxes" "define-values"))) (defsubst geiser-edit--def-re (thing) (let ((sx (regexp-quote (format "%s" thing)))) (format (concat "(%s[[:space:]]+\\(" "(%s\\_>[^)]*)\\|" "\\(\\_<%s\\_>\\) *\\([^\n]*?\\)[)\n]" "\\)") geiser-edit--def-re sx sx))) (defsubst geiser-edit--def-re* (thing) (format "(%s +([^)]*?\\_<%s\\_>" geiser-edit--def-re* (regexp-quote (format "%s" thing)))) (defun geiser-edit--find-def (symbol &optional args) (save-excursion (goto-char (point-min)) (when (or (re-search-forward (geiser-edit--def-re symbol) nil t) (re-search-forward (geiser-edit--def-re* symbol) nil t)) (cons (match-beginning 0) (and args (if (match-string 2) (let* ((v (or (match-string 3) "")) (v (and (not (string-blank-p v)) v))) (concat (match-string 2) (and v " => ") v (and v (string-prefix-p "(" v) " ..."))) (match-string 1))))))) (defsubst geiser-edit--symbol-re (thing) (format "\\_<%s\\_>" (regexp-quote (format "%s" thing)))) (defun geiser-edit--goto-location (symbol line col pos) (cond ((numberp line) (goto-char (point-min)) (forward-line (max 0 (1- line)))) ((numberp pos) (goto-char pos))) (if (not col) (when-let (pos (car (geiser-edit--find-def symbol))) (goto-char pos)) (beginning-of-line) (forward-char col) (cons (current-buffer) (point)))) (defun geiser-edit--try-edit-location (symbol loc &optional method no-error) (let ((symbol (or (geiser-edit--location-name loc) symbol)) (file (geiser-edit--location-file loc)) (line (geiser-edit--location-line loc)) (col (geiser-edit--location-column loc)) (pos (geiser-edit--location-char loc))) (when file (geiser-edit--visit-file file (or method geiser-edit-symbol-method))) (or (geiser-edit--goto-location symbol line col pos) file (unless no-error (error "Couldn't find location for '%s'" symbol))))) (defsubst geiser-edit--try-edit (symbol ret &optional method no-error) (let ((res (geiser-eval--retort-result ret))) (if (listp res) (geiser-edit--try-edit-location symbol res method no-error) (unless no-error (error "Couldn't find location for '%s'" symbol))))) ;;; Links (define-button-type 'geiser-edit--button 'action 'geiser-edit--button-action 'face 'geiser-font-lock-error-link 'follow-link t) (defun geiser-edit--button-action (button) (let ((loc (button-get button 'geiser-location)) (method (button-get button 'geiser-method))) (when loc (geiser-edit--try-edit-location nil loc method)))) (defun geiser-edit--make-link (beg end file line col &optional method) (make-button beg end :type 'geiser-edit--button 'geiser-method method 'geiser-location (geiser-edit--make-location 'error file line col) 'help-echo "Go to error location")) (defconst geiser-edit--default-file-rx "^[ \t]*\\([^<>:\n\"]+\\):\\([0-9]+\\)\\(?::\\([0-9]+\\)\\)?") (defun geiser-edit--buttonize-files (&optional rx no-fill) (let ((rx (or rx geiser-edit--default-file-rx)) (fill-column (- (window-width) 2))) (save-excursion (while (re-search-forward rx nil t) (geiser-edit--make-link (match-beginning 1) (match-end 1) (match-string 1) (match-string 2) (or (match-string 3) 0) 'window) (unless no-fill (fill-region (match-end 0) (line-end-position))))))) (defun geiser-edit--open-next (&optional n reset) (interactive) (let* ((n (or n 1)) (nxt (if (< n 0) 'backward-button 'forward-button)) (msg (if (< n 0) "previous" "next")) (n (abs n)) (p (point)) (found nil)) (when reset (goto-char (point-min))) (while (> n 0) (let ((b (ignore-errors (funcall nxt 1)))) (unless b (setq n 0)) (when (and b (eq (button-type b) 'geiser-edit--button)) (setq n (- n 1)) (when (<= n 0) (setq found t) (push-button (point)))))) (unless found (goto-char p) (error "No %s error" msg)))) ;;; Visibility (defun geiser-edit--cloak (form) (intern (format "geiser-edit-cloak-%s" form))) (defun geiser-edit--hide (form) (geiser-edit--show form) (let ((cloak (geiser-edit--cloak form))) (save-excursion (goto-char (point-min)) (while (re-search-forward (format "(%s\\b" form) nil t) (let* ((beg (match-beginning 0)) (end (progn (ignore-errors (goto-char beg) (forward-sexp)) (point)))) (when (> end beg) (overlay-put (make-overlay beg end) 'invisible cloak))))) (add-to-invisibility-spec (cons cloak t)))) (defun geiser-edit--show (form) (let ((cloak (geiser-edit--cloak form))) (remove-overlays nil nil 'invisible cloak) (remove-from-invisibility-spec (cons cloak t)))) (defun geiser-edit--show-all () (remove-overlays) (setq buffer-invisibility-spec '(t))) (defun geiser-edit--toggle-visibility (form) (if (and (listp buffer-invisibility-spec) (assoc (geiser-edit--cloak form) buffer-invisibility-spec)) (geiser-edit--show form) (geiser-edit--hide form))) ;;; Commands: (defvar geiser-edit--symbol-history nil) (defun geiser-edit-symbol (symbol &optional method marker) "Asks for a symbol to edit, with completion." (interactive (list (geiser-completion--read-symbol "Edit symbol: " nil geiser-edit--symbol-history))) (let ((cmd `(:eval (:ge symbol-location ',symbol)))) (geiser-edit--try-edit symbol (geiser-eval--send/wait cmd) method) (when marker (xref-push-marker-stack)))) (defun geiser-edit-symbol-at-point (&optional arg) "Visit the definition of the symbol at point. With prefix, asks for the symbol to locate." (interactive "P") (let* ((symbol (or (and (not arg) (geiser--symbol-at-point)) (geiser-completion--read-symbol "Edit symbol: "))) (cmd `(:eval (:ge symbol-location ',symbol))) (marker (point-marker)) (ret (ignore-errors (geiser-eval--send/wait cmd)))) (if (geiser-edit--try-edit symbol ret nil t) (when marker (xref-push-marker-stack marker)) (unless (geiser-edit-module-at-point t) (error "Couldn't find location for '%s'" symbol))) t)) (defun geiser-pop-symbol-stack () "Pop back to where \\[geiser-edit-symbol-at-point] was last invoked." (interactive) (if (fboundp 'xref-go-back) (xref-go-back) (with-no-warnings (xref-pop-marker-stack)))) (defun geiser-edit-module (module &optional method no-error) "Asks for a module and opens it in a new buffer." (interactive (list (geiser-completion--read-module))) (let ((cmd `(:eval (:ge module-location '(:module ,module))))) (geiser-edit--try-edit module (geiser-eval--send/wait cmd) method no-error))) (defun geiser-edit-module-at-point (&optional no-error) "Opens a new window visiting the module at point." (interactive) (let ((marker (point-marker))) (when (geiser-edit-module (or (geiser-completion--module-at-point) (geiser-completion--read-module)) nil no-error) (when marker (xref-push-marker-stack marker)) t))) (defun geiser-insert-lambda (&optional full) "Insert λ or lambda at point. With prefix, inserts (λ ()) or (lambda ()). See also `geiser-insert-actual-lambda'." (interactive "P") (let ((sym (if geiser-insert-actual-lambda (make-char 'greek-iso8859-7 107) "lambda"))) (if (not full) (insert sym) (insert "(" sym " ())") (backward-char 2)))) (defun geiser-squarify (n) "Toggle between () and [] for current form. With numeric prefix, perform that many toggles, forward for positive values and backward for negative." (interactive "p") (let ((pared (and (boundp 'paredit-mode) paredit-mode)) (fwd (> n 0)) (steps (abs n))) (when (and pared (fboundp 'paredit-mode)) (paredit-mode -1)) (unwind-protect (save-excursion (unless (looking-at-p "\\s(") (backward-up-list)) (while (> steps 0) (let ((p (point)) (round (looking-at-p "("))) (forward-sexp) (delete-char -1) (insert (if round "]" ")")) (goto-char p) (delete-char 1) (insert (if round "[" "(")) (setq steps (1- steps)) (backward-char) (condition-case nil (progn (when fwd (forward-sexp 2)) (backward-sexp)) (error (setq steps 0)))))) (when (and pared (fboundp 'paredit-mode)) (paredit-mode 1))))) (provide 'geiser-edit)