Используйте xslt для преобразования списка структурированных объектов в XML-таблицу с несколькими столбцами - PullRequest
1 голос
/ 19 мая 2011

У меня есть следующий фрагмент XML-файла:

<PASS>
<FIRMWARE Unit="1" Group="FIRM">
  <test name="ACPI" ID="ACPI" />
</FIRMWARE>
<NETWORK Unit="2" Group="NTWK">
  <test name="Controller Test" ID="Ctlr" />
</NETWORK>
<NETWORK Unit="1" Group="NTWK">
  <test name="Controller Test" ID="Ctlr" />
</NETWORK>
<ATA Unit="1" Group="ATA-">
  <test name="Serial Controllers" ID="SATA" />
</ATA>
<PARALLEL_PORT Unit="1" Group="LPT-">
  <test name="Verify Controller" ID="Ctrl" />
  <test name="Check Status Port" ID="Stat" />
  <test name="Interrupt Test" ID="Int-" />
</PARALLEL_PORT>
<SERIAL_PORT Unit="4" Group="COM-">
  <test name="Interrupt" ID="Intr" />
  <test name="Line Control" ID="Line" />
  <test name="Test Loopback" ID="LpBk" />
  <test name="Test Internal FIFO" ID="FIFO" />
  <test name="Test Internal Loopback" ID="ILBk" />
</SERIAL_PORT>
<SERIAL_PORT Unit="3" Group="COM-">
  <test name="Interrupt" ID="Intr" />
  <test name="Line Control" ID="Line" />
  <test name="Test Loopback" ID="LpBk" />
  <test name="Test Internal FIFO" ID="FIFO" />
  <test name="Test Internal Loopback" ID="ILBk" />
</SERIAL_PORT>
</PASS>

С существующим XSL-файлом я получаю все результаты теста в одном столбце.Это не очень удобно для печати и так далее.Как я могу создать таблицу с 2 или более столбцами?

<xsl:template match="PASS">
<div class="component">
  <h4>PASSED</h4>
  <xsl:call-template name="outputtable">
    <xsl:with-param name="result" select="'PASSED'" />
  </xsl:call-template>
</div>
</xsl:template>

<xsl:template name="outputtable">
<div class="attributes">
  <xsl:variable name="resultlist" select="*" />
  <xsl:variable name="index" select="count(child::*)" />
  <xsl:for-each select="$resultlist">
    <xsl:variable name="Unit" select="@Unit" />
    <xsl:variable name="Group" select="@Group" />
    <p class="attrtitle">
      <xsl:value-of select="name()" />
      Unit:
      <xsl:value-of select="$Unit" />
    </p>
    <xsl:variable name="testtype" select="*" />
    <xsl:for-each select="$testtype">
      <p class="attribute">
        <xsl:value-of select="@name" />
      </p>
    </xsl:for-each>
    <p />
  </xsl:for-each>
</div>

Я уже искал решение и нашел следующие ссылки: Используйте xslt для преобразования списка XML в HTMLтаблица с несколькими столбцами и http://blogs.msdn.com/b/kaevans/archive/2003/04/03/4754.aspx

Ситуация, с которой я столкнулся, довольно отличается: у меня структурированные объекты, а не просто как "Item" или "ComputerName", и другой недостаток состоит в том, что эти тесты нетого же типа (и т. д.).Этот XML-отчет автоматически создается не-OSS-программным обеспечением, поэтому у меня нет возможности изменить его во время создания.

Я буду рад любым идеям!

1 Ответ

0 голосов
/ 19 мая 2011

Это преобразование :

<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:param name="pnumCols" select="3"/>

    <xsl:template match="PASS">
        <div class="component">
            <h4>PASSED</h4>
            <div class="attributes">
             <table border="1">
                <xsl:apply-templates select=
                  "*[position() mod $pnumCols = 1]"/>
             </table>
            </div>
        </div>
    </xsl:template>

    <xsl:template match="PASS/*">
     <tr>
      <xsl:apply-templates mode="process" select=
       ".|following-sibling::*[not(position() > $pnumCols -1)]"/>
     </tr>
    </xsl:template>

    <xsl:template match="PASS/*" mode="process">
     <xsl:variable name="Unit" select="@Unit" />
     <xsl:variable name="Group" select="@Group" />
     <td>
         <p class="attrtitle">
            <xsl:value-of select="name()" />       Unit:
            <xsl:value-of select="$Unit" />
         </p>
         <xsl:variable name="testtype" select="*" />
         <xsl:for-each select="$testtype">
            <p class="attribute">
                <xsl:value-of select="@name" />
            </p>
         </xsl:for-each>
         <p />
     </td>
    </xsl:template>
</xsl:stylesheet>

при применении к предоставленному документу XML:

<PASS>
    <FIRMWARE Unit="1" Group="FIRM">
        <test name="ACPI" ID="ACPI" />
    </FIRMWARE>
    <NETWORK Unit="2" Group="NTWK">
        <test name="Controller Test" ID="Ctlr" />
    </NETWORK>
    <NETWORK Unit="1" Group="NTWK">
        <test name="Controller Test" ID="Ctlr" />
    </NETWORK>
    <ATA Unit="1" Group="ATA-">
        <test name="Serial Controllers" ID="SATA" />
    </ATA>
    <PARALLEL_PORT Unit="1" Group="LPT-">
        <test name="Verify Controller" ID="Ctrl" />
        <test name="Check Status Port" ID="Stat" />
        <test name="Interrupt Test" ID="Int-" />
    </PARALLEL_PORT>
    <SERIAL_PORT Unit="4" Group="COM-">
        <test name="Interrupt" ID="Intr" />
        <test name="Line Control" ID="Line" />
        <test name="Test Loopback" ID="LpBk" />
        <test name="Test Internal FIFO" ID="FIFO" />
        <test name="Test Internal Loopback" ID="ILBk" />
    </SERIAL_PORT>
    <SERIAL_PORT Unit="3" Group="COM-">
        <test name="Interrupt" ID="Intr" />
        <test name="Line Control" ID="Line" />
        <test name="Test Loopback" ID="LpBk" />
        <test name="Test Internal FIFO" ID="FIFO" />
        <test name="Test Internal Loopback" ID="ILBk" />
    </SERIAL_PORT>
</PASS>

производит результаты в формате $pnumCols -колонки, в данном конкретном случае pnumCols равно 3:

<div class="component">
   <h4>PASSED</h4>
   <div class="attributes">
      <table border="1">
         <tr>
            <td>
               <p class="attrtitle">FIRMWARE       Unit:
            1</p>
               <p class="attribute">ACPI</p>
               <p/>
            </td>
            <td>
               <p class="attrtitle">NETWORK       Unit:
            2</p>
               <p class="attribute">Controller Test</p>
               <p/>
            </td>
            <td>
               <p class="attrtitle">NETWORK       Unit:
            1</p>
               <p class="attribute">Controller Test</p>
               <p/>
            </td>
         </tr>
         <tr>
            <td>
               <p class="attrtitle">ATA       Unit:
            1</p>
               <p class="attribute">Serial Controllers</p>
               <p/>
            </td>
            <td>
               <p class="attrtitle">PARALLEL_PORT       Unit:
            1</p>
               <p class="attribute">Verify Controller</p>
               <p class="attribute">Check Status Port</p>
               <p class="attribute">Interrupt Test</p>
               <p/>
            </td>
            <td>
               <p class="attrtitle">SERIAL_PORT       Unit:
            4</p>
               <p class="attribute">Interrupt</p>
               <p class="attribute">Line Control</p>
               <p class="attribute">Test Loopback</p>
               <p class="attribute">Test Internal FIFO</p>
               <p class="attribute">Test Internal Loopback</p>
               <p/>
            </td>
         </tr>
         <tr>
            <td>
               <p class="attrtitle">SERIAL_PORT       Unit:
            3</p>
               <p class="attribute">Interrupt</p>
               <p class="attribute">Line Control</p>
               <p class="attribute">Test Loopback</p>
               <p class="attribute">Test Internal FIFO</p>
               <p class="attribute">Test Internal Loopback</p>
               <p/>
            </td>
         </tr>
      </table>
   </div>
</div>

Объяснение : Мы используем два шаблона - один для упаковки группы из $pnumCols последовательных результатов в строке, другой, в mode="process", для фактического создания содержимого в этой строке.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...