首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >使用XSLT动态添加HTML标记

使用XSLT动态添加HTML标记
EN

Stack Overflow用户
提问于 2015-07-18 20:15:41
回答 2查看 872关注 0票数 0

我有一个XML文档,其中包含以下内容:

代码语言:javascript
复制
<d1/>
<p1>...</p1>
<p2>...</p2>
<d2/>
<p3>...</p3>
<d3/>

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

代码语言:javascript
复制
<div>
<p1>...</p1>
<p2>...</p2>
</div>
<div>
<p3>...</p3>
</div>

我已经使用以下代码编写了一个XSLT来动态引入<div></div>标记:

代码语言:javascript
复制
<xsl:text disable-output-escaping="yes">&lt;div&gt;</xsl:text>

代码语言:javascript
复制
<xsl:text disable-output-escaping="yes">&lt;/div&gt;</xsl:text>

这在Safari上有效,但在FireFox上失败了,这让我怀疑这不是正确的方法。你有没有更好的办法在每个浏览器上都能工作?

在此之前非常感谢。

EN

回答 2

Stack Overflow用户

发布于 2015-07-18 21:59:51

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

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

XML

代码语言:javascript
复制
<root>
    <d1/>
    <p1>a</p1>
    <p2>b</p2>
    <d2/>
    <p3>c</p3>
    <d3/>
</root>

以下样式表:

XSLT 1.0

代码语言:javascript
复制
<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>

将返回:

代码语言:javascript
复制
<body>
   <div>
      <p>a</p>
      <p>b</p>
   </div>
   <div>
      <p>c</p>
   </div>
</body>
票数 2
EN

Stack Overflow用户

发布于 2015-07-18 22:09:31

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

代码语言:javascript
复制
<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>

这将在示例的末尾创建一个空的div,因此您可能希望将最后一个模板更改为

代码语言:javascript
复制
<xsl:template match="*[starts-with(local-name(), 'd')]">
    <xsl:if test="key('group', generate-id())">
      <div>
        <xsl:copy-of select="key('group', generate-id())"/>
      </div>
    </xsl:if>
</xsl:template>
票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/31491052

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档