XSLT: добавить внутренний текст узла - PullRequest
2 голосов
/ 24 мая 2010

Это небольшая версия другого вопроса, опубликованного здесь: XSLT: изменить внутренний текст узла

Представьте, что я использую XSLT для преобразования документа:

<a>
  <b/>
  <c/>
</a>

в это:

<a>
  <b/>
  <c/>
  Hello world
</a>

В этом случае я не могу использовать ни элемент

<xsl:strip-space elements="*"/> 

, ни предикат [normalize-space ()! = ''], так как неттекст в том месте, где мне нужно положить новый текст.Есть идеи?Спасибо.

Ответы [ 3 ]

2 голосов
/ 24 мая 2010

Это преобразование вставляет нужный текст (для общности) после элемента с именем a7:

<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:template match="node()|@*" name="identity">
  <xsl:copy>
   <xsl:apply-templates select="node()|@*"/>
  </xsl:copy>
 </xsl:template>

 <xsl:template match="a7">
   <xsl:call-template name="identity"/>
   <xsl:text>Hello world</xsl:text>
 </xsl:template>
</xsl:stylesheet>

при применении к этому документу XML :

<a>
  <a1/>
  <a2/>
  .....
  <a7/>
  <a8/>
</a>

желаемый результат получается :

<a>
    <a1/>
    <a2/>
  .....
    <a7/>Hello world
    <a8/>
</a>

Примечание :

  1. Использование правила идентификации для копирования каждого узла исходного XML-документа.

  2. Переопределение правила идентификации с помощью специальный шаблон , который выполняет вставку нового текста.

  3. Как правило идентификации применяется (на каждом узле) и вызывается по имени (для конкретной необходимости).

2 голосов
/ 25 мая 2010

Вот что я бы сделал:

<xsl:stylesheet 
  version="1.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
>
  <!-- identity template to copy everything unless otherwise noted -->
  <xsl:template match="node()|@*">
    <xsl:copy>
      <xsl:apply-templates select="node()|@*"/>
    </xsl:copy>
  </xsl:template>

  <!-- match the first text node directly following a <c> element -->
  <xsl:template match="text()[preceding-sibling::node()[1][self::c]]">
    <!-- ...and change its contents -->
    <xsl:text>Hello world</xsl:text>
  </xsl:template>
</xsl:stylesheet>

Обратите внимание, что текстовые узлы содержат «окружающие» пробелы - в примере XML в вопросе сопоставленный текстовый узел является только пробелом, поэтому вышеописанное работает. Он перестанет работать, как только входной документ будет выглядеть так:

<a><b/><c/></a>

потому что здесь нет текстового узла, следующего за <c>. Так что, если это слишком хрупко для вашего варианта использования, альтернативой будет:

<!-- <c> nodes get a new adjacent text node -->
<xsl:template match="c">
  <xsl:copy-of select="." />
  <xsl:text>Hello world</xsl:text>
</xsl:template>

<!-- make sure to remove the first text node directly following a <c> node-->
<xsl:template match="text()[preceding-sibling::node()[1][self::c]]" />

В любом случае такие вещи, как приведенные выше, проясняют, почему лучше не смешивать текстовые узлы и узлы элементов. Это не всегда возможно (см. XHTML). Но когда у вас есть такая возможность, а XML должен быть просто контейнером для структурных данных, отказ от смешанного контента облегчает вашу жизнь.

0 голосов
/ 24 мая 2010

edit: исправлена ​​ошибка с неправильным синтаксисом.

<xsl:template match='a'>
   <xsl:copy-of select="." />
   <xsl:text>Hello World</xsl:text>
</xsl:template>
<xsl:template match="@*|node()">
  <xsl:copy>
    <xsl:apply-templates select="@*|node()"/>
  </xsl:copy>
</xsl:template>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...