shithub: sl

Download patch

ref: 54c693d34204bae61ed9784bb9e8a8be22606711
parent: 80c38271c97b91da046f8e50bf9fb3abb7dd3fb5
author: spew <spew@cbza.org>
date: Fri Mar 28 20:27:28 EDT 2025

defstruct: add defstruct definition to help documentation

--- a/boot/sl.boot
+++ b/boot/sl.boot
@@ -64,7 +64,7 @@
   car cadr #fn("n12060:" #(#fn(gensym))) #fn(nconc) let list #fn(copy-list)
   #fn("n22001e3:" #(set!)) unwind-protect begin #fn("n22001e3:" #(set!))))  let #fn("z1q0R3B00?641<?041=?1@30q42021e12223052e124151532225052863C0268687e2e186e3@408788P:" #(#fn(nconc)
   λ #fn(map) #fn("n10B3500<:0:") #fn(copy-list) #fn("n10B3500T:7060:" #(void)) letrec))  bcode:code #fn("n1200Ee3:" #(aref))  make-label #fn("n120e1:" #(gensym))  bcode:cenv #fn("n1200r3e3:" #(aref))  quasiquote #fn("n1700E62:" #(bq-process))  > #fn("z12021e1721510e163:" #(#fn(nconc)
-  < reverse))  when #fn("z1200211Pqe4:" #(if begin))  help #fn("O100010003000W1000J60q?14W2000J7071?241;3<0422231520P13;02410e3@3007588265275882752IIIIIb;b<288;29_514288<2:_514282;?=514282<87>1?>514282=??51402>CM02?2@8<>18?2A7B26528=5252@\x12089;J5048:3\xd1082888:2C1544893@07D895147E50@30q475882F527G2H8@528A3W07E5047D2I5147E5042?2J8;>18A5247E50@30q^1^1413c07E5047D2K5147E5042?2L8;>18?2A7B26528>525247E50@30q47M50@g07D2N13<02O12P52@402Q05341JE00R3@00ZJ;07D2R51@30q47E5047M60:" #(#(:print-header
+  < reverse))  when #fn("z1200211Pqe4:" #(if begin))  help #fn("O100010003000W1000J60q?14W2000J7071?241;3<0422231520P13;02410e3@3007588265275882752IIIIIb;b<288;29_514288<2:_514282;?=514282<87>1?>514282=??51402>CM02?2@8<>18?2A7B26528=5252@$089;J5048:3\xe3082888:2C154475882D527E2F8@527E2G8@52893H07H7I2J898A535147K50@30q48B3W07K5047H2L5147K5042?2M8;>18B5247K50@30q^1^1^1413c07K5047H2N5147K5042?2O8;>18?2A7B26528>525247K50@30q47P50@g07H2Q13<02R12S52@402T05341JE00R3@00ZJ;07H2U51@30q47K5047P60:" #(#(:print-header
   0) help-print-header #fn(sym) ":doc-" doc getprop *doc* *formals-list* #0#
   #fn("n313?02021820>2162:72504738251474061:" #(#fn(for-each)
                                                 #fn("n17050471A51472F0P61:" #(newline princ print))
@@ -75,19 +75,21 @@
   #fn("n27021221>1q0537362:" #(sort #fn(table-foldl)
                                #fn("n3A051370082P:82:") <) table-keys-filter-sort) groups #fn(for-each)
   #fn("n1707105122A<7302452515347560:" #(princ caddr ": " getprop *doc* newline))
-  #fn(get) *properties* :kind princ newline *doc-extra* filter #fn("n10<20Q:" #(:doc-see)) "See also:"
-  #fn("n1A<0=700=21522263:" #(getprop *formals-list* "    ")) "Members:" #fn("n1A<070021522263:" #(getprop
-  *formals-list* "    ")) void "no help for " #fn(str) " " "" " (undefined)"))  defstruct #fn("O10005000*///W1000J7071?14W2000J60D?24W3000J60D?34W4000J60q?44W5000J60q?54z6IIb;228;230>1_5142224?<5147586518=<8==268?5127288?528<8?512912:5285;J:042902:5283;3\\0483H;3M0483DQ;3:04292;052;J504838BP;J5048382;36040e184;J:042<02=528>3s08E3=02>8E<Pe1@30q8E3I07?8E<2<2@02A53e18?53@30q47?02B8>8H5252^1@30q42C2De12E8D2Fe22G8C2Fe22H2I2J8Fe2e22K2L2FEe32J0e2e3e32M2N2Fe2268F518@Me3e4e3e18E3X02E8E2C1e12C2Je12O8F5152e12O8A5153e3@30qe12O7P2Q8;8B8A8G8D8F0>78@525164:" #(#(:constructor
+  #fn(get) *properties* :kind *doc-extra* filter #fn("n10<20Q:" #(:doc-fmt))
+  #fn("n10<20Q:" #(:doc-see)) princ foldl #fn("n20=161:") newline "See also:" #fn("n1A<0=700=21522263:" #(getprop
+  *formals-list* "    ")) "Members:" #fn("n1A<070021522263:" #(getprop *formals-list* "    ")) void
+  "no help for " #fn(str) " " "" " (undefined)"))  defstruct #fn("O10005000*///W1000J7071?14W2000J60D?24W3000J60D?34W4000J60q?44W5000J60q?54z6IIb;228;230>1_5142224?<5147586518=<8==268?5127288?528<8?512912:5285;J:042902:5283;3\\0483H;3M0483DQ;3:04292;052;J504838BP;J5048382;36040e184;J:042<02=52I222>18F8E848508?>7?H5148>3G07?02@8>2A8HPe15252@30q42B2Ce12D8D2Ee22F8C2Ee22G2H2I8Fe2e22J2K2EEe32I0e2e3e32L2M2Ee2268F518@Me3e4e3e18E3X02D8E2B1e12B2Ie12N8F5152e12N8A5153e3@30qe12N7O2P8;8B8A8G8D8F0>78@525164:" #(#(:constructor
   2 :predicate 4 NIL NIL :type 0 :named 1 :conc-name 3 NIL NIL) vec #0#
   #fn("n17005121220A>28552485:" #(cddr #fn(for-each)
                                   #fn("n17002152340q:722324A<25F2605661:" #(member (:read-only)
                                                                             error #fn(str) "invalid option in slot "
                                                                             " of struct " ": "))) slot-opts)
-  #fn("n17021062:" #(map! #fn("n10B;35040<85;J404085;35040=;J604qe186RS;J9042086513=071228652@30q423242586522087<51390q87P@408762:" #(#fn(keyword?)
+  #fn("n12021062:" #(#fn(map) #fn("n10B;35040<85;J404085;35040=;J604qe186RS;J9042086513=071228652@30q423242586522087<51390q87P@408762:" #(#fn(keyword?)
   error "invalid slot name: " #fn(list*) #fn(sym) #\:))) tokw) separate-doc-from-body #fn(length)
-  #fn(map) #fn("n10B3500<:0:") #fn(sym) #\? "make-" #fn(str) "-" :doc-see sym-set-doc "Constructor for struct `"
-  "`." #fn(append) #fn(nconc) begin def s and or not quote eq? aref = length #fn(copy-list) map-int
-  #fn("n1A<70F0525170920522193865222872324Ie2e3259423e2e2262724e228232995510Me37:2;85523O02<2=2>2?86e22@2?96e22Ae6e2@B02B232995510M24e4e4e4:" #(list-ref
+  #fn(map) #fn("n10B3500<:0:") #fn(sym) #\? "make-" #fn(str) "-" #fn("n12002152853=0220E8553@300853<02208552@40232486252627e1A78C60q@7029Ae2F360q@702:Fe292360q@802;92e2933;02<93e2@30q943;02=94e2@30q9596P578764:" #(#fn(str-find)
+  "\n\n" #fn(str-sub) "" #fn(str) "\n\n    " #fn(append) defstruct vec :type :named :constructor
+  :conc-name :predicate) fmt) sym-set-doc #fn(append) :doc-fmt #fn(nconc) begin def s and or not
+  quote eq? aref = length #fn(copy-list) map-int #fn("n1A<70F0525170920522193865222872324Ie2e3259423e2e2262724e228232995510Me37:2;85523O02<2=2>2?86e22@2?96e22Ae6e2@B02B232995510M24e4e4e4:" #(list-ref
   #fn(sym) def s v assert if void? aref #fn(length) member :read-only error str "slot " quote " in struct "
   " is :read-only" aset!))))  bcode:ctable #fn("n1200Ke3:" #(aref))  with-output-to #fn("z12021e1220e2e1e12315163:" #(#fn(nconc)
   with-bindings *io-out* #fn(copy-list)))  catch #fn("n22012122e123242522e2262722e22829e2e3262:22e20e3e42;22e22<22e2e4e3e3:" #(trycatch
--- a/src/system.sl
+++ b/src/system.sl
@@ -247,11 +247,12 @@
                                           doc-group?))
         (if (or doc formals-list)
             (begin (print-header docterm formals-list :kind kind)
-                   (when doc
-                     (princ doc)
-                     (newline))
                    (let* {[extra (getprop docterm '*doc-extra*)]
+                          [fmt (filter (λ (v) (eq? (car v) :doc-fmt)) extra)]
                           [see (filter (λ (v) (eq? (car v) :doc-see)) extra)]}
+                     (when doc
+                       (princ (foldl (λ (fmt doc) ((cdr fmt) doc)) doc fmt))
+                       (newline))
                      (when see
                        (newline)
                        (princ "See also:")
@@ -1091,20 +1092,20 @@
   (def (tokw slots)
     ; transform args list to keyworded variant.
     ; eg: (a (b 1) (c :read-only)) → ((:a NIL) (:b 1) (:c NIL :read-only))
-    (map! (λ (slot) (let* {[name-cons (and (cons? slot)
-                                           (car slot))]
-                           [name (or name-cons slot)]
-                           [tail (or (and name-cons
-                                          (cdr slot))
-                                 (list NIL))]}
-                      (when (or (not (sym? name))
-                                (keyword? name))
-                        (error "invalid slot name: " name))
-                      (list* (sym #\: name)
-                             (if (keyword? (car tail))
-                                 (cons NIL tail)
-                                 tail))))
-          slots))
+    (map (λ (slot) (let* {[name-cons (and (cons? slot)
+                                          (car slot))]
+                          [name (or name-cons slot)]
+                          [tail (or (and name-cons
+                                         (cdr slot))
+                                (list NIL))]}
+                     (when (or (not (sym? name))
+                               (keyword? name))
+                       (error "invalid slot name: " name))
+                     (list* (sym #\: name)
+                            (if (keyword? (car tail))
+                                (cons NIL tail)
+                                tail))))
+         slots))
   (let* {; first element in slots may be the doc string
          [docs+slots (separate-doc-from-body slots)]
          [docs (car docs+slots)]
@@ -1135,13 +1136,22 @@
          ; accessor prefix
          [access (or conc-name
                      (str name "-"))]}
-   (when docs
-     (let* {[docs-extra (if constructor
-                            (list (cons :doc-see (car constructor))))]}
-       (if constructor
-           (sym-set-doc (car constructor)
-                        (list (str "Constructor for struct `" name "`.")) slots))
-       (sym-set-doc name (append docs docs-extra))))
+    (def (fmt doc)
+      (let* {[cut (str-find doc "\n\n")]
+             [hd (if cut (str-sub doc 0 cut) doc)]
+             [tl (if cut (str-sub doc cut) "")]}
+        (str hd
+             "\n\n    "
+             (append (list 'defstruct)
+                     (unless (eq? type vec) (list :type type))
+                     (unless named (list :named named))
+                     (unless constructor (list :constructor constructor))
+                     (if conc-name (list :conc-name conc-name))
+                     (if predicate (list :predicate predicate))
+                     (cons name slots))
+             tl)))
+    (when docs
+      (sym-set-doc name (append docs (list (cons :doc-fmt fmt)))))
    `(begin
       ; predicate
       (def (,is? s)