xsl-list
[Top] [All Lists]

Re: Finding sequences of same element

2005-02-09 02:32:13
On Wed, 9 Feb 2005 18:58:35 +1100, Simon Kissane 
<skissane(_at_)gmail(_dot_)com> wrote:
Hi

Suppose I have an input document:
<A><B X="1"/><B X="2"/><B X="3"/><C X="4"/><B X="5"/><B X="6"/><B X="7"/></A>


Have you been using an obfuscation tool? Ever heard of formatting and
indentation?

Not that I'm trying to be picky or I'm in a bad mood, but producing
examples in the above "format" significantly decreases the number of
those, who could volunteer to decipher it (or as someone put it on his
blog, do we have to be masochists?).




Now, suppose I wish to group together consecutive B elements, giving a
result document like this:
<A><D><B X="1"/><B X="2"/><B X="3"/></D><C X="4"/><D><B X="5"/><B
X="6"/><B X="7"/></D></A>

How can I do this? (I would prefer not to use recursive templates, but
rather for-each, if that is at all possible...)

I think I can find the initial element of these sequences, like so:
 B[not(preceding-sibling::*) or preceding-sibling::*[0][not(self::B)]]
This, I think, should select all B for which there are either no
preceeding sibling elements, or for which the immediately preceeding
sibling element is not a B element. Thus, in the above example, it
would pick B[(_at_)X=1] and B[(_at_)X=5](_dot_)

But, given each initial sequence element, how can I find the remaining
nodes in the sequence?  With the initial sequence element as the
context node, I could do:
   .|following-sibling::B
But that will also pick up B[(_at_)X=5] and B[(_at_)X=6] when the context 
node is B[(_at_)X=1](_dot_)

Is there a predicate test I could use on following-sibling::B to
restrict it only to the current sequence of B elements?


This transformation:

<xsl:stylesheet version="1.0" 
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform";>

 <xsl:output omit-xml-declaration="yes" indent="yes"/>
 <xsl:strip-space elements="*"/>
 
 <xsl:key name="kNextGroup" match="B" 
   use="generate-id(following-sibling::*[not(self::B)][1])"/>
 
  <xsl:template match="node()|@*" name="identity">
    <xsl:copy>
      <xsl:apply-templates select="node()|@*"/>
    </xsl:copy>
  </xsl:template>
  
  <xsl:template match="B[not(preceding-sibling::*[1][self::B])]">
    <D>
      <xsl:copy-of select=
       "key('kNextGroup', 
               generate-id(following-sibling::*[not(self::B)][1])
               )" />
    </D>
  </xsl:template>
  
  <xsl:template match="B"/>
</xsl:stylesheet>

when applied on your source xml document and its much more readable version:

<A>
  <B X="1"/>
  <B X="2"/>
  <B X="3"/>
  <C X="4"/>
  <B X="5"/>
  <B X="6"/>
  <B X="7"/>
</A>

produces the readable version of the wanted result:

<A>
   <D>
      <B X="1"/>
      <B X="2"/>
      <B X="3"/>
   </D>
   <C X="4"/>
   <D>
      <B X="5"/>
      <B X="6"/>
      <B X="7"/>
   </D>
</A>

Hope this helped.

Cheers,
Dimitre Novatchev.

--~------------------------------------------------------------------
XSL-List info and archive:  http://www.mulberrytech.com/xsl/xsl-list
To unsubscribe, go to: http://lists.mulberrytech.com/xsl-list/
or e-mail: <mailto:xsl-list-unsubscribe(_at_)lists(_dot_)mulberrytech(_dot_)com>
--~--



<Prev in Thread] Current Thread [Next in Thread>