Search code examples
xmlxsltxslt-1.0xslt-groupingmuenchian-grouping

Nested Grouping using XSLT 1.0


I'm having some difficulties with nested grouping in XSLT 1.0. The problem is a few documents have multiple products in their metadata. First, I need to group the results by the products. Finally, grouping by types is expected as well. Actually, I gained the goals using for-each-group instruction from XSLT 2.0. But the thing is, I`m not able to change the version of XSLT in my project.

I would greatly appreciate it if someone could help me to implement this requirements in XSLT 1.0. I`m wondering if it is possible to use the Muenchian method there and what the named keys are look like.

Here is an example of input xml:

<result>
    <document>
        <metadata>
            <title>Academic Program Directors</title>
            <type>typeA</type>
            <product>Product1</product>
            <product>Product2</product>
            <product>Product3</product>
        </metadata>
    </document>
    <document>
        <metadata>
            <title>Administrative Directors</title>
            <type>typeA</type>
            <product>Product2</product>
        </metadata>
    </document>
    <document>
        <metadata>
            <title>Program Managers</title>
            <type>typeB</type>
            <product>Product1</product>
            <product>Product3</product>
        </metadata>
    </document>
</result>

The expect output would be the following:

Product1
    typeA
        Academic Program Directors
    typeB
        Program Managers

Product2
    typeA
        Academic Program Directors
        Administrative Directors

Product3
    typeA
        Academic Program Directors
    typeB
        Program Managers

Here's the XSLT 2.0 solution

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0">
    <xsl:output method="html" indent="yes"/>

    <xsl:template match="/result">
        <xsl:for-each-group select="document/metadata" group-by="product">
            <xsl:sort select="current-grouping-key()"/>
            <p style="font-weight: bold;">
                <xsl:value-of select="current-grouping-key()"/>
            </p>

            <xsl:for-each-group select="current-group()" group-by="type">
                <xsl:sort select="current-grouping-key()"/>
                <p style="padding-left: 2em;">
                    <xsl:value-of select="current-grouping-key()"/>
                </p>

                <xsl:for-each select="current-group()">
                    <p style="padding-left: 4em;">
                        <xsl:value-of select="title"/>
                    </p>
                </xsl:for-each>
            </xsl:for-each-group>
        </xsl:for-each-group>
    </xsl:template>
</xsl:stylesheet>

Solution

  • A lot of XSLT 1 processors implement set:distinct (http://exslt.org/set/functions/distinct/index.html) so with that you can use the approach in http://xsltransform.net/bEzjRKX which does

    <?xml version="1.0" encoding="UTF-8" ?>
    <xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0"
      xmlns:set="http://exslt.org/sets" exclude-result-prefixes="set">
    
        <xsl:output method="html" indent="yes"/>
    
        <xsl:template match="/result">
            <xsl:variable name="metas" select="document/metadata"/>
            <xsl:for-each select="set:distinct($metas/product)">
                <xsl:sort select="."/>
                <p style="font-weight: bold;">
                    <xsl:value-of select="."/>
                </p>
                <xsl:variable name="current-group" select="$metas[product = current()]"/>
                <xsl:variable name="types" select="set:distinct($current-group/type)"/>
                <xsl:for-each select="$types">
                    <xsl:sort select="."/>
                    <p style="padding-left: 2em;">
                        <xsl:value-of select="."/>
                    </p>
    
                    <xsl:for-each select="$current-group[type = current()]">
                        <p style="padding-left: 4em;">
                            <xsl:value-of select="title"/>
                        </p>
                    </xsl:for-each>
                </xsl:for-each>
            </xsl:for-each>
        </xsl:template>
    
    </xsl:transform>
    

    Using keys and Muenchian grouping I think the problem can be solved with

    <?xml version="1.0" encoding="UTF-8" ?>
    <xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0"
      xmlns:set="http://exslt.org/sets" exclude-result-prefixes="set">
    
        <xsl:output method="html" indent="yes"/>
    
        <xsl:template match="/result">
            <xsl:variable name="metas" select="document/metadata"/>
            <xsl:for-each select="set:distinct($metas/product)">
                <xsl:sort select="."/>
                <p style="font-weight: bold;">
                    <xsl:value-of select="."/>
                </p>
                <xsl:variable name="current-group" select="$metas[product = current()]"/>
                <xsl:variable name="types" select="set:distinct($current-group/type)"/>
                <xsl:for-each select="$types">
                    <xsl:sort select="."/>
                    <p style="padding-left: 2em;">
                        <xsl:value-of select="."/>
                    </p>
    
                    <xsl:for-each select="$current-group[type = current()]">
                        <p style="padding-left: 4em;">
                            <xsl:value-of select="title"/>
                        </p>
                    </xsl:for-each>
                </xsl:for-each>
            </xsl:for-each>
        </xsl:template>
    
    </xsl:transform>
    

    Online at http://xsltransform.net/ehVYZNZ