#lang at-exp racket/base (require scribble/manual racket/list racket/date racket/class scribble/core scribble/decode scribble/html-properties scribble/latex-properties (for-syntax syntax/parse racket/base) scheme/string setup/main-collects racket/contract) (provide define-cite author+date-style author+date-square-bracket-style number-style make-bib in-bib (rename-out [auto-bib? bib?]) author-name org-author-name (contract-out [authors (->* (content?) #:rest (listof content?) element?)] [proceedings-location (->* [any/c] [#:pages (or/c (list/c any/c any/c) #f) #:series any/c #:volume any/c] element?)] [journal-location (->* [any/c] [#:pages (or/c (list/c any/c any/c) #f) #:number any/c #:volume any/c] element?)] [book-location (->* [] [#:edition any/c #:publisher any/c] element?)] [techrpt-location (-> #:institution any/c #:number any/c element?)] [dissertation-location (->* [#:institution any/c] [#:degree any/c] element?)] [book-chapter-location (->* [any/c] [#:pages (or/c (list/c any/c any/c) #f) #:series any/c #:volume any/c #:publisher any/c] element?)]) other-authors editor abbreviate-given-names) (define abbreviate-given-names (make-parameter #f)) (define autobib-style-extras (let ([abs (lambda (s) (path->main-collects-relative (collection-file-path s "scriblib")))]) (list (make-css-addition (abs "autobib.css")) (make-tex-addition (abs "autobib.tex"))))) (define bib-single-style (make-style "AutoBibliography" autobib-style-extras)) (define bib-columns-style (make-style #f autobib-style-extras)) (define bibentry-style (make-style "Autobibentry" autobib-style-extras)) (define colbibnumber-style (make-style "Autocolbibnumber" autobib-style-extras)) (define colbibentry-style (make-style "Autocolbibentry" autobib-style-extras)) (define-struct auto-bib (author date title location url note is-book? key specific)) (define-struct bib-group (ht)) (define-struct (author-element element) (names cite)) (define-struct (other-author-element author-element) ()) (define (author-element-names* x) (and x (author-element-names x))) ;; render the use of a citation. (define (add-cite group bib-entry which with-specific? disambiguation style) (let ([key (auto-bib-key bib-entry)]) (when disambiguation (for ([bib disambiguation]) (hash-set! (bib-group-ht group) (auto-bib-key bib) bib))) (hash-set! (bib-group-ht group) key bib-entry) (make-delayed-element (lambda (renderer part ri) ;; (list which key) should be mapped to the bibliography element. (define s (resolve-get part ri `(,which ,key))) (define content (list (or s "???") (cond [(not (send style disambiguate-date?)) '()] [disambiguation ;; should be a list of bib-entries with same author/date (define disambiguation* (add-between (for/list ([bib (in-list disambiguation)]) (define key (auto-bib-key bib)) (define maybe-disambiguation (resolve-get part ri `(autobib-disambiguation ,key))) (case maybe-disambiguation [(#f) #f] [(unambiguous) #f] [else (make-link-element "AutobibLink" maybe-disambiguation `(autobib ,key))])) ",")) (cond [(not (car disambiguation*)) '()] ;; the bib was unambiguous [else disambiguation*])] [else '()]) (if with-specific? (auto-bib-specific bib-entry) ""))) (make-link-element "AutobibLink" content `(autobib ,(auto-bib-key bib-entry)))) (lambda () "(???)") (lambda () "(???)")))) (define (add-date-cites group bib-entries delimiter style sort? maybe-datestring (date-year date))))) (define (default-render-date-cite date) (make-element #f (list (number->string (date-year date))))) (define (default-date a, 1 -> b, etc. (define (default-disambiguation n) (when (>= n 26) (error 'default-disambiguation "Citations too ambiguous for default disambiguation scheme.")) (make-element #f (list (format "~a" (integer->char (+ 97 n)))))) (define author+date-style% (class object% (define/public (bibliography-table-style) bib-single-style) (define/public (entry-style) bibentry-style) (define/public (disambiguate-date?) #t) (define/public (collapse-for-date?) #t) (define/public (get-cite-open) "(") (define/public (get-cite-close) ")") (define/public (get-group-sep) "; ") (define/public (get-item-sep) ", ") (define/public (render-citation date-cite i) date-cite) (define/public (render-author+dates author dates) (list* author " " dates)) (define/public (bibliography-line i e) (list e)) (super-new))) (define author+date-style (new author+date-style%)) (define author+date-square-bracket-style (new (class author+date-style% (define/override (get-cite-open) "[") (define/override (get-cite-close) "]") (super-new)))) (define number-style (new (class object% (define/public (bibliography-table-style) bib-columns-style) (define/public (entry-style) colbibentry-style) (define/public (disambiguate-date?) #f) (define/public (collapse-for-date?) #f) (define/public (get-cite-open) "[") (define/public (get-cite-close) "]") (define/public (get-group-sep) ", ") (define/public (get-item-sep) ", ") (define/public (render-citation date-cite i) (number->string i)) (define/public (render-author+dates author dates) dates) (define/public (bibliography-line i e) (list (make-paragraph plain (make-element colbibnumber-style (list "[" (number->string i) "]"))) e)) (super-new)))) (define (gen-bib tag group sec-title style maybe-disambiguator maybe-render-date-bib maybe-render-date-cite maybe-datestring (author-element-cite (extract-bib-author a))) (content->string (author-element-cite (extract-bib-author b)))) (auto-bib-date a) (auto-bib-date b) (date=? a b))) (define bibs (sort (hash-values (bib-group-ht group)) author/datepara bib disambiguation i) (define collect-target (list (make-target-element #f (bib->entry bib style disambiguation render-date-bib i) `(autobib ,(auto-bib-key bib))))) ;; Communicate to scribble's resolve step. (define (collect ci) ;; store the author (collect-put! ci `(autobib-author ,(auto-bib-key bib)) ;; (list which key) (make-element #f (list (author-element-cite (extract-bib-author bib))))) ;; store the date (when (auto-bib-date bib) (collect-put! ci `(autobib-date ,(auto-bib-key bib)) ;; (list which key) (make-element #f (list (send style render-citation (render-date-cite (auto-bib-date bib)) i))))) ;; store how to disambiguate it from other like citations. (collect-put! ci `(autobib-disambiguation ,(auto-bib-key bib)) (or disambiguation 'unambiguous))) (send style bibliography-line i (make-paragraph plain (list (make-collect-element #f collect-target collect))))) ;; create the bibliography with disambiguations added. (define-values (last num-ambiguous rev-disambiguated*) (for/fold ([last #f] [num-ambiguous 0] [rev-disambiguated '()]) ([bib (in-list bibs)] [i (in-naturals 1)]) (define ambiguous?? (and (send style disambiguate-date?) last (ambiguous? last bib))) (define num-ambiguous* (cond [ambiguous?? (add1 num-ambiguous)] [else 0])) ;; the current entry is ambiguous with the last. Modify the last ;; to have the first disambiguation. (define rev-disambiguated* (cond [(and ambiguous?? (= 0 num-ambiguous)) (cons (bib->para last (disambiguator num-ambiguous) i) (cdr rev-disambiguated))] [else rev-disambiguated])) (define para* (bib->para bib (and ambiguous?? (disambiguator num-ambiguous*)) i)) (values bib num-ambiguous* (cons para* rev-disambiguated*)))) (reverse rev-disambiguated*))) (define (make-space) (list (make-paragraph (make-style #f '()) '("")) (make-paragraph (make-style #f '()) '("")))) (make-part #f `((part ,tag)) (list sec-title) (make-style #f '(unnumbered)) null (list (make-table (send style bibliography-table-style) (add-between #:splice? #t disambiguated (for/list ([i (in-range 1 spaces)]) (make-space))))) null)) (define (bib->entry bib style disambiguation render-date-bib i) (define-values (author date title location url note is-book?) (values (auto-bib-author bib) (auto-bib-date bib) (auto-bib-title bib) (auto-bib-location bib) (auto-bib-url bib) (auto-bib-note bib) (auto-bib-is-book? bib))) (make-element (send style entry-style) (append (if author `(,author ,@(if (ends-in-punc? author) '(" ") '(". "))) null) ;; (if is-book? null '(ldquo)) (if is-book? (list (italic title)) (decode-content (list title))) (if (ends-in-punc? title) null '(".")) ;; (if is-book? null '(rdquo)) (if location `(" " ,@(decode-content (list location)) ,(if date "," ".")) null) (if date `(" " ,@(if disambiguation `(,@(decode-content (list (render-date-bib date))) ,disambiguation) (decode-content (list (render-date-bib date)))) ".") null) (if url `(" " ,(link url (make-element 'url (list url)))) null) (if note `(" " ,note) null)))) (define-syntax (define-cite stx) (syntax-parse stx [(_ (~var ~cite id) citet:id generate-bibliography:id (~or (~optional (~seq #:style style) #:defaults ([style #'author+date-style])) (~optional (~seq #:disambiguate fn) #:defaults ([fn #'#f])) (~optional (~seq #:render-date-in-bib render-date-bib) #:defaults ([render-date-bib #'#f])) (~optional (~seq #:spaces spaces) #:defaults ([spaces #'1])) (~optional (~seq #:render-date-in-cite render-date-cite) #:defaults ([render-date-cite #'#f])) (~optional (~seq #:datestring e))) (define (understand-date inp) ;; Currently there is no string->date function. ;; Common usage of autobib has assumed that this should be the year. (cond [(or (string? inp) (number? inp)) (define year (cond [(string? inp) (string->number inp)] [else inp])) (date 0 0 0 1 1 ;; second/minute/hour/day/month year ;; week-day/year-day/daylight savings time?/timezone offset 0 0 #f 0)] [(date? inp) inp] [(not inp) #f] ;; no date is fine too. [else (error 'make-bib "Not given a value that represents a date.")])) ;; We delay making the element for the bib-entry because we may need to add ;; disambiguations during gen-bib. (define (make-bib #:title title #:author [author #f] #:is-book? [is-book? #f] #:location [location #f] #:date [date #f] #:url [url #f] #:note [note #f]) (define author* (cond [(not author) #f] [(author-element? author) author] [else (parse-author author)])) (define parsed-date (understand-date date)) (make-auto-bib author* parsed-date title location url note is-book? (content->string (make-element #f (append (if author* (list author*) null) (list title) (if location (decode-content (list location)) null) (if date (decode-content (list (default-render-date-bib parsed-date))) null) (if url (list (link url (make-element 'url (list url)))) null) (if note (list note) null)))) "")) (define (in-bib bib where) (make-auto-bib (auto-bib-author bib) (auto-bib-date bib) (auto-bib-title bib) (auto-bib-location bib) (auto-bib-url bib) (auto-bib-note bib) (auto-bib-is-book? bib) (auto-bib-key bib) ;; "where" is the only specific part of auto-bib elements currently. (string-append (auto-bib-specific bib) where))) (define (parse-author a) (cond [(author-element? a) a] [else (define s (content->string a)) ;; plain text rendering (define m (regexp-match #px"^(.*) (([\\-]|\\p{L})+)$" s)) (define given-names (and m (cadr m))) (define family-name (and m (caddr m))) (define names (cond [m (string-append family-name " " given-names)] [else s])) (define cite (cond [m (caddr m)] [else s])) (define element-content (cond [(and given-names (abbreviate-given-names)) (string-append (given-names->initials given-names) family-name)] [else a])) (make-author-element #f (list element-content) names cite)])) (define (given-names->initials str) (regexp-replace* #rx"(.)[^ ]*( |$)" str "\\1. ")) (module+ test (require rackunit) (check-equal? (given-names->initials "Matthew") "M. ") (check-equal? (given-names->initials "Matthew R.") "M. R. ") (check-equal? (given-names->initials "Matthew Raymond") "M. R. ")) (define (proceedings-location location #:pages [pages #f] #:series [series #f] #:volume [volume #f]) (let* ([s @elem{In @italic{@elem{Proc. @to-string[location]}}}] [s (if series @elem{@|s|, @to-string[series]} s)] [s (if volume @elem{@|s| volume @to-string[volume]} s)] [s (if pages @elem{@|s|, pp. @(to-string (car pages))--@(to-string (cadr pages))} s)]) s)) (define (journal-location location #:pages [pages #f] #:number [number #f] #:volume [volume #f]) (let* ([s @italic{@to-string[location]}] [s (if volume @elem{@|s| @(to-string volume)} s)] [s (if number @elem{@|s|(@(to-string number))} s)] [s (if pages @elem{@|s|, pp. @(to-string (car pages))--@(to-string (cadr pages))} s)]) s)) (define (string-capitalize str) (if (non-empty-string? str) (let ([chars (string->list str)]) (list->string (cons (char-upcase (car chars)) (cdr chars)))) str)) (define (book-location #:edition [edition #f] #:publisher [publisher #f]) (let* ([s (if edition @elem{@(string-capitalize (to-string edition)) edition} #f)] [s (if publisher (if s @elem{@|s|. @to-string[publisher]} @elem{@to-string[publisher]}) s)]) (unless s (error 'book-location "no arguments")) s)) (define (techrpt-location #:institution org #:number num) @elem{@to-string[org], @to-string[num]}) (define (dissertation-location #:institution org #:degree [degree "PhD"]) @elem{@to-string[degree] dissertation, @to-string[org]}) (define (book-chapter-location location #:pages [pages #f] #:series [series #f] #:volume [volume #f] #:publisher [publisher #f]) (let* ([s @elem{In @italic{@elem{@to-string[location]}}}] [s (if series @elem{@|s|, @to-string[series]} s)] [s (if volume @elem{@|s| volume @to-string[volume]} s)] [s (if pages @elem{@|s|, pp. @(to-string (car pages))--@(to-string (cadr pages))} s)] [s (if publisher @elem{@|s| @to-string[publisher]} s)]) s)) ;; ---------------------------------------- (define (author-name first last #:suffix [suffix #f]) (make-author-element #f (list (format "~a ~a~a" (if (abbreviate-given-names) (given-names->initials first) first) last (if suffix (format " ~a" suffix) ""))) (format "~a ~a~a" last first (if suffix (format " ~a" suffix) "")) last)) (define (org-author-name org) (make-author-element #f (list org) org org)) (define (other-authors) (make-other-author-element #f (list "Alia") (list "al" ._) (list "al" ._))) (define (authors name . names*) (define names (map parse-author (cons name names*))) (define slash-names (string-join (map (compose1 content->string author-element-names) names) " / ")) (define cite (case (length names) [(1) (author-element-cite (car names))] [(2) (if (other-author-element? (cadr names)) (list (author-element-cite (car names)) " et al" @._) (list (author-element-cite (car names)) " and " (author-element-cite (cadr names))))] [else (list (author-element-cite (car names)) " et al" ._)])) (make-author-element #f (let loop ([names names] [prefix 0]) (cond [(null? (cdr names)) (case prefix [(0) names] [(1) (if (other-author-element? (car names)) (list " et al" ._) (list " and " (car names)))] [else (if (other-author-element? (car names)) (list ", et al" ._) (list ", and " (car names)))])] [else (case prefix [(0) (list* (car names) (loop (cdr names) (add1 prefix)))] [else (list* ", " (car names) (loop (cdr names) (add1 prefix)))])])) slash-names cite)) (define (editor name) (let ([name (parse-author name)]) (make-author-element #f (append (element-content name) '(" (Ed.)")) (author-element-names name) (author-element-cite name)))) (define (to-string v) (format "~a" v))