XSLT 通过来自不同节点内的命名模板调用更改节点
XSLT changing a node via a named template call from within a different node
编辑:错误是在错误的 when
条件下。
来源xml:
<?xml version="1.0" encoding="UTF-8"?>
<mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
预期转换xml:
<?xml version="1.0" encoding="UTF-8"?>
<mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
即,当一个levelTwoNode节点嵌套到一个value节点中,属性为name遇到“triggeringValue”,文档中所有levelTwoNode节点具有相同的attributeOne和attributeTwo 作为触发节点应该注入属性 injectedAttribute="triggeringValue".
我正在使用这个 xslt:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >
<xsl:output encoding="Windows-1252" method="xml" indent="no" omit-xml-declaration="no" />
<!-- Copy the whole output -->
<xsl:template match="@* |node() ">
<xsl:copy>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:template>
<xsl:template match="//levelOneNodes/levelOneNode/value/levelTwoNodes/levelTwoNode" >
<xsl:call-template name="myTemplate">
<xsl:with-param name="myAttrOne" select=" current()/@attributeOne " />
<xsl:with-param name="myAttrTwo" select=" current()/@attributeTwo " />
<xsl:with-param name="myValue" select=" current()/../../@name " />
</xsl:call-template>
</xsl:template>
<xsl:template name="myTemplate">
<xsl:param name="myAttrOne" />
<xsl:param name="myAttrTwo" />
<xsl:param name="myValue" />
<xsl:choose>
<xsl:when test="boolean(//levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]) and boolean($myValue = 'triggeringValue')">
<xsl:copy>
<xsl:attribute name="injectedAttribute"><xsl:value-of select="$myValue"/></xsl:attribute>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
结果是 xml:
<?xml version="1.0" encoding="Windows-1252"?><mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
其中新属性injectedAttribute没有注入到第二个levelOneNode[=66=的第一个levelTwoNode ](具有子节点 value 且属性为 name “anotherValue” 的那个),即使它具有attributeOne 和 attributeTwo(值为“A”)之一 [=40=第一个 levelOneNode 节点内的 ]levelTwoNode 节点(子节点 value 属性 name“触发值”)。属性 injectedAttribute 仅在 levelTwoNode 节点中注入 triggeringValue value 节点。
为什么会出现这种情况:
<xsl:when test="boolean(//levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]) and boolean($myValue = 'triggeringValue')">
调用节点范围受限?查询 //levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]
应获取所有具有传递属性的 levelTwoNode 节点,条件 $myValue = 'triggeringValue'
不应限制该查询。
我错过了什么?
感谢您的帮助。
这很令人困惑。我根本无法理解你的代码。如果我正确理解你的描述(这是一个非常大的 if),那么你想做类似的事情:
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:strip-space elements="*"/>
<xsl:key name="trigger" match="value[@name='triggeringValue']/levelTwoNodes/levelTwoNode" use="concat(@attributeOne, '|', @attributeTwo)" />
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="levelTwoNode[key('trigger', concat(@attributeOne, '|', @attributeTwo))]">
<xsl:copy>
<xsl:attribute name="injectedAttribute">triggeringValue</xsl:attribute>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
它的工作方式是索引所有 levelTwoNode
元素,这些元素是 <value name="triggeringValue">
的后代。然后,通过添加 injectedAttribute
属性修改任何具有与其中一个索引元素(包括索引元素本身)相同的两个属性值的 levelTwoNode
元素。
你说“条件 $myValue = 'triggeringValue' 不应限制该查询”但它应该。变量 $myValue 绑定到 current()/../../@name
,我想你会发现它的值是“anotherValue”。
编辑:错误是在错误的 when
条件下。
来源xml:
<?xml version="1.0" encoding="UTF-8"?>
<mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
预期转换xml:
<?xml version="1.0" encoding="UTF-8"?>
<mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
即,当一个levelTwoNode节点嵌套到一个value节点中,属性为name遇到“triggeringValue”,文档中所有levelTwoNode节点具有相同的attributeOne和attributeTwo 作为触发节点应该注入属性 injectedAttribute="triggeringValue".
我正在使用这个 xslt:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >
<xsl:output encoding="Windows-1252" method="xml" indent="no" omit-xml-declaration="no" />
<!-- Copy the whole output -->
<xsl:template match="@* |node() ">
<xsl:copy>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:template>
<xsl:template match="//levelOneNodes/levelOneNode/value/levelTwoNodes/levelTwoNode" >
<xsl:call-template name="myTemplate">
<xsl:with-param name="myAttrOne" select=" current()/@attributeOne " />
<xsl:with-param name="myAttrTwo" select=" current()/@attributeTwo " />
<xsl:with-param name="myValue" select=" current()/../../@name " />
</xsl:call-template>
</xsl:template>
<xsl:template name="myTemplate">
<xsl:param name="myAttrOne" />
<xsl:param name="myAttrTwo" />
<xsl:param name="myValue" />
<xsl:choose>
<xsl:when test="boolean(//levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]) and boolean($myValue = 'triggeringValue')">
<xsl:copy>
<xsl:attribute name="injectedAttribute"><xsl:value-of select="$myValue"/></xsl:attribute>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:apply-templates select="@* |node() "/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
结果是 xml:
<?xml version="1.0" encoding="Windows-1252"?><mainNode>
<levelOneNodes>
<levelOneNode>
<value name="triggeringValue">
<levelTwoNodes>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="B1" attributeTwo="B1" attributeThree="B1">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode injectedAttribute="triggeringValue" attributeOne="C1" attributeTwo="C1" attributeThree="C1">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
<levelOneNode>
<value name="anotherValue">
<levelTwoNodes>
<levelTwoNode attributeOne="A" attributeTwo="A" attributeThree="A">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="B2" attributeTwo="B2" attributeThree="B2">
<textData>This is some text.</textData>
</levelTwoNode>
<levelTwoNode attributeOne="C2" attributeTwo="C2" attributeThree="C2">
<textData>This is some text.</textData>
</levelTwoNode>
</levelTwoNodes>
</value>
</levelOneNode>
</levelOneNodes>
</mainNode>
其中新属性injectedAttribute没有注入到第二个levelOneNode[=66=的第一个levelTwoNode ](具有子节点 value 且属性为 name “anotherValue” 的那个),即使它具有attributeOne 和 attributeTwo(值为“A”)之一 [=40=第一个 levelOneNode 节点内的 ]levelTwoNode 节点(子节点 value 属性 name“触发值”)。属性 injectedAttribute 仅在 levelTwoNode 节点中注入 triggeringValue value 节点。
为什么会出现这种情况:
<xsl:when test="boolean(//levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]) and boolean($myValue = 'triggeringValue')">
调用节点范围受限?查询 //levelTwoNode[(@attributeOne = $myAttrOne) and (@attributeTwo = $myAttrTwo)]
应获取所有具有传递属性的 levelTwoNode 节点,条件 $myValue = 'triggeringValue'
不应限制该查询。
我错过了什么?
感谢您的帮助。
这很令人困惑。我根本无法理解你的代码。如果我正确理解你的描述(这是一个非常大的 if),那么你想做类似的事情:
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:strip-space elements="*"/>
<xsl:key name="trigger" match="value[@name='triggeringValue']/levelTwoNodes/levelTwoNode" use="concat(@attributeOne, '|', @attributeTwo)" />
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="levelTwoNode[key('trigger', concat(@attributeOne, '|', @attributeTwo))]">
<xsl:copy>
<xsl:attribute name="injectedAttribute">triggeringValue</xsl:attribute>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
它的工作方式是索引所有 levelTwoNode
元素,这些元素是 <value name="triggeringValue">
的后代。然后,通过添加 injectedAttribute
属性修改任何具有与其中一个索引元素(包括索引元素本身)相同的两个属性值的 levelTwoNode
元素。
你说“条件 $myValue = 'triggeringValue' 不应限制该查询”但它应该。变量 $myValue 绑定到 current()/../../@name
,我想你会发现它的值是“anotherValue”。