2013-04-02 70 views
0

我想添加一个基于日期比较新的元素。如果父DATE在最近7天内,我想添加一个新元素。我编写了代码来进行日期比较,但我很难弄清楚在哪里放置它。目前,它在模板中重新格式化父DATE,但这会导致我的新元素在日期元素内。有没有办法在当前元素之外创建一个新元素?谢谢。XSLT添加新节点之外当前节点的

我输入

<?xml version="1.0" encoding="UTF-8"?> 
<NOTICES> 
    <PRESOL> 
     <DATE>03012013</DATE> 
     <AGENCY><![CDATA[Department of the Interior]]></AGENCY> 
     <OFFICE><![CDATA[Fish and Wildlife Service]]></OFFICE> 
     <LOCATION><![CDATA[CGS-WO]]></LOCATION> 
     <ZIP>97232</ZIP> 
     <CHANGES> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>US</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </PRESOL> 
    <COMBINE> 
     <DATE>03012013</DATE> 
     <AGENCY><![CDATA[Department of the Air Force]]></AGENCY> 
     <OFFICE><![CDATA[Air Force Materiel Command]]></OFFICE> 
     <LOCATION><![CDATA[Tinker OC-ALC - (Central Contracting)]]></LOCATION> 
     <ZIP>73145-3015</ZIP> 
    </COMBINE> 
    <COMBINE> 
     <DATE>03052013</DATE> 
     <AGENCY><![CDATA[Department of the Navy]]></AGENCY> 
     <OFFICE><![CDATA[Military Sealift Command]]></OFFICE> 
     <LOCATION><![CDATA[MSC Norfolk]]></LOCATION> 
    </COMBINE> 
    <COMBINE> 
     <DATE>03292013</DATE> 
     <AGENCY><![CDATA[Department of Veterans Affairs]]></AGENCY> 
     <OFFICE><![CDATA[Grand Junction VAMC)]]></OFFICE> 
     <LOCATION><![CDATA[Veterans Affairs Medical Center]]></LOCATION> 
    </COMBINE> 
    <PRESOL> 
     <DATE>03302013</DATE> 
     <AGENCY><![CDATA[Department of the Air Force]]></AGENCY> 
     <OFFICE><![CDATA[Pacific Air Forces]]></OFFICE> 
     <LOCATION><![CDATA[354 CONS - Eielson]]></LOCATION> 
     <CHANGES> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>US</COUNTRY> 
      </MOD> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>UK</COUNTRY> 
      </MOD> 
      <MOD> 
       <DATE>01142013</DATE> 
       <COUNTRY>JAPAN</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </PRESOL> 
    <FAIROPP> 
     <DATE>04012013</DATE> 
     <AGENCY><![CDATA[Department of the Navy]]></AGENCY> 
     <OFFICE><![CDATA[Bureau of Medicine and Surgery]]></OFFICE> 
     <LOCATION><![CDATA[NH Camp Pendleton]]></LOCATION> 
     <ZIP>92055</ZIP> 
     <CHANGES> 
      <MOD> 
        <DATE>02122011</DATE> 
        <COUNTRY>JAPAN</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </FAIROPP> 
</NOTICES> 

所需的输出:

<?xml version="1.0" encoding="UTF-8"?> 
<NOTICES> 
    <PRESOL> 
     <DATE>03012013</DATE> 
     <AGENCY><![CDATA[Department of the Interior]]></AGENCY> 
     <OFFICE><![CDATA[Fish and Wildlife Service]]></OFFICE> 
     <LOCATION><![CDATA[CGS-WO]]></LOCATION> 
     <ZIP>97232</ZIP> 
     <CHANGES> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>US</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </PRESOL> 
    <COMBINE> 
     <DATE>03012013</DATE> 
     <AGENCY><![CDATA[Department of the Air Force]]></AGENCY> 
     <OFFICE><![CDATA[Air Force Materiel Command]]></OFFICE> 
     <LOCATION><![CDATA[Tinker OC-ALC - (Central Contracting)]]></LOCATION> 
     <ZIP>73145-3015</ZIP> 
    </COMBINE> 
    <COMBINE> 
     <DATE>03052013</DATE> 
     <AGENCY><![CDATA[Department of the Navy]]></AGENCY> 
     <OFFICE><![CDATA[Military Sealift Command]]></OFFICE> 
     <LOCATION><![CDATA[MSC Norfolk]]></LOCATION> 
    </COMBINE> 
    <COMBINE> 
     <DATE>03292013</DATE> 
     **<mostrecent>YES</mostrecent>** 
     <AGENCY><![CDATA[Department of Veterans Affairs]]></AGENCY> 
     <OFFICE><![CDATA[Grand Junction VAMC)]]></OFFICE> 
     <LOCATION><![CDATA[Veterans Affairs Medical Center]]></LOCATION> 
    </COMBINE> 
    <PRESOL> 
     <DATE>03302013</DATE> 
     **<mostrecent>YES</mostrecent>** 
     <AGENCY><![CDATA[Department of the Air Force]]></AGENCY> 
     <OFFICE><![CDATA[Pacific Air Forces]]></OFFICE> 
     <LOCATION><![CDATA[354 CONS - Eielson]]></LOCATION> 
     <CHANGES> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>US</COUNTRY> 
      </MOD> 
      <MOD> 
       <DATE>01112013</DATE> 
       <COUNTRY>UK</COUNTRY> 
      </MOD> 
      <MOD> 
       <DATE>01142013</DATE> 
       <COUNTRY>JAPAN</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </PRESOL> 
    <FAIROPP> 
     <DATE>04012013</DATE> 
     **<mostrecent>YES</mostrecent>** 
     <AGENCY><![CDATA[Department of the Navy]]></AGENCY> 
     <OFFICE><![CDATA[Bureau of Medicine and Surgery]]></OFFICE> 
     <LOCATION><![CDATA[NH Camp Pendleton]]></LOCATION> 
     <ZIP>92055</ZIP> 
     <CHANGES> 
      <MOD> 
        <DATE>02122011</DATE> 
        <COUNTRY>JAPAN</COUNTRY> 
      </MOD> 
     </CHANGES> 
    </FAIROPP> 
</NOTICES> 

我的XSLT:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" 
xmlns:xs="http://www.w3.org/2001/XMLSchema" 
     exclude-result-prefixes="xs"> 
    <xsl:output omit-xml-declaration="yes" indent="yes" cdata-section-elements="AGENCY DESC CLASSCOD CONTACT DATE NAICS LINK OFFADD OFFICE SUBJECT ZIP AGENCY ZIP"/> 
    <xsl:strip-space elements="*"/> 
    <xsl:variable name="backdate1" select="current-date() -7*xs:dayTimeDuration('P1D')"/> 
    <xsl:variable name="backdate" select="xs:date(substring($backdate1, 1, 10))"/> 
    <!-- copy all nodes --> 
    <xsl:template match="node()|@*"> 
     <xsl:copy> 
      <xsl:apply-templates select="node()|@*"/> 
     </xsl:copy> 
    </xsl:template> 

    <xsl:template match="DATE/text()"> 
<!-- format DATE mm/dd/yyyy --> 
     <xsl:value-of select="concat(substring(., 1, 2), '/', substring(., 3, 2), '/', substring(., 5, 4))"/> 
    **<!-- add new node mostrecent if date is within the last 7 days --> 
     <xsl:variable name="subtract_date" select="days-from-duration(xs:date(concat(substring(., 5, 4), '-', substring(., 1, 2), '-', substring(., 3, 2))) - xs:date(substring($backdate1, 1, 10)))"/> 
     <xsl:if test="$subtract_date &gt;= 0"> 
     <xsl:text disable-output-escaping="yes">&lt;</xsl:text> 
     <xsl:text disable-output-escaping="yes">mostrecent</xsl:text> 
     <xsl:text disable-output-escaping="yes">&gt;</xsl:text> 
     <xsl:text disable-output-escaping="yes">&lt;/</xsl:text> 
     <xsl:text disable-output-escaping="yes">mostrecent</xsl:text> 
     <xsl:text disable-output-escaping="yes">&gt;</xsl:text> 
     </xsl:if>** 
    </xsl:template> 

    <!-- keep only the parent date node and delete all children date nodes --> 
    <xsl:template match="DATE[../ancestor::*/DATE]"/> 

    <!-- add new node type to each child node --> 
    <xsl:template match="NOTICES/child::node()"> 
    <xsl:text disable-output-escaping="yes">&lt;</xsl:text><xsl:value-of select="name(.)"/><xsl:text disable-output-escaping="yes">&gt;</xsl:text> 
    <xsl:apply-templates select="@*|node()"/> 
     <type> 
      <xsl:value-of select ="name(.)"/> 
     </type> 
    <xsl:text disable-output-escaping="yes">&lt;/</xsl:text><xsl:value-of select="name(.)"/><xsl:text disable-output-escaping="yes">&gt;</xsl:text> 
    </xsl:template> 
</xsl:stylesheet> 

回答

0

你可以做到这一点,如果你的日期元素,而不是它的文本节点上一致:

<xsl:template match="DATE"> 
     <xsl:variable name="dStr" select="string(.)" /> 
     <xsl:variable name="bdStr" select="string($backdate1)" /> 
     <!-- format DATE mm/dd/yyyy --> 
     <xsl:copy> 
      <xsl:value-of select="concat(substring($dStr, 1, 2), '/', 
             substring($dStr, 3, 2), '/', 
             substring($dStr, 5, 4))"/> 
     </xsl:copy> 
     <!-- add new node mostrecent if date is within the last 7 days --> 
     <xsl:variable name="subtract_date" 
         select="days-from-duration(xs:date(
              concat(substring($dStr, 5, 4), '-', 
               substring($dStr, 1, 2), '-', 
               substring($dStr, 3, 2))) - 
           xs:date(substring($bdStr, 1, 10)))"/> 
     <xsl:if test="$subtract_date &gt;= 0"> 
      <mostrecent>YES</mostrecent> 
     </xsl:if> 
    </xsl:template> 
+0

感谢JL,伟大的工作。问题:使用导致我的原始代码无法工作的文本节点的是什么? –

+0

它并没有因为你的方法工作,身份模板是做'DATE'元素复制到输出,然后应用_inside_该元素模板'DATE'的儿童(包括文字)的工作。因此,在'text()'模板中可以使用的任何内容都将被输出_inside_ DATE'元素。一旦你在那里,没有办法让它回到外面。 – JLRishe

+0

我试图将上述xslt从1.0转换为2.0。当我这样做时,我收到一个错误:substring()的第一个参数的必需项目类型是xs:string;提供的值具有项目类型xs:date。我不明白的是,为什么这个转换工作在1.0而不是2.0以及如何解决它。我在想,2.0需要更多的变量声明控制。此外,错误似乎来自我的if语句,因为当我注释掉它时,代码编译得很好。 –

相关问题