Antenna House Formatter 如何处理引用的脚注?

How can Antennahouse Formatter handle referenced footnotes?

我正在使用 xsl-fo(Saxon XSL 2.0、AHF V6.2)开发 pdf 印刷出版物。

我的目标是使用来自引用的静态文本元素的插入文本自动编号脚注(排除单页上的重复项)。

基本上内联脚注 (fn) 会引用静态脚注文本元素,创建内联编号并在页面底部打印相应的脚注文本。

<?xml version="1.0" encoding="UTF-8"?>
<document>
<chapter>
    <paragraph>some description...</paragraph>
    <paragraph>some description with a footnote <fn id="fn2"/></paragraph>
    <paragraph>some description with a footnote <fn id="fn2"/></paragraph>
    <paragraph>some description...</paragraph>
    <paragraph>some description with a footnote <fn id="fn1"/></paragraph>
</chapter>
<!-- this is a wrapper element that will not be displayed in the rendered pdf but only contains the needed information for different footnote texts -->
<chapter class="footnoteWrapper">
    <footnote id="fn1">
        This is the text body of footnote #1.
    </footnote>
    <footnote id="fn2">
        This is the text body of footnote #2.
    </footnote>
    <footnote id="fn3">
        This is the text body of footnote #3.
    </footnote>
</chapter>
</document>

一章中重复的内联脚注必须根据它们指向的脚注显示相同的编号。

结果应该是这样的...

是否可以通过 AHF 脚注扩展和 fo:footnote 元素实现这些目标?

如果我将 AntennaHouse Formatter 扩展用于 fn 计数,它们确实会产生奇怪的行为。他们确实继续计数 (1, 2, 3) 而不是引用所引用脚注的正确和当前编号。

这是目前为止的 XSL(只是相关的片段):

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="xs"
    version="2.0">

<xsl:template match="fn[@id = //footnote/@nodeid]"
    mode="content"
    priority="7">
    <!--+ fn link
        |
        | basic fn (inline) link template.
        |
        +-->
    <xsl:apply-templates select="//footnote[@id = current()/@id]"
        mode="content"/>
</xsl:template>

<xsl:template match="footnote"
    mode="content"
    priority="5">
    <!--+ footnote
        |
        | basic footnote template.
        |
        +-->
    <fo:footnote xsl:use-attribute-sets="fnt.footnote">
        <fo:inline baseline-shift="super">
            <axf:footnote-number id="fn_{@id}"/>
        </fo:inline>
        <fo:footnote-body space-after="1mm">
            <fo:list-block provisional-distance-between-starts="5mm"
                provisional-label-separation="2mm">
                <fo:list-item>
                    <fo:list-item-label end-indent="label-end()">
                        <fo:block>
                            <fo:inline baseline-shift="super">
                                <axf:footnote-number-citation ref-id="fn_{@id}"/>
                            </fo:inline>
                        </fo:block>
                    </fo:list-item-label>
                    <fo:list-item-body start-indent="body-start()">
                        <fo:block>
                            <xsl:apply-templates mode="content"/>
                        </fo:block>
                    </fo:list-item-body>
                </fo:list-item>
            </fo:list-block>
        </fo:footnote-body>
    </fo:footnote>
</xsl:template>
</xsl:stylesheet>

这些更改会在第一次使用脚注时生成脚注,并只为后续次数生成数字:

<xsl:key name="fn" match="fn[exists(key('footnote', @id))]" use="@id" />
<xsl:key name="fn-first" match="fn[. is key('fn', @id)[1]]" use="@id" />
<xsl:key name="footnote" match="footnote" use="@id" />

<xsl:template match="fn[exists(key('footnote', @id))][. is key('fn-first', @id)]"
    mode="content"
    priority="7">
    <xsl:apply-templates select="key('footnote', @id)"
        mode="content">
        <xsl:with-param name="number" select="count(preceding::fn[. is key('fn-first', @id)]) + 1"></xsl:with-param>
    </xsl:apply-templates>
</xsl:template>

<xsl:template match="fn[exists(key('footnote', @id))][not(. is key('fn-first', @id))]"
    mode="content"
    priority="7">
    <fo:inline baseline-shift="super">
        <xsl:value-of select="count(key('fn-first', @id)/preceding::fn[. is key('fn-first', @id)]) + 1"/>
    </fo:inline>
</xsl:template>

<xsl:template match="footnote" mode="content" priority="5">
    <xsl:param name="number" select="count(preceding-sibling::footnote) + 1" as="xs:integer" />
    <fo:footnote xsl:use-attribute-sets="fnt.footnote">
        <fo:inline baseline-shift="super">
            <xsl:value-of select="$number" />
        </fo:inline>
        <fo:footnote-body space-after="1mm">
            <fo:list-block provisional-distance-between-starts="5mm"
                provisional-label-separation="2mm">
                <fo:list-item>
                    <fo:list-item-label end-indent="label-end()">
                        <fo:block>
                            <fo:inline baseline-shift="super">
                                <xsl:value-of select="$number" />
                            </fo:inline>
                        </fo:block>
                    </fo:list-item-label>
                    <fo:list-item-body start-indent="body-start()">
                        <fo:block>
                            <xsl:apply-templates mode="content" />
                        </fo:block>
                    </fo:list-item-body>
                </fo:list-item>
            </fo:list-block>
        </fo:footnote-body>
    </fo:footnote>
</xsl:template>

您可以进一步整理它,例如,制作一个 returns 为 fncount() 值的函数,但这应该让您继续。

请参阅我的其他回答,了解如何同时使用 axf:suppress-duplicate-footnoteaxf:footnote-number,以便仅当重复项位于同一页面时才抑制重复项。

重复的脚注也有重复的 ID。来自非唯一 ID 的错误阻碍了 axf:suppress-duplicate-footnote 处理。

如果您没有链接到脚注,请根据引用它的 fn 为每个脚注生成一个唯一 ID:

<xsl:template match="fn[exists(key('footnote', @id))]" mode="content" priority="7">
    <!--+ fn link
    |
    | basic fn (inline) link template.
    |
    +-->
    <xsl:apply-templates select="key('footnote', @id)" mode="content">
        <xsl:with-param name="id" select="generate-id()" />
    </xsl:apply-templates>
</xsl:template>

<xsl:template match="footnote" mode="content" priority="5">
    <xsl:param name="id" />
    <!--+ footnote
    |
    | basic footnote template.
    |
    +-->
    <fo:footnote xsl:use-attribute-sets="fnt.footnote" axf:suppress-duplicate-footnote="true">
        <fo:inline baseline-shift="super">
            <axf:footnote-number id="{$id}" />
        </fo:inline>
        <fo:footnote-body space-after="1mm">
            <fo:list-block provisional-distance-between-starts="5mm"
                provisional-label-separation="2mm">
                <fo:list-item>
                    <fo:list-item-label end-indent="label-end()">
                        <fo:block>
                            <fo:inline baseline-shift="super">
                                <axf:footnote-number-citation ref-id="{$id}" />
                            </fo:inline>
                        </fo:block>
                    </fo:list-item-label>
                    <fo:list-item-body start-indent="body-start()">
                        <fo:block>
                            <xsl:apply-templates mode="content" />
                        </fo:block>
                    </fo:list-item-body>
                </fo:list-item>
            </fo:list-block>
        </fo:footnote-body>
    </fo:footnote>
</xsl:template>