使用param进行递归XSL模板调用

时间:2014-12-19 15:41:34

标签: xml xslt xpath recursion xslt-1.0

我尝试使用以下代码完成页面导航。 虽然这段代码对于重要的部分是愚蠢的,但应该清楚我的想法是什么。 ID可以是任何给定的名称,并且可以按任何给定的顺序出现在任何给定的数字中。

输入:定义已排序集合中上一页和下一页的页面:

<pages>
  <page id="page1" next="page2" />
  <page id="page2" prev="page1" next="page3" />
  <page id="page3" prev="page2" />
  <page id="test2" prev="test1" />
  <page id="test1" next="test2" />
  <page id="alone" />
</pages>

所需的输出(我无法生成):

<page id="page1">
  <link href="page1" class="active" />
  <link href="page2" />
  <link href="page3" />
</page>
<page id="page2">
  <link href="page1" />
  <link href="page2" class="active" />
  <link href="page3" />
</page>
<page id="page3">
  <link href="page1" />
  <link href="page2" />
  <link href="page3" class="active" />
</page>
<page id="test1">
  <link href="test1" class="active" />
  <link href="test2 />
</page>
<page id="test2">
  <link href="test1" />
  <link href="test2" class="active" />
</page>
<page id="alone">
</page>

到目前为止,我曾尝试使用XSL:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:for-each select="pages/page">
  <page id="{@id}">

    <xsl:call-template name="prev-page">
      <xsl:with-param name="id">
        <xsl:value-of select="@prev" />
      </xsl:with-param>
    </xsl:call-template>

    <link href="{@id}" class="active" />

    <xsl:call-template name="next-page">
      <xsl:with-param name="id">
        <xsl:value-of select="@next" />
      </xsl:with-param>
    </xsl:call-template>

  </page>
</xsl:for-each>

<xsl:template name="prev-page">
  <xsl:param name="id" />

  <xsl:if test="string-length($id) > 0">
    <link href="{$id}" />
    <xsl:call-template name="prev-page">
      <xsl:with-param name="id">
        <!-- I believe in here lies the problem -->
        <xsl:value-of select="(/pages/page[@id = $id])[0]/@prev" />
      </xsl:with-param>
    </xsl:call-template>
  </xsl:if>
</xsl:template>
... (same for page-next)
</xsl:stylesheet>

2 个答案:

答案 0 :(得分:4)

让我提出一种截然不同的方法。首先,让我们按要求的顺序排序页面。完成后,依次输出每个页面以及指向(所有)其他页面的链接:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>

<!-- first-pass -->
<xsl:key name="page" match="page" use="@id" />

<xsl:variable name="sorted-pages">
    <xsl:apply-templates select="pages/page[not(@prev)]" mode="first-pass"/>
</xsl:variable>

<xsl:template match="page" mode="first-pass">
    <xsl:copy-of select="."/>
    <xsl:apply-templates select="key('page', @next)" mode="first-pass"/>
</xsl:template>

<xsl:variable name="sorted-pages-set" select="exsl:node-set($sorted-pages)/page" />

<!-- final-output -->
<xsl:template match="/">
    <pages>
        <xsl:for-each select="$sorted-pages-set">
        <xsl:variable name="current-id" select="@id" />
            <page id="{$current-id}">
                <xsl:apply-templates select="$sorted-pages-set">
                    <xsl:with-param name="current-id" select="$current-id"/>
                </xsl:apply-templates>
            </page>
        </xsl:for-each>
    </pages>
</xsl:template>

<xsl:template match="page">
    <xsl:param name="current-id" />
    <link href="{@id}">
        <xsl:if test="@id=$current-id">
            <xsl:attribute name="class">active</xsl:attribute>
        </xsl:if>
    </link>
</xsl:template>

</xsl:stylesheet>

应用于以下测试输入

<pages>
    <page id="alone" />
    <page id="fourth" prev="third" next="MISSING" />
    <page id="second" prev="first" next="third" />
    <page id="ii" prev="i" next="iii" />/>  
    <page id="first" next="second" />
    <page id="third" prev="second" next="fourth" />
    <page id="solo" />
    <page id="i" next="ii" />
    <page id="b" prev="a" /> 
    <page id="iii" prev="ii" />
    <page id="PROBLEM" prev="MISSING"/>
    <page id="a" next="b" />
</pages>

生成以下结果

<?xml version="1.0" encoding="UTF-8"?>
<pages>
   <page id="alone">
      <link href="alone" class="active"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="first">
      <link href="alone"/>
      <link href="first" class="active"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="second">
      <link href="alone"/>
      <link href="first"/>
      <link href="second" class="active"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="third">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third" class="active"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="fourth">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth" class="active"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="solo">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo" class="active"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="i">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i" class="active"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="ii">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii" class="active"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="iii">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii" class="active"/>
      <link href="a"/>
      <link href="b"/>
   </page>
   <page id="a">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a" class="active"/>
      <link href="b"/>
   </page>
   <page id="b">
      <link href="alone"/>
      <link href="first"/>
      <link href="second"/>
      <link href="third"/>
      <link href="fourth"/>
      <link href="solo"/>
      <link href="i"/>
      <link href="ii"/>
      <link href="iii"/>
      <link href="a"/>
      <link href="b" class="active"/>
   </page>
</pages>

答案 1 :(得分:0)

我从michael.hor257k获得灵感,想出了自己的解决方案。我遍历链表,直到我到达@prev属性指向非现有页面的页面。然后我向前遍历列表并打印出元素,直到@next属性指向不存在的页面。

<xsl:template match="/">
  <xsl:for-each select="page">
    <page id="{@id}">
      <xsl:if test="string-length(@next) > 0 or string-length(@prev) > 0">
        <xsl:call-template name="page-navigation-reverse">
          <xsl:with-param name="id">
            <xsl:value-of select="@prev" />
          </xsl:with-param>
          <xsl:with-param name="current-id">
            <xsl:value-of select="@id" />
          </xsl:with-param>
        </xsl:call-template>
      </xsl:if>
    </page>
  </xsl:for-each>
</xsl:template>

<xsl:template name="page-navigation-reverse">
  <xsl:param name="id" />
  <xsl:param name="current-id" />

  <xsl:variable name="item-id">
    <xsl:choose>
      <xsl:when test="string-length($id) &lt;= 0">
        <xsl:value-of select="$current-id" />
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="$id" />
      </xsl:otherwise>
    </xsl:choose>
  </xsl:variable>

  <xsl:choose>

    <xsl:when test="//page[@id = $item-id]/@prev">

      <xsl:call-template name="page-navigation-reverse">
        <xsl:with-param name="id">
          <xsl:value-of select="//page[@id = $item-id]/@prev" />
        </xsl:with-param>
        <xsl:with-param name="current-id">
          <xsl:value-of select="$current-id" />
        </xsl:with-param>
      </xsl:call-template>
    </xsl:when>

    <xsl:otherwise>
      <xsl:call-template name="page-navigation-forward">
        <xsl:with-param name="id">
          <xsl:value-of select="$item-id" />
        </xsl:with-param>
        <xsl:with-param name="current-id">
          <xsl:value-of select="$current-id" />
        </xsl:with-param>
      </xsl:call-template>
    </xsl:otherwise>

  </xsl:choose>
</xsl:template>

<xsl:template name="page-navigation-forward">
  <xsl:param name="id" />
  <xsl:param name="current-id" />

  <link href="{$id}">
    <xsl:attribute name="class">
      <xsl:if test="$id = $current-id">
        <xsl:text> active</xsl:text>
      </xsl:if>
    </xsl:attribute>
  </link>

  <xsl:if test="//page[@id = $id]/@next">
    <xsl:call-template name="page-navigation-forward">
      <xsl:with-param name="id">
        <xsl:value-of select="//page[@id = $id]/@next" />
      </xsl:with-param>
      <xsl:with-param name="current-id">
        <xsl:value-of select="$current-id" />
      </xsl:with-param>
    </xsl:call-template>
  </xsl:if>
</xsl:template>