Creating dynamic XSL-FO table using XSLT
I am creating a table in xsl-fo, with undefined number of elements (indtr).
I wonder if there a smart way to do this?
Here is my input xml file
<pr-levels>
<prs>
<pr_nme><![CDATA[Level 1]]></pr_nme>
<nt><![CDATA[standards:]]></nt>
<b_is>
<indtr id="5684"><![CDATA[cell_1]]></indtr>
<indtr id="5684"><![CDATA[cell_2]]></indtr>
<indtr id="5684"><![CDATA[cell_3]]></indtr>
<indtr id="5684"><![CDATA[cell_4]]></indtr>
</b_is>
</prs>
<prs>
<pr_nme><![CDATA[Level 2]]></pr_nme>
<nt><![CDATA[standards:]]></nt>
<b_is>
<indtr id="5684"><![CDATA[cell_1]]></indtr>
</b_is>
</prs>
<prs>
<pr_nme><![CDATA[Level 3]]></pr_nme>
<nt><![CDATA[standards:]]></nt>
<b_is>
<indtr id="5684"><![CDATA[cell_1]]></indtr>
<indtr id="5684"><![CDATA[cell_2]]></indtr>
</b_is>
</prs>
<prs>
<pr_nme><![CDATA[Level 4]]></pr_nme>
<nt><![CDATA[standards:]]></nt>
<b_is>
<indtr id="5684"><![CDATA[cell_1]]></indtr>
<indtr id="5684"><![CDATA[cell_2]></indtr>
<indtr id="5684"><![CDATA[cell_3]]></indtr>
<indtr id="5684"><![CDATA[cell_4]></indtr>
<indtr id="5684"><![CDATA[cell_5]]></indtr>
<indtr id="5684"><![CDATA[cell_6]]></indtr>
<indtr id="5684"><![CDATA[cell_7]]></indtr>
</b_is>
</prs>
<prs>
<pr_nme><![CDATA[Level 5]]></pr_nme>
<nt><![CDATA[standards:]]></nt>
<b_is>
<indtr id="5684"><![CDATA[cell_1]]></indtr>
<indtr id="5684"><![CDATA[cell_2]]></indtr>
<indtr id="5684"><![CDATA[cell_3]]></indtr>
<indtr id="5684"><![CDATA[cell_4]]></indtr>
<indtr id="5684"><![CDATA[cell_5]]></indtr>
<indtr id="5684"><![CDATA[cell_6]]></indtr>
<indtr id="5684"><![CDATA[cell_7]]></indtr>
<indtr id="5684"><![CDATA[cell_8]]></indtr>
<indtr id="5684"><![CDATA[cell_9]]></indtr>
<indtr id="5684"><![CDATA[cell_10]]></indtr>
<indtr id="5684"><![CDATA[cell_11]]></indtr>
<indtr id="5684"><![CDATA[cell_12]]></indtr>
<indtr id="5684"><![CDATA[cell_13]]></indtr>
<indtr id="5684"><![CDATA[cell_14]]></indtr>
<indtr id="5684"><![CDATA[cell_15]]></indtr>
<indtr id="5684"><![CDATA[cell_16]]></indtr>
<indtr id="5684"><![CDATA[cell_17]]></indtr>
<indtr id="5684"><![CDATA[cell_18]]></indtr>
<indtr id="5684"><![CDATA[cell_19]]></indtr>
</b_is>
</prs>
</pr-levels>
Here is my XSLT template that produces the XSL-FO table:
<xsl:template match="pr-levels">
<fo:table xsl:use-attribute-sets="table_p" break-after="page" force-page-count="no-force">
<fo:table-column column-number="1" xsl:use-attribute-sets="table_col_p"/>
<fo:table-column column-number="2" xsl:use-attribute-sets="table_col_p"/>
<fo:table-column column-number="3" xsl:use-attribute-sets="table_col_p"/>
<fo:table-column column-number="4" xsl:use-attribute-sets="table_col_p"/>
<fo:table-column column-number="5" xsl:use-attribute-sets="table_col_p"/>
<fo:table-header xsl:use-attribute-sets="table_header">
<xsl:for-each select="prs/pr_nme">
<fo:table-cell>
<fo:block>
<xsl:apply-templates/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-header>
<fo:table-body >
<fo:table-row>
<xsl:for-each select="prs/nt">
<fo:table-cell>
<fo:block xsl:use-attribute-sets="nt" >
<xsl:apply-templates/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
<fo:table-row>
<xsl:for-each select="prs/b_is">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="indtr[1]"/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
<fo:table-row>
<xsl:for-each select="prs/b_is">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="indtr[2]"/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
<fo:table-row>
<xsl:for-each select="prs/b_is">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="indtr[3]"/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
<fo:table-row>
<xsl:for-each select="prs/b_is">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="indtr[4]"/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
<fo:table-row>
<xsl:for-each select="prs/b_is">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="indtr[5]"/>
</f开发者_开发知识库o:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
</fo:table-body>
</fo:table>
</xsl:template>
There could be NULL,1,69 or whatever numbers of <indtr>
s in a <b_is>
,but right now I just hard code the numbers.My question is how I can dynamically count the number of indtrs and add rows in my table.maybe a for loop in xsl?
________________________________________________________
| level 1 | level 2 | level 3 | level 4 | level 5 |
--------------------------------------------------------
|standards:|standards:|standards:|standards:|standards:|
--------------------------------------------------------
| cell 1 | cell 1 | cell 1 | cell 1 | cell 1 |
--------------------------------------------------------
| cell 2 | cell 2 | cell 2 | cell 2 | cell 2 |
--------------------------------------------------------
| | cell 3 | cell 3 | cell 3 | cell 3 |
--------------------------------------------------------
| | cell 4 | | | cell 4 |
--------------------------------------------------------
| | | | | cell 5 |
--------------------------------------------------------
| | | | | cell 6 |
--------------------------------------------------------
Just because it's an interesting problem...
Replace the multiple hard-coded fo:table-row
with:
<xsl:variable name="pr-levels" select="." />
<xsl:for-each
select="1 to max(for $prs in prs
return count($prs/b_is/indtr))">
<xsl:variable name="row" select="." as="xs:integer" />
<fo:table-row>
<xsl:for-each select="$pr-levels/prs">
<fo:table-cell padding="1pt">
<fo:block>
<xsl:value-of select="b_is/indtr[$row]"/>
</fo:block>
</fo:table-cell>
</xsl:for-each>
</fo:table-row>
</xsl:for-each>
精彩评论