使用XSLT动态添加HTML标记

使用XSLT动态添加HTML标记,html,xml,xslt,Html,Xml,Xslt,我有一个包含以下内容的XML文档: <d1/> <p1>...</p1> <p2>...</p2> <d2/> <p3>...</p3> <d3/> ... ... ... 其中pn是可能包含子元素和其他内容的元素,dn表示包装p标记的HTML DIV标记应该从哪里开始,但没有相应的结束标记,这仅由下一个dn标记隐式指示。所需的HTML输出如下: <div> <p1

我有一个包含以下内容的XML文档:

<d1/>
<p1>...</p1>
<p2>...</p2>
<d2/>
<p3>...</p3>
<d3/>

...
...
...
其中pn是可能包含子元素和其他内容的元素,dn表示包装p标记的HTML DIV标记应该从哪里开始,但没有相应的结束标记,这仅由下一个dn标记隐式指示。所需的HTML输出如下:

<div>
<p1>...</p1>
<p2>...</p2>
</div>
<div>
<p3>...</p3>
</div>

...
...
...
我编写了一个XSLT来动态地介绍
标记,使用了以下内容:

<xsl:text disable-output-escaping="yes">&lt;div&gt;</xsl:text>
div

/div
这在Safari上有效,但在FireFox上失败,这让我怀疑这不是正确的方法。 你有没有更好的办法可以在每个浏览器上使用


提前非常感谢。

您可以使用一种称为“兄弟递归”的技术

给定格式良好的输入,例如:

XML

<root>
    <d1/>
    <p1>a</p1>
    <p2>b</p2>
    <d2/>
    <p3>c</p3>
    <d3/>
</root>

A.
B
C
以下样式表:

XSLT1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="html"/>

<xsl:template match="/root">
    <body>
        <xsl:apply-templates select="*[starts-with(name(), 'd')][position()!=last()]"/>
    </body>
</xsl:template>

<xsl:template match="*[starts-with(name(), 'd')]">
    <div>
        <xsl:apply-templates select="following-sibling::*[1][not(starts-with(name(), 'd'))]"/>
    </div>
</xsl:template>

<xsl:template match="/root/*[not(starts-with(name(), 'd'))]">
    <xsl:copy-of select="."/>
    <xsl:apply-templates select="following-sibling::*[1][not(starts-with(name(), 'd'))]"/>
</xsl:template>

</xsl:stylesheet>

将返回:

<body>
   <div>
      <p>a</p>
      <p>b</p>
   </div>
   <div>
      <p>c</p>
   </div>
</body>

a

b

c


您可以使用一种称为“同级递归”的技术

给定格式良好的输入,例如:

XML

<root>
    <d1/>
    <p1>a</p1>
    <p2>b</p2>
    <d2/>
    <p3>c</p3>
    <d3/>
</root>

A.
B
C
以下样式表:

XSLT1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="html"/>

<xsl:template match="/root">
    <body>
        <xsl:apply-templates select="*[starts-with(name(), 'd')][position()!=last()]"/>
    </body>
</xsl:template>

<xsl:template match="*[starts-with(name(), 'd')]">
    <div>
        <xsl:apply-templates select="following-sibling::*[1][not(starts-with(name(), 'd'))]"/>
    </div>
</xsl:template>

<xsl:template match="/root/*[not(starts-with(name(), 'd'))]">
    <xsl:copy-of select="."/>
    <xsl:apply-templates select="following-sibling::*[1][not(starts-with(name(), 'd'))]"/>
</xsl:template>

</xsl:stylesheet>

将返回:

<body>
   <div>
      <p>a</p>
      <p>b</p>
   </div>
   <div>
      <p>c</p>
   </div>
</body>

a

b

c


Firefox不支持禁用输出转义,因为它不序列化结果树。该问题是一个分组问题,解决该问题的一种方法是使用密钥:

<xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">

<xsl:output indent="yes"/>

<xsl:key name="group" match="body/*[not(starts-with(local-name(), 'd'))]" use="generate-id(preceding-sibling::*[starts-with(local-name(), 'd')][1])"/>

<xsl:template match="body">
    <xsl:copy>
        <xsl:apply-templates select="*[starts-with(local-name(), 'd')]"/>
    </xsl:copy>
</xsl:template>

<xsl:template match="*[starts-with(local-name(), 'd')]">
    <div>
        <xsl:copy-of select="key('group', generate-id())"/>
    </div>
</xsl:template>
</xsl:transform>

Firefox不支持禁用输出转义,因为它不序列化结果树。该问题是一个分组问题,解决该问题的一种方法是使用密钥:

<xsl:transform xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">

<xsl:output indent="yes"/>

<xsl:key name="group" match="body/*[not(starts-with(local-name(), 'd'))]" use="generate-id(preceding-sibling::*[starts-with(local-name(), 'd')][1])"/>

<xsl:template match="body">
    <xsl:copy>
        <xsl:apply-templates select="*[starts-with(local-name(), 'd')]"/>
    </xsl:copy>
</xsl:template>

<xsl:template match="*[starts-with(local-name(), 'd')]">
    <div>
        <xsl:copy-of select="key('group', generate-id())"/>
    </div>
</xsl:template>
</xsl:transform>

标签真的按照你的方式编号吗?如果是,您真的想在输出中保留
p
标记的编号吗?不,这只是一个示例。它是不同标签的混合。那么如何准确地识别“分隔符”标签呢?分隔符标签是这样的,我说的是p标签。这并不能真正回答我的问题。这里的问题是
*[以(name(),'d')]开头]
(Martin Honnen和我都使用过)是否只选择分隔符标记。-还不清楚您希望如何处理其他节点;这两个答案(现在)都假设您希望按原样复制它们。标签是否真的按照您显示的方式编号?如果是,您真的想在输出中保留
p
标记的编号吗?不,这只是一个示例。它是不同标签的混合。那么如何准确地识别“分隔符”标签呢?分隔符标签是这样的,我说的是p标签。这并不能真正回答我的问题。这里的问题是
*[以(name(),'d')]开头]
(Martin Honnen和我都使用过)是否只选择分隔符标记。-还不清楚您希望如何处理其他节点;这两个答案(现在)都假设您希望按原样复制它们。