% embed all category images in postscript once #(define-markup-list-command (embed-category-images layout props)() (map (lambda (category) (interpret-markup layout props (markup #:epsfileembed (category-image-path (symbol->string (car category)))))) category-names)) % print a markup-list in columns #(define-markup-list-command (columnlayout layout props cols margin heightpair lines) (integer? number? pair? markup-list?) (let create-col-page ((line-width (- (/ (chain-assoc-get 'line-width props (ly:output-def-lookup layout 'line-width)) cols) margin )) (cols cols) (height (car heightpair)) (restlines lines)) (cons (interpret-markup layout props (make-fill-line-markup (map (lambda (foo) (make-general-align-markup Y UP (make-override-markup '(baseline-skip . 1) (make-column-markup (let add-to-col ((lines restlines) (height-left height)) (set! restlines lines) (if (null? lines) '() (let* ((line-to-stencil (lambda (line) (interpret-markup layout (cons (list (cons 'line-width line-width) (cons 'baseline-skip 1)) props) (markup line)))) (stencil-height (lambda (stencil) (interval-length (ly:stencil-extent stencil Y)))) (linestencil (line-to-stencil (car lines))) (current-line-height (stencil-height linestencil)) (new-height-left (- height-left current-line-height)) (next-line-height (if (null? (cdr lines)) current-line-height (stencil-height (line-to-stencil (cadr lines))))) (no-space-for-next-line (and (< next-line-height current-line-height) (< new-height-left next-line-height))) ) (if (or (< new-height-left 0) no-space-for-next-line) '() (cons (markup #:stencil linestencil) (add-to-col (cdr lines) new-height-left)))))))))) (make-list cols)))) (if (null? restlines) (list) (create-col-page line-width cols (cdr heightpair) restlines))))) %%%%%%%%%%%%%%%%%%%%%%% %%%Funktionen für Inhaltsverzeichnis % geklaut von da: % http://lsr.dsi.unimi.it/LSR/Snippet?id=763 % Usage: % - define and index item with \indexItem $sortstring $markup % - use \indexSection $sortstring $markup to divide the index into several sections % - display the alphabetical index with \markuplines \index % code ist mostly taken from ./ly/toc-init.ly and just renamed and slightly modfied %% defined later, in a closure #(define*-public (add-index-item! markup-symbol text sorttext #:optional label) #f) #(define-public (index-items) #f) #(let ((index-item-list (list))) (set! add-index-item! (lambda* (markup-symbol text sorttext #:optional (label (gensym "index"))) (set! index-item-list ;; We insert index items sorted from the beginning on and do ;; not sort them later - this saves pretty much computing time (insert-alphabetical-sorted! (list label markup-symbol text ;; this crazy hack is necessary because lilypond depends on guile 1.8 atm ;; and so the cool unicode conversion functions cannot be used (ly:string-substitute " " "" (ly:string-substitute "…" "" (ly:string-substitute "Č" "C" (ly:string-substitute "Đ" "D" (ly:string-substitute "Т" "T" (ly:string-substitute "Ä" "Ae" (ly:string-substitute "ä" "ae" (ly:string-substitute "Ö" "O" (ly:string-substitute "ö" "oe" (ly:string-substitute "Ü" "U" (ly:string-substitute "ü" "ue" sorttext)))))))))))) index-item-list)) (make-music 'EventChord 'page-marker #t 'page-label label 'elements (list (make-music 'LabelEvent 'page-label label))))) (set! index-items (lambda () index-item-list))) #(define (insert-alphabetical-sorted! iitem ilist) (if (null? ilist) (list iitem) (if (string-cisymbol category)) (catlist (hashq-ref category-index-hash catsym (list (list label 'indexCategoryMarkup category))))) (if (assq catsym category-names) (hashq-set! category-index-hash catsym (cons (list label markup-symbol text) catlist)) (ly:error "song: <~a> category ~a is not defined!" (markup->string text) category)))) categories) (make-music 'EventChord 'page-marker #t 'page-label label 'elements (list (make-music 'LabelEvent 'page-label label))))) (set! category-index-items (lambda () (append-map (lambda (kv) (reverse (hashq-ref category-index-hash (car kv) (list)))) category-names)))) % code for author index #(define*-public (add-author-index-item! authorIDs markup-symbol text #:optional label) #f) #(define-public (author-index-items) #f) #(let ((author-index-hash (make-hash-table))) (set! add-author-index-item! (lambda* (authorIDs markup-symbol text #:optional (label (gensym "index"))) (for-each (lambda (authorID) (let* ((authorsym (string->symbol authorID)) (authorlist (hashq-ref author-index-hash authorsym (list (list label 'indexAuthorMarkup authorID))))) (hashq-set! author-index-hash authorsym (cons (list label markup-symbol text) authorlist)) )) authorIDs) (make-music 'EventChord 'page-marker #t 'page-label label 'elements (list (make-music 'LabelEvent 'page-label label))))) (set! author-index-items (lambda () (hash-fold (lambda (authorsym authorlist previous-author-items) (append previous-author-items (reverse authorlist))) '() author-index-hash)))) #(define-markup-command (with-link-symbol-ref layout props symbol arg) (symbol? markup?) "call with-link with the label referenced by symbol" (let ((label (chain-assoc-get symbol props))) (interpret-markup layout props (markup #:with-link label arg)))) #(define-markup-command (category-image-symbol-ref layout props size symbol) (number? symbol?) "call category-image with the category referenced by symbol" (let ((category (chain-assoc-get symbol props))) (interpret-markup layout props (markup #:category-image size category)))) #(define-markup-command (category-name-symbol-ref layout props symbol) (symbol?) "get the name of a category referenced by symbol" (let* ((category (chain-assoc-get symbol props)) (catname (assq (string->symbol category) category-names))) (interpret-markup layout props (markup #:override (cons 'baseline-skip 3.5) (if catname (make-left-column-markup (string-split (cadr catname) #\newline)) category))))) #(define-markup-command (index-item-with-pattern layout props)() (let ( (text (chain-assoc-get 'index:text props)) (page (chain-assoc-get 'index:page props)) (width (- (chain-assoc-get 'line-width props) (interval-length (ly:stencil-extent (interpret-markup layout props "XXXX") X)))) ) (interpret-markup layout props (make-column-markup (let ((revlist (if (markup? text) (list text) (reverse (map (lambda (stil) (markup #:stencil stil)) (wordwrap-string-internal-markup-list layout (cons (if (chain-assoc-get 'alternative text) (list (cons 'line-width width) (cons 'font-shape 'italic)) (list (cons 'line-width width))) props) #f (chain-assoc-get 'rawtext text)))))) (target-size-markup (make-column-markup (list (make-simple-markup "Agj") (make-vspace-markup 0.2)))) ) (reverse (map (lambda (m) (make-size-box-to-box-markup #f #t m target-size-markup)) (cons (make-with-link-symbol-ref-markup 'index:label (make-fill-with-pattern-markup 1 RIGHT "." (car revlist) page)) (cdr revlist))))))))) \paper { indexItemMarkup = \markup { \index-item-with-pattern } indexSectionMarkup = \markup \override #'(baseline-skip . 1.5) \left-column { \sans \bold \fontsize #3 \fromproperty #'index:text \null } indexCategoryMarkup = \markup \override #'(baseline-skip . 1.5) \column { \fill-line { \line { \vcenter \category-image-symbol-ref #7 #'index:text \hspace #3 \vcenter \sans \bold \fontsize #3 \category-name-symbol-ref #'index:text } \null } \vspace #.4 } indexAuthorMarkup = \markup \override #'(baseline-skip . 1.5) \left-column { \vspace #1 \sans \bold \fontsize #3 #(make-on-the-fly-markup (lambda (layout props m) (interpret-markup layout props (make-justify-string-markup (format-author layout (chain-assoc-get 'index:text props #f) #f)))) (make-null-markup)) \vspace #.4 } } #(define (prepare-item-markup items layout) (map (lambda (index-item) (let ((label (car index-item)) (index-markup (cadr index-item)) (text (caddr index-item))) (markup #:override (cons 'index:label label) #:override (cons 'index:page (markup #:custom-page-number label -1)) #:override (cons 'index:text text) (ly:output-def-lookup layout index-markup)))) (items))) #(define-markup-list-command (index-in-columns-with-title layout props index-type title-markup) (symbol? markup?) ( _i "Outputs index alphabetical sorted or in categories" ) (let ((items (case index-type ((alphabetical) index-items) ((categories) category-index-items) ((authors) author-index-items))) (title (interpret-markup layout props title-markup))) (cons title (interpret-markup-list layout props (make-columnlayout-markup-list songTocColumns 2 (let ((h (- (ly:output-def-lookup layout 'paper-height) 12))) (cons (- h (interval-length (ly:stencil-extent title Y))) h)) (prepare-item-markup items layout)))))) indexItem = #(define-music-function (parser location sorttext text) (string? markup?) "Add a line to the alphabetical index, using the @code{indexItemMarkup} paper variable markup." (add-index-item! 'indexItemMarkup text sorttext)) indexSection = #(define-music-function (parser location sorttext text) (string? markup?) "Add a section line to the alphabetical index, using @code{indexSectionMarkup} paper variable markup. This can be used to divide the alphabetical index into different sections, for example one section for each first letter." (add-index-item! 'indexSectionMarkup text sorttext)) #(define (extract-and-check-vars-from-header bookheader varlist) (let* ((headervars (hash-map->list cons (struct-ref (ly:book-header bookheader) 0))) (extract-var-and-check (lambda (headervar) (let* ((variableref (assoc-ref headervars headervar)) (extracted (if variableref (variable-ref variableref) #f))) (if (and extracted (not (and (string? extracted) (string-null? extracted)))) extracted #f))))) (map (lambda (varsymbol) (cons varsymbol (extract-var-and-check varsymbol)) ) varlist))) headerToTOC = #(define-music-function (parser location header label) (ly:book? symbol?) (define (all-author-ids authors) (let ((poetIds (find-author-ids-by 'text authors)) (translatorIds (find-author-ids-by 'translation authors)) (versePoetData (find-author-id-with-part-numbers 'verse authors)) (composerIds (find-author-ids-by 'melody authors)) (verseComposerData (find-author-id-with-part-numbers 'meloverse authors)) (voiceComposerData (find-author-id-with-part-numbers 'voice authors)) (compositionIds (find-author-ids-by 'composition authors)) (bridgeIds (find-author-ids-by 'bridge authors)) (interludeIds (find-author-ids-by 'interlude authors))) (delete-duplicates (append poetIds translatorIds (map car versePoetData) composerIds (map car verseComposerData) (map car voiceComposerData) compositionIds bridgeIds interludeIds)) )) (let* ( (extractedheadervars (extract-and-check-vars-from-header header '(title starttext alttitle altalttitle categorytitle categories authors))) (title (assq-ref extractedheadervars 'title)) (starttext (assq-ref extractedheadervars 'starttext)) (alttitle (assq-ref extractedheadervars 'alttitle)) (altalttitle (assq-ref extractedheadervars 'altalttitle)) (categorytitle (assq-ref extractedheadervars 'categorytitle)) (categories (assq-ref extractedheadervars 'categories)) (authors (assq-ref extractedheadervars 'authors)) (add-to-toc! (lambda (toctitle tocmarkup) (add-index-item! 'indexItemMarkup tocmarkup toctitle label))) ) (if categories (add-category-index-item! (string-tokenize categories) 'indexItemMarkup (cons (list (cons 'rawtext (if categorytitle categorytitle title))) '()) label)) (if authors (add-author-index-item! (all-author-ids authors) 'indexItemMarkup (cons (list (cons 'rawtext (if categorytitle categorytitle title))) '()) label)) (if starttext (add-to-toc! starttext (cons (list (cons 'rawtext starttext) (cons 'alternative #t)) '()))) (if alttitle (add-to-toc! alttitle (cons (list (cons 'rawtext alttitle) (cons 'alternative #t)) '()))) (if altalttitle (add-to-toc! altalttitle (cons (list (cons 'rawtext altalttitle) (cons 'alternative #t)) '()))) (if title (add-to-toc! title (cons (list (cons 'rawtext title)) '())) #{ #}) )) %% https://github.com/NalaGinrut/guile-csv/blob/master/csv/csv.scm #(define* (sxml->csv sxml port #:key (delimiter #\,)) (let* ((d (string delimiter)) (csv (map (lambda (l) (string-join l d)) sxml))) (for-each (lambda (l) (format port "~a~%" l)) csv))) #(define csv-write sxml->csv) #(define-markup-command (write-toc-csv layout props) () (define (csv-escape field) (if (string-null? field) field (string-append "\"" (ly:string-substitute "\n" "\\n" (ly:string-substitute "\"" "\\\"" field)) "\""))) (define (format-authors authorIds) (string-join (map (lambda (authorId) (format-author layout authorId #f)) authorIds) ", ")) (define cr-regex (ly:make-regex "\r")) (define crlf-regex (ly:make-regex "\r\n")) (define para-sep-regex (ly:make-regex "\n[ \t\n]*\n[ \t\n]*")) (define whitespace-regex (ly:make-regex "[ \t\n]+")) (define leading-whitespace-regex (ly:make-regex "^[ \t\n]+")) (define trailing-whitespace-regex (ly:make-regex "[ \t\n]+$")) (define (cleanup-whitespaces str) (ly:regex-replace leading-whitespace-regex (ly:regex-replace trailing-whitespace-regex (ly:regex-replace whitespace-regex str " ") "") "")) (define (format-info-paragraphs text) (let* ((para-strings (ly:regex-split para-sep-regex (ly:regex-replace cr-regex (ly:regex-replace crlf-regex text "\n") "\n"))) (para-lines (map cleanup-whitespaces para-strings))) (string-join para-lines "\n"))) (define (generate-toc-csv labelPageTable) (let ((song-lines (map (lambda (song) (let* ((filename (symbol->string (car song))) (songvars (cdr song)) (page-number (number->string (assoc-get (assq-ref songvars 'label) labelPageTable))) (extractedheadervars (extract-and-check-vars-from-header (assq-ref songvars 'header) '(title starttext alttitle altalttitle categorytitle categories authors year_text year_melody year_translation year_composition infotext translation pronunciation copyright source))) (headervar-or-empty (lambda (varsym) (let ((extracted (assq-ref extractedheadervars varsym))) (if extracted extracted "")))) (authors (assq-ref extractedheadervars 'authors)) (poetIds (find-author-ids-by 'text authors)) (translatorIds (find-author-ids-by 'translation authors)) (versePoetData (find-author-id-with-part-numbers 'verse authors)) (composerIds (find-author-ids-by 'melody authors)) (verseComposerData (find-author-id-with-part-numbers 'meloverse authors)) (voiceComposerData (find-author-id-with-part-numbers 'voice authors)) (compositionIds (find-author-ids-by 'composition authors)) (bridgeIds (find-author-ids-by 'bridge authors)) (interludeIds (find-author-ids-by 'interlude authors))) (map csv-escape (list filename page-number (headervar-or-empty 'title) (headervar-or-empty 'starttext) (headervar-or-empty 'alttitle) (headervar-or-empty 'altalttitle) (headervar-or-empty 'categorytitle) (headervar-or-empty 'categories) (format-authors (append poetIds (map car versePoetData))) (format-authors translatorIds) (format-authors (append composerIds compositionIds bridgeIds interludeIds (map car voiceComposerData) (map car verseComposerData))) (headervar-or-empty 'year_text) (headervar-or-empty 'year_melody) (headervar-or-empty 'year_translation) (headervar-or-empty 'year_composition) (headervar-or-empty 'copyright) (headervar-or-empty 'source) (format-info-paragraphs (headervar-or-empty 'infotext)) (format-info-paragraphs (headervar-or-empty 'translation)) (format-info-paragraphs (headervar-or-empty 'pronunciation)) )))) (alist-delete 'imagePage (alist-delete 'emptyPage song-list))))) (call-with-output-file "toc.csv" (lambda (port) (csv-write (cons '( "filename" "page-number" "title" "starttext" "alttitle" "altalttitle" "categorytitle" "categories" "poets" "translators" "composers" "year_text" "year_melody" "year_translation" "year_composition" "copyright" "source" "infotext" "translation" "pronunciation" ) song-lines) port)) ))) ; we use a delayed stencil to have all the page references available (ly:make-stencil `(delay-stencil-evaluation ,(delay (let* ((table (ly:output-def-lookup layout 'label-page-table))) (generate-toc-csv (if (list? table) table '())) empty-stencil)))))