reynaldo(_dot_)rizzo(_at_)aciworldwide(_dot_)com wrote:
The desired output is a delimited file shown as follows:
'output.csv'
root,itemCode,itemName,aCode,aDescription,bCode,bDescription,cCode,cDescription
01,name0,10,description0,100,description2,996,description4
01,name0,10,description0,100,description2,997,description5
01,name0,10,description0,200,description3,996,description4
01,name0,10,description0,200,description3,997,description5
02,name1,20,description6,null,null,998,description10
02,name1,20,description6,null,null,999,description11
Output is the Cartesian product of each item with its own sub-sequences.
Here is an XSLT 2.0 stylesheet that produces the above output, with the
exception of the first line where I have left out the 'root,' at the
beginning.
<xsl:stylesheet
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:mf="http://example.com/2009/mf"
version="2.0">
<xsl:output method="text"/>
<xsl:param name="lf" select="' '" as="xs:string"/>
<xsl:param name="sep" select="','" as="xs:string"/>
<xsl:param name="default" select="'null'" as="xs:string"/>
<xsl:template match="/">
<xsl:variable name="max-items" select="max(root/item/count((*
except (code, name))))"/>
<xsl:variable name="item-list" select="root/item[count(* except
(code, name)) eq $max-items][1]/(* except (code, name))"/>
<xsl:value-of select="('itemCode', 'itemName',
$item-list/(concat(substring-before(name(), 'List'), 'Code'),
concat(substring-before(name(), 'List'), 'Description')))"
separator="{$sep}"/>
<xsl:value-of select="$lf"/>
<xsl:for-each select="root/item">
<xsl:call-template name="mf:make-lines">
<xsl:with-param name="item" select="."/>
<xsl:with-param name="item-list" select="$item-list"/>
</xsl:call-template>
</xsl:for-each>
</xsl:template>
<xsl:template name="mf:make-lines">
<xsl:param name="item" as="element()"/>
<xsl:param name="item-list" as="element()*"/>
<xsl:param name="line" as="item()*" select="()"/>
<xsl:variable name="current-list" as="element()?"
select="$item/*[node-name(.) eq node-name($item-list[1])]"/>
<xsl:choose>
<xsl:when test="count($item-list) eq 1">
<xsl:choose>
<xsl:when test="$current-list">
<xsl:for-each select="$current-list/*">
<xsl:value-of select="($item/code, $item/name, $line,
code, description)" separator="{$sep}"/>
<xsl:value-of select="$lf"/>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="($item/code, $item/name, $line,
$default, $default)" separator="{$sep}"/>
<xsl:value-of select="$lf"/>
</xsl:otherwise>
</xsl:choose>
</xsl:when>
<xsl:otherwise>
<xsl:choose>
<xsl:when test="$current-list">
<xsl:for-each select="$current-list/*">
<xsl:call-template name="mf:make-lines">
<xsl:with-param name="item" select="$item"/>
<xsl:with-param name="item-list"
select="$item-list[position() gt 1]"/>
<xsl:with-param name="line" select="($line, code,
description)"/>
</xsl:call-template>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="mf:make-lines">
<xsl:with-param name="item" select="$item"/>
<xsl:with-param name="item-list"
select="$item-list[position() gt 1]"/>
<xsl:with-param name="line" select="($line, $default,
$default)"/>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
To decide which *List elements to output the stylesheet looks for the
maximum number of child elements of an 'item' element, then for each
'item' element the named template mf:make-lines is called, passing in
the item element and the item list.
That template recursively builds each line to be output as a sequence,
shortening the item list each time.
--
Martin Honnen
http://msmvps.com/blogs/martin_honnen/
--~------------------------------------------------------------------
XSL-List info and archive: http://www.mulberrytech.com/xsl/xsl-list
To unsubscribe, go to: http://lists.mulberrytech.com/xsl-list/
or e-mail: <mailto:xsl-list-unsubscribe(_at_)lists(_dot_)mulberrytech(_dot_)com>
--~--