You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

223 lines
7.4 KiB

;;; examples-to-docs.el --- Extract dash.el's doc from examples.el -*- lexical-binding: t -*-
;; Copyright (C) 2012-2021 Free Software Foundation, Inc.
;; 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:
;; FIXME: Lots of duplication with examples-to-info.el.
;;; Code:
(require 'dash)
(require 'help-fns)
(require 'lisp-mnt)
(defvar functions ())
(defun example-to-string (example)
(-let (((actual sym expected) example)
(print-quoted t))
(--> (cond
((eq sym '=>) (format "=> %S" expected))
((eq sym '~>) (format "~> %S" expected))
((eq sym '!!>) "Error")
((error "Invalid test case: %S" example)))
(format "%S ;; %s" actual it)
(replace-regexp-in-string "\\\\\\?" "?" it t t)
(replace-regexp-in-string
"[^\n[:print:]]"
(lambda (s) (concat "\\" (text-char-description (string-to-char s))))
it t t))))
(defun dash--describe (fn)
"Return the (ARGLIST DOCSTRING) of FN symbol.
Based on `describe-function-1'."
(with-temp-buffer
(pcase-let* ((text-quoting-style 'grave)
(`(,real-fn ,def ,_alias ,real-def)
(help-fns--analyze-function fn))
(buf (current-buffer))
(doc-raw (documentation fn t))
(doc (help-fns--signature fn doc-raw real-def real-fn buf)))
(goto-char (1+ (point-min)))
(delete-region (point) (progn (forward-sexp) (1+ (point))))
(downcase-region (point) (point-max))
(backward-char)
(list (read buf) doc))))
(defmacro defexamples (cmd &rest examples)
`(push (cons ',cmd
(nconc (dash--describe ',cmd)
(list (mapcar #'example-to-string
(-partition 3 ',examples)))))
functions))
(defmacro def-example-group (group desc &rest examples)
`(progn
(push ,(concat "### " group) functions)
(when ,desc
(push ,desc functions))
,@examples))
(defun format-link (string-name)
(-let* ((name (intern string-name))
((_ signature _ _) (assoc name functions)))
(if signature
(format "[`%s`](#%s)" name (github-id name signature))
(format "`%s`" name))))
(defun dash--quote-argnames ()
"Downcase and quote arg names in current buffer for Markdown."
(let ((beg (point-min)))
(while (setq beg (text-property-any beg (point-max)
'face 'help-argument-name))
(goto-char beg)
(insert ?`)
(goto-char (or (next-single-property-change (point) 'face)
(point-max)))
(downcase-region (1+ beg) (point))
(insert ?`)
(setq beg (point)))))
(defun dash--quote-metavars ()
"Downcase and quote metavariables in current buffer for Markdown."
(goto-char (point-min))
(while (re-search-forward (rx bow (group (in upper) (* (in upper ?-)) (* num))
(| (group ?\() (: (group (? "th")) eow)))
nil t)
(unless (match-beginning 2)
(let* ((suf (match-string 3))
(var (format "`%s`%s" (downcase (match-string 1)) suf)))
(replace-match var t t)))))
(defun dash--quote-hyperlinks ()
"Convert hyperlinks in current buffer from Elisp to Markdown."
(goto-char (point-min))
(while (re-search-forward (rx ?` (+? (not (in " `"))) ?\') nil t)
(replace-match (format-link (substring (match-string 0) 1 -1)) t t)))
(defun dash--indent-blocks ()
"Indent example blocks in current buffer for Markdown."
(goto-char (point-min))
(while (re-search-forward (rx bol " ") nil t)
(replace-match " " t t)))
(defun dash--format-docstring (docstring)
(with-temp-buffer
(let ((case-fold-search nil))
(insert docstring)
(dash--quote-argnames)
(dash--quote-metavars)
(dash--quote-hyperlinks)
(dash--indent-blocks)
(buffer-string))))
(defun function-to-md (function)
(if (stringp function)
(concat "\n" (s-replace "### " "## " function) "\n")
(-let [(command-name signature docstring examples) function]
(format "#### %s `%s`\n\n%s\n\n```el\n%s\n```\n"
command-name
signature
(dash--format-docstring docstring)
(mapconcat #'identity (-take 3 examples) "\n")))))
(defun docs--chop-prefix (prefix s)
"Remove PREFIX if it is at the start of S."
(let ((pos (length prefix)))
(if (and (>= (length s) (length prefix))
(string= prefix (substring s 0 pos)))
(substring s pos)
s)))
(defun docs--chop-suffix (suffix s)
"Remove SUFFIX if it is at end of S."
(let ((pos (- (length suffix))))
(if (and (>= (length s) (length suffix))
(string= suffix (substring s pos)))
(substring s 0 pos)
s)))
(defun github-id (command-name signature)
(docs--chop-suffix
"-"
(replace-regexp-in-string "[^a-zA-Z0-9-]+" "-" (docs--chop-prefix
"!"
(format "%S %S" command-name signature)))))
(defun s-replace (old new s)
"Replace OLD with NEW in S."
(replace-regexp-in-string (regexp-quote old) new s t t))
(defun function-summary (function)
(if (stringp function)
(concat "\n" function "\n")
(let ((command-name (car function))
(signature (cadr function)))
(format "* [%s](#%s) `%s`" command-name (github-id command-name signature) signature))))
(defun simplify-quotes ()
(goto-char (point-min))
(while (re-search-forward (rx (or "'nil" "(quote nil)")) nil t)
(replace-match "'()" t t))
(goto-char (point-min))
(while (search-forward "(quote " nil t)
(forward-char -7)
(let ((p (point)))
(forward-sexp 1)
(delete-char -1)
(goto-char p)
(delete-char 7)
(insert "'")))
(goto-char (point-min))
(while (search-forward "(function " nil t)
(forward-char -10)
(let ((p (point)))
(forward-sexp 1)
(delete-char -1)
(goto-char p)
(delete-char 10)
(insert "#'"))))
(defun goto-and-remove (s)
(goto-char (point-min))
(search-forward s)
(delete-char (- (length s))))
(defun goto-and-replace-all (s replacement)
(while (progn (goto-char (point-min)) (search-forward s nil t))
(delete-char (- (length s)))
(insert replacement)))
(defun create-docs-file ()
(let ((functions (nreverse functions)))
(with-temp-file "./README.md"
(insert-file-contents "./readme-template.md")
(goto-and-remove "[[ function-list ]]")
(insert (mapconcat 'function-summary functions "\n"))
(goto-and-remove "[[ function-docs ]]")
(insert (mapconcat 'function-to-md functions "\n"))
(dolist (pkg '(dash dash-functional))
(goto-and-replace-all (format "[[ %s-version ]]" pkg)
(lm-version (format "%s.el" pkg))))
(simplify-quotes))))
;;; examples-to-docs.el ends here