Как вычислить сумму значения атрибута colspec / @ colwidth в совпадении Атрибут namest для nameend в записи Element- XSLT - PullRequest
0 голосов
/ 25 апреля 2020

Я пытаюсь вычислить сумму tgroup/colspec/@colwidth, когда значение namest to nameend совпадает с tgroup/colspec/@colname. Я вычислил только ширину атрибута в colname = "COLSPEC0" и colname = "1". Отсутствует ширина в colname = "COLSPEC1". Ввод. XML

<?xml version="1.0" encoding="UTF-8"?>
<table frame="all" orient="portrait">
    <title>MEDICARE</title>
    <tgroup align="center" cols="4">
        <colspec colname="COLSPEC0" colwidth="6.86pi"/>
        <colspec colname="COLSPEC1" colwidth="6.31pi"/>
        <colspec colname="1" colwidth="5.75pi"/>
        <colspec colname="2" colwidth="5.87pi"/>
        <tbody>
            <row>
                <entry nameend="1" namest="COLSPEC0">
                    <para flow="new">If beneficiary files</para>
                </entry>
                <entry align="center" colname="2" colsep="1" morerows="0" nameend="2" namest="2" rowsep="1">
                    <para flow="new">Monthly premium in 2019 is:</para>
                </entry>
            </row>
        </tbody>
    </tgroup>
</table>

Мой XSLT-код

    <?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:output method="xml"/>

    <xsl:template match="@*|node()">
        <xsl:copy>
            <xsl:apply-templates select="@*|node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="entry">
        <xsl:copy>
            <xsl:copy-of select="@*"/>
        <xsl:variable name="namest" select="@namest"/>
        <xsl:variable name="nameend" select="@nameend"/>
            <xsl:variable name="namestPos" select="number(substring-before(ancestor::tgroup/colspec[@colname=$namest]/@colwidth, 'pi'))"/>
            <xsl:variable name="nameendPos" select="number(substring-before(ancestor::tgroup/colspec[@colname=$nameend]/@colwidth, 'pi'))"/>
        <xsl:if test="@namest != @nameend">
            <xsl:attribute name="width">
                    <xsl:value-of select="$namestPos + $nameendPos"/>
            </xsl:attribute>
        </xsl:if>
            <xsl:apply-templates/>
        </xsl:copy>
    </xsl:template>

</xsl:stylesheet>

Ожидаемый вывод

    <?xml version="1.0" encoding="UTF-8"?><table frame="all" orient="portrait">
    <title>MEDICARE</title>
    <tgroup align="center" cols="4">
        <colspec colname="COLSPEC0" colwidth="6.86pi"/>
        <colspec colname="COLSPEC1" colwidth="6.31pi"/>
        <colspec colname="1" colwidth="5.75pi"/>
        <colspec colname="2" colwidth="5.87pi"/>
        <tbody>
            <row>
                <entry nameend="1" namest="COLSPEC0" width="18.92">
                    <para flow="new">If beneficiary files</para>
                </entry>
                <entry align="center" colname="2" colsep="1" morerows="0" nameend="2" namest="2" rowsep="1">
                    <para flow="new">Monthly premium in 2019 is:</para>
                </entry>
            </row>
        </tbody>
    </tgroup>
</table>

Если я запустил этот код, создайте значение суммы по ширине = "12,61".

1 Ответ

1 голос
/ 25 апреля 2020

Я бы сделал это так:

  <xsl:template match="entry[@namest != @nameend]">
      <xsl:copy>
          <xsl:copy-of select="@*"/>
          <xsl:attribute 
            name="width" 
            select="let $colspecs := ancestor::tgroup/colspec,
                    $names := data($colspecs/@colname)
                    return sum($colspecs[position() = index-of($names, current()/@namest) to index-of($names, current()/@nameend)]/@colwidth/xs:decimal(substring-before(., 'pi')))"/>
          <xsl:apply-templates/>                                    
      </xsl:copy>
  </xsl:template>

https://xsltfiddle.liberty-development.net/bEzknsT

Если у вас есть только поддержка XSLT 2, вам нужно использовать два xsl:variable объявления для colspecs и names вместо встроенного XPath let, который я использовал.

...