列表元素的以下兄弟姐妹

Posted

技术标签:

【中文标题】列表元素的以下兄弟姐妹【英文标题】:following-sibling for list-element 【发布时间】:2019-03-27 17:07:40 【问题描述】:

我想创建一个列表项,并且格式正确。但是,我想将字符串 'or' 添加到替代项中。 我的 XPath 控件无法正常工作。如果我有一个List 和一个底层SmallList,则该函数将被忽略。

如果列表不存在,则一切正常。但是,我没有得出正确的表达方式。我正在使用 XSLT 1.0。下面是一个例子。

XML:

<University>
    <Class>
        <Student>John Doe</Student>
        <List>
            <SmallList>
                <Student>Jane Doe</Student>
            </SmallList>
        </List>
    </Class>
    <AlternativeClass>
        <Class>
            <Student>Richard Roe</Student>
        </Class>
    </AlternativeClass>
</University>

结果应该是什么样子:

<div>
   <p data-role="heading">University-Heading</p>
   <ul>
      <li>John Doe
         <ul>
            <li>Jane Doe</li>
         </ul>
      </li>
      <li class="parablock bold_">or</li>
      <li>Richard Roe</li>
   </ul>
</div>

正如我提到的,&lt;li class="parablock bold_&gt;or&lt;li&gt; 不见了。 如果我想删除List-Block,结果看起来符合预期(例如本例中的 Jane Doe)。

<div>
   <p data-role="heading">University-Heading</p>
   <ul>
      <li>John Doe</li>
      <li class="parablock bold_">or</li>
      <li>Richard Roe</li>
   </ul>
</div>

following-sibling 的结构应该是这样的(BBB 的following-sibling 应该是XXX):

<AAA>
    <BBB>
        <CCC/>
    </BBB>
    <XXX>
        <DDD/>
        <EEE/>
    </XXX>
</AAA>

我的 XSLT 如下所示:

<xsl:template match="University">
    <div>
        <!-- HEADING FOR UNIVERSITY -->
        <p data-role="heading">
            <xsl:text>University-Heading</xsl:text>
        </p>
        <ul>
            <xsl:apply-templates/>
        </ul>
    </div>
</xsl:template>
<!-- CLASS -->
<xsl:template match="Class">
    <xsl:apply-templates/>
    <!-- 'OR' BEFORE ALTERNATIVE CLASS -->
    <!-- I GUESS THAT THE MISTAKE IS IN HERE -->
    <xsl:if test="following-sibling::*[1][self::AlternativeClass]">
        <li class="parablock bold_">
            <xsl:text>or</xsl:text>
        </li>
    </xsl:if>
</xsl:template>
<!-- CLASS-NODE WITH LIST -->
<xsl:template match="Class[List]">
    <xsl:apply-templates select="Student"/>
</xsl:template>
<!-- ALTERNATIVECLASS -->
<xsl:template match="AlternativeClass">
    <xsl:apply-templates/>
</xsl:template>
<!-- STUDENT IN CLASS AND SMALLLIST -->
<xsl:template match="SmallList/Student | Class/Student">
    <li>
        <xsl:if test="name(preceding-sibling::*[1]) = 'Para'">
            <xsl:attribute name="class">parablock</xsl:attribute>
        </xsl:if>
        <xsl:apply-templates/>
        <xsl:apply-templates select="following-sibling::*[1][self::List]"/>
    </li>
</xsl:template>
<!-- LIST -->
<xsl:template match="List">
    <ul>
        <xsl:apply-templates/>
    </ul>
</xsl:template>
<!-- SMALLLIST -->
<xsl:template match="SmallList">
    <xsl:apply-templates/>
</xsl:template>

.NET XSLT 小提琴:https://xsltfiddle.liberty-development.net/bdxtqV

【问题讨论】:

【参考方案1】:

您对以下兄弟的陈述实际上是正确的。问题是Class确实有一个跟随AlternativeClass,也有一个子List元素,这意味着它被这个模板匹配(因为它有额外的条件,模板获得更高的优先级)

<xsl:template match="Class[List]">
    <xsl:apply-templates select="Student"/>
</xsl:template>

由于这没有以下兄弟检查,因此不会创建您的预期输出。

你也可以在这个语句中添加条件

<xsl:template match="Class[List]">
    <xsl:apply-templates select="Student"/>
    <xsl:if test="following-sibling::*[1][self::AlternativeClass]">
        <li class="parablock bold_">
            <xsl:text>or</xsl:text>
        </li>
    </xsl:if>
</xsl:template>

虽然这有点重复代码,但您可以将该逻辑放在命名模板中,或者您可以将两个“类”模板合并为一个...

<xsl:template match="Class">
    <xsl:choose>
        <xsl:when test="List">
            <xsl:apply-templates select="Student"/>
        </xsl:when>
        <xsl:otherwise>
            <xsl:apply-templates />
        </xsl:otherwise>
    </xsl:choose>
    <xsl:if test="following-sibling::*[1][self::AlternativeClass]">
        <li class="parablock bold_">
            <xsl:text>or</xsl:text>
        </li>
    </xsl:if>
</xsl:template>

【讨论】:

我只是忘记了额外的条件 - 更高的优先级。我的整个 XSLT 解释了代码重复,因为它更大更复杂。但是,我想将描述分解为一个最小且易于理解的示例。不过,谢谢!

以上是关于列表元素的以下兄弟姐妹的主要内容,如果未能解决你的问题,请参考以下文章

使用 React Hooks 更改元素类名称和兄弟姐妹

显示动态 WooCommerce 兄弟姐妹和直接子类别列表

动画列表中的 div,保持兄弟姐妹静态

PHP Wordpress显示子页面和兄弟姐妹的列表,如果有的话

R语言基础知识|apply函数家族中的兄弟姐妹

如果值匹配,XSLT 会更改兄弟元素的子元素值