【问题标题】:How to generate Bar Charts in PDF Documents with XSLT and Apache FOP如何使用 XSLT 和 Apache FOP 在 PDF 文档中生成条形图
【发布时间】:2018-11-12 16:10:30
【问题描述】:

我正在寻找一种使用 Java Apache FOP 在 XSLT 中生成 条形图 的方法。

我正在尝试生成具有以下 XML 和 XSLT 的条形图,但它生成的空 .PDF 文件没有任何图表?

data.xml

<?xml version="1.0" encoding="UTF-8"?>
<sales>
 <region>
  <title>Eastern Region Quarterly Sales (Second/'04)</title>
  <key1 area="New York Area">.95</key1>
  <key2 area="Virginia Area">.89</key2>
  <key3 area="Maryland Area">.67</key3>
  <key4 area="Connecticut Area">.65</key4>
  <key5 area="Delaware Area">.45</key5>
 </region>
</sales>

bar.xsl

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output method="xml" indent="yes" />
    <xsl:template match="sales">
        <svg width="650" height="500">
            <g id="axis" transform="translate(0 500) scale(1 -1)">
                <line id="axis-y" x1="30" y1="20" x2="30" y2="450" style="fill:none;stroke:rgb(0,0,0);stroke-width:2" />
                <line id="axis-x" x1="30" y1="20" x2="460" y2="20" style="fill:none;stroke:rgb(0,0,0);stroke-width:2" />
            </g>
            <xsl:apply-templates select="region" />
        </svg>
    </xsl:template>
    <xsl:template match="region">
        <g id="bars" transform="translate(30 479) scale(1 -430)">
            <rect x="30" y="0" width="50" height="{key1}" style="fill:rgb(255,0,0);stroke:rgb(0,0,0);stroke-width:0" />
            <rect x="100" y="0" width="50" height="{key2}" style="fill:rgb(0,255,0);stroke:rgb(0,0,0);stroke-width:0" />
            <rect x="170" y="0" width="50" height="{key3}" style="fill:rgb(255,255,0);stroke:rgb(0,0,0);stroke-width:0" />
            <rect x="240" y="0" width="50" height="{key4}" style="fill:rgb(0,255,255);stroke:rgb(0,0,0);stroke-width:0" />
            <rect x="310" y="0" width="50" height="{key5}" style="fill:rgb(0,0,255);stroke:rgb(0,0,0);stroke-width:0" />
        </g>
        <g id="scale" transform="translate(29 60)">
            <text id="scale1" x="0px" y="320px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$25K</text>
            <text id="scale2" x="0px" y="215px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$50K</text>
            <text id="scale3" x="0px" y="107.5px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$75K</text>
            <text id="scale4" x="0px" y="0px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$100K</text>
        </g>
        <g id="key">
            <rect id="key1" x="430" y="80" width="25" height="15" style="fill:rgb(255,0,0);stroke:rgb(0,0,0);stroke-width:1" />
            <rect id="key2" x="430" y="100" width="25" height="15" style="fill:rgb(0,255,0);stroke:rgb(0,0,0);stroke-width:1" />
            <rect id="key3" x="430" y="120" width="25" height="15" style="fill:rgb(255,255,0);stroke:rgb(0,0,0);stroke-width:1" />
            <rect id="key5" x="430" y="140" width="25" height="15" style="fill:rgb(0,255,255);stroke:rgb(0,0,0);stroke-width:1" />
            <rect id="key4" x="430" y="160" width="25" height="15" style="fill:rgb(0,0,255);stroke:rgb(0,0,0);stroke-width:1" />
        </g>
        <text id="key1-text" x="465px" y="92px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
            <xsl:value-of select="key1/@area" />
        </text>
        <text id="key2-text" x="465px" y="112px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
            <xsl:value-of select="key2/@area" />
        </text>
        <text id="key3-text" x="465px" y="132px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
            <xsl:value-of select="key3/@area" />
        </text>
        <text id="key4-text" x="465px" y="152px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
            <xsl:value-of select="key4/@area" />
        </text>
        <text id="key5-text" x="465px" y="172px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
            <xsl:value-of select="key5/@area" />
        </text>
        <g id="title">
            <text x="325px" y="20px" style="text-anchor:middle;fill:rgb(0,0,0);font-size:24;font-family:Arial">
                <xsl:value-of select="title" />
            </text>
        </g>
    </xsl:template>
</xsl:stylesheet>

我不知道为什么我得到空的 pdf 并且我也尝试生成 svg 文件 - 它也没有显示任何图表。任何帮助将不胜感激。

【问题讨论】:

  • 您可以使用fo:instream-foreign-object 在 XSL-FO 中创建和嵌入 SVG。这是一个示例:stackoverflow.com/questions/4121501/a-beginner-question-on-xslt/… 这并不是您想要做的,因为在那个答案中 SVG 已经存在。您必须在 XSLT 中创建 SVG 才能在 fo:instream-foreign-object 中使用。
  • 感谢您的回复!如何使用 xml 或 xslt 创建 SVG 图表?
  • Daniel Haley 的建议很好。在 XSLT 中创建 条形图 条的坐标,将它们输出为 SVG,然后在 Apache FOP 中的 fo:instream-foreign-object 中使用它们以将它们嵌入到 PDF 中。这听起来像是一个不错的小练习,如果你在这里发布一个工作示例,我会赞成。
  • @zx485 - 在我的回答中添加了一个工作示例。 :-)
  • @DanielHaley:看起来不错,因此我赞成 :-)

标签: java xml xslt charts apache-fop


【解决方案1】:

您可以使用fo:instream-foreign-object 在 XSL-FO 中创建和嵌入 SVG。

由于您想使用 XSLT 和 FOP,我假设您有 XML 格式的输入数据。 (如果不是,也许 XSLT 和 FOP 不是适合这项工作的工具?)

因此,XSLT 将从您的 XML 输入创建 XSL-FO 和嵌入的 SVG。 FOP 将从 XSL-FO 创建 PDF。

示例...

XML 输入

<list>
    <fruit>
        <type>Apples</type>
        <qty>5</qty>
    </fruit>
    <fruit>
        <type>Oranges</type>
        <qty>2</qty>
    </fruit>
    <fruit>
        <type>Bananas</type>
        <qty>7</qty>
    </fruit>
    <fruit>
        <type>Peaches</type>
        <qty>9</qty>
    </fruit>
</list>

XSLT 1.0

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" 
  xmlns:fo="http://www.w3.org/1999/XSL/Format">
  <xsl:output indent="yes"/>
  <xsl:strip-space elements="*"/>

  <xsl:template match="/">
    <fo:root>
      <fo:layout-master-set>
        <fo:simple-page-master master-name="my-page" page-width="8.5in" page-height="11in">
          <fo:region-body margin="1in" margin-top="1.5in" margin-bottom="1.5in"/>
        </fo:simple-page-master>
      </fo:layout-master-set>
      <fo:page-sequence master-reference="my-page">
        <fo:flow flow-name="xsl-region-body">
          <xsl:apply-templates/>
        </fo:flow>
      </fo:page-sequence>
    </fo:root>
  </xsl:template>

  <xsl:template match="list">
    <fo:block>Example embedded SVG:</fo:block>
    <fo:block space-before="4pt">
      <fo:instream-foreign-object>
        <svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="420" height="150">
          <title id="title">List of Fruit</title>
          <xsl:apply-templates select="fruit">
            <xsl:sort select="type" order="ascending" data-type="text"/>
          </xsl:apply-templates>
        </svg>
      </fo:instream-foreign-object>
    </fo:block>
  </xsl:template>

  <xsl:template match="fruit">
    <xsl:variable name="y" select="((position() - 1) * 20) + (position() * 5 - 5)"/>
    <g xmlns="http://www.w3.org/2000/svg">
      <rect width="{qty * 10}" height="20" y="{$y}"/>
      <text x="100" y="{$y + 10}" dy=".35em">
        <xsl:value-of select="concat(qty,' ',type)"/>
      </text>
    </g>    
  </xsl:template>

</xsl:stylesheet>

XSL-FO 输出

<fo:root xmlns:fo="http://www.w3.org/1999/XSL/Format">
   <fo:layout-master-set>
      <fo:simple-page-master master-name="my-page" page-width="8.5in" page-height="11in">
         <fo:region-body margin="1in" margin-top="1.5in" margin-bottom="1.5in"/>
      </fo:simple-page-master>
   </fo:layout-master-set>
   <fo:page-sequence master-reference="my-page">
      <fo:flow flow-name="xsl-region-body">
         <fo:block>Example embedded SVG:</fo:block>
         <fo:block space-before="4pt">
            <fo:instream-foreign-object>
               <svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="420" height="150">
                  <title id="title">List of Fruit</title>
                  <g>
                     <rect width="50" height="20" y="0"/>
                     <text x="100" y="10" dy=".35em">5 Apples</text>
                  </g>
                  <g>
                     <rect width="70" height="20" y="25"/>
                     <text x="100" y="35" dy=".35em">7 Bananas</text>
                  </g>
                  <g>
                     <rect width="20" height="20" y="50"/>
                     <text x="100" y="60" dy=".35em">2 Oranges</text>
                  </g>
                  <g>
                     <rect width="90" height="20" y="75"/>
                     <text x="100" y="85" dy=".35em">9 Peaches</text>
                  </g>
               </svg>
            </fo:instream-foreign-object>
         </fo:block>
      </fo:flow>
   </fo:page-sequence>
</fo:root>

呈现的 PDF 将是一个 8.5" x 11" 的页面,包含以下内容:

注意事项:

  • 我使用 Saxon-HE 进行 XSLT 转换。 Saxon 是一个 XSLT 3.0 处理器,所以如果您决定使用它,您就不会受 XSLT 1.0 的限制。我只使用了 XSLT 1.0,因为这是一个非常基本的示例。
  • 我使用 FOP 2.1 版(包含在 oXygen XML 编辑器中)来处理 XSL-FO 以创建 PDF 输出。

从修改后的问题更新 XSLT:

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

  <xsl:template match="/">
    <fo:root xmlns:fo="http://www.w3.org/1999/XSL/Format">
      <fo:layout-master-set>
        <fo:simple-page-master master-name="my-page" page-width="8.5in" page-height="11in">
          <fo:region-body margin="1in" margin-top="1.5in" margin-bottom="1.5in"/>
        </fo:simple-page-master>
      </fo:layout-master-set>
      <fo:page-sequence master-reference="my-page">
        <fo:flow flow-name="xsl-region-body"> 
          <fo:block>
            <fo:instream-foreign-object>
              <xsl:apply-templates/>
            </fo:instream-foreign-object>
          </fo:block>
        </fo:flow>
      </fo:page-sequence>
    </fo:root>
  </xsl:template>

  <xsl:template match="sales">
    <svg width="650" height="500">
      <g id="axis" transform="translate(0 500) scale(1 -1)">
        <line id="axis-y" x1="30" y1="20" x2="30" y2="450" style="fill:none;stroke:rgb(0,0,0);stroke-width:2" />
        <line id="axis-x" x1="30" y1="20" x2="460" y2="20" style="fill:none;stroke:rgb(0,0,0);stroke-width:2" />
      </g>
      <xsl:apply-templates select="region" />
    </svg>
  </xsl:template>
  <xsl:template match="region">
    <g id="bars" transform="translate(30 479) scale(1 -430)">
      <rect x="30" y="0" width="50" height="{key1}" style="fill:rgb(255,0,0);stroke:rgb(0,0,0);stroke-width:0" />
      <rect x="100" y="0" width="50" height="{key2}" style="fill:rgb(0,255,0);stroke:rgb(0,0,0);stroke-width:0" />
      <rect x="170" y="0" width="50" height="{key3}" style="fill:rgb(255,255,0);stroke:rgb(0,0,0);stroke-width:0" />
      <rect x="240" y="0" width="50" height="{key4}" style="fill:rgb(0,255,255);stroke:rgb(0,0,0);stroke-width:0" />
      <rect x="310" y="0" width="50" height="{key5}" style="fill:rgb(0,0,255);stroke:rgb(0,0,0);stroke-width:0" />
    </g>
    <g id="scale" transform="translate(29 60)">
      <text id="scale1" x="0px" y="320px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$25K</text>
      <text id="scale2" x="0px" y="215px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$50K</text>
      <text id="scale3" x="0px" y="107.5px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$75K</text>
      <text id="scale4" x="0px" y="0px" style="text-anchor:end;fill:rgb(0,0,0);font-size:10;font-family:Arial">$100K</text>
    </g>
    <g id="key">
      <rect id="key1" x="430" y="80" width="25" height="15" style="fill:rgb(255,0,0);stroke:rgb(0,0,0);stroke-width:1" />
      <rect id="key2" x="430" y="100" width="25" height="15" style="fill:rgb(0,255,0);stroke:rgb(0,0,0);stroke-width:1" />
      <rect id="key3" x="430" y="120" width="25" height="15" style="fill:rgb(255,255,0);stroke:rgb(0,0,0);stroke-width:1" />
      <rect id="key5" x="430" y="140" width="25" height="15" style="fill:rgb(0,255,255);stroke:rgb(0,0,0);stroke-width:1" />
      <rect id="key4" x="430" y="160" width="25" height="15" style="fill:rgb(0,0,255);stroke:rgb(0,0,0);stroke-width:1" />
    </g>
    <text id="key1-text" x="465px" y="92px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
      <xsl:value-of select="key1/@area" />
    </text>
    <text id="key2-text" x="465px" y="112px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
      <xsl:value-of select="key2/@area" />
    </text>
    <text id="key3-text" x="465px" y="132px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
      <xsl:value-of select="key3/@area" />
    </text>
    <text id="key4-text" x="465px" y="152px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
      <xsl:value-of select="key4/@area" />
    </text>
    <text id="key5-text" x="465px" y="172px" style="fill:rgb(0,0,0);font-size:18;font-family:Arial">
      <xsl:value-of select="key5/@area" />
    </text>
    <g id="title">
      <text x="325px" y="20px" style="text-anchor:middle;fill:rgb(0,0,0);font-size:24;font-family:Arial">
        <xsl:value-of select="title" />
      </text>
    </g>
  </xsl:template>
</xsl:stylesheet>

PDF 输出

【讨论】:

  • 谢谢,丹尼尔! - 我试过你的例子 - 它有效。我尝试了另一个条形图示例,但我得到的是空的 pdf。我已经用示例 XML 和 XSLT 更新了原始帖子。你能说出我错过了什么吗?
  • @learngroovy - 我认为您的 PDF 是空的,因为您没有创建任何 XSL-FO(除非您试图让 FOP 仅从 SVG 创建 PDF,但我不确定是可能的(但我也不经常使用 FOP))。为了让您的样式表正常工作,我添加了一个模板来输出 XSL-FO (match="/"),并将 svg 命名空间 (xmlns="http://www.w3.org/2000/svg") 添加到 xsl:stylesheet。请查看我的编辑。
  • 嗨丹尼尔 - 我喜欢你的例子,我有类似的要求 - 如何使用动态数据创建条形图和折线图?我正在寻找这样的图表peltiertech.com/images/2009-11/BarLine_ColumnLine.png 如果您可以重用相同的示例以动态方式填充数据并生成折线图,那将非常感激。谢谢。
  • @mark - 我的第一个示例是相当动态的,因为大多数值都在生成。例如,我可以添加更多 fruit 而不必对样式表进行任何更改。我建议手动创建一个示例 svg 图表,然后找出如何使用输入 xml 在样式表中生成该 svg。
猜你喜欢
  • 1970-01-01
  • 2022-11-25
  • 2019-08-26
  • 1970-01-01
  • 2011-02-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-06-15
相关资源
最近更新 更多