XSLT: вариация проблемы нумерации страниц - PullRequest
3 голосов
/ 06 апреля 2010

Я должен преобразовать некоторые данные XML в разбитый на страницы список полей.Вот пример.

Входной XML:

<?xml version="1.0" encoding="UTF-8"?>
<data>
    <books>
        <book title="t0"/>
        <book title="t1"/>
        <book title="t2"/>
        <book title="t3"/>
        <book title="t4"/>
    </books>
    <library name="my library"/>
</data>

Желаемый вывод:

<?xml version="1.0" encoding="UTF-8"?>
<pages>
    <page number="1">
        <field name="library_name" value="my library"/>
        <field name="book_1" value="t0"/>
        <field name="book_2" value="t1"/>
    </page>
    <page number="2">
        <field name="book_1" value="t2"/>
        <field name="book_2" value="t3"/>
    </page>
    <page number="3">
        <field name="book_1" value="t4"/>
    </page>
</pages>

В приведенном выше примере я предполагаю, что я хочу максимум 2 поля с именем book_nn в диапазоне от 1 до 2) на страницу.Теги <page> должны иметь атрибут number.Наконец, поле с именем library_name должно отображаться только первым <page>.

Вот мое текущее решение с использованием XSLT:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet 
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    version="2.0"
    exclude-result-prefixes="trx xs">

    <xsl:output method="xml" indent="yes" omit-xml-declaration="no" />

    <xsl:variable name="max" select="2"/>

    <xsl:template match="//books">

        <xsl:for-each-group select="book" group-ending-with="*[position() mod $max = 0]">

            <xsl:variable name="pageNum" select="position()"/>

            <page number="{$pageNum}">

                <xsl:for-each select="current-group()">
                    <xsl:variable name="idx" select="if (position() mod $max = 0) then $max else position() mod $max"/>
                    <field value="{@title}">
                        <xsl:attribute name="name">book_<xsl:value-of select="$idx"/>
                        </xsl:attribute>
                    </field>                
                </xsl:for-each>

                <xsl:if test="$pageNum = 1">
                    <xsl:call-template name="templateFor_library"/>            
                </xsl:if>    

            </page>

        </xsl:for-each-group>    

    </xsl:template>

    <xsl:template name="templateFor_library">
        <xsl:for-each select="//library">
            <field name="library_name" value="{@name}" />
        </xsl:for-each>        
    </xsl:template>

</xsl:stylesheet> 

Есть ли лучший / более простой способ выполнить этопреобразование

1 Ответ

4 голосов
/ 06 апреля 2010

Да, есть.

<xsl:param name="pagesize" select="2" />

<xsl:template match="data">
  <pages>
    <xsl:apply-templates mode="page" select="
      books/book[position() mod $pagesize = 1]
    " />
  </pages>
</xsl:template>

<xsl:template match="book" mode="page">
  <page number="{position()}">
    <xsl:apply-templates select="
      . | following-sibling::book[position() &lt; $pagesize]
    " />
  </page>
</xsl:template>

<xsl:template match="book">
  <field name="book_{position()}" value="{@title}" />
</xsl:template>

РЕДАКТИРОВАТЬ # 1: Выше соответствует XSLT 1.0. Вы все еще можете изменить его, чтобы использовать XSLT 2.0 <xsl:for-each-group>, если хотите. Лично я нахожу отдельные шаблоны более привлекательными, чем одна большая толстая вложенная для каждой конструкции YMMV.

РЕДАКТИРОВАТЬ # 2: Согласно запросу в комментариях. Чтобы что-то появилось только для первой страницы, измените один шаблон:

<xsl:template match="book" mode="page">
  <page number="{position()}">
    <xsl:if test="position() = 1">
      <xsl:attribute name="library_name`>
        <xsl:value-of select="ancestor::data/library/@name" />
      </xsl:attribute>
    </xsl:if>
    <xsl:apply-templates select="
      . | following-sibling::book[position() &lt; $pagesize]
    " />
  </page>
</xsl:template>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...