Нормализация модели списка смежности XML в список дерева HTML - PullRequest
2 голосов
/ 21 июня 2011

Я видел примеры преобразования "модели смежности" XML, но ни один из них не сделал бы это совершенно правильно для списка маркеров ul / li. Может ли кто-нибудь дать мне подсказку? Было бы здорово, если бы решение могло поддерживать типичные требования модели смежности и иметь дело с многоуровневой вложенностью / рекурсией.

Если XML:

<?xml version="1.0" encoding="ISO-8859-1"?>
<root>
  <row Id="2" Name="data" />
  <row Id="3" Name="people" />
  <row Id="4" Name="person" ParentId="3" />
  <row Id="6" Name="folder" ParentId="2" />
  <row Id="7" Name="thing" ParentId="3" />
  <row Id="8" Name="web" />
  <row Id="9" Name="link" ParentId="8" />
</root>

И я использую что-то вроде:

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

<xsl:template match="/">
<ul id="someid" class="menu">
  <xsl:apply-templates select="root/row[not(@ParentId)]"/>
</ul>
</xsl:template>

<xsl:template match="row">
<ul>
    <li>
    <xsl:variable name="ID" select="@Id"/>
      <xsl:attribute name="rel">
      <xsl:value-of select="@Id"/>
      </xsl:attribute>
      <xsl:value-of select="@Name"/>
      <xsl:apply-templates select="//row[@ParentId=$ID]"/>
   </li>
</ul>
</xsl:template>

</xsl:stylesheet>   

Тогда я получаю:

<ul id="someid" class="menu">
  <ul>
    <li rel="2">
      data<ul>
        <li rel="6">folder</li>
      </ul>
    </li>
  </ul>
  <ul>
    <li rel="3">
      people
      <ul>
        <li rel="4">person</li>
      </ul><ul>
        <li rel="7">thing</li>
      </ul>
    </li>
  </ul>
  <ul>
    <li rel="8">
      web<ul>
        <li rel="9">link</li>
      </ul>
    </li>
  </ul>
</ul>

Обратите внимание, что дополнительные теги ul / close между open и person не должны быть. Я понимаю, почему это происходит, но не знаю, как изменить код, чтобы исправить это.

Спасибо.

Ответы [ 2 ]

3 голосов
/ 21 июня 2011

Обновлено с учетом новых запросов OP

Это рекурсивный шаблон, который создает HTML-совместимый вложенный список в соответствии с деинфекцией в спецификации W3C.

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

    <xsl:output indent="yes"/>

    <xsl:template match="/root">
        <ul id="someid" class="menu">
            <xsl:apply-templates select="row[not(@ParentId)]"/>
        </ul>
    </xsl:template>

    <xsl:template match="row">
        <li rel="{@Id}">
            <xsl:value-of select="@Name"/>
            <xsl:if test="count(../row[@ParentId=current()/@Id])>0">
                <ul>
                    <xsl:apply-templates select="../row[@ParentId=current()/@Id]"/>
                </ul>
            </xsl:if>
        </li>
    </xsl:template>

</xsl:stylesheet>

Применяется к этому входу:

<root>
  <row Id="1" Name="data" />
  <row Id="2" Name="data" />
  <row Id="3" Name="people" />
  <row Id="4" Name="person" ParentId="3" />
  <row Id="6" Name="folder" ParentId="2" />
  <row Id="7" Name="thing" ParentId="3" />
  <row Id="8" Name="web" />
  <row Id="9" Name="link" ParentId="8" />
    <row Id="10" Name="anotherone" ParentId="9" />
    <row Id="11" Name="anotherone" ParentId="9" />
    <row Id="12" Name="anotherone" ParentId="9" />
    <row Id="13" Name="anotherone" ParentId="3" />
</root>

Производит:

<ul id="someid" class="menu">
   <li rel="1">data</li>
   <li rel="2">data<ul>
         <li rel="6">folder</li>
      </ul>
   </li>
   <li rel="3">people<ul>
         <li rel="4">person</li>
         <li rel="7">thing</li>
         <li rel="13">anotherone</li>
      </ul>
   </li>
   <li rel="8">web<ul>
         <li rel="9">link<ul>
               <li rel="10">anotherone</li>
               <li rel="11">anotherone</li>
               <li rel="12">anotherone</li>
            </ul>
         </li>
      </ul>
   </li>
</ul>

Если у вас возникли проблемы с созданием списка, попробуйте их по адресу W3School .

2 голосов
/ 21 июня 2011

Вы действительно близки, вам просто нужно переместить ваше <xsl:variable name="ID" select="@Id"/> и ваше <xsl:apply-templates select="//row[@ParentId=$ID]"/> за пределы вашего <ul>...</ul> заявления.

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

  <xsl:template match="/">
    <ul id="someid" class="menu">
      <xsl:apply-templates select="root/row[not(@ParentId)]"/>
    </ul>
  </xsl:template>

  <xsl:template match="row">
    <xsl:variable name="ID" select="@Id"/>
      <li>
        <xsl:attribute name="rel">
          <xsl:value-of select="@Id"/>
        </xsl:attribute>
        <xsl:value-of select="@Name"/>
      </li>
    <xsl:if test="//row[@ParentId=$ID]">
      <ul>
        <xsl:apply-templates select="//row[@ParentId=$ID]"/>
      </ul>
    </xsl:if>
  </xsl:template>

</xsl:stylesheet>

это дает результат, который, я думаю, вы ищете

<ul id="someid" class="menu">
  <li rel="2">data</li>
  <ul>
    <li rel="6">folder</li>
  </ul>
  <li rel="3">people</li>
  <ul>
    <li rel="4">person</li>
    <li rel="7">thing</li>
  </ul>
  <li rel="8">web</li>
  <ul>
    <li rel="9">link</li>
  </ul>
</ul>

это также удаляет лишние <ul> до верхнего уровня <li rel="2">

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