File: racket-scribble-anchor.el

package info (click to toggle)
racket-mode 20251013~git.b9a4f51-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 2,036 kB
  • sloc: lisp: 17,282; makefile: 106
file content (113 lines) | stat: -rw-r--r-- 4,382 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
;;; racket-scribble-anchor.el -*- lexical-binding: t -*-

;; Copyright (c) 2022-2024 by Greg Hendershott.
;; Portions Copyright (C) 1985-1986, 1999-2013 Free Software Foundation, Inc.

;; Author: Greg Hendershott
;; URL: https://github.com/greghendershott/racket-mode

;; SPDX-License-Identifier: GPL-3.0-or-later

(require 'cl-macs)
(require 'ring)
(require 'seq)
(require 'shr)
(require 'racket-back-end)
(require 'racket-describe)
(require 'racket-scribble)

(defun racket--company-doc-buffer (how str)
  (pcase (racket--cmd/await (racket--repl-session-id)
                            `(describe ,(racket-how-front-to-back how) ,str))
    (`(,(and path (pred stringp)) . ,anchor)
     (let ((path (racket-file-name-back-to-front path))
           (name "*racket-company-doc-buffer*"))
       (when-let (buf (get-buffer name))
         (when (buffer-live-p buf)
           (kill-buffer buf)))
       (with-current-buffer (get-buffer-create name)
         (goto-char (point-min))
         (racket--scribble-path+anchor-insert path anchor)
         (goto-char (point-min))
         (setq buffer-read-only t)
         (current-buffer))))))

(defvar racket--path+anchor-ring (make-ring 16)
  "A small MRU cache of the N most recent strings.
Each ring item is (cons (cons path anchor) str).")

(defun racket--path+anchor->string (path anchor)
  "A wrapper for `racket--scribble-path+anchor-insert'.
Uses `racket--path+anchor-cache'."
  (pcase (seq-some (lambda (item)
                     (and (equal (car item) (cons path anchor))
                          item))
                   (ring-elements racket--path+anchor-ring))
    ((and `(,_path+anchor . ,str) item)
     ;; Re-insert as newest.
     (ring-remove+insert+extend racket--path+anchor-ring item)
     str)
    (_
     (let* ((str (with-temp-buffer
                   (racket--scribble-path+anchor-insert path anchor)
                   (buffer-string)))
            (item (cons (cons path anchor) str)))
       ;; Insert as newest; oldest discarded when ring full.
       (ring-insert racket--path+anchor-ring item)
       str))))

(defun racket--scribble-path+anchor-insert (path anchor)
  (let* ((tramp-verbose 2) ;avoid excessive tramp messages
         (dom (racket--html-file->dom path))
         (dom (racket--elements-for-anchor dom anchor))
         (dom (racket--massage-scribble-dom path
                                            (file-name-directory path)
                                            dom)))
    (ignore tramp-verbose)
    (save-excursion
      (let ((shr-use-fonts nil)
            (shr-external-rendering-functions `((span . ,#'racket-render-tag-span)))
            (shr-width 76))
        (shr-insert-document dom)))
    (while (re-search-forward (string racket--scribble-temp-nbsp) nil t)
      (replace-match " " t t))))

(defun racket--elements-for-anchor (dom anchor)
  "Return the subset of DOM elements pertaining to ANCHOR."
  (cl-flet
      ((anchor-p (node name)
         (dom-search node
                     (lambda (node)
                       (and (eq 'a (dom-tag node))
                            (equal name (dom-attr node 'name))))))
       (boxed-p (node)
         (dom-search node
                     (lambda (node)
                       (and (eq 'table (dom-tag node))
                            (equal "boxed RBoxed" (dom-attr node 'class))))))
       (heading-p (node)
         (memq (dom-tag node) '(h1 h2 h3 h4 h5 h6))))
    ;; Consider immediate children of the "main" div.
    (let ((result nil)
          (xs (dom-children
               (dom-search (dom-child-by-tag dom 'body)
                           (lambda (node)
                             (and (eq 'div (dom-tag node))
                                  (equal "main" (dom-attr node 'class))))))))
      ;; Discard elements before the one containing a matching anchor.
      (while (and xs (not (anchor-p (car xs) anchor)))
        (setq xs (cdr xs)))
      ;; Accumulate result up to an element containing an RBoxed table
      ;; or heading.
      (when xs
        (push (car xs) result)
        (setq xs (cdr xs))
        (while (and xs (not (or (heading-p (car xs))
                                (boxed-p (car xs)))))
          (push (car xs) result)
          (setq xs (cdr xs))))
      `(div () ,@(reverse result)))))

(provide 'racket-scribble-anchor)

;; racket-scribble-anchor.el ends here