XSLT选择包含强标记的文本

时间:2015-07-13 13:15:43

标签: xml xslt

我尝试使用XSLT对某些XML进行转换。

我的xml是;

    <body>
      <slideshow data-uuid="z3970z88-0475-11dz-8603-00144zeabdc1/>
      <p>Some <strong>strong</strong> text</p>
    </body>

我希望将其转换为;

    <xsl:template match="/body/p[a[substring(@href, string-length(@href) - 6) = '#slide0' and string-length(text()) = 0] and count(*) = 1]">
      <xsl:apply-templates select="a" />
      <xsl:if test="string-length(text()) > 0">
        <p>
          <xsl:value-of select="text()"/>
        </p>
      </xsl:if>
    </xsl:template>

    <xsl:template match="a[substring(@href, string-length(@href) - 6) = '#slide0' and string-length(text()) = 0]">
      <slideshow>
        <xsl:attribute name="data-uuid">
          <xsl:value-of select="substring-before(substring(@href, string-length(@href) - 47), '.html#slide0')" />
        </xsl:attribute>
    </xsl:template>

到目前为止我所拥有的是什么;

<strong>

但这仅适用于文字没有任何子项的情况,例如<a>标记或其他int index_to_del[$]; index_to_del = que.find_first_index(x) with ( x == task_to_del ); que.delete(index_to_del[0]); 标记。

有没有人有解决方案可以全力以赴。

2 个答案:

答案 0 :(得分:0)

我已经想出了这个XSLT。我不确定是否符合你能想到的每一个目的,但它确实解决了你在这里给出的例子:

<?xml version="1.0"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:template match="/body/p[a]">
        <xsl:apply-templates select="a" />
        <p>
            <xsl:copy-of select="./node()[not(self::a)]" />
        </p>
    </xsl:template>

    <xsl:template match="a[substring(@href, string-length(@href) - 6) = '#slide0' and string-length(text()) = 0]">
        <slideshow>
            <xsl:attribute name="data-uuid">
                <xsl:value-of select="substring-before(substring(@href, string-length(@href) - 47), '.html#slide0')" />
            </xsl:attribute>
        </slideshow>
    </xsl:template>
</xsl:stylesheet>

我们的想法是复制<p>代码中<a>标记的内容。 我在第一个apply-templates上改变了条件,但你不必这样做。它让我的内容更具可读性。

答案 1 :(得分:0)

如果您只转换部分XML,通常以Identity Template

开头是一种好方法
<xsl:template match="@*|node()" >
    <xsl:copy>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

这意味着,不要在模板中使用xsl:value-of,而是使用xsl:apply-templates并让身份标识

    <xsl:if test="string-length(text()) > 0">
        <p>
            <xsl:apply-templates select="@*|node()[not(self::a)]"/>
        </p>
    </xsl:if>

试试这个XSLT

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

    <xsl:template match="p[a[substring(@href, string-length(@href) - 6) = '#slide0' and string-length(text()) = 0] and count(*) = 1]">
        <xsl:apply-templates select="a" mode="slideshow" />
        <xsl:if test="string-length(text()) > 0">
            <p>
                <xsl:apply-templates select="@*|node()[not(self::a)]"/>
            </p>
        </xsl:if>
    </xsl:template>

    <xsl:template match="a" mode="slideshow">
      <slideshow>
        <xsl:attribute name="data-uuid">
          <xsl:value-of select="substring-before(substring(@href, string-length(@href) - 47), '.html#slide0')" />
        </xsl:attribute>
      </slideshow>
    </xsl:template>

    <xsl:template match="@*|node()" >
        <xsl:copy>
            <xsl:apply-templates select="@*|node()"/>
        </xsl:copy>
    </xsl:template>
</xsl:stylesheet>

请注意在匹配a的模板中使用“模式”。这纯粹是为了避免两次编码冗长的条件。

另外,您可以在a模板中使用Attribute Value Templates来简化它

<xsl:template match="a" mode="slideshow">
      <slideshow data-uuid="{substring-before(substring(@href, string-length(@href) - 47), '.html#slide0')}" />
</xsl:template>