文章目录
一、判断一个文件是否是pdf文件
pdf文件头部位置,包含着PDF版本信息,可以通过这个来判断:
@PostMapping(value = "/upload")
public void uploadFile(@RequestParam MultipartFile file) {
InputStream inputStream = null;
try {
inputStream = file.getInputStream();
if (file.getOriginalFilename().endsWith("pdf") || file.getOriginalFilename().endsWith("PDF")) {
// pdf检查
if (!checkPdfType(inputStream)) {
logger.info("pdf文件检查失败");
return result.failed().template(StandardI18nTemplate.MSG_Error).arguments("请上传合法的pdf文件");
}
// 流重置
inputStream.reset();
}
} catch (Exception e) {
logger.error("上传失败", e);
} finally {
if (inputStream != null) {
try {
inputStream.close();
} catch (IOException e) {
logger.error("关闭inputStream失败", e);
}
}
}
}
// 读取前几个字符,判断有没有PDF关键字
public boolean checkPdfType(InputStream inputStream) {
try {
byte[] bytes = new byte[20];
inputStream.read(bytes);
String string = new String(bytes);
if (!string.contains("PDF")) {
return false;
}
} catch (Exception e) {
logger.error("checkPdfType error", e);
return false;
}
return true;
}
二、多个pdf文件合并为一个
1、需要的包
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.13.3</version>
</dependency>
2、实现
import com.itextpdf.text.*;
import com.itextpdf.text.pdf.*;
public static void main(String[] args) {
List<String> pdfs = new ArrayList<>();
pdfs.add("E:\\test.pdf");
pdfs.add("E:\\test2.pdf");
mergePdf(pdfs, "E:\\merge.pdf");
}
/**
* 将多个pdf合并成一个pdf文件
*/
public static void mergePdf(List<String> pdfs, String mergeFilePath) {
List<PdfReader> readers = new ArrayList<>();
try {
// 创建一个新的文档
Document document = new Document();
// 创建一个PDF复制对象
PdfCopy copy = new PdfCopy(document, new FileOutputStream(mergeFilePath));
// 打开文档
document.open();
// 读取需要合并的PDF文件
for (String file : pdfs) {
readers.add(new PdfReader(file));
}
for (PdfReader reader : readers) {
// 将读取的PDF文件合并到新的文档中
copy.addDocument(reader);
}
// 关闭文档
document.close();
logger.info("PDF合并完成!");
} catch (Exception e) {
logger.error("PDF合并失败!", e);
} finally {
for (PdfReader reader : readers) {
if (reader != null) {
reader.close();
}
}
}
}
三、使用itextpdf5手绘pdf
1、需要的包
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.13.3</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itext-asian</artifactId>
<version>5.2.0</version>
</dependency>
官网:https://kb.itextpdf.com/home/it7kb/ebooks
2、代码
import com.itextpdf.text.*;
import com.itextpdf.text.pdf.BaseFont;
import com.itextpdf.text.pdf.PdfPCell;
import java.io.IOException;
public class PdfHelper {
// 宋体 汉字支持
public static BaseFont STSongLight;
static {
try {
STSongLight = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
} catch (DocumentException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 汉字支持,宋体支持
*/
public static BaseFont getSTSongLightBaseFont() {
return STSongLight;
}
public static PdfPCell getCell(String text, int colspan, int rowspan) {
Font font = new Font(STSongLight, 16);
Chunk chunk = new Chunk(text, font);
Paragraph paragraph = new Paragraph(chunk);
PdfPCell cell = new PdfPCell(paragraph);
cell.setHorizontalAlignment(Element.ALIGN_CENTER);
cell.setColspan(colspan);
cell.setRowspan(rowspan);
return cell;
}
}
import com.itextpdf.text.*;
import com.itextpdf.text.pdf.PdfPCell;
import com.itextpdf.text.pdf.PdfPTable;
import com.itextpdf.text.pdf.PdfWriter;
import java.io.FileOutputStream;
import java.io.IOException;
public class PdfTest {
public static void main(String[] args) throws IOException, DocumentException {
// 定义pdf大小为A4纸大小,并且上下左右边距都为50
Document document = new Document(PageSize.A4, 50, 50, 50, 50);
PdfWriter.getInstance(document, new FileOutputStream("G:\\test.pdf"));
document.open();
// 1、输出一个段落
document.add(new Paragraph("Hello World!"));
// 2、添加空行
document.add(Chunk.NEWLINE);
// 3、添加列表
// 字体可以设置 大小,加粗/倾斜下划线等,颜色 ,可以通过构造方法一行生成,也可以调用set方法
Font font3 = new Font(PdfHelper.getSTSongLightBaseFont(), 18f, Font.BOLD, new BaseColor(0, 0, 255));
// 文本 加上这个才会显示中文!!
Chunk chunk = new Chunk("列表输出:", font3);
// 段落
Paragraph paragraph3 = new Paragraph(chunk);
paragraph3.setFont(font3);// 字体
paragraph3.setPaddingTop(10);
paragraph3.setAlignment(Element.ALIGN_CENTER); // 居中
document.add(paragraph3);
// Create a List
List list3 = new List();
list3.setSymbolIndent(12);
list3.setListSymbol("\u2022"); // 列表符号
// Add ListItem objects 可以设置字体等
ListItem listItem3 = new ListItem("Never gonna give you up");
listItem3.setFont(font3);
list3.add(listItem3);
list3.add(new ListItem("Never gonna let you down"));
list3.add(new ListItem("Never gonna run around and desert you"));
list3.add(new ListItem("Never gonna make you cry"));
list3.add(new ListItem("Never gonna say goodbye"));
list3.add(new ListItem("Never gonna tell a lie and hurt you"));
// Add the list
document.add(list3);
document.add(Chunk.NEWLINE);
// 4.添加图片
// 段落
Paragraph paragraph4 = new Paragraph();
paragraph4.add(new Chunk("我是java搬运工,我会", new Font(PdfHelper.getSTSongLightBaseFont(), 18f)));
Image image4_1 = Image.getInstance("G:\\docker.png"); // 也可以传递网络图片
// image4_1.setAlignment(Element.ALIGN_CENTER);
image4_1.scaleToFit(50, 50); // 宽高限制
paragraph4.add(image4_1);
paragraph4.add(new Chunk("我还会", new Font(PdfHelper.getSTSongLightBaseFont(), 18f)));
Image image4_2 = Image.getInstance("G:\\mybatis.jpg"); // 也可以传递网络图片
// image4_2.setAlignment(Element.ALIGN_CENTER);
image4_2.scaleToFit(50, 50); // 宽高限制
paragraph4.add(image4_2);
document.add(paragraph4);
document.add(Chunk.NEWLINE);
// 5.表格
// 7列的表格
PdfPTable table5 = new PdfPTable(7);
// 每一列表格的宽度比例
float[] columnWidths = {3, 3, 2, 3, 2, 4, 3};
table5.setWidths(columnWidths);
table5.setWidthPercentage(95);// 表格占页面的比例
PdfPCell defaultCell5 = table5.getDefaultCell();
defaultCell5.setBorder(1); // 边框
defaultCell5.setPadding(2);
table5.addCell(PdfHelper.getCell("姓名", 1, 1));
table5.addCell(PdfHelper.getCell("张三", 1, 1));
table5.addCell(PdfHelper.getCell("性别", 1, 1));
table5.addCell(PdfHelper.getCell("男", 1, 1));
table5.addCell(PdfHelper.getCell("年龄", 1, 1));
table5.addCell(PdfHelper.getCell("18", 1, 1));
Image img = Image.getInstance("G:\\docker.png");
img.scaleToFit(83, 168);
PdfPCell imageCell = new PdfPCell(img);
imageCell.setBorderWidth(0);
imageCell.setHorizontalAlignment(Element.ALIGN_LEFT); // 设置单元格内容水平对齐方式
imageCell.setColspan(1);
imageCell.setRowspan(3);
imageCell.setPadding(2);
table5.addCell(imageCell);
document.add(table5);
// 关闭文档
document.close();
}
}
4、效果
四、根据pdf模板导出pdf
1、引包
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.13.3</version>
</dependency>
2、准备pdf模板
(1)我们使用wps打开原始PDF
注意,wps编辑表单需要会员,可以使用其他软件(Adobe Acrobat Pro或者福昕等)
(2)编辑表单
最终的结果:双击黑框可以编辑文字格式:
3、编码
import com.itextpdf.text.*;
import com.itextpdf.text.pdf.*;
import java.io.ByteArrayOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
/**
* Created by cxf on 2019/11/12.
* https://www.cnblogs.com/wangpeng00700/p/8418594.html
* 用Adobe Acrobat Pro创建表单
*/
public class ItextPdfUtils {
// 利用模板生成pdf
public static void pdfout(Map<String,Object> o) {
// 模板路径
String templatePath = "E:\\mytest.pdf";
// 生成的新文件路径
String newPDFPath = "E:\\testout1.pdf";
PdfReader reader;
FileOutputStream out;
ByteArrayOutputStream bos;
PdfStamper stamper;
try {
BaseFont bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
Font FontChinese = new Font(bf, 5, Font.NORMAL);
out = new FileOutputStream(newPDFPath);// 输出流
reader = new PdfReader(templatePath);// 读取pdf模板
bos = new ByteArrayOutputStream();
stamper = new PdfStamper(reader, bos);
AcroFields form = stamper.getAcroFields();
//文字类的内容处理
Map<String,String> datemap = (Map<String,String>)o.get("datemap");
form.addSubstitutionFont(bf);
for(String key : datemap.keySet()){
String value = datemap.get(key);
form.setField(key,value);
}
//图片类的内容处理
Map<String,String> imgmap = (Map<String,String>)o.get("imgmap");
for(String key : imgmap.keySet()) {
String value = imgmap.get(key);
String imgpath = value;
int pageNo = form.getFieldPositions(key).get(0).page;
Rectangle signRect = form.getFieldPositions(key).get(0).position;
float x = signRect.getLeft();
float y = signRect.getBottom();
//根据路径读取图片
Image image = Image.getInstance(imgpath);
//获取图片页面
PdfContentByte under = stamper.getOverContent(pageNo);
//图片大小自适应
image.scaleToFit(signRect.getWidth(), signRect.getHeight());
//添加图片
image.setAbsolutePosition(x, y);
under.addImage(image);
}
stamper.setFormFlattening(true);// 如果为false,生成的PDF文件可以编辑,如果为true,生成的PDF文件不可以编辑
stamper.close();
Document doc = new Document();
Font font = new Font(bf, 32);
PdfCopy copy = new PdfCopy(doc, out);
doc.open();
PdfImportedPage importPage = copy.getImportedPage(new PdfReader(bos.toByteArray()), 1);
copy.addPage(importPage);
doc.close();
} catch (IOException e) {
System.out.println(e);
} catch (DocumentException e) {
System.out.println(e);
}
}
public static void main(String[] args) {
Map<String,String> map = new HashMap();
map.put("name","张三");
map.put("age","永远18");
map.put("weather","晴朗");
Map<String,String> map2 = new HashMap();
map2.put("img","E:\\mybatis.jpg");
Map<String,Object> o=new HashMap();
o.put("datemap",map);
o.put("imgmap",map2);
pdfout(o);
}
}
五、html转pdf(不支持js)
1、导包
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>html2pdf</artifactId>
<version>3.0.2</version>
</dependency>
<!-- 中文字体支持 -->
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>font-asian</artifactId>
<version>7.1.13</version>
</dependency>
2、水印和页码
import com.itextpdf.kernel.colors.WebColors;
import com.itextpdf.kernel.events.Event;
import com.itextpdf.kernel.events.IEventHandler;
import com.itextpdf.kernel.events.PdfDocumentEvent;
import com.itextpdf.kernel.font.PdfFont;
import com.itextpdf.kernel.font.PdfFontFactory;
import com.itextpdf.kernel.geom.Rectangle;
import com.itextpdf.kernel.pdf.PdfDocument;
import com.itextpdf.kernel.pdf.PdfPage;
import com.itextpdf.kernel.pdf.canvas.PdfCanvas;
import com.itextpdf.layout.Canvas;
import com.itextpdf.layout.element.Paragraph;
import com.itextpdf.layout.property.TextAlignment;
import com.itextpdf.layout.property.VerticalAlignment;
import java.io.IOException;
/**
* 水印
*/
public class WaterMarkEventHandler implements IEventHandler {
/**
* 水印内容
*/
private String waterMarkContent;
/**
* 一页中有几列水印
*/
private int waterMarkX;
/**
* 一页中每列有多少水印
*/
private int waterMarkY;
public WaterMarkEventHandler(String waterMarkContent) {
this(waterMarkContent, 5, 5);
}
public WaterMarkEventHandler(String waterMarkContent, int waterMarkX, int waterMarkY) {
this.waterMarkContent = waterMarkContent;
this.waterMarkX = waterMarkX;
this.waterMarkY = waterMarkY;
}
@Override
public void handleEvent(Event event) {
PdfDocumentEvent documentEvent = (PdfDocumentEvent) event;
PdfDocument document = documentEvent.getDocument();
PdfPage page = documentEvent.getPage();
Rectangle pageSize = page.getPageSize();
PdfFont pdfFont = null;
try {
pdfFont = PdfFontFactory.createFont("STSongStd-Light", "UniGB-UCS2-H", false);
} catch (IOException e) {
e.printStackTrace();
}
PdfCanvas pdfCanvas = new PdfCanvas(page.newContentStreamAfter(), page.getResources(), document);
Paragraph waterMark = new Paragraph(waterMarkContent).setOpacity(0.5f);
Canvas canvas = new Canvas(pdfCanvas, pageSize)
.setFontColor(WebColors.getRGBColor("lightgray"))
.setFontSize(16)
.setFont(pdfFont);
for (int i = 0; i < waterMarkX; i++) {
for (int j = 0; j < waterMarkY; j++) {
canvas.showTextAligned(waterMark, (150 + i * 300), (160 + j * 150), document.getNumberOfPages(), TextAlignment.CENTER, VerticalAlignment.BOTTOM, 120);
}
}
canvas.close();
}
}
import com.itextpdf.kernel.events.Event;
import com.itextpdf.kernel.events.IEventHandler;
import com.itextpdf.kernel.events.PdfDocumentEvent;
import com.itextpdf.kernel.font.PdfFont;
import com.itextpdf.kernel.font.PdfFontFactory;
import com.itextpdf.kernel.geom.Rectangle;
import com.itextpdf.kernel.pdf.PdfDocument;
import com.itextpdf.kernel.pdf.PdfPage;
import com.itextpdf.kernel.pdf.canvas.PdfCanvas;
import com.itextpdf.layout.Canvas;
import com.itextpdf.layout.element.Paragraph;
import com.itextpdf.layout.property.TextAlignment;
import java.io.IOException;
/**
* 页码
*/
public class PageEventHandler implements IEventHandler {
@Override
public void handleEvent(Event event) {
PdfDocumentEvent documentEvent = (PdfDocumentEvent) event;
PdfDocument document = documentEvent.getDocument();
PdfPage page = documentEvent.getPage();
Rectangle pageSize = page.getPageSize();
PdfFont pdfFont = null;
try {
pdfFont = PdfFontFactory.createFont("STSongStd-Light", "UniGB-UCS2-H", false);
} catch (IOException e) {
e.printStackTrace();
}
PdfCanvas pdfCanvas = new PdfCanvas(page.getLastContentStream(), page.getResources(), document);
Canvas canvas = new Canvas(pdfCanvas, pageSize);
float x = (pageSize.getLeft() + pageSize.getRight()) / 2;
float y = pageSize.getBottom() + 15;
Paragraph paragraph = new Paragraph("第" + document.getPageNumber(page) + "页/共" + document.getNumberOfPages() + "页")
.setFontSize(10)
.setFont(pdfFont);
canvas.showTextAligned(paragraph, x, y, TextAlignment.CENTER);
canvas.close();
}
}
3、工具类
import com.itextpdf.html2pdf.ConverterProperties;
import com.itextpdf.html2pdf.HtmlConverter;
import com.itextpdf.io.font.PdfEncodings;
import com.itextpdf.kernel.events.PdfDocumentEvent;
import com.itextpdf.kernel.font.PdfFont;
import com.itextpdf.kernel.font.PdfFontFactory;
import com.itextpdf.kernel.geom.PageSize;
import com.itextpdf.kernel.pdf.PdfDocument;
import com.itextpdf.kernel.pdf.PdfWriter;
import com.itextpdf.layout.font.FontProvider;
import java.io.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Itext7转换工具类
*/
public class HtmlToPdfUtils {
/**
* html转pdf
*
* @param inputStream 输入流
* @param waterMark 水印
* @param fontPath 字体路径,ttc后缀的字体需要添加<b>,0<b/>
* @param outputStream 输出流
* @date : 2022/11/15 14:07
*/
public static void convertToPdf(InputStream inputStream, String waterMark, String fontPath, OutputStream outputStream) throws IOException {
PdfWriter pdfWriter = new PdfWriter(outputStream);
PdfDocument pdfDocument = new PdfDocument(pdfWriter);
//设置为A4大小
pdfDocument.setDefaultPageSize(PageSize.A4);
//添加水印
pdfDocument.addEventHandler(PdfDocumentEvent.END_PAGE, new WaterMarkEventHandler(waterMark));
//添加中文字体支持
ConverterProperties properties = new ConverterProperties();
FontProvider fontProvider = new FontProvider();
// 设置字体
/*PdfFont sysFont = PdfFontFactory.createFont("STSongStd-Light", "UniGB-UCS2-H", false);
fontProvider.addFont(sysFont.getFontProgram(), "UniGB-UCS2-H");*/
//添加自定义字体,例如微软雅黑
if (fontPath != null) {
PdfFont microsoft = PdfFontFactory.createFont(fontPath, PdfEncodings.IDENTITY_H, false);
fontProvider.addFont(microsoft.getFontProgram(), PdfEncodings.IDENTITY_H);
}
properties.setFontProvider(fontProvider);
// 读取Html文件流,查找出当中的 或出现类似的符号空格字符
inputStream = readInputStrem(inputStream);
if (inputStream != null) {
// 生成pdf文档
HtmlConverter.convertToPdf(inputStream, pdfDocument, properties);
pdfWriter.close();
pdfDocument.close();
return;
} else {
System.out.println(("转换失败!"));
}
}
/**
* 读取HTML 流文件,并查询当中的 或类似符号直接替换为空格
*
* @param inputStream
* @return
*/
private static InputStream readInputStrem(InputStream inputStream) {
// 定义一些特殊字符的正则表达式 如:
String regEx_special = "\\&[a-zA-Z]{1,10};";
try {
//<1>创建字节数组输出流,用来输出读取到的内容
ByteArrayOutputStream baos = new ByteArrayOutputStream();
//<2>创建缓存大小
byte[] buffer = new byte[1024]; // 1KB
//每次读取到内容的长度
int len = -1;
//<3>开始读取输入流中的内容
while ((len = inputStream.read(buffer)) != -1) { //当等于-1说明没有数据可以读取了
baos.write(buffer, 0, len); //把读取到的内容写到输出流中
}
//<4> 把字节数组转换为字符串
String content = baos.toString();
//<5>关闭输入流和输出流
// inputStream.close();
baos.close();
// log.info("读取的内容:{}", content);
// 判断HTML内容是否具有HTML的特殊字符标记
Pattern compile = Pattern.compile(regEx_special, Pattern.CASE_INSENSITIVE);
Matcher matcher = compile.matcher(content);
String replaceAll = matcher.replaceAll("");
// log.info("替换后的内容:{}", replaceAll);
// 将字符串转化为输入流返回
InputStream stringStream = getStringStream(replaceAll);
//<6>返回结果
return stringStream;
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
/**
* 将一个字符串转化为输入流
* @param sInputString 字符串
* @return
*/
public static InputStream getStringStream(String sInputString) {
if (sInputString != null && !sInputString.trim().equals("")) {
try {
ByteArrayInputStream tInputStringStream = new ByteArrayInputStream(sInputString.getBytes());
return tInputStringStream;
} catch (Exception e) {
e.printStackTrace();
}
}
return null;
}
}
4、测试
import java.io.*;
public class Test {
public static void main(String[] args) throws IOException {
long startTime = System.currentTimeMillis();
// html文件所在相对路径
String htmlFile = "E:\\pdf\\test.htm";
// pdf文件存储相对路径
String pdfFile = "E:\\pdf\\test.pdf";
// 自定义水印
String waterMarkText = "";
InputStream inputStream = new FileInputStream(htmlFile);
OutputStream outputStream = new FileOutputStream(pdfFile);
//微软雅黑在windows系统里的位置如下,linux系统直接拷贝该文件放在linux目录下即可
// String fontPath = "src/main/resources/font/STHeiti Light.ttc,0";
// windows字体都在这个目录下
String fontPath = "C:\\Windows\\Fonts\\simsun.ttc,0";
HtmlToPdfUtils.convertToPdf(inputStream, waterMarkText, fontPath, outputStream);
System.out.println(("转换结束,耗时:" + (System.currentTimeMillis() - startTime) + "ms"));
}
}
六、PDFBox操作pdf
https://www.yiibai.com/pdfbox
七、(有小坑)使用wkhtmltopdf导出包含js的html
0、参考资料
https://gitee.com/syxyw/html2file
1、wkhtmltopdf简介
灵活性:wkhtmltopdf支持多种输入格式,包括本地HTML文件、远程URL和HTML片段。它还提供了丰富的选项和参数,使用户能够自定 义生成的PDF文件的布局、样式和其他属性。
跨平台支持:wkhtmltopdf可在多个操作系统上运行,包括Windows、Linux和Mac OS X。这使得它成为开发人员和用户在不同平台上生成PDF文件的理想选择。
高质量输出:由于基于WebKit引擎,wkhtmltopdf能够准确地呈现HTML和CSS样式,生成高质量的PDF输出。它支持各种排版功能,如分页、页眉页脚、表格、图像等,并能处理复杂的CSS样式和布局。
扩展性:wkhtmltopdf可以通过插件和自定义脚本进行扩展。用户可以编写自定义的JavaScript脚本来处理页面上的元素,或者使用插件来添加额外的功能,如水印、页码等。
命令行界面:wkhtmltopdf提供了命令行界面,使用户能够轻松地将HTML文件或网页转换为PDF。这使得它可以与其他工具或脚本集成 ,并可以在自动化流程中使用。
总之,wkhtmltopdf是一个功能强大、灵活且易于使用的工具,可用于将HTML内容转换为高质量的PDF文档。无论是开发人员还是普通用户,都能够从中受益并满足各种需要。
关键时可以运行js,将js渲染的数据也生成在pdf中
2、安装
(1)官网
官网:https://wkhtmltopdf.org/
下载:https://wkhtmltopdf.org/downloads.html
需要根据操作系统,下载对应不同的安装包进行安装:
(2)windows系统安装
下载windows系统对应的版本,然后一直安装即可,记住我们的安装位置。
这两个exe,分别是html生成图片、html生成pdf的:
(3)linux
下载对应版本安装即可。
3、测试
(1)准备一个测试html模板(关闭动画)
<!DOCTYPE html>
<html lang="zh-CN" xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta charset="UTF-8"/>
<script src="https://cdn.jsdelivr.net/npm/echarts@5.5.0/dist/echarts.min.js"></script>
<style type="text/css">
body {
font-family: FangSong,HanaMinB;
line-height: 1.2;
/*设置背景色*/
/*background: #00FF00 ;*/
/*设置背景图片*/
/*background-image:url(data:image/gif;base64,AAAA) no-repeat fixed top;*/
}
/** 指定pdf纸张大小 **/
@page {
size: A4 ;
margin: 1cm;
/*margin-bottom: 1cm;*/
/*border: thin solid black;*/
}
</style>
</head>
<body>
<div style="text-align: center;">
<span style="font-size: 14px;">HelloWorld_1</span>
<br/>
<div id="main" style="width: 600px;height:400px;"></div>
<script type="text/javascript">
// 基于准备好的dom,初始化echarts实例
var myChart = echarts.init(document.getElementById('main'));
// 指定图表的配置项和数据
var option = {
title: {
text: 'ECharts 入门示例'
},
tooltip: {},
legend: {
data: ['销量']
},
xAxis: {
data: ['衬衫', '羊毛衫', '雪纺衫', '裤子', '高跟鞋', '袜子']
},
yAxis: {},
series: [
{
name: '销量',
type: 'bar',
data: [5, 20, 36, 10, 10, 20]
}
],
animation: false
};
// 使用刚指定的配置项和数据显示图表。
myChart.setOption(option);
</script>
</div>
</body>
</html>
(2)打开cmd
# 生成
.\wkhtmltopdf.exe E:\\test.html E:\\test.pdf
4、Java对接
(1)代码
java对接很简单,只需要调用一个cmd命令即可
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
/**
* 用于处理Runtime.getRuntime().exec产生的错误流及输出流
*/
public class StreamGobbler extends Thread {
/**
* 输入流
*/
private InputStream is;
/**
* 输出流
*/
private OutputStream os;
/**
* 流类型
*/
private String type;
public StreamGobbler(InputStream is, String type) {
this(is, type, null);
}
StreamGobbler(InputStream is, String type, OutputStream redirect) {
this.is = is;
this.type = type;
this.os = redirect;
}
@Override
public void run() {
InputStreamReader isr = null;
BufferedReader br = null;
PrintWriter pw = null;
try {
if (os != null) {
pw = new PrintWriter(os);
}
isr = new InputStreamReader(is);
br = new BufferedReader(isr);
String line = null;
while ((line = br.readLine()) != null) {
if (pw != null) {
pw.println(line);
}
System.out.println(type + ">" + line);
}
if (pw != null) {
pw.flush();
}
} catch (IOException ioe) {
ioe.printStackTrace();
} finally {
try {
if (pw != null) {
pw.close();
}
if (br != null) {
br.close();
}
if (isr != null) {
isr.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
/**
* cmd服务类
*/
public class CmdUtils {
/**
* 执行cmd命令
*
* @param cmdStr 命令字符串
* @return 成功失败
*/
public static boolean excute(String cmdStr) {
// 利用Runtime输出流读取
Runtime rt = Runtime.getRuntime();
try {
System.out.println("Command:" + cmdStr);
Process p = rt.exec(cmdStr);
StreamGobbler errorGobbler = new StreamGobbler(p.getErrorStream(),
"ERROR");
// 开启屏幕标准错误流
errorGobbler.start();
StreamGobbler outGobbler = new StreamGobbler(p.getInputStream(),
"STDOUT");
// 开启屏幕标准输出流
outGobbler.start();
int w = p.waitFor();
int v = p.exitValue();
if (w == 0 && v == 0) {
return true;
}
} catch (Exception e) {
e.printStackTrace();
return false;
}
return false;
}
public static void main(String[] args) {
boolean excute = excute("D:\\Program Files\\wkhtmltopdf\\bin\\wkhtmltopdf.exe E:\\\\test.html E:\\\\test.pdf");
System.out.println(excute);
}
}
(2)结果
5、(小坑)关于JS的支持
由于项目中需要画图表用到Echarts,一定要将echarts的动画
关掉才行!!!
其它方面的js没怎么试过,canvas应该也不太支持的,可以尝试一下。
八、使用jvppeteer抓取页面
https://gitee.com/fanyong920/jvppeteer
九、图片生成pdf
使用itext
public static void createPdfFromImage(String imagePath, String outputFileName) throws IOException, DocumentException {
try {
// 获取图片尺寸
Image image = Image.getInstance(imagePath);
Rectangle pageSize = new Rectangle(image.getWidth(), image.getHeight());
// 创建文档
Document document = new Document(pageSize);
PdfWriter.getInstance(document, Files.newOutputStream(Paths.get(outputFileName)));
document.open();
// 添加图片
image.setAbsolutePosition(0, 0);
image.scaleToFit(pageSize.getWidth(), pageSize.getHeight());
document.add(image);
document.close();
logger.info("success {}", outputFileName);
} catch (Exception e) {
logger.error(e);
}
}