我正在使用XSLT将包含复杂数据结构,自由文本等的xml文档转换为HTML文档。 我处理的文档可以带有或不带结构项,如果存在,结构标签可以任意嵌套。数据标签可以参考 任何类型的商品,所以我事先并不知道XML doc的内容。
文档看起来像
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<?xml-stylesheet type="text/xsl" href="my.xsl"?>
<doc>
<elementStructure type="tag">
The following items are cars:
<Car id="12" type="data" value="Ferrari"> this is a sport car.
<structureLev2 type="tag">
<Model id="432" value="458" />
<Rim id="55" value="wheels of car" type="data">
<Tire id="234" value="front" type="data">
<Note id="33" value="special tire" type="data"/> size of front is less that rear.
<TypeTire id="44" value="radial tire" type="data"/>
</Tire>
</Rim>
</structureLev2>
</Car>
</elementStructure>
<elementStructure type="tag">
Other text
<Car id="22" type="data" value="Ford">
this is a family car.
<structureLev2 type="tag">
<Model id="872" value="Mondeo" />
<Rim id="45" value="wheels of car" type="data">
<Tire id="734" value="front" type="data">
<Note id="63" value="normal tire" type="data"/> spare tire could be replaced by run-flat.
<TypeTire id="84" value="tubeless tire" type="data"/>
</Tire>
</Rim>
</structureLev2>
</Car>
</elementStructure>
</doc>
desiderata HTML文档应如下所示:
<section>
Some text....
<ul>
<li>Car - Ferrari (this is a sport car.)</li>
<li>Rim - wheels of the car:</li>
<ul>
<li>Tire - front Note: special tire (size of front is less than rear).</li>
<li>TypeTire - radial tire
</ul>
</ul>
</section>
实际上我不知道它是不是一个好的解决方案,但由于我不知道我的xml是否包含结构标签,我使用“开关”在两个主模板之间进行选择。
<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output method="html"/>
<xsl:strip-space elements="*"/>
<xsl:template match="doc">
<html>
<body>
<xsl:choose>
<xsl:when test="//*[@type='tag']">
<xsl:call-template name="stuct" />
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="plain" />
</xsl:otherwise>
</xsl:choose>
</body>
</html>
</xsl:template>
<xsl:template match="doc/*[@type='tag']" name="stuct">
<h3>
<xsl:value-of select="text()"/>
</h3>
<xsl:apply-templates mode="str"/>
</xsl:template>
<xsl:template match="doc/*[@type='tag']" mode="str">
<ul>
<li>
<xsl:call-template name="dataElem"/>
<xsl:apply-templates mode="str"/>
</li>
</ul>
</xsl:template>
<xsl:template match="doc/*[@type='data']" name="dataElem">
<xsl:for-each select="descendant::node()[not(@type='tag')]">
<xsl:value-of select="name(.)"/>: <xsl:value-of select="@value" /> (<xsl:value-of select="text()"/>)<br/>
</xsl:for-each>
</xsl:template>
<xsl:template match="doc/*[@type='data']" name="plain">
<xsl:call-template name="dataElem" />
</xsl:template>
</xsl:stylesheet>
在我遇到的所有问题中,从代码本身开始:-)出现的一个问题是关于文本。我希望标签之间的文本可能只出现一次,并且在“已解析”元素之后出现。有可能吗?
答案 0 :(得分:0)
很难从单个不完整的例子中辨别出所需转换的逻辑。此外,我怀疑所提供的输出实际上是不正确的 - 因为没有明显的理由推广&#34; Rim&#34;成为&#34; Car&#39;的兄弟姐妹在原始文件中,它是一个孩子。
也许这样的事情对你有用:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<html>
<body>
<xsl:apply-templates select="*"/>
</body>
</html>
</xsl:template>
<xsl:template match="elementStructure">
<section>
<xsl:value-of select="normalize-space(text())"/>
<ul>
<xsl:apply-templates select="*"/>
</ul>
</section>
</xsl:template>
<xsl:template match="*[@type='data']">
<li>
<xsl:value-of select="concat(name(), ' - ', @value)"/>
<xsl:if test="text()">
<xsl:value-of select="concat(' (', normalize-space(text()), ')')"/>
</xsl:if>
<xsl:if test="descendant::*[@type='data']">
<ul>
<xsl:apply-templates select="*"/>
</ul>
</xsl:if>
</li>
</xsl:template>
</xsl:stylesheet>
应用于示例输入时,结果将为:
<html>
<body>
<section>The following items are cars:
<ul>
<li>Car - Ferrari (this is a sport car.)
<ul>
<li>Rim - wheels of car
<ul>
<li>Tire - front (size of front is less that rear.)
<ul>
<li>Note - special tire</li>
<li>TypeTire - radial tire</li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
</ul>
</section>
<section>Other text
<ul>
<li>Car - Ford (this is a family car.)
<ul>
<li>Rim - wheels of car
<ul>
<li>Tire - front (spare tire could be replaced by run-flat.)
<ul>
<li>Note - normal tire</li>
<li>TypeTire - tubeless tire</li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
</ul>
</section>
</body>
</html>
呈现为: