2

我的xml如下:

<valid_path>
    <document var_name="some_value">
        <processControls>
            <p>Lots of good text here ...</p>
            <ul class="unIndentedList">
                <li> Graphical display of system</li>
                <li> Other bulleted items ...</li>
            </ul>
            <p>etc. etc. etc.</p>
        </processControls>
    </document>
</valid_path>

我的输入是由这个决定的:

<xsl:variable name="processControlsValue" select="/valid_path/document[@var_name='some_value']/processControls" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"/>

我正在调用递归模板来替换在某些输入中找到的给定字符串。调用如下。

<xsl:call-template name="bulRep">
    <xsl:with-param name="text" select="$processControlsValue"/>
    <xsl:with-param name="replace" select="'Graphical'"/>
    <xsl:with-param name="by" select="'foofoobars'"/>
</xsl:call-template>

这是模板。

<xsl:template name="bulRep">
    <xsl:param name="text"/>
    <xsl:param name="replace"/>
    <xsl:param name="by"/>
    <xsl:choose>
        <xsl:when test="contains($text, $replace)">
            <xsl:value-of select="substring-before($text,$replace)"/>
            <xsl:value-of select="$by"/>
            <xsl:call-template name="bulRep">
                <xsl:with-param name="text" select="substring-after($text,$replace)"/>
                <xsl:with-param name="replace" select="$replace"/>
                <xsl:with-param name="by" select="$by"/>
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:value-of select="$text"/>
            <!--<xsl:apply-templates select="$text" />-->
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>

该模板的工作原理是替换找到的文本。(例如,'Graphical' 确实被'foofoobars' 取代) 我遇到的问题是丢失了带有 value-of 或 copy-of 的格式。我意识到 value-of 将返回文本,而不是文本和格式,并且我发现许多帖子指示其他人使用 copy-of 来保留格式。但是,这不起作用。输出只是一个连续的文本行。

现在,如果我使用 apply-templates 行(当前已注释掉)而不是 value-of 行,并确保我点击了模板的“否则”部分,我会得到所需的输出。我会得到段落、项目符号列表等。但是使用 value-of 或 copy-of 只会获取文本,如果存在字符串匹配,使用 apply-templates 会中断。

我的最终结果是使用 xsl-fo 的 PDF。

我现在看到的:

这里有很多好文字... foofoobars 系统显示 其他项目符号... 等等 等等 等等

我想看到的:

这里有很多好文...

• foofoobars 系统显示

• 其他项目符号...

等等等等等等

4

1 回答 1

2

问题是,如果您使用的查找和替换模板仅用于处理作为参数传入的文本。如果您要传入节点列表,那么当您执行“包含”之类的字符串函数时,它将只使用第一个节点的文本。或者,如果您传入具有多个后代的单个元素,则字符串值将被视为所有后代文本节点的串联。

我认为您可能需要做的是在这里使用身份转换,这可能是您在提到xsl:apply-templates时已经在做的事情,但是有一个单独的模板来匹配文本节点,您可以在其中然后调用查找和替换模板

<xsl:template ....>
  <xsl:variable name="processControlsValue" 
                select="/valid_path/document[@var_name='some_value']/processControls" />
  <xsl:apply-templates select="$processControlsValue" mode="replace">
    <xsl:with-param name="replace" select="'Graphical'"/>
    <xsl:with-param name="by" select="'foofoobars'"/>
  </xsl:apply-templates>
</xsl:template>

<xsl:template match="@*|node()[not(self::text())]" mode="replace">
  <xsl:param name="replace"/>
  <xsl:param name="by"/>
  <xsl:copy>
    <xsl:apply-templates select="@*|node()" mode="replace">
      <xsl:with-param name="replace" select="$replace"/>
      <xsl:with-param name="by" select="$by"/>
    </xsl:apply-templates>
  </xsl:copy>
</xsl:template>

<xsl:template match="text()" mode="replace">
  <xsl:param name="replace"/>
  <xsl:param name="by"/>
  <xsl:call-template name="bulRep">
    <xsl:with-param name="text" select="."/>
    <xsl:with-param name="replace" select="$replace"/>
    <xsl:with-param name="by" select="$by"/>
  </xsl:call-template>
</xsl:template>

请注意,我曾经mode="replace"防止与您可能​​拥有的模板发生潜在冲突。如果没有这样的冲突,可能会删除该模式。

所以,这个想法在这两个模板中,它复制任何现有的元素和属性,但是当它找到一个文本节点时,它只对文本进行查找和替换。

顺便说一句,如果您使用的是 XSLT 2.0,则可以只使用替换函数而不是命名模板

<xsl:template match="text()" mode="replace">
   <xsl:param name="replace"/>
   <xsl:param name="by"/>
   <xsl:value-of select="replace(., $replace, $by)" />
</xsl:template>

编辑:作为一个完整的工作示例,试试这个 XSLT

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

   <xsl:template match="/">
      <xsl:variable name="processControlsValue" select="/valid_path/document[@var_name='some_value']/processControls"/>
      <xsl:apply-templates select="$processControlsValue" mode="replace">
         <xsl:with-param name="replace" select="'Graphical'"/>
         <xsl:with-param name="by" select="'foofoobars'"/>
      </xsl:apply-templates>
   </xsl:template>

   <xsl:template match="@*|node()[not(self::text())]" mode="replace">
      <xsl:param name="replace"/>
      <xsl:param name="by"/>
      <xsl:copy>
         <xsl:apply-templates select="@*|node()" mode="replace">
            <xsl:with-param name="replace" select="$replace"/>
            <xsl:with-param name="by" select="$by"/>
         </xsl:apply-templates>
      </xsl:copy>
   </xsl:template>

   <xsl:template match="text()" mode="replace">
      <xsl:param name="replace"/>
      <xsl:param name="by"/>
      <xsl:call-template name="bulRep">
         <xsl:with-param name="text" select="."/>
         <xsl:with-param name="replace" select="$replace"/>
         <xsl:with-param name="by" select="$by"/>
      </xsl:call-template>
   </xsl:template>

   <xsl:template name="bulRep">
      <xsl:param name="text"/>
      <xsl:param name="replace"/>
      <xsl:param name="by"/>
      <xsl:choose>
         <xsl:when test="contains($text, $replace)">
            <xsl:value-of select="substring-before($text,$replace)"/>
            <xsl:value-of select="$by"/>
            <xsl:call-template name="bulRep">
               <xsl:with-param name="text" select="substring-after($text,$replace)"/>
               <xsl:with-param name="replace" select="$replace"/>
               <xsl:with-param name="by" select="$by"/>
            </xsl:call-template>
         </xsl:when>
         <xsl:otherwise>
            <xsl:value-of select="$text"/><!--<xsl:apply-templates select="$text" />-->
         </xsl:otherwise>
      </xsl:choose>
   </xsl:template>
</xsl:stylesheet>

当应用于您的示例 XML 时,将输出以下内容。可以看出,文本中的“图形”一词已替换为“foofoobars”。

<processControls>
<p>Lots of good text here ...</p>
<ul class="unIndentedList">
<li> foofoobars display of system</li>
<li> Other bulleted items ...</li>
</ul>
<p>etc. etc. etc.</p>
</processControls>
于 2013-09-20T17:22:10.993 回答