我有一个元素列表:
<vehiciles>
<vehicile value="_CAR">CAR</vehicile>
<vehicile value="01">vehicile1</vehicile>
<vehicile value="02">vehicile2</vehicile>
<vehicile value="03">vehicile3</vehicile>
<vehicile value="_TRUCK">TRUCK</vehicile>
<vehicile value="04">vehicile4</vehicile>
<vehicile value="05">vehicile5</vehicile>
<vehicile value="06">vehicile6</vehicile>
</vehiciles>
不幸的是我不能改变结构,但是我必须按照车辆指示的类别对它进行分组(在html select / optgroup标签中),该值以下划线开头。
我希望实现的结果:
<select>
<optgroup label="CAR">
<option value="01">vehicile1</option>
<option value="02">vehicile2</option>
<option value="03">vehicile3</option>
</optgroup>
<opgroup label="TRUCK">
<option value="04">vehicile4</option>
<option value="05">vehicile5</option>
<option value="06">vehicile6</option>
</optgroup>
</select>
我试过的是:
<xsl:template match="field" mode="dropdown_list">
<select>
<xsl:choose>
<xsl:when test="vehiciles/vehicile[starts-with(@value, '_')]">
<xsl:for-each select="vehiciles/vehicile[starts-with(@value, '_')]">
<xsl:variable name="lastValue" select="following-sibling::*[starts-with(@value, '_')][@value]" />
<optgroup>
<xsl:attribute name="label">
<xsl:value-of select="text()"/>
</xsl:attribute>
<xsl:for-each select="following-sibling::*[not(preceding::vehicile[1][@value = $lastValue])]">
<option value="{@value}">
<xsl:value-of select="text()"/>
</option>
</xsl:for-each>
</optgroup>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<!-- something here -->
</xsl:otherwise>
</xsl:choose>
</select>
</xsl:template>
它输出第二个循环不错,但首先包含所有元素。试着弄清楚几个小时没有运气。
试图通过递归来做但是失败以及Muenchian分组。
有没有办法从某个节点查找符合条件的第一个兄弟节点?或者另一种方式?
任何帮助表示感谢。
答案 0 :(得分:1)
使用
<xsl:template match="vehiciles">
<select>
<xsl:apply-templates select="vehicile[starts-with(@value, '_')]"/>
</select>
</xsl:template>
<xsl:template match="vehicile[starts-with(@value, '_')]">
<optgroup label="{substring(@value, 2)}">
<xsl:apply-templates select="key('k1', generate-id())"/>
</optgroup>
</xsl:template>
<xsl:key name="k1" match="vehicile[not(starts-with(@value, '_'))]"
use="generate-id(preceding-sibling::vehicile[starts-with(@value, '_')][1])"/>
<xsl:template match="vehicile[not(starts-with(@value, '_'))]">
<xsl:variable name="num">
<xsl:number count="vehicile[not(starts-with(@value, '_'))]"/>
</xsl:variable>
<option value="{format-number($num, '00')}"><xsl:value-of select="."/></option>
</xsl:template>
请参阅http://xsltransform.net/jyH9rMi以获取完整示例。
答案 1 :(得分:1)
以这种方式尝试:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:key name="vehicile" match="vehicile[not(starts-with(@value, '_'))]" use="generate-id(preceding-sibling::vehicile[starts-with(@value, '_')][1])" />
<xsl:template match="/vehiciles">
<select>
<xsl:for-each select="vehicile[starts-with(@value, '_')]">
<optgroup label="{substring-after(@value, '_')}">
<xsl:for-each select="key('vehicile', generate-id())">
<option value="{@value}">
<xsl:value-of select="."/>
</option>
</xsl:for-each>
</optgroup>
</xsl:for-each>
</select>
</xsl:template>
</xsl:stylesheet>