diff --git a/collects/scribble/base-render.ss b/collects/scribble/base-render.ss index e918f79d..20e797c3 100644 --- a/collects/scribble/base-render.ss +++ b/collects/scribble/base-render.ss @@ -278,11 +278,7 @@ [(element? i) (cond [(link-element? i) - (let-values ([(dest ext?) (resolve-get/where d ri (link-element-tag i))]) - (when ext? - (hash-table-put! (resolve-info-undef ri) - (tag-key (link-element-tag i) ri) - #t)))]) + (resolve-get d ri (link-element-tag i))]) (for-each (lambda (e) (resolve-element e d ri)) (element-content i))])) diff --git a/collects/scribble/html-render.ss b/collects/scribble/html-render.ss index f4ce24eb..86dbf2f4 100644 --- a/collects/scribble/html-render.ss +++ b/collects/scribble/html-render.ss @@ -150,8 +150,17 @@ ,@(render-onthispage-contents d ri top) ,@(apply append (map (lambda (t) - (render-table t d ri)) - (filter auxiliary-table? (flow-paragraphs (part-flow d))))))))) + (let loop ([t t]) + (if (table? t) + (render-table t d ri) + (loop (delayed-flow-element-flow-elements t ri))))) + (filter (lambda (e) + (let loop ([e e]) + (or (and (auxiliary-table? e) + (pair? (table-flowss e))) + (and (delayed-flow-element? e) + (loop (delayed-flow-element-flow-elements e ri)))))) + (flow-paragraphs (part-flow d))))))))) (define/private (render-onthispage-contents d ri top) (if (ormap (lambda (p) (part-whole-page? p ri)) diff --git a/collects/scribble/manual.ss b/collects/scribble/manual.ss index 35140712..7115ebbd 100644 --- a/collects/scribble/manual.ss +++ b/collects/scribble/manual.ss @@ -8,7 +8,8 @@ (lib "string.ss") (lib "list.ss") (lib "class.ss") - (lib "stxparam.ss")) + (lib "stxparam.ss") + (lib "serialize.ss")) (require-for-syntax (lib "stxparam.ss")) (require-for-label (lib "lang.ss" "big") (lib "class.ss")) @@ -214,14 +215,15 @@ (elem (method a b) " in " (scheme a))])) (define (*method sym id) - (let ([tag (method-tag (register-scheme-definition id #t) - sym)]) - (make-element - "schemesymbol" - (list (make-link-element - "schemevaluelink" - (list (symbol->string sym)) - tag))))) + (**method sym (register-scheme-definition id #t))) + + (define (**method sym tag) + (make-element + "schemesymbol" + (list (make-link-element + "schemevaluelink" + (list (symbol->string sym)) + (method-tag tag sym))))) (define (method-tag vtag sym) (list 'meth @@ -376,7 +378,7 @@ [(_ name ([field field-contract] ...) immutable? transparent? desc ...) (*defstruct (quote-syntax/loc name) 'name '([field field-contract] ...) (list (lambda () (schemeblock0 field-contract)) ...) - #t #t (lambda () (list desc ...)))])) + immutable? transparent? (lambda () (list desc ...)))])) (define-syntax (defform*/subs stx) (syntax-case stx () [(_ #:literals (lit ...) [spec spec1 ...] ([non-term-id non-term-form ...] ...) desc ...) @@ -543,7 +545,7 @@ (define (annote-exporting-library e) (make-delayed-element (lambda (render p ri) - (let ([from (resolve-get p ri '(exporting-libraries #f))]) + (let ([from (resolve-get/tentative p ri '(exporting-libraries #f))]) (if (and from (pair? from)) (list (make-hover-element @@ -890,6 +892,12 @@ (define (*defstruct stx-id name fields field-contracts immutable? transparent? content-thunk) (define spacer (hspace 1)) (define to-flow (lambda (e) (make-flow (list (make-paragraph (list e)))))) + (define (field-name f) (if (pair? (car f)) + (caar f) + (car f))) + (define (field-view f) (if (pair? (car f)) + (make-shaped-parens (car f) #\[) + (car f))) (make-splice (cons (make-table @@ -914,13 +922,18 @@ (list 'make- name) (append (map (lambda (f) - (list name '- (car f))) + (list name '- (field-name f))) fields) (if immutable? null - (map (lambda (f) - (list 'set- name '- (car f) '!)) - fields))))))]) + (filter + values + (map (lambda (f) + (if (and (pair? (car f)) + (memq '#:immutable (car f))) + #f + (list 'set- name '- (field-name f) '!))) + fields)))))))]) (if (pair? name) (to-element (list just-name (make-just-context (cadr name) stx-id))) @@ -928,12 +941,18 @@ [short-width (apply + (length fields) 8 - (map (lambda (s) - (string-length (symbol->string s))) - (append (if (pair? name) - name - (list name)) - (map car fields))))]) + (append + (map (lambda (s) + (string-length (symbol->string s))) + (append (if (pair? name) + name + (list name)) + (map field-name fields))) + (map (lambda (f) + (if (pair? (car f)) + (+ 3 2 (string-length (keyword->string (cadar f)))) + 0)) + fields)))]) (if (and (short-width . < . max-proto-width) (not immutable?) (not transparent?)) @@ -942,7 +961,7 @@ (to-element `(,(schemeparenfont "struct") ,the-name - ,(map car fields))))) + ,(map field-view fields))))) (make-table #f (append @@ -958,8 +977,8 @@ (schemeparenfont "("))))) (to-flow (if (or (null? fields) (short-width . < . max-proto-width)) - (to-element (map car fields)) - (to-element (caar fields)))))) + (to-element (map field-view fields)) + (to-element (field-view (car fields))))))) (if (short-width . < . max-proto-width) null (let loop ([fields fields]) @@ -971,7 +990,7 @@ (to-flow spacer) (to-flow spacer) (to-flow - (let ([e (to-element (car fld))]) + (let ([e (to-element (field-view fld))]) (if (null? (cdr fields)) (make-element #f @@ -1033,7 +1052,7 @@ #f (list (list (to-flow (hspace 2)) - (to-flow (to-element (car v))) + (to-flow (to-element (field-name v))) (to-flow spacer) (to-flow ":") (to-flow spacer) @@ -1322,9 +1341,9 @@ ;; ---------------------------------------- (provide defclass - define-class-doc + defclass/title definterface - define-interface-doc + definterface/title defconstructor defconstructor/make defconstructor*/make @@ -1333,141 +1352,135 @@ defmethod* methspec methimpl - this-obj - include-class-section - include-class) + this-obj) (define-syntax-parameter current-class #f) - (define-struct decl (name super intfs mk-head body methods)) + (define-struct decl (name super intfs mk-head body)) (define-struct constructor (def)) (define-struct meth (name mode desc def)) (define-struct spec (def)) (define-struct impl (def)) - (define-for-syntax (class-id->class-doc-info-id id) - (datum->syntax-object id - (string->symbol (format "class-doc-info:~a" (syntax-e id))) - id)) + (define-serializable-struct cls/intf (name-element super intfs methods)) - (define-syntax (define-class-doc-info stx) - (syntax-case stx () - [(_ id val) - (with-syntax ([id (class-id->class-doc-info-id #'id)]) - #'(begin - (provide id) - (define id val)))])) - - (define-syntax (class-doc-info stx) - (syntax-case* stx (object%) module-label-identifier=? - [(_ object%) #'#f] - [(_ id) (class-id->class-doc-info-id #'id)])) - - (define (collect-inherited supers ht) - (let* ([supers (let loop ([supers supers][accum null]) - (cond - [(null? supers) (reverse accum)] - [(memq (car supers) accum) - (loop (cdr supers) accum)] - [else - (let ([super (car supers)]) - (loop (append (reverse (decl-intfs super)) - (if (decl-super super) - (list (decl-super super)) - null) - (cdr supers)) - (cons super accum)))]))] + (define (make-inherited-table r d ri decl) + (let* ([start (let ([key (register-scheme-definition (decl-name decl))]) + (list (cons key (lookup-cls/intf d ri key))))] + [supers (cdr + (let loop ([supers start][accum null]) + (cond + [(null? supers) (reverse accum)] + [(memq (car supers) accum) + (loop (cdr supers) accum)] + [else + (let ([super (car supers)]) + (loop (append (map (lambda (i) + (cons i (lookup-cls/intf d ri i))) + (reverse (cls/intf-intfs (cdr super)))) + (let ([s (cls/intf-super (cdr super))]) + (if s + (list (cons s (lookup-cls/intf d ri s))) + null)) + (cdr supers)) + (cons super accum)))])))] + [ht (let ([ht (make-hash-table)]) + (for-each (lambda (i) + (when (meth? i) + (hash-table-put! ht (meth-name i) #t))) + (decl-body decl)) + ht)] [inh (apply append (map (lambda (super) (let ([inh (filter values - (hash-table-map - (decl-methods super) - (lambda (k v) - (let ([v (hash-table-get ht k)]) - (and (eq? (car v) (decl-name super)) - (cons (symbol->string k) - (*method k (car v))))))))]) + (map + (lambda (k) + (if (hash-table-get ht k #f) + #f + (begin + (hash-table-put! ht k #t) + (cons (symbol->string k) + (**method k (car super)))))) + (cls/intf-methods (cdr super))))]) (if (null? inh) null (cons (make-element #f (list (make-element "inheritedlbl" '("from ")) - (to-element (decl-name super)))) + (cls/intf-name-element (cdr super)))) (map cdr (sort inh (lambda (a b) (stringstring (syntax-e (decl-name decl)))) + tag))) + (and (decl-super decl) + (not (module-label-identifier=? #'object% (decl-super decl))) + (register-scheme-definition (decl-super decl))) + (map register-scheme-definition (decl-intfs decl)) + (map (lambda (m) + (meth-name m)) + (filter meth? (decl-body decl)))))))))) + + (define (build-body decl body) + (append + (map (lambda (i) + (cond + [(constructor? i) ((constructor-def i))] + [(meth? i) + ((meth-def i) (meth-desc i))] + [else i])) + body) + (list + (make-delayed-flow-element + (lambda (r d ri) + (make-inherited-table r d ri decl)))))) + + (define (*include-class/title decl) + (make-splice + (list* (title #:style 'hidden (to-element (decl-name decl))) + (make-decl-collect decl) + (build-body decl + (append + ((decl-mk-head decl) #t) + (decl-body decl)))))) (define (*include-class decl) (make-splice - (append - ((decl-mk-head decl) #f) - (list - (make-blockquote - "leftindent" - (flow-paragraphs - (decode-flow - (map (lambda (i) - (cond - [(constructor? i) ((constructor-def i))] - [(meth? i) - ((meth-def i) (meth-desc i))] - [else i])) - (decl-body decl))))))))) + (cons + (make-decl-collect decl) + (append + ((decl-mk-head decl) #f) + (list + (make-blockquote + "leftindent" + (flow-paragraphs + (decode-flow + (build-body decl (decl-body decl)))))))))) - (define-syntax include-class-section - (syntax-rules () - [(_ id) (*include-class-section (class-doc-info id))])) - - (define-syntax include-class - (syntax-rules () - [(_ id) (*include-class (class-doc-info id))])) - - (define (*define-class-doc stx-id super intfs whole-page?) + (define (*class-doc stx-id super intfs whole-page?) (let ([spacer (hspace 1)]) (make-table 'boxed @@ -1476,7 +1489,7 @@ (list (make-flow (list (make-paragraph - (list (let ([tag (register-scheme-definition stx-id #t)] + (list (let ([tag (register-scheme-definition stx-id)] [content (list (annote-exporting-library (to-element stx-id)))]) (if tag ((if whole-page? @@ -1521,51 +1534,57 @@ (make-flow (list (make-paragraph (list (to-element i))))))) (cdr intfs))))))))))))) - (define-syntax define-class-doc + (define-syntax *defclass (syntax-rules () - [(_ name super (intf ...) body ...) - (define-class-doc-info name + [(_ *include-class name super (intf ...) body ...) + (*include-class (syntax-parameterize ([current-class (quote-syntax name)]) - (register-class (quote-syntax/loc name) - (class-doc-info super) - (list (class-doc-info intf) ...) - (lambda (whole-page?) - (list - (*define-class-doc (quote-syntax/loc name) - (quote-syntax super) - (list (quote-syntax intf) ...) - whole-page?))) - (list body ...))))])) + (make-decl (quote-syntax/loc name) + (quote-syntax/loc super) + (list (quote-syntax/loc intf) ...) + (lambda (whole-page?) + (list + (*class-doc (quote-syntax/loc name) + (quote-syntax super) + (list (quote-syntax intf) ...) + whole-page?))) + (list body ...))))])) (define-syntax defclass (syntax-rules () - [(_ name . rest) - (begin - (define-class-doc name . rest) - (include-class name))])) + [(_ name super (intf ...) body ...) + (*defclass *include-class name super (intf ...) body ...)])) - (define-syntax define-interface-doc + (define-syntax defclass/title (syntax-rules () - [(_ name (intf ...) body ...) - (define-class-doc-info name - (syntax-parameterize ([current-class (quote-syntax name)]) - (register-class (quote-syntax/loc name) - #f - (list (class-doc-info intf) ...) - (lambda (whole-page?) - (list - (*define-class-doc (quote-syntax/loc name) - #f - (list (quote-syntax intf) ...) - whole-page?))) - (list body ...))))])) + [(_ name super (intf ...) body ...) + (*defclass *include-class/title name super (intf ...) body ...)])) + (define-syntax *definterface + (syntax-rules () + [(_ *include-class name (intf ...) body ...) + (*include-class + (syntax-parameterize ([current-class (quote-syntax name)]) + (make-decl (quote-syntax/loc name) + #f + (list (quote-syntax/loc intf) ...) + (lambda (whole-page?) + (list + (*class-doc (quote-syntax/loc name) + #f + (list (quote-syntax intf) ...) + whole-page?))) + (list body ...))))])) + (define-syntax definterface (syntax-rules () - [(_ name . rest) - (begin - (define-interface-doc name . rest) - (include-class name))])) + [(_ name (intf ...) body ...) + (*definterface *include-class name (intf ...) body ...)])) + + (define-syntax definterface/title + (syntax-rules () + [(_ name (intf ...) body ...) + (*definterface *include-class/title name (intf ...) body ...)])) (define-syntax (defconstructor*/* stx) (syntax-case stx () @@ -1628,7 +1647,7 @@ [(override) "Overrides "] [(extend) "Extends "] [(augment) "Augments "]) - (*xmethod/super (class-doc-info cname) 'name1) "."))] + (*xmethod/super (quote-syntax/loc cname) 'name1) "."))] [else null])]) #'(make-meth 'name1 @@ -1671,17 +1690,43 @@ (with-syntax ([cname (syntax-parameter-value #'current-class)]) #'(*this-obj 'cname))])) - (define (*xmethod/super decl name) - (let ([super (ormap (lambda (decl) - (and decl - (let ([m (hash-table-get (decl-methods decl) name #f)]) - (and m (car m))))) - (cons (decl-super decl) - (decl-intfs decl)))]) - (make-element #f - (list (*method name super) - " in " - (to-element super))))) + (define (*xmethod/super cname name) + (let ([get + (lambda (d ri key) + (let ([v (lookup-cls/intf d ri key)]) + (if v + (cons (cls/intf-super v) + (cls/intf-intfs v)) + null)))]) + (make-delayed-element + (lambda (r d ri) + (let loop ([search (get d ri (register-scheme-definition cname))]) + (cond + [(null? search) + (make-element #f "")] + [(not (car search)) + (loop (cdr search))] + [else + (let ([v (lookup-cls/intf d ri (car search))]) + (if v + (if (member name (cls/intf-methods v)) + (list + (make-element #f + (list (**method name (car search)) + " in " + (cls/intf-name-element v)))) + (loop (append (cdr search) (get d ri (car search))))) + (loop (cdr search))))]))) + (lambda () (format "~a in ~a" (syntax-e cname) name)) + (lambda () (format "~a in ~a" (syntax-e cname) name))))) + + (define (lookup-cls/intf d ri name) + (let ([v (resolve-get d ri `(cls/intf ,name))]) + (or v + (make-cls/intf "unknown" + #f + null + null)))) ;; ---------------------------------------- ) diff --git a/collects/scribble/scheme.ss b/collects/scribble/scheme.ss index 60e3a569..59cb5bd2 100644 --- a/collects/scribble/scheme.ss +++ b/collects/scribble/scheme.ss @@ -78,7 +78,7 @@ (lambda (renderer sec ri) (let* ([vtag `(def ,tag)] [stag `(form ,tag)] - [sd (resolve-get sec ri stag)]) + [sd (resolve-get/tentative sec ri stag)]) (list (cond [sd @@ -541,11 +541,15 @@ (car b)) (cadr b))))) + (define (register-scheme/invent stx warn-if-no-label?) + (or (register-scheme stx warn-if-no-label?) + (format ":UNKNOWN:~a" (syntax-e stx)))) + (define (register-scheme-definition stx [warn-if-no-label? #f]) - `(def ,(register-scheme stx warn-if-no-label?))) + `(def ,(register-scheme/invent stx warn-if-no-label?))) (define (register-scheme-form-definition stx [warn-if-no-label? #f]) - `(form ,(register-scheme stx warn-if-no-label?))) + `(form ,(register-scheme/invent stx warn-if-no-label?))) (define syntax-ize-hook (make-parameter (lambda (v col) #f))) diff --git a/collects/scribble/struct.ss b/collects/scribble/struct.ss index 24001b98..0d608a43 100644 --- a/collects/scribble/struct.ss +++ b/collects/scribble/struct.ss @@ -38,6 +38,14 @@ (values v #t))])))) (define (resolve-get part ri key) + (let-values ([(v ext?) (resolve-get/where part ri key)]) + (when ext? + (hash-table-put! (resolve-info-undef ri) + (tag-key key ri) + #t)) + v)) + + (define (resolve-get/tentative part ri key) (let-values ([(v ext?) (resolve-get/where part ri key)]) v)) @@ -47,7 +55,7 @@ part-collected-info collect-put! resolve-get - resolve-get/where) + resolve-get/tentative) ;; ---------------------------------------- diff --git a/collects/scribblings/scribble/manual.scrbl b/collects/scribblings/scribble/manual.scrbl index 37e53114..2aa7510b 100644 --- a/collects/scribblings/scribble/manual.scrbl +++ b/collects/scribblings/scribble/manual.scrbl @@ -167,9 +167,13 @@ in a form definition.} @; ------------------------------------------------------------------------ @section{Definition Reference} -@defform[(defproc (id arg-spec ...) - result-contract-expr-datum - pre-flow ...)]{ +@defform/subs[(defproc (id arg-spec ...) + result-contract-expr-datum + pre-flow ...) + ([arg-spec (arg-id contract-expr-datum) + (arg-id contract-expr-datum default-expr) + (keyword arg-id contract-expr-datum) + (keyword arg-id contract-expr-datum default-expr)])]{ Produces a sequence of flow elements (encaptured in a @scheme[splice]) to document a procedure named @scheme[id]. The @scheme[id] is indexed, @@ -368,56 +372,73 @@ at once, aligned around the @litchar{=} and @litchar{|}.} @; ------------------------------------------------------------------------ @section{Classes and Interfaces} -@defform[(define-class-doc id super-id (intf-id ...) pre-flow ...)]{ +@defform[(defclass id super-id (intf-id ...) pre-flow ...)]{ -Binds @schemeidfont{class-doc-info:}@scheme[id] to documentation for -the class @scheme[id]. If @scheme[super-id] is not @scheme[object%], -then @schemeidfont{class-doc-info:}@scheme[super-id] must be bound to -documentation for the superclass (so that links can be created to -inherited methods, etc.). Similarly, -@schemeidfont{class-doc-info:}@scheme[intf-id] must be bound to -documentation for interfaces implemented by the class. At the same -time, @scheme[id], @scheme[super-id], and the @scheme[int-id]s must -have for-label bindings that are used for hyperlinks in the usual way. +Creates documentation for a class @scheme[id] that is a subclass of +@scheme[super-id] and implements each interface @scheme[intf-id]. Each +@scheme[super-id] (except @scheme[object%]) and @scheme[intf-id] must +be documented somewhere via @scheme[defclass] or @scheme[definterface]. The decoding of the @scheme[pre-flow] sequence should start with general documentation about the class, followed by constructor definition (see @scheme[defconstructor]), and then field and method -definitions (see @scheme[defmethod]). +definitions (see @scheme[defmethod]). In rendered form, the +constructor and method specification are indented to visually group +them under the class definition.} -A @scheme[define-class-doc] form is a Scheme-level definition. It does -not produce documentation directly. Instead, @scheme[(include-class -id)] or @scheme[(include-class-section id)] should be used later to -produce the documentation.} +@defform[(defclass/title id super-id (intf-id ...) pre-flow ...)]{ -@defform[(include-class id)]{ +Like @scheme[defclass], also includes a @scheme[title] declaration +with the style @scheme['hidden]. In addition, the constructor and +methods are not left-indented. -Generates inline documentation based on the information bound to -@schemeidfont{class-doc-info:}@scheme[id]. Constructor and method -specification are indented to visually group them under the class -definition.} +This form is normally used to create a section to be rendered on its +own HTML. The @scheme['hidden] style is used because the definition +box serves as a title.} -@defform[(include-class-section id)]{ +@defform[(definterface id (intf-id ...) pre-flow ...)]{ -Generates documentation based on the information bound to -@schemeidfont{class-doc-info:}@scheme[id] as a new section. The -@scheme[id] is used as the section title, but the title is not -rendered in HTML output, as the definition box serves as a title. With -the expectation that the section will have its own page, constructor -and method specifications are not indented (unlike the result of -@scheme[include-class]).} +Like @scheme[defclass], but for an interfaces. Naturally, +@scheme[pre-flow] should not generate a constructor declaration.} -@defform[(defclass id super-id (intf-id ...) pre-flow ...)]{ +@defform[(definterface/title id (intf-id ...) pre-flow ...)]{ -Combines @scheme[define-class-doc] and @scheme[include-class].} +Like @scheme[definterface], but for single-page rendering as in +@scheme[defclass/title].} -@defform[(defconstructor)]{ +@defform/subs[(defconstructor (arg-spec ...) pre-flow ...) + ([arg-spec (arg-id contract-expr-datum) + (arg-id contract-expr-datum default-expr)])]{ -TBD.} +Like @scheme[defproc], but for a constructor declaration in the body +of @scheme[defclass], so no return contract is specified. Also, the +@scheme[new]-style keyword for each @scheme[arg-spec] is implicit from +the @scheme[arg-id].} -@defform[(defmethod)]{ +@defform[(defconstructor/make (arg-spec ...) pre-flow ...)]{ -TBD.} +Like @scheme[defconstructor], but specifying by-position +initialization arguments (for use with @scheme[make-object]) instead +of by-name arguments (for use with @scheme[new]).} + +@defform[(defconstructor*/make [(arg-spec ...) ...] pre-flow ...)]{ + +Like @scheme[defconstructor/make], but with multiple constructor +patterns analogous @scheme[defproc*].} + +@defform[(defmethod (id arg-spec ...) + result-contract-expr-datum + pre-flow ...)]{ + +Like @scheme[defproc], but for a method within a @scheme[defclass] or +@scheme[definterface] body.} + +@defform[(defmethod* ([(id arg-spec ...) + result-contract-expr-datum] ...) + pre-flow ...)]{ + +Like @scheme[defproc*], but for a method within a @scheme[defclass] or +@scheme[definterface] body.} @; ------------------------------------------------------------------------ @section{Various String Forms}