我們這里是maven項目,導入相應jar包:
<dependency>
<groupId>org.freemarker</groupId>
<artifactId>freemarker</artifactId>
<version>2.3.23</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.4.1</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itext-asian</artifactId>
<version>5.2.0</version>
</dependency>
<dependency>
<groupId>com.itextpdf.tool</groupId>
<artifactId>xmlworker</artifactId>
<version>5.4.1</version>
</dependency>
注意:建議 itextpdf 和 tool 包版本一致,不然有可能會出現未知錯誤(我遇到一次~)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8"/>
<title>Title</title>
<style>
body{
font-family:SimHei;
}
.red{
color: red;
}
</style>
</head>
<body>
<div class="red">
你好,${name}
</div>
</body>
</html>
package com.springdemo.job;
import java.io.File;
import java.io.IOException;
import com.itextpdf.text.Document;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.pdf.PdfWriter;
import com.itextpdf.tool.xml.XMLWorkerFontProvider;
import com.itextpdf.tool.xml.XMLWorkerHelper;
import freemarker.template.Configuration;
public class Test {
private static final String DEST = "H:\\test\\test.pdf";
private static final String HTML = "freemarker.html";
private static final String FONT = "simhei.ttf";
private static Configuration freemarkerCfg = null;
static {
freemarkerCfg = new Configuration();
// 獲取freemarker的模板目錄
try {
freemarkerCfg.setDirectoryForTemplateLoading(new File("H:\\test\\").getCanonicalFile());
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* html渲染為pdf
*
* @param data
* 變量
* @param htmlTmp
* 模板文件名
* @param pdftemp
* pdf導出路徑
* @return
*/
public static String freeMarkerRender(Map<String, Object> data, String htmlTmp, String pdftemp) {
// 獲取模板,並設置編碼方式
Template template = freemarkerCfg.getTemplate(htmlTmp);
template.setEncoding("UTF-8");
StringWriter out = new StringWriter();
// 合並模板跟數據
template.process(data, out);
// htmlData 模板字符流
String htmlData = out.toString();
// 設置文檔格式,數字邊距
Document document = new Document(PageSize.A4, 30, 30, 30, 30);
PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream(pdftemp));
// 添加頁碼
PDFBuilder builder = new PDFBuilder();
writer.setPageEvent(builder);
// 打開文檔
document.open();
XMLWorkerHelper.getInstance().parseXHtml(writer, document, new ByteArrayInputStream(htmlData.getBytes()), null,
new MyFontsProvider());
// 關閉文檔
document.close();
}
public static void main(String[] args) throws Exception {
Map<String, Object> data = new HashMap();
data.put("name", "test");
Test.freeMarkerRender(data, HTML, DEST);
}
}
PDF設置字符集
package com.springdemo.job;
/**
* 設置字符集
*/
public static class MyFontsProvider extends XMLWorkerFontProvider {
public MyFontsProvider(){
super(null, null);
}
@Override
public Font getFont(final String fontname, String encoding, float size, final int style) {
String fntnames = fontname;
Font FontChinese = null;
if (fntnames == null) {
fntnames = "宋體";
}
if (size == 0) {
size = 4;
}
try{
BaseFont bfChinese = BaseFont.createFont("STSong-Light",
"UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
FontChinese = new Font(bfChinese, 12, Font.NORMAL);
}catch (Exception e){
e.printStackTrace();
}
if(FontChinese==null){
FontChinese = super.getFont(fntnames, encoding, size, style);
}
return FontChinese;
}
}
設置自動分頁類
package com.springdemo.job;
import java.io.IOException;
import com.itextpdf.text.Document;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.Element;
import com.itextpdf.text.Font;
import com.itextpdf.text.PageSize;
import com.itextpdf.text.Phrase;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.BaseFont;
import com.itextpdf.text.pdf.ColumnText;
import com.itextpdf.text.pdf.PdfContentByte;
import com.itextpdf.text.pdf.PdfPageEventHelper;
import com.itextpdf.text.pdf.PdfTemplate;
import com.itextpdf.text.pdf.PdfWriter;
public class PDFBuilder extends PdfPageEventHelper {
/**
* 頁眉
*/
public String header = "";
/**
* 文檔字體大小,頁腳頁眉最好和文本大小一致
*/
public int presentFontSize = 12;
/**
* 文檔頁面大小,最好前面傳入,否則默認為A4紙張
*/
public Rectangle pageSize = PageSize.A4;
// 模板
public PdfTemplate total;
// 基礎字體對象
public BaseFont bf = null;
// 利用基礎字體生成的字體對象,一般用於生成中文文字
public Font fontDetail = null;
/**
*
* Creates a new instance of PdfReportM1HeaderFooter 無參構造方法.
*
*/
public PdfReportM1HeaderFooter() {
}
/**
*
* Creates a new instance of PdfReportM1HeaderFooter 構造方法.
*
* @param yeMei
* 頁眉字符串
* @param presentFontSize
* 數據體字體大小
* @param pageSize
* 頁面文檔大小,A4,A5,A6橫轉翻轉等Rectangle對象
*/
public PdfReportM1HeaderFooter(String yeMei, int presentFontSize, Rectangle pageSize) {
this.header = yeMei;
this.presentFontSize = presentFontSize;
this.pageSize = pageSize;
}
public void setHeader(String header) {
this.header = header;
}
public void setPresentFontSize(int presentFontSize) {
this.presentFontSize = presentFontSize;
}
/**
*
* TODO 文檔打開時創建模板
*
* @see com.itextpdf.text.pdf.PdfPageEventHelper#onOpenDocument(com.itextpdf.text.pdf.PdfWriter,
* com.itextpdf.text.Document)
*/
public void onOpenDocument(PdfWriter writer, Document document) {
total = writer.getDirectContent().createTemplate(50, 50);// 共 頁 的矩形的長寬高
}
/**
*
* TODO 關閉每頁的時候,寫入頁眉,寫入'第幾頁共'這幾個字。
*
* @see com.itextpdf.text.pdf.PdfPageEventHelper#onEndPage(com.itextpdf.text.pdf.PdfWriter,
* com.itextpdf.text.Document)
*/
public void onEndPage(PdfWriter writer, Document document) {
try {
if (bf == null) {
bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", false);
}
if (fontDetail == null) {
fontDetail = new Font(bf, presentFontSize, Font.NORMAL);// 數據體字體
}
} catch (DocumentException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
// 1.寫入頁眉
ColumnText.showTextAligned(writer.getDirectContent(), Element.ALIGN_LEFT, new Phrase(header, fontDetail),
document.left(), document.top() + 20, 0);
// 2.寫入前半部分的 第 X頁/共
int pageS = writer.getPageNumber();
String foot1 = "第 " + pageS + " 頁 /共";
Phrase footer = new Phrase(foot1, fontDetail);
// 3.計算前半部分的foot1的長度,后面好定位最后一部分的'Y頁'這倆字的x軸坐標,字體長度也要計算進去 = len
float len = bf.getWidthPoint(foot1, presentFontSize);
// 4.拿到當前的PdfContentByte
PdfContentByte cb = writer.getDirectContent();
// 5.寫入頁腳1,x軸就是(右margin+左margin + right() -left()- len)/2.0F
// 再給偏移20F適合人類視覺感受,否則肉眼看上去就太偏左了
// ,y軸就是底邊界-20,否則就貼邊重疊到數據體里了就不是頁腳了;注意Y軸是從下往上累加的,最上方的Top值是大於Bottom好幾百開外的。
ColumnText.showTextAligned(cb, Element.ALIGN_CENTER, footer,
(document.rightMargin() + document.right() + document.leftMargin() - document.left() - len) / 2.0F
+ 20F,
document.bottom() - 20, 0);
// 6.寫入頁腳2的模板(就是頁腳的Y頁這倆字)添加到文檔中,計算模板的和Y軸,X=(右邊界-左邊界 - 前半部分的len值)/2.0F +
// len , y 軸和之前的保持一致,底邊界-20
cb.addTemplate(total,
(document.rightMargin() + document.right() + document.leftMargin() - document.left()) / 2.0F + 20F,
document.bottom() - 20); // 調節模版顯示的位置
}
/**
*
* TODO 關閉文檔時,替換模板,完成整個頁眉頁腳組件
*
* @see com.itextpdf.text.pdf.PdfPageEventHelper#onCloseDocument(com.itextpdf.text.pdf.PdfWriter,
* com.itextpdf.text.Document)
*/
public void onCloseDocument(PdfWriter writer, Document document) {
// 7.最后一步了,就是關閉文檔的時候,將模板替換成實際的 Y 值,至此,page x of y 制作完畢,完美兼容各種文檔size。
total.beginText();
total.setFontAndSize(bf, presentFontSize);// 生成的模版的字體、顏色
String foot2 = " " + (writer.getPageNumber() - 1) + " 頁";
total.showText(foot2);// 模版顯示的內容
total.endText();
total.closePath();
}
}
第三步:
如果仔細度過那個超入門的朋友會知道設置表頭有一個固定表頭的功能,比如表格很長的時候PDF自動分頁了,原生代碼可以做到每一頁都顯示表頭。
實際上看了下iText的源碼發現HTML轉PDF的時候也是可以的,iText支持一個CSS屬性,只需要給你需要重復表頭表尾的table標簽設置css屬性“repeat-header:yes”或“repeat-footer:yes”即可,然后將你需要重復的表頭放在thead標簽內,表尾放在tfoot標簽內。
例:
<table style="repeat-header:yes;repeat-footer:yes;">
<thead>
<tr>
<th>如果表格過長自動分頁了,我是重復的表頭1</th>
</tr>
<tr>
<th>如果表格過長自動分頁了,我是重復的表頭2</th>
</tr>
</thead>
<tbody>
<tr>
<td>內容....</td>
</tr>
<!-- ..... -->
<tr>
<td>內容....</td>
</tr>
</tbody>
<tfoot>
<tr>
<th>如果表格過長自動分頁了,我是重復的表尾1</th>
</tr>
<tr>
<th>如果表格過長自動分頁了,我是重復的表尾2</th>
</tr>
</tfoot>
</table>
使用代碼寫pdf的時候,如果某個小章節完畢了,我們需要另起一頁直接new Page()即可。在HTML生成pdf的時候也是有特殊的css屬性來控制的,分別是page-break-after:always和page-break-before:always(還有一個page-break-inside屬性,但測試沒發現有什么特殊的效果)
<span style="page-break-after:always;">這段文字后面會重新分頁</span>
<p>正常的文字流,但是因為前面元素強制在后面分頁了,后面的文字也強制在前面分頁了,因此這段文字會是頂頭文字,也是這一頁的唯一的文字。</p>
<span style="page-break-before:always;">這段文字前面會重新分頁</span>
