在XPath 2.0中创建JavaScript文字对象

时间:2012-05-05 11:14:28

标签: javascript json xslt xslt-2.0 xpath-2.0

后台:我需要在Web浏览器中使用XSLT 2.0来执行另一个XSLT 2.0转换 - 这将用于XSLT处理器的测试驱动程序。处理器的JavaScript API允许您构建一个文本命令对象,该对象作为参数传递给 run 函数以执行转换。

从JavaScript构建命令对象非常简单,但是从XSLT开始,我需要使用内置的JavaScript扩展和用户定义的JavaScript函数将XSLT数据转换为JavaScript对象。有一个问题,XPath 2.0适用于项目序列,但不允许序列序列。我正在使用的方法显示在以下XSLT代码段中,该代码段声明了一个变量 cmd

XSLT code to create a JavaScript object

等效的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 是内置于处理器中的扩展函数会不会更好?

2 个答案:

答案 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(), ':&#9;'"/>
         <xsl:apply-templates/>

         <xsl:sequence select="','[$vNotLast], '&#xA;'"/>
 </xsl:template>

 <xsl:template match="*[not(self::js:*)]/text()">
  <xsl:sequence select='concat("&apos;", ., "&apos;")'/>
 </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)"/>