[all pages:] introduction message/location/muli format dtd xantlr tdom ops paisley metajava umod option auxiliaries d2d downloads & licenses people bibliography APPENDICES:: white papers white papers 2 white papers 3 project struct proposal SOURCE:option.dtd SOURCE:dtd.umod DOC:deliverables.ddf DOC-DE:deliverables.ddf DOC:mtdocpage.ddf DOC-DE:mtdocpage.ddf DOC-EN:lablog.ddf SOURCE:basic.dd2 DOC:xslt.ddf SOURCE:xslt.dd2 DOC:meta.ddf [site map]
All pages: introduction message/location/muli format dtd xantlr tdom ops paisley metajava umod option auxiliaries d2d downloads & licenses people bibliography APPENDICES:: white papers white papers 2 white papers 3 project struct proposal SOURCE:option.dtd SOURCE:dtd.umod DOC:deliverables.ddf DOC-DE:deliverables.ddf DOC:mtdocpage.ddf DOC-DE:mtdocpage.ddf DOC-EN:lablog.ddf SOURCE:basic.dd2 DOC:xslt.ddf SOURCE:xslt.dd2 DOC:meta.ddf [site map]
DOC-EN:lablog.ddf | bandm meta_tools | DOC:xslt.ddf |
APPENDIX: The ddf Document Type Definition File for the General Purpose Document Architecture d2d_gp
The following text shows the d2d text type definition (according to the d2d ddf text type definition mechanism), which defines the general purpose document architecture d2d_gp. Its basic outlines are explained in in a dedicated sectionof the d2d user doc.
The top-level definitions are either to be used directly, or to be instantiated, parametrized and taylored to the user's need by the parametrization mechanimsms described in another dedicated section of the d2d user doc.
Please refer also to
the automatically generated html documentation.
Please note that this docu is a comment on the instantiated("dynamic")
case:
The top-level module "basic.deliverables" (to be found at the end
of the following source text) does import other modules
(from the preceeding parts of the following source text)
not in an unaltered way, but by parameterizing
them. And so do the other modules, on lower instantiation level, recursively.
This is to be seen clearly in the following sources in the "import ..."
at the start of each module.
So the dynamic documentation differs substantially from the following
file in all regular expressions which are affected by parameterization.
// basic.dd2, General Purpose and Simple Publishing Architecture //======================================================== module basic //======================================================== xmlns a = "http://bandm.eu/doctypes/d2d_gp/basic" is default // is used by several modules (inlineElements, table, image), but // may appear only once as an xslt:param declaration: docu to_xhtml_1_0 = "#param user.showlabels" docu to_xhtml_1_0 = #d2d #param user.refTitle #xp 'inline' // maybe "toolTip" "inline" or none of these //is needed by interdocument AND simpleImage #template#name copyAltFromContext #choose #when a:alt #attribute alt #foreach a:alt#call a:getOneLineText #when a:image_alt // FIXME ???? DISLOC ?? #attribute alt #foreach a:image_alt#call a:getOneLineText #when @a:alt #attribute alt#valueof @a:alt #when ancestor::a:floating/a:caption #attribute alt#foreach ancestor::a:floating/a:caption#call a:getOneLineText #other#message#terminate ERROR, neither direct nor indirect \"alt\" text found for embedded object #valueof . #/d2d docu user_en = #d2d #p This module is the central starting point for all our current applications of #link ../d2d.html#text d2d#/link, and a convenient starting point for any user. It contains (a) sub-modules and (b) functional libraries. #p The sub-modules are listed the beginning of this file. Their sequential order in the #link 0/bandm.eu/metatools/docs/usage/file_basic_ddf.html#text source file#/link // #link ../file_basic_ddf.html#text source file#/link is bottom-up in complexity: the very first sub-modules define infra-structure (like #link #loc __MODULE__basic.sets#text set#/link); the middle ones model different organizations of text (starting with #link#loc __MODULE__basic.inlineElements#text inlineElements#/link); at the end these are combined to a complete text corpus architecture (starting with #link #loc __MODULE__basic.structure#text structure#/link). #p The modules are in very different state of development: Some of them are fully operative (like #link #loc __MODULE__basic.citation#text citation#/link); others are operative, but utmost simple (like #link #loc __MODULE__basic.simpleLists#text simpleLists#/link); few are currently not employed at all, but only for demonstrative purpose (like #link #loc __MODULE__basic.personal_names_de#text personal_names_de#/link); #p Xslt code is currently (20150423) only given into xhtml_1_0, and into source files for different processors, allowing to embed the graphical output of lilypond, musixtex and LaTeX source directly. This is supported by the module #link #loc __MODULE__basic.interDocuments#text interDocuments#/link together with a dedicated Makefile system. #p The xslt code in the following modules assumes some procedures (for string and file name manipulation, etc.) to be provided by the basic library "libbasic.xslt". #nl On the other side of the scale it assumes that the top-level module, which finally plugs all the sub-modules together, provides also some global services like a template with name "muliDefault", to find the translation into the selected language, and some parameter definitions, which have default values depending on the top-level intentions ($user.p_kind_filter, GEHT ABER DOCHNICHT ZURZEITetc.) #p ((FIXME The basic lib could be moved HERE, as an inline code.)) #nl ((FIXME The muliDefault could be moved from top-level to lowest level (REALLY?) )) #/d2d //------------------------------------------------------ module sets //------------------------------------------------------ chars all = 0x0001 .. 0xFFFE docu user_en all = "Currently only four hex digits are decoded for character values, so that the higher unicode planes (beyond 0x0_FFFF) can currently (201200208) not be described on the d2d #emph!input! side. (They can be output by xslt)." chars decimalDigit = '0' .. '9' chars upto1 = '01' chars upto2 = upto1 U '2' chars upto3 = upto2 U '3' chars decimalPositive = @decimalDigit ~+ chars ascii_lower = 'a'..'z' chars ascii_upper = 'A'..'Z' chars ascii_alpha = ascii_lower U ascii_upper chars alpha = @ascii_alpha // FIXME alpha_lower U alpha_upper chars alpha_lower = @ascii_lower // FIXME plus additonal non-ascii lower case letters chars alpha_upper = @ascii_upper // FIXME plus additonal non-ascii upper case letters chars identFirstChar = @alpha chars identInnerChar = alpha U '_' U '-' U decimalDigit chars ident = @identFirstChar ~ @identInnerChar ~* chars whitespace = ' ' U 0x0009 U 0x000a U 0x000D chars non_whitespace = all - whitespace end module // sets //------------------------------------------------------ module xmlInfra plain text is trimmed xmlns xml = "http://www.w3.org/XML/1998/namespace" is default import S from sets //------------------------------------------------------ docu user_en = #d2d Xml infrastructure, definition of ubiquituous attributes and their data type. Corresponds mostly to the xml namespace and the definitions in #link 0/www.w3.org/XML/1998/namespace#/link, namely w.r.t. "xml:lang", "xml:space", "xml:base" and "xml:id" #/d2d chars nameStartChar = ':_' U 'A'..'Z' U 'a'..'z' U (0x00C0..0x02FF - 0x00D7) - 0x00F7 U (0x0370..0x1FFF - 0x027E) U 0x200C U 0x200D U 0x2070..0x218F U 0x2C00..0x2FEF U 0x3001..0xD7FF U 0xF900..0xFDCF U 0xFDF0..0xFFFD chars nameChar = nameStartChar U '-.' U '0'..'9' U 0x00B7 U 0x0300..0x036F U 0x203F..0x2040 chars name = @nameStartChar ~ @nameChar ~* chars nmtoken = @nameChar ~* docu user_en name = #d2d #commentchar \ This definitions closely follows #link http://www.w3.org/TR/2008/REC-xml-20081126/#loc NT-Name#/link, but still excludes 0x0001,0000..0x000E,FFFF, because the d2d input parser currently (20120308) supports only sixteen bit characters. #commentchar / #/d2d public chars id = @name with xmlrep att chars lang = @S.ascii_lower~@S.ascii_lower ~( "-" ~ @S.ascii_upper~@S.ascii_upper)? with xmlrep att = "xml:lang" chars langs = @lang~('+'~@lang)~* with xmlrep att = "xml:lang" docu user_en lang = #d2d The xml namespace description in #link 0/www.w3.org/XML/1998/namespace#/ refers to #link 0/www.w3.org/TR/xml/#loc sec-lang-tag#/link for the xml:lang attribute, which in turn refers to [IETF BCP 47], which shall consist of "RFC 4646: Tags for Identifying Languages", #link 0/tools.ietf.org/html/rfc4646#/link and "RFC 4647: Matching of Language Tags", #link 0/tools.ietf.org/html/rfc4647#/link. This is currently (20150423) already obsoleted by #link 0/tools.ietf.org/html/rfc5646#/link. #nl The second does define the processing of LISTS of language selectors, but not their syntax. Anyhow, xml:lang seems to be forseen only for one(1) such language code. Our handling deliberately violates the specification and allows more than one language, cf. parser "langs". #nl The precedessor is [IETF RFC 3066] #link 0/www.ietf.org/rfc/rfc3066.txt#/link. #nl The character parser implemented here is a very primitive variant of the older "two-(plus-two)" format of the ISO language code, and stores it as the xml standard attribute "xml:lang". #/d2d docu user_en langs = #d2d Character parser for more than one language, written without spaces and separated by #src!+! character. They are ALSO stored into ONE SINGLE xml.lang attribute, because the spec is not totally clear about the cardinality of language values, ie. of tokens in this attribute. INDEED, more than one language value are frequently required. The xslt library libbasic.d2d provides a template "#src!firstLangInList(String p-list)!" used for extracting the first language. This is normally treated as dominant/main/default language. #/d2d enum space = default, preserve with xmlrep att public chars base = #chars // FIXME more exact structure def: URL with xmlrep trimmed att end module // xmlInfra //------------------------------------------------------ module physical //------------------------------------------------------ docu user_en = "Physical mark-up is mostly deprecated. It follows the (old!) html way of physical mark-up. In this basic model this can only be applied to physical mark-up itself/recursively. The declaration is naive, allowing multiple (nested) application of the same modifier. It can NOT be applied to semantic mark-up. Additionally this module defines some html/xml specific elements." tags MIXTURE = bold| ital| ulin| blink| strikeout| scaps| hh | larger| smaller| indexlo | indexhi | spanstyle docu user_en MIXTURE = "What is recursively subject to physical mark-up" public tags EXPORTMIXTURE = @MIXTURE | src | verbatimXml docu user_en EXPORTMIXTURE = "What can be adressed from outside, from some module using it." tags bold, ital, ulin, blink, strikeout, scaps, larger, smaller, indexlo, indexhi = (#chars | @ MIXTURE)* tags spanstyle = #implicit style, (#chars | @ MIXTURE)* tags style = #chars with xmlrep att tags hh = #empty docu to_xhtml_1_0 hh = "#valueof $a:const.hashmark" docu user_en hh = "Draws a hashmark sign. Is currently the only element which can be included in #link #loc __DECLARATION__basic.physical:src #text #src!src!." public tags verbatimXml = #chars with input verbatim docu user_en verbatimXml = "A by-pass when generating html/xhtml documents. This element is intended to contain special html fragments which shall be included verbatim in the output. This is achieved simply by an xslt rule which disables output escaping." docu to_xhtml_1_0 verbatimXml = "#valueof .#noescape" tags hr = #empty docu to_xhtml_1_0 hr = "#hr/" docu user_en hr = "Draws a horizontal line, leaned from html." docu to_xhtml_1_0 spanstyle = "#span #style {@a:style} #apply#xp (*|text())" docu user_en spanstyle = "Inserts an html-<span style='xxx'>" docu to_xhtml_1_0 bold = "#b#apply" docu user_en bold = "Explicitly select bold text. Deprecated. Consider to use #src!basic.inlineElements:xemph! instead." docu to_xhtml_1_0 ital = "#i#apply" docu user_en ital = "Explicitly select italic text. Deprecated. Consider to use #src!basic.inlineElements:emph! instead." docu to_xhtml_1_0 ulin = "#span #style text-decoration:underline; #apply" docu user_en ital = #d2d Explicitly select underlined text. Deprecated. #nl The xslt rule generates a #src!<span style="text-decoration:underline;">!, because the old #src!<u>! tag has not been taken over to xhtml 1.0 strict. #/d2d docu to_xhtml_1_0 blink = "#span #style text-decoration:blink; #apply" docu user_en blink = #d2d Explicitly select blinking text. Not only deprecated, but also bad manners. #nl The xslt rule generates a #src!<span style="text-decoration:blink;">!, because the old #src!<blink>! tag has not been taken over to xhtml 1.0 strict. #/d2d docu to_xhtml_1_0 strikeout = "#span #style text-decoration:line-through; #apply" docu user_en blink = "Explicitly select stroke-out text. Deprecated." docu to_xhtml_1_0 scaps = "#span #style font-variant:small-caps; #apply" docu user_en scaps = "Explicitly select small caps. Deprecated. Consider to use #src!basic.inlineElements:opus! or #src!basic.inlineElements:pers! instead." docu to_xhtml_1_0 larger = "#span #style font-size:130%; #apply" docu user_en larger = "Explicitly select larger font. Deprecated." docu to_xhtml_1_0 smaller = "#span #style font-size:75%; #apply" docu user_en smaller = "Explicitly select smaller font. Deprecated." docu to_xhtml_1_0 indexlo = "#sub #apply" docu user_en indexlo = "Explicitly select subscript. Deprecated." docu to_xhtml_1_0 indexhi = "#sup #apply" docu user_en indexhi = "Explicitly select superscript. Deprecated." tags src = (#chars | hh)* docu user_en src = "Small piece of program source text, character data only! Is rendered inline and in tele-type fonts. Can include #emph!only character data!, or #link #loc __DECLARATION__basic.physical:hh #text #src!#hh hh/! #/link to include a hashmark sign \"#hh\"." docu to_xhtml_1_0 src = "#span#class source #apply" end module // physical //------------------------------------------------------ module inlineElements xmlns xml = "http://www.w3.org/XML/1998/namespace" local node xmlrep naming = join upcased plain text is trimmed import S from sets import X from xmlInfra //------------------------------------------------------ docu user_en = "This module contains semantic (and a few physical) elements/text particles. Most of them can appear in every context. This module must be instantiated with the contents for emphasizing." public tags MIXTURE = nl | ldots | label | ref | pers | opus | persop | emph | xemph docu user_en MIXTURE = "Refers to all tags defined in this module. Meant as an export mechanism." tags emph = (EMPH_CONTENTS)* tags xemph = (EMPH_CONTENTS)* tags EMPH_CONTENTS = #GENERIC docu user_en emph = "Something emphasized." docu user_en xemph = "Something strongly emphasized." docu user_en EMPH_CONTENTS = "To be replaced with an expression allowing all those things which can appear as subject to emphasis. (Should include plain pcdata.)" // xhtml:i and xhtml:span have same content model docu to_xhtml_1_0 emph = "#i #apply" docu to_xhtml_1_0 xemph = "#b #apply" docu to_latex emph = " tags nl = #empty docu to_xhtml_1_0 nl = "#br/" docu user_en nl = "Forces a line break, but not a new paragraph." tags ldots = #empty docu to_xhtml_1_0 ldots = "#text ...#/text" docu user_en ldots = "Draws three low dots surrounded by spaces." chars label = @S.ident docu user_en label = #d2d Such a label binds an identifier to that certain location of the text body where itself happens to appear. #nl By setting the swich $user.showlabels="yes", these labels and others will be printed in the output, for proof reading. #nl Putting a label is possible nearly anywhere. If it is in the middle of some text flow, it just marks this special text position. But if it is in a structured object, e.g. the caption of a diagram, its meaning is possibly referring to this diagram. This interpretation and the way a reference is generated may depend on the "ref" elements. #/d2d docu to_xhtml_1_0 label = #d2d #if $user.showlabels='yes' #x-var tx #xp text() #if preceding::a:label[text()=$tx] or following::a:label[text()=$tx] #span #class error #i multiple label // FIXME font-size:75%; #/if #span #class debug #valueof text() #/if #a #name {text()} #/template #/d2d chars ref = @S.ident docu user_en ref = #d2d Refering is possible nearly anywhere. Mostly it is to a label. #nl A hyper-ref link will be generated and its textual appearance will be constructed automatically. There are many different strategies, eg. generating text like "in this list" or "in section 2.1.2 (TITLE OF THE SECTION)". These strategies are selectable and respect the current base language of the text. #nl (This is achieved by calling the named template "muliDefault", which makes translations into the "default" language.) #nl The #emph!implementation! of each rendering strategy starts always with the Xslt template matching "a:ref" and works as follows: #list #i Identify the positional context of the reference, e.g. the innermost enclosing section, table, list, etc. #i Find the label element referred to. #i If not found, an error is generated and an alerting meta-element inserted in the output. #i Otherwise, start a #src!mode=getreftext! visiting of the label, with the refering context stored to parameters. #i This visiting traverses upward, up to a nearest common parent, while accumulating the text corresponding to the downward navigation so far. #i In case of lists and tables as part of TEXT FLOW, the generation of navigation text is delayed, until the next hosting level is found and numbering/counting will be possible. #p The text for sections, chapters, parts etc. is constructed to #source Abschnitt 17.3.1 ("<title-of-the section>") ^ ^ iff user.refTitle=='inline' ^ (otherwise it may be rendered as toolTip) ^ translated to doc's default language #/s=====ource #p "floatings" are identified by their genuine reference mechanism. #nl "lists", "list points", "tables" and "table rows" by counting relative to the nearest "container", which is a list, a table, a floating, a section, etc. #nl Iff reference and label are in a common container (list, table, or lowest level section), this part of the reference text is abbreviated to "this list"/"this table"/"this section", etc. #nl There is currently NO further abbreviation implemented, like "next section" or "preceding chapter". #p If you want to determine the refering text appearance #emph!freely! by yourself, using #emph!all mark-up! currently available, you can always use an explicit "#src!#hh link!" construct, using only the local part of the link target ("fragment identifier", in w3c speak) #nl So you can link to "mylabel" directly by #nl #commandchar$ "$src ... and #link#loc mylabel#text here is the #emph!real! text#/link ...$/src" $commandchar# #/d2d // FIXME whether container elements are transparent or count (ie. have an impact in // the "mode=getreftext" traversal, should be controled by meta-attributes (constant per type). // FIXME: In an multi-lingual setting, the language of the paragraph of the // REFERENCE must be used for translation. (see template "find_innermost_lang") docu to_xhtml_1_0 ref = #d2d #x-var tx #xp text() #x-var lb #xp /descendant::a:label[text()=$tx][1] #if $user.showlabels="yes" #span#class debug #valueof concat("(",$tx,")") #/if #choose #when $lb #a#href {concat($a:const.hashmark,text())} // FIXME darf kein NL dazwischen?? #if $user.refTitle='toolTip' #x-var TX #apply #mode getreftextTitle #xp $lb #/x-var #if $TX #attribute title #valueof $TX #/if #/if #apply #mode getreftext #xp $lb #arg referer #xp . #other #message#text label #valueof $tx #text not found.#/message #span #class error #text [label #valueof $tx #text not found ???]#/text #/span #template #match * #mode getreftextTitle #apply #mode getreftextTitle #xp parent::* #template #match * #mode getreftext #param referer#/param #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #apply #xp parent::node() #mode getreftext //ATTENTION: "*" does not visit root #arg referer#xp $referer #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #template #match / #mode getreftext #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #call muliDefault#arg p-key REF_thisDoc#/call #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #template #name refSameTest #param referer#/param // is label element, test whether it is under $other #param other#/param // is id=string, means current addressing base //#message IN refSameTest >>#valueof!$referer!<< // >>#valueof!generate-id($referer)!<< >>#valueof!$other!<<#/message #choose #when generate-id($referer)=$other #valueof 'true' #when $referer/parent::* #call refSameTest #arg referer #xp $referer/parent::* #arg other #xp $other #other // empty rtf ?? #/template /* S /|\ \ / | T /\ /\ t t t \ /\ tNO NO tNO */ // Called with current-node = current ref-base = S. // Returns the number of all "t" and the position of "T". // Descends in neither or these. // $textSingle may be empty string, then no comma must be generated. #template #name reftextByCount #param referred#/param // = T #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #x-var position #foreach child::*[1] #call countref_right #arg referred #xp $referred #arg nsuri #xp $nsuri #arg localname #xp $localname #/x-var // /**/#message POSITION =>>#valueof!$position!<<#/message #choose #when string-length($position)=1 #if $textSingle #text!, ! #call muliDefault #arg p-key #xp $textSingle #other #text!, ! #call muliDefault #arg p-key #xp $textMultiple #valueof concat(' ', string(string-length(substring-before($position, 'x'))+1)) // Counts all descendants of the given type (excluding their descendants). // Two results: number of all matching elements, and position of the referred, // encoded as string "----x--" : #template #name countref_right // currentnode = cursor; starts with first child of // current ref-base (which has been reached by upward traversal) #param referred#/param #param nsuri#/param #param localname #/param // /**/#message count ref at #valueof concat(local-name(),' ',$localname)#/message #x-var right #choose #when ./following-sibling::* #foreach ./following-sibling::*[1] #call countref_right #arg referred #xp $referred #arg nsuri #xp $nsuri #arg localname #xp $localname #other #valueof '' #/x-var #x-var down #choose #when local-name()=$localname and namespace-uri()=$nsuri #choose #when generate-id()=$referred #valueof "x" #other #valueof "-" #/choose #other #foreach ./child::*[1] #call countref_right #arg referred #xp $referred #arg nsuri #xp $nsuri #arg localname #xp $localname #/x-var #valueof concat($down, $right) #/template // countref_right #/d2d tags pers = #chars docu user_en pers = "Personal name. (This is a candidate for replacement with some more sophisticated, parser based, but locale specific definition.) Our standard to_xhtml_1_0 script renders it to a span/class=pers, which normally results in small caps." docu to_xhtml_1_0 pers = "#span #class pers #apply" docu to_css pers = "span.pers : {font-variant:small-caps;}" tags opus = #chars docu user_en opus = "Title of an opus, a work of art, or any product, (This is a candidate for replacement with some more sophisticated, parser based definition.) Our standard to_xhtml_1_0 script renders it to a span/class=opus, which normally results in slanted small caps." docu to_xhtml_1_0 opus = "#span #class opus #apply" chars persop = [ppers (S.all - '!')*] ~ "!" ~ [popus (S.all - '!')*] ~ "!" docu user_en persop = "Name of its author and name of an opus. Behaves like the sequence of #hh pers and #hh opus. Any separation char must be appended to the former." docu to_xhtml_1_0 persop = "#span #class pers #apply#xp ppers#/span#text! !#span #class opus #apply#xp popus#/span" tags xlang = #implicit X.lang, (XLANG_CONTENTS)* tags XLANG_CONTENTS = #GENERIC docu user_en xlang = "A short sentence/sequence of words in some foreign lanugage, -- less than a whole paragraph. Our standard to_xhtml_1_0 rendering creates a span element with an xml:lang attribute, which you may render by css as you like." docu to_xhtml_1_0 xlang = "#span #call copyLang_direct #apply" end module // inlineElements // ------------------------------------------------------ module calendaric_de import S from sets // ------------------------------------------------------ docu user_en = #d2d This module contains the #xemph!German! form of entering calendaric dates. This moduls is likely to be replaced when instantiating calling modules in a different language. The field 'UTC' may and should be used for the normalized and international format, independent of the language specific front-end representations. Currently a java-coded postprocessor "eu.bandm.tools.d2d2.postproc.Calendaric" fills in this field automatically. #nl #bold Currently not used anywhere,#/ for demonstration purpose only. #/d2d enum monate = jan 1, januar 1, feb 2, februar 2, mar 3, maer 3, "mär" 3, "märz" 3, maerz 3, apr 4, april 4, mai 5, jun 6, juni 6, jul 7, juli 7, aug 8, august 8, sep 9, sept 9, september 9, okt 10, oktober 10, nov 11, november 11, dez 12, dezember 12 with xmlrep element numeric chars calendaric = [dayofmonth @S.upto3? ~ @S.decimalDigit] ~ ".", ([month @S.upto1? ~ @S.decimalDigit] ~"." | [month /*@ FIXME */monate]), [year ("-"?) ~ @S.decimalDigit? ~ @S.decimalDigit? ~ @S.decimalDigit? ~ @S.decimalDigit] public chars date = @calendaric, [UTC "XXXXX"]? with postproc "eu.bandm.tools.d2d2.postproc.Calendaric", xmlrep = calendaric-date docu to_xhtml_1_0 date = "#valueof a:dayofmonth#text .#valueof a:month#text .#valueof a:year" chars time = [hour S.upto2? ~ S.decimalDigit] , ":" , [minute S.decimalDigit ~ S.decimalDigit], ("Uhr"?) end module // calendaric_de // ------------------------------------------------------ module interDocuments local node xmlrep naming = no mangling import S from sets // ------------------------------------------------------ docu user_en = "Primitive elements for embedding of and linking to other documents." tags link = #implicit url, ( text? & loc? & refdate?) // PRE 20240902 ( (blank|top|inframe|framename)? & text? & loc? & refdate?) with local tags text = LINKTEXT* // PRE 20240902 tags blank, top, inframe = #empty // PRE 20240902 tags framename = #chars with xmlrep trimmed tags refdate = #chars end local tags LINKTEXT = #GENERIC docu user_en LINKTEXT = "For the visual representation to contain any mark-up, this must be declared by replacing this generic reference." tags url = #chars with xmlrep trimmed tags loc = #chars with xmlrep trimmed docu user_en link = "Inter document link, very similar to classical html #src!<a>! element. But it is a little bit higher level!" docu user_en url = #d2d The url of a link or a resource.#nl (This is currently NOT YET parsed, but can be any sequence of characters. Errors will occur later, when the link is used !-) #nl Iff the url starts with a #emph!digit=n!, then there must be an n-th entry in the self-structured list in the stylesheet parameter #src!$user.linkurlprefices!, and the digit is replaced by this. #/d2d docu user_en link.text = #d2d The text to print when the link is rendered. It is defined by a generic parameter, so anything may appear there, depending on the instantiation of this module. #nl If this field is left out, then the url will be taken as text.#nl In this case, if the url starts with a digit, then the prefix replacing this digit is taken from the stylesheet parameter #src!$user.linktextprefices!. This allows more readable, more symbolic references to frequently used mount points.#nl If the text is present and starts with an exclamation mark "#src(!)", then this is also replaced by this prefix. #/d2d docu user_en loc = "The location/fragment in the document (what normally appears behind a hashmark!-) A link must contain either an url or a loc entry, or both." docu user_en link.refdate = "The calendaric date when the meant contents of the link had been retrieved." // PRE 20240902 // docu user_en link.blank, link.top, link.inframe, link.framename = // #d2d The target where to open the new link. // #xemph(This cannot be rendered in xhtml 1.0 strict!) // A javascript based implementation must be provided by the user // whenever this "target" feature shall be used in xhtml 1.0 strict. // The standard xslt script will ignore it and emit a warning // iff #src($user.xhtmlVariant) ist set to "strict". // #/d2d docu to_xhtml_1_0 link = #d2d #x-var l-start #xp substring(a:url,1,1) #choose #when string-length(normalize-space(a:url))=0 #if string-length(normalize-space(a:loc))=0 #message ERROR invalid link, both url and fragment location are empty #/if #call makelink #arg p-url #xp '' #arg p-urlprefix #xp '' #arg p-textprefix #xp '' #when contains('0123456789',$l-start) #call makelink #arg p-url #xp substring(a:url,2) #arg p-urlprefix #call a:splitbyfirst #arg p-select #xp $l-start #arg p-list #xp $user.linkurlprefices #arg p-errormsg #xp a:url #/call #arg p-textprefix #call a:splitbyfirst #arg p-select #xp $l-start #arg p-list #xp $user.linktextprefices #arg p-errormsg #xp a:url #other #call makelink #arg p-url #xp a:url #arg p-urlprefix #xp '' #arg p-textprefix #xp '' #/template #param user.linkurlprefices #xp '!http:/!' #/param #param user.linktextprefices#xp '!http:/!' #/param // PRE 20230902 #param user.xhtmlVariant#xp 'strict'#/param #template #name makelink #param p-url#/param #param p-urlprefix#/param #param p-textprefix#/param #param l-loc #choose #when a:loc#valueof concat($a:const.hashmark,normalize-space(a:loc)) // #when a:loc#valueof concat($a:const.hashmark,normalize-space(a:loc/text())) #other #valueof '' #/param //PRE 20240902 // #param l-target // #choose // #when a:top #valueof '_top' // #when a:blank #valueof '_blank' // #when a:inframe #valueof '_self' // #when a:framename #valueof a:framename/text() // #other #valueof '' // #/param #a #href {concat($p-urlprefix,normalize-space($p-url),$l-loc)} //PRE 20240902 // #choose // #when $user.xhtmlVariant='strict' // #if not($l-target='') // #message Trying to set link target in strict xhtml 1.0 is ignored // #other // #if not($l-target='') // #attribute target#valueof $l-target // #/choose #choose #when a:text and starts-with(normalize-space(a:text),'!') #valueof concat($p-textprefix,substring-after(a:text,'!')) #when a:text #apply #xp a:text #other #valueof $p-textprefix #valueof normalize-space($p-url) #valueof normalize-space($l-loc) #/a #/template #/d2d tags treeInclude, treeInclude_PROVIS = #implicit url, path, mode? with xmlrep = treeInclude docu user_en treeInclude = #d2d Includes an xml text corpus into the rendering process. "url" gives the file position, and "mode" gives the mode parameter to the "apply-templates" call. #nl #bold CURRENTLY PATH is not yet operative#/, because there are problems serializing the dynamic xpath extension required for its execution. #/d2d docu user_en treeInclude_PROVIS = "Same as treeInclude, but callable when parsing VERTICALLY stacked text elements." tags path, mode = #chars with xmlrep trimmed docu to_xhtml_1_0 treeInclude = #d2d // NO, w3cdom Xerces does not find base url for text node !?!?!? // #foreach document(./a:url/text()) #foreach document(./a:url) #apply /*==== NO, mode is not a "AVT"-attribute; cannot be set DYNAMICALLY; "#mode {a:mode}" will be rejected; dynamic extension would be needed: #choose #when a:mode #apply #mode a:mode // #apply #mode {a:mode} #other #apply == */ #/d2d tags embed = #implicit type, (alt? & body) with local chars type = @S.ident with xmlrep attribute=embedType tags body = #chars with input verbatim, xmlrep element=embedBody tags alt = EMBED_ALT* end local tags EMBED_ALT = #GENERIC docu user_en embed.type = "Suffix of the generated temporary filename in which the body will be written. Used to select the tool for subsequent processing." docu user_en embed = #d2d Embeds any third-party source text into the source and the rendering result into the output.#nl (Eg. "lilypond" source resulting to a ".png" which is included into the xhtml document.) #nl This processing must be executed by the #bold make system#/ which controls the overall document processing pipeline.#nl The xslt code does only #list #i generate a unique file name #i generate an #src!<img>! element when generating xhtml. #i piping all the thirs-party-source into a bash-script, which will auto-split into source files for these tools. #nl This is done by a second conversion target, "#src!to_embedScript!". #nl The filename suffix is taken from the "type" attribute, and indicates to the further processing which tool(s) has/have to be started (e.g. for creating the png included by the html, or sim.) #/list An #src!alt! #emph!may! be given, but if omitted, it will be inherited from the containing element. In many cases the containing element be a "floating"! #/d2d // does assume a "make" script which makes "png"s of ALL embeds. // EXCEPT those of type "pdfSourceRef", where the pdf is displayed using an // xhtml object tag. // EXCEPT those type "pdfRef", where the "body" is the path to a ready-made pdf, docu to_xhtml_1_0 embed = #d2d #choose #when @a:embedType="pdfSourceRef" #x-var l-url #call create-fileid-for-embedded-image #text!.pdf!#/x-var #p #object #type application/pdf //FIXME SOME back-ends lack escaping of ampersand ? // #data {$user.currentkey}_picts/{$l-url}{$a:const.hashmark}statusbar=0&toolbar=0&navpanes=0 #data {$user.currentkey}_picts/{$l-url} #width 711 #height 956 #br/#br/ YOUR BROWSER DOES NOT SUPPORT PDF EMBEDDING, CLICK #a #href!{$user.currentkey}_picts/{$l-url}! HERE #/a TO VIEW .pdf FILE ! #br/#br/ #when @a:embedType="pdfRef" #p #object #type application/pdf #data {normalize-space(a:embedBody)} #width 711 #height 956 #br/#br/ YOUR BROWSER DOES NOT SUPPORT EMBEDDING; CLICK #a #href!{normalize-space(a:embedBody)}! HERE #/a TO VIEW FILE #valueof a:body#text!.! #br/#br/ #other #div #img #attribute src #valueof $user.currentkey #text!_picts/! #call create-fileid-for-embedded-image #text!.png! #/attribute #call copyAltFromContext #template #name create-fileid-for-embedded-image #text!image!#number #count a:embed#level any #/template #param user.currentkey#/param #/d2d docu to_embedScript embed = #d2d #text! cat > !#call create-fileid-for-embedded-image#text!.!#valueof @a:embedType #text#noescape <<"EOF" #/text #valueof a:embedBody #text! EOF ! #template #match text() //NOP #template #name create-fileid-for-embedded-image #text!image!#number #count a:embed#level any #/template #/d2d end module // interDocuments // ------------------------------------------------------ module personal_names_de import S from sets // ------------------------------------------------------ docu user_en = #d2d Character Parsers for personal names, including honours, middle initials, etc. Will be replaced when the locale/cultur context changes. #nl #bold Currently not used anywhere,#/ for demonstration purpose only. #/d2d chars namepart = S.ascii_upper~S.ascii_lower~* chars nameprefix = S.ascii_lower~* chars personalName = [honours (@namepart~".")+]?, [prename @namepart +], [middleInitials (@namepart~".")*], [familyname "\""~ @namepart] //TEST DQUOTE | [family @namepart +], "," , [honours (@namepart~".")+]?, [prename @namepart +], [middleInitials (@namepart~".")] end module //names_de // ------------------------------------------------------ module citation local node xmlrep naming = no mangling plain text is trimmed import S from sets import X from xmlInfra import LINK from interDocuments // ------------------------------------------------------ /* === TEST 20241113 docu user_en = #d2d A first small solution for bibliographic data, citation keys and bibliographic lists.#nl Its structure is based on "bibTeX" and our experiences in bib-style programming, but heavily trying to modernize and orthogonalize.#nl The grammar definition of the entries is (of course!-) independent of the "kind" of entry, because the complex requirements and implications on the semantical level cannot adequately be expressed by a regular expression. But we decided to #emph!coarsely! reflect these requirements by grouping the components of an entry. Inside most groups the fields may be permutated. #list #i First we require key(implicit), kind, title and one or more authors. #i Additionally subtitle, latextitle (later:htmltitle !?!) #i Then may follow references to copies of the text, which have formalized structure, namely url(s), isbn(s), issn(s), doi(s), etc. #i Then follows #emph!either! a simple crossref to the collection containing the article, #i #emph!or! the standard bibliographic co-ordinates in case of stand-alone publications (place, date, publisher, etc.) #i At last there may follow abstracts and keywords in different languages, and a list of library signatures where a printed copy is available. #/list #p If there is only one single sub-document cited from a collection, its bibliographic co-ordinates may be inlined into the entry for the article. But in contrast to BibTex, we recommend the solution with two(2) separate entries. Nevertheless, we put the "formal references" in the middle of both groups, because in the single-entry case these may either describe just the article, i.e. the head of the entry, or the whole volume, i.e. the tail of the entry. #p #bold Please note#/ that the items contained in #src!editors! are tagged with #src!author!, as the items in #src!authors!. Firstly, the code for rendering is inherited, and secondly editors are professional authors, in most cases. #p #bold Please note#/ that the problem of a generic, standardized and widely accepted bib-entry format is still unsolved. We plan a whitepaper on this subject, and a preliminary link collection is found in metatools/doc/papers/recherche/recherche_biblio.html. #/d2d === */ chars key = (S.ascii_alpha U '-_' U S.decimalDigit)~+ docu user_en key = #d2d The identifier for a bibliographic entry.#nl This could be replaced by something more restricted, like "alpha~+ ~ digit~*". The key must appear in the bib entry and in all cite-elements. When using this standard rendering script it is also #emph!immediately the visible representation! of the citation/link. This is a signficant difference to BibTex, and could of course be done differently by some more elaborated rendering script. #/d2d chars cite = key ~ ('!'~[locus (@S.all - '!')~+]~'!')? docu user_en cite = #d2d A citation is a reference to a bibliographic entry, plus possibly a location indication ("pg 12 to 13"). CURRENTLY both is free text, only excluding the delimiter "#src(!)".#nl Every cite is rendered as a xhtml link to the bibliographic entry. This is assumed to be in the same file, OR in the file given by the stylesheet parameter "#src!$user.bibLocation!". #/d2d docu to_xhtml_1_0 cite = #d2d #call renderCite #/template // "biblio.xml" and "biblio.html" must both be adressable #param user.bibLocation#/param // used by render cite and render crossref // looks for local entries, and for entries in separate file // FIXME more formats/styles of rendering selectable #template #name renderCite #x-var entry #xp /descendant::a:bibentry[a:key/text()=current()/a:key/text()] #/x-var #x-var entryExtern #if (not($entry) and $user.bibLocation) #valueof document(concat($user.bibLocation,'.xml'),.)/descendant::a:bibentry[a:key/text()=current()/a:key/text()] #/x-var #x-var prefix #if not($entry) #valueof concat($user.bibLocation,'.html') #/x-var #text![! #a#href {$prefix}{$a:const.hashmark}__bib{a:key/text()}#/href #title {$entry/a:author} {$entry/a:year} #valueof a:key/text() #/a #if a:locus #text(, )#i#valueof a:locus/text()#/if // #message ENTRY IS >>#valueof concat($entry,'<<')#/message #if not($entry or $entryExtern) #span#class error#text not found#/if #text!]! /* == OLD OK #template #name renderCite #text![! #a#href {$user.bibLocation}{$a:const.hashmark}__bib{a:key/text()}#/href #valueof a:key/text() #/a #if a:locus #text(, )#i#valueof a:locus/text()#/if] == */ #/d2d public tags biblist = ( X.lang? & sortMethod? & abstractLangs? & abstractHiding? ), bibentry + docu user_en biblist = "A list of bibliographic entries, rendered as a whole, e.g. concering sort order.#nl The \"lang\" attribute is taken as the #emph!default language! for the contained bibentries. (See the other components for documentation!)" enum sortMethod = none, chronAsc, chronDesc, key, author docu user_en sortMethod = #d2d How the biblist will be sorted in rendering.#nl CURRENTLY there are #bold two Fixmes#/ with the chronological sort methods:#nl (a) only "year" and "month" are respected, since the internal structure of "date" is not defined.#nl (b) these dates do not carry over via "crossref" #/d2d tags abstractLangs = #chars docu user_en abstractLangs = "A list of language codes, determins the selection and sequential order of included abstracts. The list must be \"self-structured\", i.e. start and end with an arbitrary item separator character." tags abstractHiding = #empty docu user_en abstractHiding = "If present, the abstracts will be invisible, and a visibility toggle will be added. A corresponding java-script procedure must be present, as well as the translations for the button texts." docu to_xhtml_1_0 biblist = #d2d #table #tbody #choose #when a:sortMethod/text()='chronAsc' #apply #xp a:bibentry #sort #xp a:year/text() #data-type number #order ascending #sort #xp a:month/text() #data-type number #order ascending #when a:sortMethod/text()='chronDesc' #apply #xp a:bibentry #sort #xp a:year/text() #data-type number #order descending #sort #xp a:month/text() #data-type number #order descending #when a:sortMethod/text()='key' #apply #xp a:bibentry #sort #xp a:key #data-type text #order ascending #when a:sortMethod/text()='author' #apply #xp a:bibentry #sort #xp a:author #data-type text #order ascending #when not(a:sortMethod) #apply #xp a:bibentry #other #message Invalid sort method selection in biblist: #valueof a:sortMethod #/tbody#/table #/template #template #name bib-getlang // focus=biblist #choose #when @xml:lang #valueof @xml:lang #other #valueof $baseLanguage #template #name bib-get-author // focus=bibentry FIXME #template #name bib-get-year // focus=bibentry if not year, but date, then excerpt FIXME #template #name bib-get-month // focus=bibentry if not month, but year, then excerpt FIXME #/d2d tags bibentry = #implicit key, kind, ( title & subtitle? & latextitle? & X.lang? & (author|authors)? ), ( url | isbn | issn | doi)* , ( ( crossref, pages?) | ( ((year, month?) | date) & booktitle? & publisher? & place & (editor|editors)? & (series|journal)? & volume? & number? & pages? ) ), (abstract | note | available ) * tags title, subtitle = (TITLE_CONTENTS)* tags url = @LINK.url with xmlrep = bib-url docu user_en url = #d2d Whenever the target format permits, the rendering is to a hyperlink. The visible text is determined by the stylesheet parameter #src!$user.biblistHideUrl!: When this is set, only the suffix is shown, intended to represent the document's type. Otherwise the whole URL is shown, what may cause inappropriate visual noise. #nl FIXME better make it parameter of the biblist structure !? #nl For the url value, and the displayed url in the latter case, the leading digits #src!0!..#src!9! will be replaced by the mount points in the stylesheet parameter #src!$user.linkurlprefices! and #src!$user.linktextprefices!, similar as with "base.interdocument:link" elements. #/d2d // FIXME re-use of xslt code from "LINK:url" not possible ? docu to_xhtml_1_0 url = #d2d #x-var l-start #xp substring(.,1,1) // XALAN: FIXME #x-var l-start #xp substring(text(),1,1) #choose #when contains('0123456789',$l-start) #x-var l-urlprefix #call a:splitbyfirst #arg p-select #xp $l-start #arg p-list #xp $user.linkurlprefices #arg p-errormsg #xp . // text() #/call #/x-var #x-var l-textprefix #call a:splitbyfirst #arg p-select #xp $l-start #arg p-list #xp $user.linktextprefices #arg p-errormsg #xp . // XALAN text() #/call #/x-var #x-var l-rest #xp substring(. /*text()*/ ,2) #/x-var #a#href {concat ($l-urlprefix,$l-rest)} #tt #valueof concat ($l-textprefix,$l-rest) #other #a#href {.} #tt #valueof . // #a#href {text()} #tt #valueof text() #/d2d tags TITLE_CONTENTS = #chars docu user_en TITLE_CONTENTS = "For allowing mark-up in title, subtitle, etc., this parameter can be overridden when importing this module." tags latextitle = #chars docu user_en latextitle = "Characters only, intended for over-riding LaTeX-bibTex entries, because these are hard to synthesize automatically." tags author = #chars tags editor = @author tags authors = author+ tags editors = author+ tags year, month, date = #chars tags publisher, place = #chars tags series, journal, volume, number, pages = #chars tags booktitle = @title chars crossref = @cite tags isbn, issn, doi = #chars docu user_en isbn, issn, doi = "Currently not yet parsed internally, but whenever the target format permits, rendered as a \"clickable\" link to some service provider." enum kind = inproceedings, incollection, report, article, proceedings, collection, thesis, masterthesis, phdthesis, book, technicalreport, manual, webpage, unpublished, musicsheet, fullscore, vocalscore, radiofeature, audiorecording, tvcast, lecture with xmlrep att = "kind" docu user_en kind = #d2d The different kinds of publication, as historically introduced by "bibtex".#nl We added further alternatives.#nl Please note that in the default rendering there is no relation between this kind value and the syntax of entries or the format of the rendering. Nevertheless, indicating the "kind" is always required. #/d2d tags abstract = #implicit X.lang, (ABSTRACT_CONTENTS)* docu user_en abstract = "An abstract in one certain language." tags ABSTRACT_CONTENTS = #GENERIC docu user_en ABSTRACT_CONTENTS = "Generic parameter which defines what an abstract will be composed of." docu to_xhtml_1_0 abstract = #d2d #choose #when ../../a:abstractHiding #x-var tableId #xp concat('abstract_',./@xml:lang,'_',../a:key) #x-var nichtdatext #call muliLookUp #arg p-key #xp 'abstract' #arg p-lang #xp ./@xml:lang // <- correct, language of the abstract #/x-var #x-var datext #call muliLookUp #arg p-key #xp 'hideAbstract' #arg p-lang #xp ./@xml:lang #/x-var #span #class linkLike #id switch_{$tableId} #lang {@xml:lang} #onclick toggleAbstract(this,'{$datext}','{$nichtdatext}') #onmouseover this.style.textDecoration='underline'; #onmouseout this.style.textDecoration='none'; #valueof concat('[',$nichtdatext,']') #/span #table #id {$tableId} #style display:none; #tbody #tr #td #class abstractText #lang {@xml:lang} #apply #other #table #tbody #tr // #td #class abstractText #lang {@xml:lang} // NEU 20141115 xml-tag a:abstract kann auch aus "deliverables" kommen, // da wäre dann "lang=''", was nicht durch w3c verifier geht ! #td #class abstractText #if @xml:lang #attribute lang#valueof @xml:lang #/attribute // ??? why #attribute xml:lang#valueof @xml:lang #/if #apply #template #name renderAbstract // focus isa bibentry #param p-langs #/param #param p-sep #/param #if string-length($p-langs) > 1 #x-var l-currentlang #xp substring-before($p-langs,$p-sep) #apply #xp a:abstract[@xml:lang=$l-currentlang] #call renderAbstract #arg p-langs #xp substring-after($p-langs,$p-sep) #arg p-sep #xp $p-sep #/d2d tags note = ABSTRACT_CONTENTS* docu user_en note = "Additional remarks to a bib entry, eg. comments, protocols where it has been used, hints for a rendering process, etc." tags available = #chars docu user_en available = "Indication of one library where to printed text can be found, and the signature they use." docu to_xhtml_1_0 bibentry = #d2d #x-var l-lang // only needed for <author> "und" <author>, could use "&" or ", " #choose #when @xml:lang #valueof @xml:lang #when parent::a:biblist/@xml:lang #valueof parent::a:biblist/@xml:lang #other #valueof $baseLanguage #/x-var #tr#td #if a:note[normalize-space(text())='recommended'] // SPECIAL for ML FIXME // better give it a "class/pkind" attribute !? #attribute class #text recommended #/if #a#name __bib{a:key/text()} #id __bib{a:key/text()} #span #class printKey #text![!#valueof a:key #text!]! #/a #br/ #if a:author|a:authors #apply #xp a:author|a:authors/a:author #arg p-lang #xp $l-lang #br/ #/if #span #class!publicationTitle! #call copyLang_direct #apply #xp a:title/node() // #apply #xp a:title // wrong, runs into (section-)title ! #/span #br/ #if a:subtitle #span #class!publicationSubtitle! #call copyLang_direct #apply #xp a:subtitle/node() #/span #br/ #/if #if a:booktitle or a:crossref or a:journal #i #call muliDefault #arg p-key inCollectionPrefix#/call #if a:booktitle #valueof a:booktitle/text()#text!, ! #/if #if a:journal #valueof a:journal/text()#text!, ! #/if #if a:series #valueof a:series/text()#text!, ! #/if #foreach a:crossref #call renderCite#/foreach #if a:number #text!Nr.! #valueof a:number/text()#text!, ! #/if #if a:volume #text!Vol.! #valueof a:volume/text()#text!, ! #/if #if a:pages #text!pg.! #valueof a:pages/text()#text!, ! #/if #/if #if a:editor #i #valueof a:editor // NOT a:editor/text() SEE COMMENT BELOW #call muliLookUp #arg p-key editorsAbbrev#arg p-lang #xp $l-lang #/call#/i #/if #if a:editors #i #apply #xp a:editors/a:author #arg p-lang #xp $l-lang // insert "and", as with authors #call muliLookUp #arg p-key editorsAbbrev#arg p-lang #xp $l-lang #/i #/if #if a:editor|a:editors|a:booktitle|a:crossref #br/ #/if #if a:place #apply #xp a:publisher #if a:publisher #text!, !#/if #apply #xp a:place #text!, ! #if a:month #valueof a:month/text() #text! !#/if #apply #xp a:year #apply #xp a:date #br #/if #apply#xp a:isbn|a:issn|a:doi|a:link #choose #when $user.biblistHideUrl #foreach a:bib-url #a #href {.} #text![.! #call a:extractFileSuffix #arg p-string #xp . #call muliLookUp #arg p-key#xp 'URL_document' #arg p-lang #xp $l-lang #text!]! #/a // NO #br !! #other #apply #xp a:bib-url #/choose //#apply #xp a:note #apply #xp a:note[normalize-space(text())!='recommended']//SPECIAL for ML FIXME #if a:abstract #if ../a:abstractLangs #call renderAbstract #arg p-sep #xp substring(../a:abstractLangs/text(),1,1) #arg p-langs #xp substring(../a:abstractLangs/text(),2)#/arg #br/ #/if #/if #br/ #/td#/tr #/template #param user.biblistHideUrl#/param #x-var const.ISBN_OFFICIAL_CATAOLOG_RQ #xp concat($a:const.httpPrefix, 'www.worldcat.org/search?q=') #/d2d docu to_xhtml_1_0 issn = #d2d #a #href {$const.ISBN_OFFICIAL_CATAOLOG_RQ}{.} #text!ISSN !#valueof . #br #/d2d docu to_xhtml_1_0 isbn = #d2d #a #href {$const.ISBN_OFFICIAL_CATAOLOG_RQ}{.} #text!ISBN !#valueof . #br // attention: "#valueof text()" yields a node set of text nodes and converts the FIRST ONLY! //value-of[xslt1.0:7.6.1] SEE string()[xslt1.0:4.2] SEE string-value[xslt1.0:5.2] ("concatenation") #/d2d docu to_xhtml_1_0 doi = #d2d #a #href {$a:const.httpPrefix}dx.doi.org/{.} #text!doi:!#valueof . // attention: "#valueof text()" yields a node set of text nodes and converts the FIRST ONLY! #br #/d2d docu to_xhtml_1_0 author = #d2d #param p-lang #/param #span #class!author! #valueof . #/span #if following-sibling::a:author #text! ! #call muliLookUp #arg p-key#xp 'and' #arg p-lang #xp $p-lang #text! ! #/d2d end module // citation // ------------------------------------------------------ module simpleLists local node xmlrep naming = join upcased import S from sets import IN from inlineElements // ------------------------------------------------------ docu user_en = "A very simple list module." tags CONTENTS = #GENERIC docu user_en CONTENTS = "To be replaced by what is allowed as contents of an <i>." public tags list = (symbol? & (startnum|continues)? & IN.label?), i+ with local chars startnum = @S.decimalPositive chars symbol = @listSymbol chars continues = @S.ident tags i = (CONTENTS)* with xmlrep el=listitem end local docu user_en list.startnum = #d2d CURRENTLY the starting number of a numbered list CANNOT be set anymore! Once, in former html versions, this was possible. NOW it is advised to use CSS-three for this purpose, but this is not yet on the market.#nl THEREFORE we ignore the value of "startnum" and of "continues".#nl IT COULD be built by rendering the list-items explicitly, not using the <ol> construct, but <xsl:number> and <html:dd> instead. But we are not sure whether this is sensible. #/d2d enum listSymbol = i, I, a, A, alpha, arab, plus, minus, longminus, bullet, openBullet, square, openSquare, diamond, openDiamond docu user_en listSymbol = #d2d longminus, openSquare, diamond, openDiamond are not yet supported. #/d2d docu user_en listSymbol.openDiamond, listSymbol.longminus, listSymbol.diamond, listSymbol.openSquare = "not yet supported #bold!FIXME!" docu to_xhtml_1_0 list = #d2d #x-var listchar #xp normalize-space(./a:listSymbol)#/x-var #choose #when $listchar='' or $listchar='arab' #ol#style list-style-type:decimal; #call descendList #when $listchar='i' #ol#style list-style-type:lower-roman; #call descendList #when $listchar='I' #ol#style list-style-type:upper-roman; #call descendList #when $listchar='a' #ol#style list-style-type:lower-alpha; #call descendList #when $listchar='A' #ol#style list-style-type:upper-alpha; #call descendList #when $listchar='alpha' #ol#style list-style-type:lower-greek; #call descendList #when $listchar='bullet' #ul#style list-style-type:disc; #call descendList #when $listchar='openBullet' #ul#style list-style-type:circle; #call descendList #when $listchar='square' #ol#style list-style-type:square; #call descendList #other #ul#style list-style-type:disc; #call descendList // current() is possibly a child of a p, then the lang and kind info must be copied: #template #name descendList #if parent::a:p #call copyLang_1_indirect #call copy_kind_to_class #/if #apply#xp a:listitem #template #match a:listitem #li#apply #template #match a:listitem #mode getreftext #param referer#/param #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #apply #xp parent::a:list #mode getreftext #arg referer#xp $referer #text!, !# #call muliDefault#arg p-key listitem#/call #text! ! #x-var listchar #xp normalize-space(parent::a:list/a:listSymbol/text())#/x-var #choose #when $listchar='a' or $listchar='A' or $listchar='i' or $listchar='I' #number #format {$listchar} #count a:listitem // XALAN BUG WORK-AROUND, xsl:number needs "count" parameter. #when $listchar='alpha' or $listchar='α' // U+03B1 (945) = klein-alpha = α #number #format α #count a:listitem #other #number #format 1 #count a:listitem #/choose #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #template #match a:list #mode getreftext #param referer#/param #x-var sameList #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when boolean(string($sameList)) // ATTENTION #call muliDefault#arg p-key REF_sameList#/call #other #apply #xp parent::* #mode getreftext #arg referer#xp $referer #arg referred#xp generate-id(.) #arg nsuri#xp namespace-uri() #arg localname#xp local-name() #arg textMultiple#xp 'list' #arg textSingle#xp '' #/d2d end module // simpleLists // ------------------------------------------------------ module simpleTables local node xmlrep naming = no mangling import IN from inlineElements //Neu 20201201: import SETS from sets // ------------------------------------------------------ tags CONTENTS = #GENERIC tags table = (border? & IN.label?), tr* tags tr = (IN.label)?, (td*|th*) // tags td = (CONTENTS)* //Neu 20201201: tags td = c?, (CONTENTS)* chars c = @SETS.identInnerChar ~+ with xmlrep = "tdCls" tags th = (CONTENTS)* docu user_en table = #d2d Most simple table model; sufficient for many use cases, but can of course be replaced by something more elaborate, when necessary. #nl The nomenclature follows html. #p Labels at the explicit positions refer to the table or the table row; references to labels appearing anywhere in the text will contain the full address path "table <T>, row <R>, column <C>". Numbering starts with one(1). #/d2d tags border = #chars docu user_en border = "Values for kind, style, width of cell borders, following CSS2.0 syntax. Valid for the whole table." docu to_xhtml_1_0 table = #d2d #table #if a:border #attribute border#valueof a:border #/if // current() is possibly a child of a p, then lang and kind info must be copied: #if parent::a:p #call copyLang_1_indirect #call copy_kind_to_class #/if #tbody #apply #xp a:tr #template #match a:td|a:th #mode getreftext #param referer#/param #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #x-var sameTd #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when boolean(string($sameTd)) // ATTENTION #call muliDefault #arg p-key REF_same_td#/call #other #x-var table #xp ancestor::a:table #apply #xp parent::* #mode getreftext #arg referer #xp $referer #/apply #text!, ! #call muliDefault#arg p-key REF_table_col#/call #text! ! #valueof count(preceding-sibling::a:tr | preceding-sibling::a:td)+1 #/choose #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #/call // can only come from td, or refer to the row as such. #template #match a:tr #mode getreftext #param referer#/param #x-var sameTr #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when boolean(string($sameTr)) // ATTENTION #call muliDefault #arg p-key REF_same_tr#/call #other #apply #xp parent::* #mode getreftext #arg referer #xp $referer #/apply #text!, ! #call muliDefault#arg p-key REF_table_row#/call #text! ! #valueof count(preceding-sibling::a:tr)+1 #/choose #template #match a:table #mode getreftext #param referer#/param #x-var sameTable #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when boolean(string($sameTable)) // ATTENTION #call muliDefault #arg p-key REF_same_table#/call #other #apply #xp parent::* #mode getreftext #arg referer #xp $referer #arg referred#xp generate-id(.) #arg nsuri#xp namespace-uri() #arg localname#xp local-name() #arg textMultiple#xp 'REF_one_of_many_tables' #arg textSingle#xp 'REF_single_table' #/apply #/d2d // "tr" may e empty in d2d, but not in xhtml: docu to_xhtml_1_0 tr = #d2d #choose #when ./td #tr#apply #other #tr#td // generate one emtpy dummy element #/d2d docu to_xhtml_1_0 th = "#th#apply" docu to_xhtml_1_0 c = "" docu to_xhtml_1_0 td = #d2d #td #if a:tdCls #attribute class #valueof concat('tdCls_',a:tdCls/text()) #/if #apply #/d2d end module // simpleTables // ------------------------------------------------------ module simpleImage local node xmlrep naming = join by "_" import LINK from interDocuments // ------------------------------------------------------ tags image = file & alt? & width? & height? & align? & showLabel? with local tags file = @ LINK.url with xmlrep trimmed tags width, height = #chars with xmlrep trimmed tags align = #chars with xmlrep trimmed tags alt = ALT tags showLabel = #empty end local tags ALT = #GENERIC docu user_en image.alt = "If you leave it out, some alternative text from the enclosing elements must be found for rendering. This searched for by the named template 'copyAltFromContext', which otherwise will fail with an error." docu to_xhtml_1_0 image = #d2d #if $user.showlabels='yes' or a:image_showLabel #span#class debug #valueof a:image_file #/span#br #/if #img #src {a:image_file} #call copyAltFromContext #if a:image_width #attribute width #valueof a:image_width #/if #if a:image_height #attribute height #valueof a:image_height #/if #/d2d end module // simpleImage // ====================================================================== module floatings local node xmlrep naming = join by "_" import S from sets import E from inlineElements // ====================================================================== tags tableOfContents = caption? & switchOnText? & switchOffText? tags listOf = #implicit series, ( caption & numberingScope? & switchOnText? & switchOffText? & initiallyHidden? ) tags series = #chars with xmlrep trimmed tags caption, shortCaption = captiontext* chars numberingScope = "h"~@S.decimalDigit tags switchOnText, switchOffText = switchtext* tags initiallyHidden = #empty docu user_en tableOfContents = #d2d Will be expanded to a complete table of contents, of the hierarchical structure of the whole document. #/d2d docu user_en listOf = #d2d Will be expanded to a list of the floating objects of a certain series. #/d2d docu user_en caption = #d2d The headline for the "table of contents", or for any floating object. #/d2d docu user_en shortCaption = #d2d A shorter version of the caption of a floating object. Iff present, it replaces the longer caption in the "listOf" survey. #/d2d docu user_en switchOnText = #d2d A short text line. When clicked, it will pop-up/open the "listOf" survey. #/d2d docu user_en switchOffText = #d2d A short text line. When clicked, it will hide/close the "listOf" survey. If a switchOff text is given, a switch will be installed and a switchOnText is required, or at least a "caption", which will be used as such. #/d2d docu user_en numberingScope = #d2d Indication of the structure level (like "h1", "h2", ...) in which the numbering of the floatings will be based on. #/d2d docu user_en series = #d2d The series is NOT an identifier, but can contain arbitrary characters. This is usefule because it is #emph!visible! to the end user. E.g. in the German language you want to include a list of "Gemälde" and floating objects called "Gemälde 1.2" #/d2d tags switchtext = #GENERIC tags captiontext = #GENERIC tags floating = #implicit series, (caption? & shortCaption? & E.label? & width? & height? & positioning? & body) with local tags positioning = #chars with xmlrep trimmed tags width, height = #chars with xmlrep trimmed tags body = (floating.BODY)* // just @ BODY does not work !! FIXME ??? tags BODY = #GENERIC end local docu to_xhtml_1_0 = #d2d /* ATTENTION docu to_xhtml_1_0 tableOfContents = ... DOES NOT WORK because in basic.structure "@listOf" does not copy xslt code. FIXME */ #template #match a:tableOfContents #call makeVisibilitySwitch #arg id-switch #xp'_switch_TOC' #arg id-table #xp'__div_TOC' #p#id __div_TOC #a #id TOC #span #class tocTitle #apply #xp a:caption#/a #br/ #choose #when /descendant::a:part #apply #xp /descendant::a:part #mode maketoc #other #apply #xp /descendant::a:h1 #mode maketoc #/p #/d2d docu to_xhtml_1_0 = #d2d /* ATTENTION docu to_xhtml_1_0 listOf = ... DOES NOT WORK because in basic.structure "@listOf" does not copy xslt code. FIXME */ #template #match a:listOf #x-var myseries#xp a:series/text() #hr/ #call makeVisibilitySwitch #arg id-switch #xp concat('__switch_listOf_',$myseries) #arg id-table #xp concat('__div_listOf_',$myseries) #div #id __div_listOf_{$myseries} #attribute style #choose #when a:initiallyHidden #text!display:none;! #other #text!display:block;! #/attribute #span #class tocTitle #a#id {concat('__listOf_',$myseries)} #name {concat('__listOf_',$myseries)} #apply #xp a:caption #/span #table #tbody #foreach /descendant::a:floating[a:series/text()=$myseries] #x-var mynum #call makeFloatingNumber #/x-var #x-var mylab #call makeFloatingLabel #/x-var ///**/ #message FLOAT TABLE LABEL IS #valueof $mylab #text!---! /* #valueof . */ #/message //FIXME falsche node-list be count "any" ?? /**/#message#valueof concat(" MYNUM ",$mynum,"---",.)#/message #tr #valign top #td #a #href {$a:const.hashmark}{$mylab} #valueof $mynum #/a #/td #td #a #href {$a:const.hashmark}{$mylab} #choose #when a:shortCaption #apply#xp a:shortCaption#mode intoc #other#apply#xp a:caption#mode intoc #/a #/foreach #tr#td // dummy, since tbody may not be emtpy #/div #hr/ #template #name makeFloatingLabel // "." points to a "floating" #choose #when a:label #valueof a:label/text() ///**/ #message FLOAT MAKE LABEL xxxxxxxxxxxxxxxx IS #valueof a:label/text() #text!---! /* #valueof . */ #/message #other #x-var mynum #call makeFloatingNumber #/x-var #valueof concat('_', a:series, '_', $mynum) #template #name makeFloatingNumber // "." points to a "floating" #x-var myseries#xp a:series/text() #x-var numberbase #xp /descendant::a:listOf[a:series/text()=$myseries]/a:numberingScope #choose #when $numberbase #apply #xp ancestor::*[local-name()=$numberbase] #mode getnumberComplete // = with partnumber,if necessary. Cannot be used in a postappendix. #text!.! #number #count a:floating[a:series/text()=$myseries] #from *[local-name()=$numberbase] #level any #other #number #count a:floating[a:series/text()=$myseries] #level any #template #name makeVisibilitySwitch #param id-switch#/param #param id-table#/param #if a:switchOffText and (not(a:switchOnText or a:caption)) #message ERROR, switchOfftext, but neither switchOnText nor caption in #valueof . #/if #if a:switchOffText #x-var switchontx #choose #when a:switchOnText #foreach a:switchOnText#call a:getOneLineText #other #foreach a:caption#call a:getOneLineText #/x-var #x-var switchofftx# #foreach a:switchOffText#call a:getOneLineText #/x-var #p#span #id {$id-switch}#class switchText #onclick toggleVisibility('{$id-switch}', '{$id-table}', '{$switchontx}', '{$switchofftx}') #onmouseover this.style.textDecoration='underline'; #onmouseout this.style.textDecoration='none'; #choose #when a:initiallyHidden #valueof $switchontx #other #valueof $switchofftx #/d2d docu to_xhtml_1_0 floating = #d2d #x-var mylab #call makeFloatingLabel#/x-var ///**/ #message FLOAT LABEL IS #valueof $mylab#/message #x-var myseries #valueof a:series/text()#/x-var #table #if a:width #attribute width #valueof a:width #/if #if a:height #attribute height#valueof a:height #/if #choose #when a:positioning/text()='floatleft' #attribute style#text float:left;margin:10pt; #when a:positioning/text()='floatright' #attribute style#text float:right;margin:10pt; #/choose // PRE 20201117 #tbody #tr #td #tbody #tr #td #class floatbody_{$myseries} #a #name {$mylab} #id {$mylab}#/a #apply #xp a:floating_body #/tr #if a:caption #tr #td #class floatingCaptionField #if $user.showlabels="yes" #span #class debug #valueof $mylab #/if #choose #when /descendant::a:listOf[a:series/text()=$myseries] #a #href {$a:const.hashmark}__listOf_{$myseries} #valueof a:series#text#noescape xA0;#call makeFloatingNumber #other #valueof a:series#text#noescape xA0;#call makeFloatingNumber #/choose #text!: ! #apply #xp a:caption // reflects already the hierarchy of sections as prepended number sequence. #template #match a:floating #mode getreftext #param referer#/param #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #x-var sameFloat #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when sameFloat #call muliDefault #arg p-key #xp concat('this ',./a:series/text()) #other #valueof a:series#text#noescape xA0;#call makeFloatingNumber #if $user.refTitle='inline' #valueof concat(' ("', a:caption, '")') #/choose #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #/call #template #match a:floating #mode getreftextTitle #apply #xp ./a:caption #mode puretext #/d2d end module // floatings // ====================================================================== module structure local node xmlrep naming = join by "-" // ====================================================================== import SET from sets import PHY from physical import X from xmlInfra import E from inlineElements ^(@horizontalText / EMPH_CONTENTS) ^(@horizontalText / XLANG_CONTENTS) import LINK from interDocuments ^( (@horizontalText ^(#none/LINK.link)) / LINKTEXT) ^(@horizontalText / EMBED_ALT) import BIB from citation ^( (@verticalText ^(#none / IM.image)) | (@verticalObjects ^(#none / PHY.hr) ^(#none / LINK.treeInclude) ) / ABSTRACT_CONTENTS) ^( (@horizontalText^(#none/LINK.link)) / TITLE_CONTENTS) // ATTENTION treeInclude is in verticalText (via horizontalText) AND in verticalObjects import LI from simpleLists ^( (@verticalText | p) / CONTENTS) import TA from simpleTables ^( (@verticalText | p) / CONTENTS) import IM from simpleImage ^( #chars / ALT) ^( LINK / LINK ) // FIXME necessary when doing whole-module-rewrite, due to: // error: an insertion of the generic (placeholder) definition named "basic.interDocuments:EMBED_ALT" should have been eliminated by a concrete definition, or by "#none". import FLOAT from floatings ^( (@horizontalText^(#none/LINK.link)^(#none/BIB.cite)) / switchtext) ^( @verticalObjects / floating.BODY ) ^( @horizontalText / captiontext ) // ------------------------------------------------------ docu user_en = "This module defines the structural elements like paragraphs, chapters, etc. It starts plugging the sub-modules together, and distributing parametrization, without yet defining the very top of the corpus structure." tags DOMAINSPECIFIC_LINEAR_ELEMENTS = #GENERIC docu user_en DOMAINSPECIFIC_LINEAR_ELEMENTS = "Additional domain specific abbreviations, etc. Their xhtml rendering must fit under a xhtml:span, e.g. for a d2d:emph or d2d:xemph. Normally they may not be used under physical mark-up like d2d:ital." tags DOMAINSPECIFIC_VERTICAL_ELEMENTS = #GENERIC public tags horizontalText = #chars | @PHY.EXPORTMIXTURE | @E.MIXTURE | E.xlang | DOMAINSPECIFIC_LINEAR_ELEMENTS | LINK.treeInclude | LINK.link | BIB.cite | IM.image docu user_en horizontalText = #d2d All possible structured horizontal (ie. linear) text. Must be restricted for certain use cases, e.g. links are not allowed in links, images not in abstracts, but possibly in chapter titles. When rendered to xhtml, these yield "inline-elements", i.e. they can be enclosed in an xhtml:span. (Is already too rich to appear in pure-character data, like those rendered as attribute values.) #/d2d // =@p tags verticalText = @horizontalText | LI.list | TA.table | footnote | footnotemark | footnotetext // | @SET.ident // TESTWEISE --> uber 10 fehlermeldungen ! besser abbrechen?? FIXME docu user_en verticalText = #d2d This is the flow contents of a #emph!paragraph!, as defined by the #link #loc __DECLARATION__basic.structure:p #text #src!#hh p! element #/link, without the leading attributes kind, lang, etc.#nl This is both, horizontal (=linear) text and vertically structured text elements (lists and tables, etc.), but not elements which are between paragraphs, like floats, biblists and (larger) embedded third-party objects.#nl They cannot stand under xhtml:span, but under xhtml:div.#nl Since d2d_gp uses the p as the central means of text organisation, eg. wrt. language, kind, etc., the d2d:p contains more than the xhtml:p. Therefore the "p-rotation algorithm" is applied. #/d2d tags verticalObjects = p | LINK.embed | PHY.hr | xvolute | source | FLOAT.floating | BIB.biblist | DOMAINSPECIFIC_VERTICAL_ELEMENTS | LINK.treeInclude // PRE20240302 | LINK.treeInclude_PROVIS // FIXME necessary!?!? RAUS tags p = kind?, X.lang?, (@verticalText)* docu user_en p = #d2d #p The paragraph is the central means for organizing text, eg. wrt. language and kind. #p When rendering to xhtml, the items contained in "kind" are prepended with "paragraphkind_" and then stored as "css class names", so that visual appearance can be controlled via css. #nl Furthermore, setting the variable user.p_kind_filter lets only pass those paragraphs with a non-empty intersection with this value. #nl ATTENTION: the values of "kind" are currently not restricted by an enumeration type (since enumeration are not yet rewritable!), but are free identifiers. So typos are not recognized and can cause missing or ill-rendered text. #p Due to its central role in organization, the contents of p also includes tables and lists, which cannot be rendered in xhtml:p. The difference is bridged by the rotation algorithm. The transition between the possible contents is like this: #source d2d:h2 xhtml: | p p list p table / / | \ \ / \ | chars link list chars table chars link chars #/s=====ource #p In the other direction, a d2d:p can be contained also in containers like list item or table cell. Luckily, this is identical to xhtml: #source table/tr/td | p list/i = ul(ol)/li | | p p #/s=====ource #/d2d chars kind = [pkind @SET.ident]~("+"~[pkind @SET.ident])~* docu to_xhtml_1_0 p = #d2d #choose #when a:kind #if contains($p_kind_filter, ' * ') or contains($p_kind_filter, concat(' ', normalize-space(a:kind),' ')) //FIXME only ONE of the a:kind/a:pkind should be necessary to be in common! #foreach child::a:kind/following-sibling::node()[position()=1] #call new_p #other #foreach child::node()[position()=1] #call new_p #/template #param user.p_kind_filter#/param // ATTENTION undefined var $user.p_kind_filter, must be declared BY CALLER, // e.g. in basic.deliverables, because different default values may be sensible. #x-var p_kind_filter #choose #when normalize-space($user.p_kind_filter)="" #valueof ' * ' #other #valueof concat(' ',$user.p_kind_filter,' ') #/d2d docu to_xhtml_1_0 = #d2d // focus is a child of p, growing output a new structure (p or table or ..) // copy kind to class #template #name copy_kind_to_class #if parent::a:p/a:kind #attribute class #foreach parent::a:p/a:kind/a:pkind[position()=1] #call collect_p_kind #template #name collect_p_kind // focus isA pkind #valueof concat('paragraphkind_',text(),' ') #foreach following-sibling::a:pkind[position()=1] #call collect_p_kind // starts a new xhtml:p and stores there all contents as long as possible. // current() is a child of d2d:p #template #name new_p #choose #when local-name()='list' or local-name()='table' #foreach . #call from_p // switch to outside html:p #other #p #call copy_kind_to_class #call copyLang_1_indirect #call in_p // consumes all, up to the first non-p #/p #foreach (following-sibling::a:table|following-sibling::a:list)[position()=1] #call from_p // easier would be "following-sibling-or-self", but this does not exist !?!? // loops internally as long current object can be contained INSIDE xhtml:p #template #name in_p //#message IN P reached, current focus = #valueof . #/message #if local-name()!='list' and local-name()!='table' #apply #xp . #foreach following-sibling::node()[position()=1] #call in_p // loops internally as long current object must be OUTSIDE of xhtml:p, then // starts a new html-p #template #name from_p #apply #xp . // must call copy_kind_to_class and copyLang_1_indirect explicitly, on its own. #foreach following-sibling::node()[position()=1] #choose #when local-name()='list' or local-name()='table' #call from_p #other #call new_p #/d2d tags footnote, footnotetext = (symbol)?, (@p | p+) tags footnotemark = symbol? chars symbol = (SET.all - SET.whitespace)~+ with xmlrep att // FIXME footnotenmark, footnotetext not yet supported docu to_xhtml_1_0 footnote = #d2d #x-var symbolV #choose#when @a:symbol #valueof @a:symbol #other #valueof count(preceding::a:footnote[not(@a:symbol)])+1 #/x-var #x-var symbolA #choose#when @a:symbol #valueof concat('SYMB_',count(preceding::a:footnote[@a:symbol])+1) #other #valueof $symbolV #/x-var #a#href {$a:const.hashmark}__footnote{$symbolA} #name __etontoof{$symbolA} #id __etontoof{$symbolA} #sup #valueof $symbolV #template#match a:footnote#mode footn #x-var symbolV #choose#when @a:symbol #valueof @a:symbol #other #valueof count(preceding::a:footnote[not(@a:symbol)])+1 #/x-var #x-var symbolA #choose#when @a:symbol #valueof concat('SYMB_',count(preceding::a:footnote[@a:symbol])+1) #other #valueof $symbolV #/x-var #choose#when a:p #p #call copyLang_direct #call copy_kind_to_class #a#href {$a:const.hashmark}__etontoof{$symbolA} #name __footnote{$symbolA} #id __footnote{$symbolA} #sup #valueof $symbolV #/a #foreach a:p[1]/node()[1] #call in_p#/foreach #/p #foreach a:p[1] #foreach (a:table|a:list)[position()=1] #call from_p #/foreach #/foreach #foreach a:p[position()>1] #foreach node()[1] #call new_p #other #p // no lang or pkind info is copied from the main text into the footnote! #a#href {$a:const.hashmark}__etontoof{$symbolA} #name __footnote{$symbolA} #id __footnote{$symbolA} #sup #valueof $symbolV #/a #foreach ./node()[1] #call in_p#/foreach #/p #foreach (a:table|a:list)[position()=1] #call from_p #template#match a:footnote #mode getreftext #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #call muliDefault #arg p-key REF_footnote#/call #valueof count(preceding::a:footnote)+1 #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #/d2d tags xvolute = #empty docu to_xhtml_1_0 xvolute = " #p#style text-align:center#img#alt Volute#src volute148.png " // neu treeInclude 20160424 GEHT NICHT FIXME (siehe "imtaktVorwort.d2d" / "xx.SCO") tags source = (#chars | E.label | LINK.treeInclude)* // tags source = (#chars | E.label)* with xmlrep el = SRC, input verbatim docu to_xhtml_1_0 source = " #table #class source #tbody #tr #td #pre #apply" docu user_en source = #d2d Larger unit of program source text, rendered as a text block on its own. Its input mode is "verbatim". This means (1) respecting whitespace and line breaks, (2) must be ended by an explicit end token, (3) all command characters will be ignored, except of those DIRECTLY followed by a tag of an explicitly mentioned direct sub-element (here: "label"). #/d2d docu to_xhtml_1_0 = #d2d /* strategy for toc-generation: * In this default style, each toc line is the title prefixed by the complete * number of the section. * (Realized as one line, not in a table, but with constant distance.) * The mode maketoc descends the total structure. The modes getnumber<> ascend and collect the hierarchy number. (getnumberComplete with prepended part, or NN<n> for post-appendices, getnumberVisible RELATIVE to part, and empty for post-appendices, so not usable in anchors and identification.) The mode intoc renders the title of the collected sections IN the toc, where links and references must be left out. */ #template #match a:title #mode intoc #choose #when parent::a:h1 or parent::a:part #b #apply #mode intoc #other #apply #mode intoc #template #match a:label|a:footnote #mode intoc // NOP #template #match a:link #mode intoc #choose #when a:text #apply #xp a:text #mode intoc #other #valueof a:url #template #match text() #mode intoc #valueof . // Create appearances of "pure text", where ALL decoration must be left out // (e.g. for values of an xml attribute like "<img alt='XXX'>" #template #match text() #mode puretext #valueof . #template #match * #mode puretext #apply #mode puretext #template #match a:link #mode puretext #choose#when a:text #apply #xp a:text #mode puretext #other #valueof a:url #template #match a:label|a:footnote #mode puretext //NOP #template #name maketocline #if local-name()='part' #br#/if #x-var numberV #apply #mode getnumberVisible #xp . #/x-var #x-var numberC #apply #mode getnumberComplete #xp . #/x-var // start with "NN" indicates intended invisibility of numbering #if $numberV #a #href {$a:const.hashmark}HIER{$numberC} #valueof $numberV #/if #text#noescape xA0;xA0;xA0;xA0;xA0;xA0;xA0;xA0; #a #href {$a:const.hashmark}HIER{$numberC} #choose #when a:toctitle #apply #xp a:toctitle #mode intoc #other #apply #xp a:title #mode intoc #/a #if local-name()='part' #br#/if #br #template #match a:part #mode maketoc #call maketocline #apply #xp a:h1 #mode maketoc #template #match a:h1 #mode maketoc #call maketocline #apply #xp a:h2 #mode maketoc #template #match a:h2 #mode maketoc #call maketocline #apply #xp a:h3 #mode maketoc #template #match a:h3 #mode maketoc #call maketocline #apply #xp a:h4 #mode maketoc #template #match a:h4 #mode maketoc #call maketocline #apply #xp a:h5 #mode maketoc #template #match a:h5 #mode maketoc #call maketocline #apply #xp a:h6 #mode maketoc #template #match a:h6 #mode maketoc #call maketocline /* Mode getnumberComplete constructs the following patterns: * for #h1 .. #h1 .. #h2 deliver 2.1 * for #part .. #h1 .. #h2 deliver I.2.1 (IF $user.partResetsChapterNumber) * for #appendix #h1 .. #h1 .. #h2 deliver B.1 * for #postappendix .. #h1 .. #h1 deliver NN2 * (Postappendices (like "bibliography" or "Bildnachweise") thus cannot be * referred to by numbering.) * * The result of getnumberVisible is used for printing toc-lines and chapter headers: * Empty for postappendix, and part numbers do not appear. */ #template #match a:part #mode getnumberComplete #number #format I #count a:part #template #match a:part #mode getnumberVisible #number #format I #count a:part #template #match a:h1 #mode getnumberComplete #choose #when parent::a:postappendices #text NN_#number #count a:h1 #when parent::a:appendices #number #format A #count a:h1 #when $user.partResetsChapterNumber="yes" #if parent::a:part #apply #xp parent::a:part#mode getnumberComplete #text!.! #/if #number #count a:h1 #other #number #count a:h1 #level any #/template #template #match a:h1 #mode getnumberVisible #choose #when parent::a:postappendices #when parent::a:appendices #number #format A #count a:h1 #when $user.partResetsChapterNumber="yes" #number #count a:h1 #other #number #count a:h1 #level any #/template #param user.partResetsChapterNumber #xp "no" #/param #template #match a:h2 #mode getnumberVisible #apply #mode getnumberVisible #xp ancestor::a:h1#text .#number #count a:h2 #template #match a:h3 #mode getnumberVisible #apply #mode getnumberVisible #xp ancestor::a:h2#text .#number #count a:h3 #template #match a:h4 #mode getnumberVisible #apply #mode getnumberVisible #xp ancestor::a:h3#text .#number #count a:h4 #template #match a:h5 #mode getnumberVisible #apply #mode getnumberVisible #xp ancestor::a:h4#text .#number #count a:h5 #template #match a:h6 #mode getnumberVisible #apply #mode getnumberVisible #xp ancestor::a:h5#text .#number #count a:h6 #template #match a:h2 #mode getnumberComplete #apply #mode getnumberComplete #xp ancestor::a:h1#text .#number #count a:h2 #template #match a:h3 #mode getnumberComplete #apply #mode getnumberComplete #xp ancestor::a:h2#text .#number #count a:h3 #template #match a:h4 #mode getnumberComplete #apply #mode getnumberComplete #xp ancestor::a:h3#text .#number #count a:h4 #template #match a:h5 #mode getnumberComplete #apply #mode getnumberComplete #xp ancestor::a:h4#text .#number #count a:h5 #template #match a:h6 #mode getnumberComplete #apply #mode getnumberComplete #xp ancestor::a:h5#text .#number #count a:h6 #template #match a:h1|a:h2|a:h3|a:h4|a:h5|a:h6 #mode getreftext #param referer#/param #param referred#/param #param nsuri#/param #param localname #/param #param textMultiple #/param #param textSingle #/param #x-var sameH #call refSameTest #arg referer #xp $referer #arg other #xp generate-id(.) #/x-var #choose #when boolean(string($sameH)) // ATTENTION non-empty node set, but empty string! #choose #when local-name()="h1" #call muliDefault #arg p-key REF_sameChapter #other #call muliDefault #arg p-key REF_sameSection #other #choose #when local-name()="h1" and parent::a:appendices #call muliDefault #arg p-key appendix#/call#text! ! #number#format A #count a:h1 #when local-name()="h1" and parent::a:postappendices #apply #mode getreftextTitle // "Glossary" "Index" have no number #when local-name()="h1" #call muliDefault #arg p-key chapter#/call#text! ! #apply #xp .#mode getnumberComplete #other #call muliDefault #arg p-key #xp 'Section' #text! ! #apply #xp .#mode getnumberComplete // subsections of postappendices = NN1_... are not useful! #/choose #if $user.refTitle='inline' and not (local-name()="h1" and parent::a:postappendices) #text! ("!#apply#xp . #mode getreftextTitle#text!")! #/choose #if $localname #call reftextByCount #arg referred#xp $referred #arg nsuri#xp $nsuri #arg localname#xp $localname #arg textMultiple#xp $textMultiple #arg textSingle#xp $textSingle #/call #template #match a:h1|a:h2|a:h3|a:h4|a:h5|a:h6 #mode getreftextTitle #x-var text #choose #when a:reftitle #apply #xp a:reftitle#mode intoc #other #apply #xp a:title #mode intoc #/x-var #valueof normalize-space($text) // #choose #when a:reftitle #apply #xp a:reftitle#mode intoc // #other #apply #xp a:title #mode intoc #template #match a:reftitle|a:toctitle #apply #xp label // not intoc mode => simply NOP, but proofmode print of labels #/d2d tags title, toctitle, reftitle = (@horizontalText)* with xmlrep trimmed docu user_en reftitle = "The text with which references to this corpus part will be rendered. Defaults to the (normal) title." docu user_en toctitle = "The text with which this corpus part will be rendered in the Table Of Contents. Defaults to the (normal) title." tags htmlTitle = #chars with xmlrep trimmed docu user_en htmlTitle = "The text used for html header, DC meta tags, etc, where only plain text is allowed. Defaults to the (normal) title stripped from all mark-up." tags abstract = @p | p+ tags hierarchie = title, (toctitle? & reftitle?), (@verticalObjects)*, SUBH* with local tags SUBH = #GENERIC end local tags part = @hierarchie ^ (h1 / hierarchie.SUBH) tags h1 = @hierarchie ^ (h2 / hierarchie.SUBH) tags h2 = @hierarchie ^ (h3 / hierarchie.SUBH) tags h3 = @hierarchie ^ (h4 / hierarchie.SUBH) tags h4 = @hierarchie ^ (h5 / hierarchie.SUBH) tags h5 = @hierarchie ^ (h6 / hierarchie.SUBH) tags h6 = @hierarchie ^ (#none / hierarchie.SUBH) tags appendices, postappendices = h1+ docu user_en part = "Parts are an optional top-level grid and identified with roman numbers. The chapter numbers beneath are reset with a new part, iff $user.partResetsChapterNumber = 'yes'." docu user_en appendices = "There may be one such part, under which the h1-elements are numbered with capital latin characters instead of numbers." docu user_en postappendices = "There may be one such part, under which (according to tradition) the h1-elements do not carry any numbering, like \"index\", \"bibliography\", \"glossary\", etc." /* In this default format, all titles are prefixed with the complete section number * (collected by mode=getNumberVisible), in the text body as well as in the toc. */ docu to_xhtml_1_0 title = #d2d #choose #when parent::a:webpage #span #class pagetitle #apply #other #x-var number #apply #xp parent::* #mode getnumberVisible #/x-var #x-var technicalNumber #apply #xp parent::* #mode getnumberComplete #/x-var #x-var elname // = the name for the xhtml representation #choose #when local-name(parent::*)='part' #valueof 'h1' #other #valueof local-name(parent::*) #/x-var #element {$elname}#namespace!{$a:const.httpPrefix}www.w3.org/1999/xhtml! #if local-name(parent::*)='part' #attribute class #valueof 'partTitle' #/if #if $hastoc-internal='true' #a #href {$a:const.hashmark}TOC #class backToToc #sup #text!^!#call muliDefault #arg p-key tocAbbrev #/if #a #name {concat('HIER',$technicalNumber)} #valueof $number // = is empty for postappendices and does not name the part #/a #text #/text #apply #/element #/d2d docu to_xhtml_1_0 = #d2d #x-var hastoc-internal #xp boolean(/descendant::a:tableOfContents) #/d2d tags metaContentsTag = #implicit name, content with local tags name, content = #chars with xmlrep trimmed end local docu to_xhtml_1_0 metaContentsTag = "#x-var text #foreach a:metaContentsTag-content #call a:getOneLineText#/x-var #meta #name {a:metaContentsTag-name} #content {$text} " // ============================================================== // module structure, UBIQUITUOUS general subroutines // ATTENTION, named templates must only be included ONCE! // ============================================================== docu to_xhtml_1_0 = // for ONLY TO LOAD ONCE !!! match() seems idempotent, // but duplicate name() yields error!!!??? #d2d #param user.mulitable #xp concat('http:/','/bandm.eu/doctypes/d2d_gp/gp_translations.xml') #/param #param user.defaultLang#/param #x-var baseLanguage #choose #when /child::*[@xml:lang] #call firstLangInList #arg p-list #xp /child::*[@xml:lang][1]/@xml:lang #when not($user.defaultLang='') #call firstLangInList #arg p-list #xp $user.defaultLang #other #message#terminate no base language defined for document ? #/choose #/x-var #template #name muliDefault #param p-key#/param #call muliLookUp #arg p-key #xp $p-key #arg p-lang #xp $baseLanguage // #template #name muliLookUp_with_fallback MISSING FIXME #template #name muliLookUp #param p-lang#/param #param p-key#/param // #message MULITABLE LOOK UP #valueof $user.mulitable#/message // #choose #when not($user.mulitable) #valueof $p-key #when document($user.mulitable)/a:translationfile /a:entry[@key=$p-key and @lang=$p-lang] #valueof $p-key #when not(document($user.mulitable)/a:translationfile/a:entry[@key=$p-key]) #message no muli translation entry for key #text >>#valueof $p-key#text << #/message #valueof $p-key #when not(document($user.mulitable)/a:translationfile/a:entry[@key=$p-key] /a:text[@lang=$p-lang]) #message no muli translation entry for key #text >>#valueof $p-key#text << with language >>#valueof $p-lang#text << #/message //deliver the first "entry/text" as a fall-back. #valueof document($user.mulitable)/a:translationfile/a:entry[@key=$p-key] /a:text[position()=1]/text() #other #valueof document($user.mulitable)/a:translationfile/a:entry[@key=$p-key] /a:text[@lang=$p-lang]/text() #/template #/d2d end module // structure // ====================================================================== module deliverables local node xmlrep naming = join by "_" import E from inlineElements import BIB from citation ^ ( #none / ABSTRACT_CONTENTS) import STR from structure ^ ( #none /DOMAINSPECIFIC_LINEAR_ELEMENTS) ^ ( #none /DOMAINSPECIFIC_VERTICAL_ELEMENTS) import X from xmlInfra // ====================================================================== docu user_en = "This module defines top-level containers of documents, i.e. certain structures which represent products as a whole, in a certain tradition of publishing." public tags webpage = ( STR.title & subtitle? & STR.htmlTitle? & (BIB.author|BIB.authors)? & (STR.metaContentsTag*) & date? & STR.abstract? & (X.lang | X.langs)? ), ( STR.FLOAT.tableOfContents | STR.FLOAT.listOf | @STR.verticalObjects)*, (STR.h1+ | STR.part+)?, STR.appendices?, STR.postappendices? tags subtitle = @STR.title with xmlrep trimmed tags date = #chars with xmlrep trimmed // MISSING FIXME difference to web-page : no meta-tags, etc. public tags article = @webpage docu to_latex article = " public tags pseq = STR.p* // ------------------------------------------------------ docu to_xhtml_1_0 webpage = #d2d #html #head // according to http://schneegans.de/web/xhtml, this should be the FIRST // entry in // the head, and it should use "text/html", not "application/xml+xhtml", // e.g. to be effective for Lynx: #meta #http-equiv content-type #content text/html; charset={$user.outputEncoding}#/meta // needed for html/xss validation: #meta #name referrer #content unsafe-url#/meta // PRE 20240902 // // FIXME stimmt noch nicht, wird IMMER erzeugt !?! : // #if descendant::a:link[a:blank|a:top|a:inframe|a:framename] // #pi d2d_gp xhtml_transitional due to a/@target #/pi // #/if #if $user.iconUrl #x-var l-iconType #call a:extractFileSuffix #arg p-string #xp $user.iconUrl#/x-var #link #rel icon #href {$user.iconUrl} #type image/{$l-iconType} #/if #x-var l-njs #xp normalize-space($user.jsUrls) #if string-length($l-njs)>2 #call processJsUrls #arg p-separator #xp substring($l-njs,1,1) #arg p-list #xp substring($l-njs,2) #/if #x-var l-css #xp normalize-space($user.cssUrls) #if string-length($l-css)>2 #call processCssUrls #arg p-separator #xp substring($l-css,1,1) #arg p-list #xp substring($l-css,2) #/if // The "DC.<>" meta tags come from "Dublin Core", as described in // http://de.selfhtml.org/html/kopfdaten/meta.htm#dublin_core // and http://dublincore.org/documents/dcmi-terms/ #if a:author #meta #name DC.creator #content {a:author} #/if // FIXME check FORMAT/schema of date encoding, etc. #if $date_source #meta #name DC.date #content {$date_source} #meta #name date #content {$date_source} #/if // ATTENTION currently MORE THAN ONE language is permitted, like "#langs de+en" // but these are (a) also stored in d2d:(xml:)lang, and (b) only the first // language code is extracted into $baseLanguage. #meta #name DC.language #content {$baseLanguage} #/meta #apply #xp a:metaContentsTag #choose #when a:htmlTitle #title #valueof a:htmlTitle #meta #name DC.title #content {a:htmlTitle} #other #x-var oneLineTitle #foreach ./a:title #call a:getOneLineText #/x-var #title #valueof $oneLineTitle #/title #meta #name DC.title #content {$oneLineTitle} #/choose #/head #body #call make-header #hr/ #p #apply#xp ./a:title #if a:subtitle #br #span#class pageSubtitle #apply#xp ./a:subtitle#/span #/if #br/ #choose #when a:author and $date_source #apply#xp a:author#text!, !#valueof $date_source #other #apply#xp a:author #valueof $date_source #/choose #hr/ // statt dessen positiv-liste "h1|p" !?!?!? FIXME #apply#xp *[local-name()!='title' and local-name()!='htmlTitle' and local-name()!='subtitle' and local-name()!='author' and local-name()!='date' and local-name()!='metaContentsTag' ] #p#br #hr/ // does not know that "if" does not deliver contents to "hr" #if /descendant-or-self::node()/a:footnote #apply #xp /descendant-or-self::node()/a:footnote #mode footn #p#br #hr #p#br #hr #/if #call make-footer /* === 20221009 moved to ./libbasic.xslt.d2d == GEHT NICHT, da ist kein xhtml bekannt !?!!!! */ /* === ./libbasic_xhtml.xslt.d2d FIXME komische RESOLVER BUG ?!?!? */ #template #name processJsUrls #param p-separator#/param #param p-list#/param #if string-length($p-list)>1 #script #type text/javascript #src {substring-before($p-list, $p-separator)} // WORKAROUND for browser bugs: #commandchar! !text!noescape ￯!/text!commandchar# #/script #call processJsUrls #arg p-separator #xp $p-separator #arg p-list #xp substring-after($p-list, $p-separator) #template #name processCssUrls #param p-separator#/param #param p-list#/param #if string-length($p-list)>1 #link #rel stylesheet #type text/css #href {substring-before($p-list, $p-separator)} #/link #call processCssUrls #arg p-separator #xp $p-separator #arg p-list #xp substring-after($p-list, $p-separator) /* == */ #template #name make-header // still missing #template #name make-footer #hr #p #if $date_translation or $user.host or $user.user #span #style font-size:x-small; #text#noescape made xA0;xA0; #valueof $date_translation #text#noescape xA0;xA0; #if $user.user #text#noescape by xA0;xA0; #valueof $user.user #text#noescape xA0;xA0; #/if #if $user.host #text#noescape on xA0;xA0; #valueof $user.host #text#noescape xA0;xA0; #/if #/span #/if #if not($user.footerNoValidator) #text#noescape xA0;xA0;xA0;xA0;xA0;xA0; #a #href {$a:const.httpPrefix}validator.w3.org/check?uri=referer #img #src valid-xhtml10-blue-66x23.png #alt Valid XHTML 1.0 Strict #style border:0; position:relative; top:5px #/a #a #href {$a:const.httpPrefix}jigsaw.w3.org/css-validator/check/referer #img #src valid-css2-blue-66x23.png #alt Valid CSS 2.1 #style border:0; width:66px; position:relative; top:5px #/a #/if #commentchar\ #if //a:embed[starts-with(@a:embedType,'lily')] #br/ #span#style color:red#text Note setting thanks to #a#href!https://lilypond.org!LilyPond#/a #/if #if //a:embed[contains(@a:embedType,'mutex')] #br/ #span#style color:red#text Note setting thanks to #a#href!https://ctan.org/pkg/musixtex!musixTex#/a #/if #commentchar/ #br/ #if $user.footerSignet #valueof $user.footerSignet #noescape #text#noescape xA0;xA0; #/if // ++++ #span #style font-size:x-small; #text produced with #a#href {$a:const.httpPrefix}bandm.eu/metatools/docs/usage/d2d.html #code #style font-size:medium #text eu.bandm.metatools.d2d #/a #text#noescape xA0;xA0; and xA0;xA0; #a#href {$a:const.httpsPrefix}www.w3.org/TR/xslt-10/ #text XSLT 1.0 // PRE20240913 #a#href {$a:const.httpPrefix}https://www.w3.org/TR/xslt-10/.w3.org/TR/xslt #text XSLT #/a #if $user.pageSource #text#noescape xA0;xA0; #call muliDefault #arg p-key fyi_view#/call #text#noescape xA0; #a #href {$user.pageSource} #/href #text page source text #/a #/if #/span#/p #hr #/template #/d2d docu to_xhtml_1_0 = #d2d #x-var date_source // the date in the d2d source document, otherwise the call date #choose #when /a:webpage/a:date #valueof /a:webpage/a:date #when $user.date #valueof $user.date #/x-var #x-var date_translation // the date of the rendering call, otherwise the doc date #choose #when $user.date #valueof $user.date #when /a:webpage/a:date #valueof /a:webpage/a:date #/x-var #param user.date#/param #param user.user#/param #param user.host#/param #param user.cssUrls#/param #param user.iconUrl#/param #param user.jsUrls#/param #param user.pageSource#/param #param user.footerSignet#/param #param user.footerNoValidator#/param #/d2d // --------------------------------------------------------------- docu to_xhtml_1_0 = #d2d #commentchar \ #output #method xml #encoding utf-8 #indent yes #omit-xml-declaration no #doctype-public -//W3C//DTD XHTML 1.0 Strict//EN #doctype-system http://bandm.eu/doctypes/xhtml/xhtml_1_0.dtd #commentchar / #param user.outputEncoding #xp 'utf-8' #/d2d // --------------------------------------------------------------- // FIXME DISLOC !?!?! template name rootToEmbed ??? match /node() ???? docu to_embedScript webpage = #d2d #apply #template #match h1|h2|h3|h4|h5|h6 // only HERE embeds may reside, not IN p-s #apply #output #method text #encoding utf-8 #indent no #omit-xml-declaration yes #/d2d end module // deliverables end module // basic // eof \n |
DOC-EN:lablog.ddf | bandm meta_tools | DOC:xslt.ddf |
made
2025-01-16_09h57 by
lepper on
happy-ubuntu
produced with
eu.bandm.metatools.d2d
and
XSLT
FYI view
page d2d source text