最近又用到HTML转PDF的功能,而且需要对PDF进行定制,需要页眉页脚的
经过调查最终还是感觉wkhtmltopdf比较好,中间还经历了使用Google浏览器直接转pdf的,但是google不支持定制,不能接受参数,以及又自带的内容,最后选择了wkhtmltopdf,功能比较强大,支持参数,中间又遇到svg代码导致转换不成功的问题,后来把svg代码单独提出来然后再进行引入解决了转换的问题,
关于在程序里面调用wkhtmltopdf
调用也很简单,就是通过程序调用cmd窗口进行执行命令,关键代码如下
private static String toPdfTool = "C:\\Program Files\\wkhtmltopdf\\bin\\wkhtmltopdf.exe";
/**
* html转pdf
* @param srcPath html路径,可以是硬盘上的路径,也可以是网络路径
* @param destPath pdf保存路径
* @param isCard
* @return 转换成功返回true
*/
public static boolean convert(String srcPath, String destPath, boolean isCard){
File file = new File(destPath);
File parent = file.getParentFile();
// 如果pdf保存路径不存在,则创建路径
if (!parent.exists()) {
parent.mkdirs();
}
StringBuilder cmd = new StringBuilder();
if (System.getProperty("os.name").indexOf("Windows") == -1) {
// 非windows 系统
toPdfTool = "/usr/local/bin/wkhtmltopdf";
}
cmd.append(toPdfTool);
cmd.append(" --margin-bottom 10mm");
cmd.append(" --margin-left 0mm");
cmd.append(" --margin-right 0mm");
//cmd.append(" --margin-top 0mm");
cmd.append(" --disable-smart-shrinking");
//cmd.append(" -–print-media-type");
//cmd.append(" --header-line");// 页眉下面的线
//cmd.append(" --header-center 这里是页眉这里是页眉这里是页眉这里是页眉 ");//页眉中间内容
//cmd.append(" --header-spacing 5 ");// (设置页眉和内容的距离,默认0)
//设置在中心位置的页脚内容
cmd.append(" --footer-center 答题纸页码:[topage]—[page] ");
cmd.append(" --header-right 页码:[topage]—[page] ");
cmd.append(" --header-font-name 微软雅黑 ");
cmd.append(" --footer-font-name 微软雅黑 ");
//cmd.append(" --footer-line");
// * 显示一条线在页脚内容上)
//cmd.append(" --footer-spacing 5 ");
// (设置页脚和内容的距离)
cmd.append(srcPath);
cmd.append(" ");
cmd.append(destPath);
boolean result = true;
try{
Process process = Runtime.getRuntime().exec(cmd.toString());
//记录进程缓存错误信息
final StringBuffer errorLog = new StringBuffer();
//处理InputStream的线程,可以共同的抽出来,然后线程使用线程池
HtmlToPdfInterceptor error = new HtmlToPdfInterceptor(process.getErrorStream());
HtmlToPdfInterceptor output = new HtmlToPdfInterceptor(process.getInputStream());
error.start();
output.start();
process.waitFor();
}catch(Exception e){
result = false;
e.printStackTrace();
}
return result;
}
wkhtmltopdf的参数就不多做解释了,官网或者别的博客都可以看到
还有一个获取执行进程的错误流的类
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
public class HtmlToPdfInterceptor extends Thread {
private InputStream is;
public HtmlToPdfInterceptor(InputStream is){
this.is = is;
}
//获取执行进程的错误流
@Override
public void run(){
try{
InputStreamReader isr = new InputStreamReader(is, "utf-8");
BufferedReader br = new BufferedReader(isr);
String line = null;
while ((line = br.readLine()) != null) {
System.out.println(line.toString()); //输出内容
}
}catch (IOException e){
e.printStackTrace();
}
}
}