Как проверить, существует ли тег в XSLT?

У меня есть следующий шаблон

<h2>one</h2>
<xsl:apply-templates select="one"/>
<h2>two</h2>
<xsl:apply-templates select="two"/>
<h2>three</h2>
<xsl:apply-templates select="three"/>

Я хотел бы отображать только заголовки (один,два, три), если есть хотя бы один член соответствующего шаблона. Как это проверить?

2 ответов


<xsl:if test="one">
  <h2>one</h2>
  <xsl:apply-templates select="one"/>
</xsl:if>
<!-- etc -->

кроме того, вы можете создать именованный шаблон

<xsl:template name="WriteWithHeader">
   <xsl:param name="header"/>
   <xsl:param name="data"/>
   <xsl:if test="$data">
      <h2><xsl:value-of select="$header"/></h2>
      <xsl:apply-templates select="$data"/>
   </xsl:if>
</xsl:template>

а затем позвоните как:

  <xsl:call-template name="WriteWithHeader">
    <xsl:with-param name="header" select="'one'"/>
    <xsl:with-param name="data" select="one"/>
  </xsl:call-template>

но, честно говоря, для меня это больше похоже на работу... полезно, только если рисование заголовка является сложным... для простого <h2>...</h2> у меня было бы искушение оставить его встроенным.

Если заголовок заголовка всегда является именем узла, вы можете упростить шаблон, удалив arg "$header " и использовать вместо этого:

<xsl:value-of select="name($header[1])"/>

мне нравится пользоваться функциональное аспекты XSL, которые приводят меня к следующей реализации:

<?xml version="1.0" encoding="UTF-8"?>

<!-- test data inlined -->
<test>
    <one>Content 1</one>
    <two>Content 2</two>
    <three>Content 3</three>
    <four/>
    <special>I'm special!</special>
</test>

<!-- any root since take test content from stylesheet -->
<xsl:template match="/">
    <html>
        <head>
            <title>Header/Content Widget</title>
        </head>
        <body>
            <xsl:apply-templates select="document('')//test/*" mode="header-content-widget"/>
        </body>
    </html>
</xsl:template>

<!-- default action for header-content -widget is apply header then content views -->
<xsl:template match="*" mode="header-content-widget">
    <xsl:apply-templates select="." mode="header-view"/>
    <xsl:apply-templates select="." mode="content-view"/>
</xsl:template>

<!-- default header-view places element name in <h2> tag -->
<xsl:template match="*" mode="header-view">
    <h2><xsl:value-of select="name()"/></h2>
</xsl:template>

<!-- default header-view when no text content is no-op -->
<xsl:template match="*[not(text())]" mode="header-view"/>

<!-- default content-view is to apply-templates -->
<xsl:template match="*" mode="content-view">
    <xsl:apply-templates/>
</xsl:template>

<!-- special content handling -->
<xsl:template match="special" mode="content-view">
    <strong><xsl:apply-templates/></strong>
</xsl:template>

раз в тело все элементы, содержащиеся в тест элемент applied (в порядке следования документов).

по умолчанию шаблон ( " * " ) впервые применяет затем применяется content-view к текущему элементу.

по умолчанию шаблон помещает текущий элемент в имя в теге h2. Значение по умолчанию content-view применяет общие правила обработки.

когда нет содержания, по мнению [не(текст())] предикат не выводится для элемента.

один специальные дела легко обрабатываются.