java-使用 flying-saucer 通过 xhtml 生成 pdf 文档支持 css 和 图片

原创 2017年04月26日 20:26:34


java-使用 flying-saucer 通过 xhtml 生成 pdf 文档支持 css 和 图片


重要说明:
1.使用 xhtml 生成 pdf ,对于 xhtml 的语法要求非常严格;
2.中文支持,需要手动加载中文字体文件,还需要在 css 中指定 body 中指定字体,加载的字体文件和css中指定的字体不对应时,中文不显示;
3.图片路径,需要手动指定图片的相对路径地址,否则图片不显示;
4.路劲格式,使用 uri 路径格式,uri 格式不正确是生成的内容将是空,正确格式为(file:/D:/Temp/pdf/template/about.xhtml);


项目地址:
https://github.com/flyingsaucerproject
用的官网 v9.1.5 版本:
https://github.com/flyingsaucerproject/flyingsaucer/releases/tag/v9.1.5
源码下载:
https://github.com/flyingsaucerproject/flyingsaucer/archive/v9.1.5.zip


提示:在 eclipse 中打开时,建议使用 阿里云 的maven 私服,源码 20M 构建需要下载 20M 的依赖包;


演示内容:通过 xhtml 生成带有中文、图片的 pdf 文档;


一、添加 maven 依赖包
<!-- pdf 生成 -->
<dependency>
	<groupId>org.xhtmlrenderer</groupId>
	<artifactId>flying-saucer-pdf</artifactId>
	<version>9.1.5</version>
</dependency>	




二、junit 测试代码

import static org.junit.Assert.*;


import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.util.HashMap;
import java.util.Iterator;


import org.junit.Before;
import org.junit.Test;
import org.springframework.web.bind.annotation.InitBinder;
import org.w3c.dom.Document;
import org.xhtmlrenderer.pdf.ITextOutputDevice;
import org.xhtmlrenderer.pdf.ITextRenderer;
import org.xhtmlrenderer.pdf.ITextUserAgent;
import org.xhtmlrenderer.resource.XMLResource;
import org.xml.sax.InputSource;


import com.lowagie.text.pdf.BaseFont;
import com.sun.xml.internal.xsom.impl.scd.Iterators.Map;


/**
 *
 * @author xxj
 */
public class PdfTest {


	@Before
	public void Init() {
	}


	/**
	 * 指定 xhtml 路径形式
	 * 
	 */	
	@Test
	public void craetePdf() {
		String url="D:/Temp/pdf/template/about.xhtml";
		String pdf="D:/Temp/pdf/craetePdf.pdf";


		
		OutputStream os = null;
		try {
	        if (url.indexOf("://") == -1) {
	            // maybe it's a file
	            File f = new File(url);
	            if (f.exists()) {
	                url = f.toURI().toURL().toString();
	            }
	        }
	        System.err.println("xhtml 地址:"+url);
	        System.err.println("pdf 地址:"+url);
	        
			os = new FileOutputStream(pdf);


			/*
			 * standard approach ITextRenderer renderer = new ITextRenderer();
			 * 
			 * renderer.setDocument(url); renderer.layout();
			 * renderer.createPDF(os);
			 */


			ITextRenderer renderer = new ITextRenderer();
			ResourceLoaderUserAgent callback = new ResourceLoaderUserAgent(renderer.getOutputDevice());
			callback.setSharedContext(renderer.getSharedContext());
			renderer.getSharedContext().setUserAgentCallback(callback);


			Document doc = XMLResource.load(new InputSource(url)).getDocument();
			
			renderer.setDocument(doc, url);
			
			renderer.layout();
			renderer.createPDF(os);


			os.close();
			os = null;
		} catch(Exception ex){
			ex.printStackTrace();
		}
		finally {
			if (os != null) {
				try {
					os.close();
				} catch (IOException e) {
					// ignore
				}
			}
		}
	}


	/**
	 * 将 xhtml 转为 string 形式
	 * 
	 */
	@Test
	public void craetePdfByHtmlStr() {
		String url="D:/Temp/pdf/template/about.xhtml";
		String pdf="D:/Temp/pdf/craetePdfByHtmlStr.pdf";
		String font1 ="C:/Windows/Fonts/simsun.ttc";//宋体(对应css中的 属性 font-family: SimSun; /*宋体*/)


		String html = readFile(url);
		
		OutputStream os = null;
		try {
	        if (url.indexOf("://") == -1) {
	            File f = new File(url);
	            if (f.exists()) {
	                url = f.toURI().toURL().toString();
	            }
	        }
	        System.err.println("xhtml 地址:"+url);
	        System.err.println("pdf 地址:"+pdf);
	        
			os = new FileOutputStream(pdf);


			ITextRenderer renderer = new ITextRenderer();
			ResourceLoaderUserAgent callback = new ResourceLoaderUserAgent(renderer.getOutputDevice());
			renderer.getSharedContext().setUserAgentCallback(callback);
			callback.setSharedContext(renderer.getSharedContext());
			//添加中文字体
			renderer.getFontResolver().addFont(font1, BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
			
			
			renderer.setDocumentFromString(html,url);
			
			renderer.layout();
			renderer.createPDF(os);


			os.close();
			os = null;
		} catch(Exception ex){
			ex.printStackTrace();
		}
		finally {
			if (os != null) {
				try {
					os.close();
				} catch (IOException e) {
					// ignore
				}
			}
		}
	}
	/**
	 * 资源加载代理
	 *
	 */
	private static class ResourceLoaderUserAgent extends ITextUserAgent {
		public ResourceLoaderUserAgent(ITextOutputDevice outputDevice) {
			super(outputDevice);
		}


		protected InputStream resolveAndOpenStream(String uri) {
			InputStream is = super.resolveAndOpenStream(uri);
			System.out.println("加载资源文件: " + uri);
			return is;
		}
	}
	/**
	 * 将文件转为 字符串
	 * @param fileName
	 * @return
	 */
	public static String readFile(String fileName) {
        FileInputStream file = null;
        BufferedReader reader = null;
        InputStreamReader inputFileReader = null;
        String content = "";
        String tempString = null;
        try {
            file = new FileInputStream(fileName);
            inputFileReader = new InputStreamReader(file, "utf-8");
            reader = new BufferedReader(inputFileReader);
            // 一次读入一行,直到读入null为文件结束
            while ((tempString = reader.readLine()) != null) {
                content += tempString;
            }
            reader.close();
        } catch (IOException e) {
            e.printStackTrace();
            return null;
        } finally {
            if (reader != null) {
                try {
                    reader.close();
                } catch (IOException e1) {
                }
            }
        }
        return content;
    }
}




三、编写 xhtml 模板
目录结构:D:\Temp\pdf\template
template
|-images
|-+-flyingsaucer.png
|-about.xhtml


about.xhtml 文件内容
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html
    PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN"
    "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">


<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
  <head>
    <title>Flying Saucer: CSS List Support</title>
    <link rel="stylesheet" type="text/css" href="general.css" title="Style" media="screen" />
    <style>
      html {
        background-color: transparent;
      }


      body {
        background-color: white;
        border: 0px;
        margin: 0;
        padding: 15;
        text-align: center;
		/*font-family: Arial Unicode MS;*/
		font-family: SimSun; /*宋体,对应 simsun.ttc 字体文件*/
      }
      p {
        font-size: 10pt;
      }
    </style>
  </head>
  <body>
    <div style="border: 1px dotted blue; padding: 10px">
    <p id="fslogo">Flying Saucer</p>
    <img src="images/flyingsaucer.png" />
    <p id="pagebyline">Browser Application Demo</p>


    <p>An example of integrating Flying Saucer in a real application.</p>
    <p><em>(This is not a real web browser)</em></p>


    <p>Licensed under the GNU Lesser General Public License.</p>
    </div>
	<h1>hello word!</h1>
	<h1>你好,世界!</h1>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
	<div style="border:1px solid red;color:red;">
		<h1>你好,世界!</h1>
		<h2>你好,世界!</h2>
		<h3>你好,世界!</h3>
		<h4>你好,世界!</h4>
		<h5>你好,世界!</h5>
		<h6>你好,世界!</h6>		
	</div>
  </body>
</html>


版权声明:本文为博主原创文章,未经博主允许不得转载。

使用 itext、flying-saucer 实现html转PDF

itext-2.1.7高于此版本已经更新为AGPL授权,所有不建议使用。 所需jar包: 使用maven就简单多了,只需将一下配置加入poom.xml即可 org.xhtmlrenderer...
  • mhouwei62
  • mhouwei62
  • 2016年05月13日 13:48
  • 3892

iText和flying saucer结合生成pdf的技术

原博文地址 http://blog.csdn.net/shanliangliuxing/article/details/6833471 下面是我自己利用flying sau...
  • jimmy609
  • jimmy609
  • 2013年10月15日 12:40
  • 18030

基于iText和flying saucer结合freemark java生成pdf 范例

基于iText和flying saucer结合freemark生成pdf 范例
  • lemonzone2010
  • lemonzone2010
  • 2013年11月26日 16:38
  • 7670

flying saucer技术生成pdf文档

下面是我自己利用flying saucer技术生成pdf文档的实现代码: Servlet方式: html代码: [html] view plaincopyprint? %@ page lan...
  • ouyhong123
  • ouyhong123
  • 2015年03月19日 18:22
  • 8026

Flying-Saucer使用HTML或者FTL(Freemarker模板)生成PDF

PDF导出工具有itext,但是itext对中文支持不好,还有样式CSS支持也不好,使用IReport比较复杂,上手不太容易,怎么办? 幸好有Flying-Saucer这个项目,帮助我们解决了以...
  • hybaym
  • hybaym
  • 2016年03月17日 13:54
  • 5427

Flying Saucer 支持中文宋体win,Linux,直接使用

  • 2016年12月08日 21:30
  • 1.04MB
  • 下载

iText和flying saucer结合生成pdf的技术

下面是我自己利用flying saucer技术生成pdf文档的实现代码: Servlet方式: html代码: ...
  • shanliangliuxing
  • shanliangliuxing
  • 2011年09月29日 12:56
  • 22719

Freemarker+Flying sauser +Itext 整合生成PDF

  • 2011年09月21日 16:31
  • 7.42MB
  • 下载

Flying-Saucer jar包

  • 2012年06月30日 22:02
  • 3.92MB
  • 下载

利用Flying Saucer 和 java iText 实现HTMl转PDF

一直对java iText 感兴趣,今天终于找到了一篇好文章,记录一下: http://blog.lichengwu.cn/java/2010/10/27/using-flying-saucer-an...
  • baohuan_love
  • baohuan_love
  • 2013年11月05日 11:22
  • 2262
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:java-使用 flying-saucer 通过 xhtml 生成 pdf 文档支持 css 和 图片
举报原因:
原因补充:

(最多只允许输入30个字)