;;; org-ref-ref-links.el --- cross-reference links for org-ref -*- lexical-binding: t; -*- ;; ;; Copyright (C) 2021-2024 John Kitchin ;; Author: John Kitchin ;; Keywords: convenience ;; 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: ;; ;;; Code: (eval-and-compile (require 'org-macs)) (eval-and-compile (require 'ol)) (require 'hydra) (defcustom org-ref-default-ref-type "ref" "Default ref link type to use when inserting ref links." :type 'string :group 'org-ref) (defcustom org-ref-activate-ref-links t "If non-nil use font lock to activate ref links. Activation can be slow in large documents with a lot of ref links. Set this to nil to turn off activation." :type 'boolean :group 'org-ref) (defface org-ref-ref-face `((t (:inherit org-link :foreground "dark red"))) "Face for ref links in org-ref." :group 'org-ref-faces) (defvar org-ref-label-re (rx-to-string '(group-n 1 (one-or-more (any word "-.:?!`'/*@+|(){}<>&_^$#%~")))) "Regexp for labels.") (defvar org-ref-label-link-re (rx-to-string `(seq "label:" (group-n 1 (one-or-more (any word "-.:?!`'/*@+|(){}<>&_^$#%~"))))) "Regexp for label links.") (defvar org-ref-ref-label-regexps (list (concat ":ID:\\s-+" org-ref-label-re "\\_>") ;; CUSTOM_ID in a heading (concat ":CUSTOM_ID:\\s-+" org-ref-label-re "\\_>") ;; #+name (concat "^\\s-*#\\+name:\\s-+" org-ref-label-re "\\_>") ;; labels in latex (concat "\\\\label{" org-ref-label-re "}") ;; A target, code copied from org-target-regexp and group 1 numbered. (let ((border "[^<>\n\r \t]")) (format "<<\\(?1:%s\\|%s[^<>\n\r]*%s\\)>>" border border border)) ;; A label link (concat "label:" org-ref-label-re "\\_>") "\\\\lstset{.*label=\\(?1:.*?\\),.*}") "List of regular expressions to labels. The label should always be in group 1.") (defvar org-ref-ref-types '(("ref" "A regular cross-reference to a label") ("eqref" "A cross-reference to an equation") ("pageref" "to the page number a label is on") ("nameref" "to the name associated with a label (e.g. a caption)") ("autoref" "from hyperref, adds automatic prefixes") ("Autoref" "from hyperref, capitalized version of autoref") ("cref" "from cleveref, adds automatic prefixes, and condenses multiple refs") ("Cref" "from cleveref, capitalized version of cref") ("crefrange" "from cleveref, makes a range of refs from two refs with a prefix") ("Crefrange" "from cleveref, capitalized prefix version of crefrange")) "List of ref link types (type description).") (defun org-ref-select-ref-type () "Select a ref type with annotated completion." (let* ((type-annotation (lambda (s) (let ((item (assoc s minibuffer-completion-table))) (when item (concat (make-string (- 12 (length s)) ? ) "-- " (cl-second item)))))) (completion-extra-properties `(:annotation-function ,type-annotation))) (completing-read "Type: " org-ref-ref-types))) (defun org-ref-change-ref-type (new-type) "Change the ref type to NEW-TYPE." (interactive (list (org-ref-select-ref-type))) (let* ((cite-link (org-element-context)) (old-type (org-element-property :type cite-link)) (begin (org-element-property :begin cite-link)) (end (org-element-property :end cite-link)) (bracketp (eq 'bracket (org-element-property :format cite-link))) (path (org-element-property :path cite-link)) (deltap (- (point) begin))) ;; note this does not respect brackets (cl--set-buffer-substring begin end (concat (if bracketp "[[" "") new-type ":" path (if bracketp "]]" ""))) ;; try to preserve the character the point is on. (goto-char (+ begin deltap (- (length new-type) (length old-type)))))) (defvar-local org-ref-label-cache nil "Buffer-local cache variable for labels.") (defvar-local org-ref-buffer-chars-modified-tick nil "Buffer-local variable to hold `buffer-chars-modified-tick'.") (defun org-ref-get-labels () "Return a list of referenceable labels in the document. You can reference: A NAME keyword A CUSTOM_ID property on a heading A LaTeX label A target. A label link A setting in lstset See `org-ref-ref-label-regexps' for the patterns that find these. Returns a list of cons cells (label . context). It is important for this function to be fast, since we use it in font-lock." (if (or ;; if we have not checked we have to check (null org-ref-buffer-chars-modified-tick) ;; Now check if buffer has changed since last time we looked. We check ;; this with the buffer-chars-modified-tick which keeps track of changes. ;; If this hasn't changed, no chars have been modified. (not (= (buffer-chars-modified-tick) org-ref-buffer-chars-modified-tick))) ;; We need to search for all the labels either because we don't have them, ;; or the buffer has changed since we looked last time. (let ((case-fold-search t) (rx (string-join org-ref-ref-label-regexps "\\|")) (labels '()) oe ;; org-element context) (save-excursion (org-with-wide-buffer (goto-char (point-min)) (while (re-search-forward rx nil t) (save-match-data ;; Here we try to get some relevant context for different things you ;; might reference. (setq oe (org-element-context) context (string-trim (pcase (car oe) ('latex-environment (buffer-substring (org-element-property :begin oe) (org-element-property :end oe))) ;; figure ('paragraph (buffer-substring (org-element-property :begin oe) (org-element-property :end oe))) ('table (buffer-substring (org-element-property :begin oe) (org-element-property :end oe))) ;; Headings fall here. (_ (buffer-substring (line-beginning-position) (line-end-position))))))) (cl-pushnew (cons (match-string-no-properties 1) context) labels)))) ;; reverse so they are in the order we find them. (setq org-ref-buffer-chars-modified-tick (buffer-chars-modified-tick) org-ref-label-cache (delete-dups (reverse labels)))) ;; retrieve the cached data org-ref-label-cache)) (defun org-ref-ref-jump-to (&optional path) "Jump to the target for the ref link at point." (interactive) (let ((case-fold-search t) (label (get-text-property (point) 'org-ref-ref-label)) (labels (split-string path ",")) (rx (string-join org-ref-ref-label-regexps "\\|"))) (when (null label) (pcase (length labels) (1 (setq label (cl-first labels))) (_ (setq label (completing-read "Label: " labels))))) (when label (org-mark-ring-push) (widen) (goto-char (point-min)) (catch 'found (while (re-search-forward rx) (when (string= label (match-string-no-properties 1)) (save-match-data (org-mark-ring-push)) (goto-char (match-beginning 1)) (org-fold-show-entry) (substitute-command-keys "Go back with (org-mark-ring-goto) \`\\[org-mark-ring-goto]'.") (throw 'found t))))))) (defun org-ref-ref-help-echo (_win _obj position) "Tooltip for context on a ref label. POSITION is the point under the mouse I think." (cdr (assoc (get-text-property position 'org-ref-ref-label) (org-ref-get-labels)))) (defun org-ref-ref-activate (start _end path _bracketp) "Activate a ref link. The PATH should be a comma-separated list of labels. Argument START is the start of the link. Argument END is the end of the link." (when org-ref-activate-ref-links (let ((labels (mapcar 'car (org-ref-get-labels)))) (goto-char start) (cl-loop for label in (split-string path ",") do (search-forward label) ;; store property so we can follow it later. (put-text-property (match-beginning 0) (match-end 0) 'org-ref-ref-label label) (unless (member label labels) (put-text-property (match-beginning 0) (match-end 0) 'face 'font-lock-warning-face) (put-text-property (match-beginning 0) (match-end 0) 'help-echo "Label not found")))))) (defun org-ref-ref-export (cmd keyword _desc backend) "An export function for ref links. Argument CMD is the LaTeX command to export to. Argument KEYWORD is the path of the ref link. Argument BACKEND is the export backend. This is meant to be used with `apply-partially' in the link definitions." (cond ((eq backend 'latex) (format "\\%s{%s}" cmd keyword)))) (defun org-ref-complete-link (refstyle &optional _arg) "Complete a ref link to an existing label." (concat refstyle ":" (completing-read "Label: " (org-ref-get-labels)))) (defun org-ref-store-ref-link (&optional reftype) "Store a ref link to a label. The output will be a ref to that label." ;; First we have to make sure we are on a label link. (unless reftype (setq reftype "ref")) (let* ((object (and (eq major-mode 'org-mode) (org-element-context))) (label (cond ;; here literally on a label link. ((and (equal (org-element-type object) 'link) (equal (org-element-property :type object) "label")) (org-element-property :path object)) ;; here on a file link. if it has a caption with a label in it, we store ;; it. ((and (equal (org-element-type object) 'link) (equal (org-element-property :type object) "file") (org-file-image-p (org-element-property :path object))) (if (org-element-property :name object) (org-element-property :name object) ;; maybe we have a caption to get it from. (let* ((parent (org-element-property :parent object))) (when (and parent (equal (org-element-type parent) 'paragraph)) (if (org-element-property :name parent) ;; caption paragraph may have a name which we use if it is there (org-element-property :name parent) ;; else search caption (let ((caption (s-join "" (mapcar 'org-no-properties (org-export-get-caption parent))))) (when (string-match org-ref-label-re caption) (match-string 1 caption)))))))) ;; here on a paragraph (eg in a caption of an image). it is a paragraph with a caption ;; in a caption, with no name, but maybe a label ((equal (org-element-type object) 'paragraph) (if (org-element-property :name object) (org-element-property :name object) ;; See if it is in the caption name (let ((caption (s-join "" (mapcar 'org-no-properties (org-export-get-caption object))))) (when (string-match org-ref-label-re caption) (match-string 1 caption))))) ;; If you are in a table, we need to be at the beginning to ;; make sure we get the name. Note when in a caption it appears ;; you are in a table but org-at-table-p is nil there. ((or (equal (org-element-type object) 'table) (org-at-table-p)) (save-excursion (goto-char (org-table-begin)) (let* ((table (org-element-context)) (label (org-element-property :name table)) (caption (s-join "" (mapcar 'org-no-properties (org-export-get-caption table))))) (when (null label) ;; maybe there is a label in the caption? (when (string-match org-ref-label-link-re caption) (match-string 1 caption)))))) ;; and to #+namel: lines ((and (equal (org-element-type object) 'paragraph) (org-element-property :name object)) (org-element-property :name object)) ;; in a latex environment ((equal (org-element-type object) 'latex-environment) (let ((value (org-element-property :value object))) (when (string-match "\\\\label{\\(?1:[+a-zA-Z0-9:\\._-]*\\)}" value) (match-string-no-properties 1 value)))) ;; Match targets, like <