Como obter as informações e tipo, um subconjunto de elementos com apenas identificadores usando XSLT?
Pergunta
Eu sou novo para XSLT e eu não posso resolver o seguinte problema:
Eu tenho um arquivo xml como este:
<root>
<subset>
<e id="A"></e>
<e id="C"></e>
</subset>
<data>
<info id="A" order="3" name="ANode"></info>
<info id="B" order="4" name="BNode"></info>
<info id="C" order="1" name="CNode"></info>
<info id="D" order="2" name="DNode"></info>
</data>
</root>
E eu quero produzir este:
<root>
<newnode id="C" order="1" name="CNode"></newnode>
<newnode id="A" order="3" name="ANode"></newnode>
</root>
Como você pode ver, a idéia é "completa" o subconjunto de ids, recuperando a respectiva informação, e classificá-los de acordo com sua ordem especificada na <data>
.
Observe que <data>
é suposto ter um monte de nós filhos.
Além disso, estou separando os nós com as informações sobre o elemento <data>
dos subconjuntos de ids, porque vou ter muitos subconjuntos e eu não quero ter informações repetidas "em todos os lugares".
Agradecemos antecipadamente.
Solução
Você poderia off também fundamental do subconjunto dependendo o que mais você pode precisar de fazer com o seu real dos dados.
<xsl:output indent="yes" />
<xsl:key name="subset" match="e" use="@id" />
<xsl:template match="/">
<root>
<xsl:for-each select="//info[count(key('subset',@id)[1]) > 0]">
<xsl:sort select="@order" data-type="number" />
<newnode id="{@id}" order="{@order}" name="{@name}" />
</xsl:for-each>
</root>
</xsl:template>
Outras dicas
As referências cruzadas pedir usando teclas:
<xsl:stylesheet
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
version="1.0">
<xsl:output indent="yes"/>
<xsl:key name="k1" match="info" use="@id"/>
<xsl:template match="root">
<xsl:copy>
<xsl:apply-templates select="subset/e">
<xsl:sort select="key('k1', @id)/@order" data-type="number"/>
</xsl:apply-templates>
</xsl:copy>
</xsl:template>
<xsl:template match="e">
<newnode id="{@id}" order="{key('k1', @id)/@order}" name="{key('k1', @id)/@name}"/>
</xsl:template>
</xsl:stylesheet>
Isso deve fazer para a amostra de entrada que você apresentou.