后台:我需要在Web浏览器中使用XSLT 2.0来执行另一个XSLT 2.0转换 - 这将用于XSLT处理器的测试驱动程序。处理器的JavaScript API允许您构建一个文本命令对象,该对象作为参数传递给 run 函数以执行转换。
从JavaScript构建命令对象非常简单,但是从XSLT开始,我需要使用内置的JavaScript扩展和用户定义的JavaScript函数将XSLT数据转换为JavaScript对象。有一个问题,XPath 2.0适用于项目序列,但不允许序列序列。我正在使用的方法显示在以下XSLT代码段中,该代码段声明了一个变量 cmd :
等效的JavaScript如下所示,供参考:
var cmd= {
initialTemplate: initialTemplate,
stylesheet: stylesheet,
source: 'uk-maps.xml',
parameters: {
country: 'UK',
city: 'Cheltenham',
color: [28, 329, 767]
}
};
当进入JavaScript时,XSLT处理器将序列转换为JavaScript对象数组。用户定义的JavaScript函数 js:object 处理数组并为奇数项创建属性,并从相应的偶数项中指定属性值。可以递归调用 js:object 函数,以将JavaScript对象分配给新对象的属性。我的另一个解决方法是用户定义的js:array函数,它在JavaScript对象中包装XPath序列,以允许它作为序列项嵌入。 js:object 函数必须检测并解包这些js:array对象。
所以,问题是:这是从XSLT 2.0中构建JavaScript文字对象的一种方法,但它涉及一些可能不适合所有人的解决方法。还可以使用哪些其他方法?也许我应该使用现有的JavaScript库函数将XML转换为文字JavaScript对象? XSLT 3.0映射(此处理器中尚未提供)是否会提供更好的解决方案?在提出的JSON / XSLT兼容性功能方面取得了哪些进展?如果 js:object 和 js:array 是内置于处理器中的扩展函数会不会更好?
答案 0 :(得分:1)
以下是此类任务的一种方法:
给出以下XML文档:
<t xmlns:js="js:aux">
<initialTemplate>
<js:var>initialTemplate</js:var>
</initialTemplate>
<stylesheet>
<js:var>stylesheet</js:var>
</stylesheet>
<source>uk-maps.xml</source>
<parameters>
<js:object>
<country>UK</country>
<city>Cheltenham</city>
<colours>
<js:array>
<js:num>28</js:num>
<js:num>329</js:num>
<js:num>767</js:num>
</js:array>
</colours>
</js:object>
</parameters>
</t>
此转型:
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:js="js:aux">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/*">
{
<xsl:apply-templates/>
}
</xsl:template>
<xsl:template match="*/*[not(self::js:*)]">
<xsl:variable name="vNotLast" select="exists(following-sibling::*[1])"/>
<xsl:sequence select="name(), ':	'"/>
<xsl:apply-templates/>
<xsl:sequence select="','[$vNotLast], '
'"/>
</xsl:template>
<xsl:template match="*[not(self::js:*)]/text()">
<xsl:sequence select='concat("'", ., "'")'/>
</xsl:template>
<xsl:template match="js:object">
{
<xsl:apply-templates/>
}
</xsl:template>
<xsl:template match="js:array">
[
<xsl:apply-templates/>
]
</xsl:template>
<xsl:template match="js:array/*">
<xsl:next-match/>
<xsl:variable name="vNotLast" select="exists(following-sibling::*[1])"/>
<xsl:sequence select="','[$vNotLast]"/>
</xsl:template>
</xsl:stylesheet>
<强>产生强>:
{
initialTemplate : initialTemplate,
stylesheet : stylesheet,
source : 'uk-maps.xml' ,
parameters :
{
country : 'UK' ,
city : 'Cheltenham' ,
colours :
[
28,329,767
]
}
}
然后,这个结果只需要提供给Javascript eval()
函数。
答案 1 :(得分:0)
我目前的想法是最初使用问题中概述的XPath 2.0的原始解决方案。然后,当处理器支持XPath 3.0映射时,可以使用新的map constructor,因此等效代码将是:
<xsl:variable
name="cmd"
select="
map {
'initialTemplate' := $initialTemplate;
'stylesheet' := $stylesheet;
'source' := 'uk-maps.xml';
'parameters' := map {
'country' := 'UK';
'city' := 'Cheltenham';
'color' := (28, 329, 767);
};
};
"/>
我假设地图可以将序列作为值项处理,并且它还可以包含其他地图作为值项,因此对JavaScript函数的调用将是唯一的非标准部分:
<xsl:variable name="result" as="document-node()" select="js:run($cmd)"/>