xsl-list
[Top] [All Lists]

Re: [xsl] XSLT/XPath Question (Grouping Authors by First Character of Last Name)

2007-03-04 13:27:19
I looked at the example.  I couple of notes:

1) The text boxes holding the XML and XSLT code did not work in
FF2.0.0.2 - they were very small (20% or 15% of what they should have
been).  I viewed the page in IE7 and they appeared OK (the content of
the boxes was the same in both instances)

2) The stylesheet seems to be missing the </xsl:stylesheet> --- At
least I had to add it before it would run for me.

I looked at them, but don't yet really understand them.  I will have
to spend some time studying the example.

Thanks....

On 3/3/07, Dimitre Novatchev <dnovatchev(_at_)gmail(_dot_)com> wrote:
> Perfect.  Thanks.  Just out of curiosity, is this possible
> (practical?) in XSLT 1.0?

Yes and yes -- read about the Muenchian method for grouping.

Actually, this reminds me that six years ago I published a snippet in
topxml.com solving exactly your problem in XSLT 1.0:


   http://www.topxml.com/code/default.asp?p=3&id=v20010129150851



--
Cheers,
Dimitre Novatchev
---------------------------------------
Truly great madness cannot be achieved without significant intelligence.
---------------------------------------
To invent, you need a good imagination and a pile of junk
-------------------------------------
You've achieved success in your field when you don't know whether what
you're doing is work or play


On 3/3/07, Kevin Grover <kevin(_at_)kevingrover(_dot_)net> wrote:
> Perfect.  Thanks.  Just out of curiosity, is this possible
> (practical?) in XSLT 1.0?
>
> On 3/3/07, Dimitre Novatchev <dnovatchev(_at_)gmail(_dot_)com> wrote:
> > You are not using XSLT 2.0 to its full potential.
> >
> > The solution is as easy as this:
> >
> > <xsl:stylesheet version="2.0"
> >  xmlns:xsl="http://www.w3.org/1999/XSL/Transform";
> >  >
> >   <xsl:output method="text"/>
> >
> >  <xsl:template match="/">
> >    <xsl:for-each-group select="*/author"
> >      group-by="substring(name/@file-as,1,1)"
> >     >
> >      <xsl:text>&#xA;&#xA;</xsl:text>
> >      <xsl:value-of select="current-grouping-key()"/>
> >      <xsl:text>&#xA;</xsl:text>
> >
> >      <xsl:for-each select="current-group()">
> >        <xsl:value-of select="name/@file-as"/>
> >        <xsl:text>&#xA;</xsl:text>
> >      </xsl:for-each>
> >    </xsl:for-each-group>
> >  </xsl:template>
> > </xsl:stylesheet>
> >
> > and  when applied on the provided xml document it produces the wanted 
result:
> >
> >
> >
> > A
> > Adams, Douglas
> > Anderson, Kevin J.
> > Anthony, Piers
> > Archer, Jeffrey
> >
> >
> > B
> > Baldacci, David
> > Ball, Margaret
> > Bradley, Marion Zimmer
> >
> >
> > C
> > Carcaterra, Lorenzo
> > Card, Orson Scott
> > Chalker, Jack L.
> >
> >
> > D
> > Dahl, Roald
> > Daley, Brian
> > Dann, Jack
> >
> >
> >
> > --
> > Cheers,
> > Dimitre Novatchev
> > ---------------------------------------
> > Truly great madness cannot be achieved without significant intelligence.
> > ---------------------------------------
> > To invent, you need a good imagination and a pile of junk
> > -------------------------------------
> > You've achieved success in your field when you don't know whether what
> > you're doing is work or play
> >
> >
> >
> >
> > On 3/3/07, Kevin Grover <kevin(_at_)kevingrover(_dot_)net> wrote:
> > > I have an XSLT/XPath Question that I've been beating my head against
> > > for a while.
> > >
> > > I have a list of authors with a name element and a file-as attribute
> > > (XPATH: /booklist/author/name/@file-as).  I currently build a Table of
> > > Contents sorted by the author/name/@file-as attribute.  That works OK.
> > >  I want to add further markup (by putting authors in groups by the
> > > first character of their last name).
> > >
> > > I can get the first character of the last name, I've even created a
> > > key that allows me to access the nodes of authors by the first
> > > character in the last name (see the commented out section of the b.xsl
> > > file included below).  However, I can not figure out how to get a list
> > > of distinct characters (from the last names) so that I can iterate
> > > over them and generate my list.
> > >
> > > Here are stripped down example source files:
> > >
> > > Exmaple author file: (file a.xml)
> > >
> > > <?xml version="1.0" encoding="UTF-8"?>
> > > <booklist>
> > >   <author>
> > >      <name file-as="Adams, Douglas">Douglas Adams</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Anderson, Kevin J.">Kevin J. Anderson</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Anthony, Piers">Piers Anthony</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Archer, Jeffrey">Jeffrey Archer</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Baldacci, David">David Baldacci</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Ball, Margaret">Margaret Ball</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Bradley, Marion Zimmer">Marion Zimmer Bradley</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Carcaterra, Lorenzo">Lorenzo Carcaterra</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Card, Orson Scott">Orson Scott Card</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Chalker, Jack L.">Jack L. Chalker</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Dahl, Roald">Roald Dahl</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Daley, Brian">Brian Daley</name>
> > >   </author>
> > >   <author>
> > >      <name file-as="Dann, Jack">Jack Dann</name>
> > >   </author>
> > > </booklist>
> > >
> > >
> > > I can extract the authors and generate a table of contents with the
> > > following XSLT: (b.xsl)
> > >
> > > <?xml version="1.0"?>
> > > <xsl:stylesheet version="1.0"
> > >                xmlns:xsl="http://www.w3.org/1999/XSL/Transform";
> > >                >
> > >    <xsl:output method="xml"/>
> > >    <xsl:output indent="yes"/>
> > >    <xsl:strip-space elements="*"/>
> > >
> > >    <xsl:key name="alet" match="//author" 
use="substring(name/@file-as,1,1)"/>
> > >
> > >    <xsl:template match="/booklist">
> > >        <xsl:call-template name="gen-toc"/>
> > >    </xsl:template>
> > >
> > >    <!-- Generate Table of Contents -->
> > >    <xsl:template name="gen-toc">
> > >      <div id="toc" class="toc">
> > > <!--
> > >      <h1>Test</h1>
> > >      <ul>
> > >      <xsl:for-each select="key('alet','A')">
> > >        <li><xsl:value-of select="name/@file-as"/></li>
> > >      </xsl:for-each>
> > >      </ul>
> > > -->
> > >      <h1>Table of Contents (by Author)</h1>
> > >      <ul>
> > >      <xsl:for-each select="//author/name">
> > >        <xsl:sort select="@file-as"/>
> > >        <li><a href="#{generate-id(..)}"><xsl:value-of
> > > select="@file-as"/></a></li>
> > >      </xsl:for-each>
> > >      </ul>
> > >      </div>
> > >    </xsl:template>
> > >
> > > </xsl:stylesheet>
> > >
> > >
> > > This works.  It generates this: (out.xml)
> > >
> > > <?xml version="1.0"?>
> > > <div id="toc" class="toc">
> > >  <h1>Table of Contents (by Author)</h1>
> > >  <ul>
> > >    <li>
> > >      <a href="#id91524">Adams, Douglas</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91570">Anderson, Kevin J.</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91578">Anthony, Piers</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91588">Archer, Jeffrey</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91597">Baldacci, David</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91607">Ball, Margaret</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91617">Bradley, Marion Zimmer</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id91627">Carcaterra, Lorenzo</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id93622">Card, Orson Scott</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id93633">Chalker, Jack L.</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id93643">Dahl, Roald</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id93653">Daley, Brian</a>
> > >    </li>
> > >    <li>
> > >      <a href="#id93663">Dann, Jack</a>
> > >    </li>
> > >  </ul>
> > > </div>
> > >
> > >
> > > I want to change that so that I have the authors in subsections, with
> > > the first letter of their last names as the the main entry, like so
> > >
> > >        A
> > >                //author/name/@file-as[.=='A']
> > >        B
> > >                //author/name/@file-as[.=='B']
> > >        etc..
> > >
> > > Like this output: (out-desired.xml)
> > >
> > > [This was hand edited, I have not yet figured out how to automatically
> > > generate it.]
> > >
> > > <?xml version="1.0"?>
> > > <div id="toc" class="toc">
> > >  <h1>Table of Contents (by Author)</h1>
> > >  <ul>
> > >    <li>
> > >      <a name="A">A</a>
> > >      <ul>
> > >        <li>
> > >          <a href="#id91524">Adams, Douglas</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id91570">Anderson, Kevin J.</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id91578">Anthony, Piers</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id91588">Archer, Jeffrey</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id91597">Baldacci, David</a>
> > >        </li>
> > >      </ul>
> > >    </li>
> > >    <li>
> > >      <a name="B">B</a>
> > >      <ul>
> > >        <li>
> > >          <a href="#id91607">Ball, Margaret</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id91617">Bradley, Marion Zimmer</a>
> > >        </li>
> > >      </ul>
> > >    </li>
> > >    <li>
> > >      <a name="C">C</a>
> > >      <ul>
> > >        <li>
> > >          <a href="#id91627">Carcaterra, Lorenzo</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id93622">Card, Orson Scott</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id93633">Chalker, Jack L.</a>
> > >        </li>
> > >      </ul>
> > >    </li>
> > >    <li>
> > >      <a name="D">D</a>
> > >      <ul>
> > >        <li>
> > >          <a href="#id93643">Dahl, Roald</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id93653">Daley, Brian</a>
> > >        </li>
> > >        <li>
> > >          <a href="#id93663">Dann, Jack</a>
> > >        </li>
> > >      </ul>
> > >    </li>
> > >  </ul>
> > > </div>
> > >
> > >
> > > I've tried some XPath 2 stuff and XQuery.  I can get a list of first
> > > letters using XQuery (and thus XPath 2)
> > >
> > > For example: (file fl.xq)
> > >
> > > for $a in //author/name/substring(@file-as,1,1)
> > > return
> > >  <c>{$a}</c>
> > >
> > >
> > > Generates a list of the characters of the last name (although not unique).
> > >
> > > I tried embedding something like the above "for $a in.." into an XSLT
> > > 2.0 stylesheet, and assign it to a variable, but I get an error from
> > > saxon when I tried it.
> > >
> > > Something like this is desired.... (with UNIQUE-SET-OF-FIRST-LETTERS)
> > > replaced with something that works.  I set this to XSLT 2, and tried
> > > "$letters//l" in place of that place-holder text
> > > (UNIQUE-SET-OF-FIRST-LETTERS).  It generated the Letter items A B C
> > > .., but all of the sub <ul>s where empty.
> > >
> > > <?xml version="1.0"?>
> > > <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"; 
version="1.0">
> > >  <xsl:output method="xml"/>
> > >  <xsl:output indent="yes"/>
> > >  <xsl:strip-space elements="*"/>
> > >
> > >  <xsl:key name="alet" match="//author" 
use="substring(name/@file-as,1,1)"/>
> > >
> > > <!--
> > >  <xsl:variable name="letters">
> > >    <l>A</l>
> > >    <l>B</l>
> > >    <l>C</l>
> > >    <l>D</l>
> > >    <l>E</l>
> > >  </xsl:variable>
> > > -->
> > >  <xsl:template match="/booklist">
> > >    <xsl:call-template name="gen-toc"/>
> > >  </xsl:template>
> > >
> > >  <!-- Generate Table of Contents -->
> > >  <xsl:template name="gen-toc">
> > >    <div id="toc" class="toc">
> > >      <h1>Table of Contents (by Author)</h1>
> > >      <ul>
> > >        <xsl:for-each select="UNIQUE-SET-OF-FIRST-LETTERS">
> > >          <xsl:variable name="lc" select="."/>
> > >          <li>
> > >            <a>
> > >              <xsl:attribute name="name" select="$lc"/>
> > >              <xsl:value-of select="$lc"/>
> > >            </a>
> > >            <ul>
> > >              <xsl:for-each select="key('alet',$lc)">
> > >                <xsl:sort select="name/@file-as"/>
> > >                <li>
> > >                  <a href="#{generate-id(.)}">
> > >                    <xsl:value-of select="name/@file-as"/>
> > >                  </a>
> > >                </li>
> > >              </xsl:for-each>
> > >            </ul>
> > >          </li>
> > >        </xsl:for-each>
> > >      </ul>
> > >    </div>
> > >  </xsl:template>
> > > </xsl:stylesheet>
> > >
> > > Hopefully, I didn't include too much detail.
> > >
> > > Any help appreciated.  Thanks.
> > >
> > > --~------------------------------------------------------------------
> > > 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>
> > > --~--
> > >
> > >
> >
> > --~------------------------------------------------------------------
> > 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>
> > --~--
> >
> >
>
> --~------------------------------------------------------------------
> 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>
> --~--
>
>

--~------------------------------------------------------------------
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>
--~--



--~------------------------------------------------------------------
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>
--~--