生成PDF報表是不少企業系統常見的需求, 有些對外提供報表的系統還須要對生成的pdf文件添加水印, 本文將介紹以上2個問題簡單又免費的技術方案 ( 商業收費可見: 最新版ItextPdf )html
免費方案要用到的 第三方依賴有:前端
本文的方案基於spring boot開發, 簡化了許多thymleaf的配置, 固然你也能夠手動配置.vue
如下是maven依賴,java
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.11</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itext-asian</artifactId>
<version>5.2.0</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>com.itextpdf.tool</groupId>
<artifactId>xmlworker</artifactId>
<version>5.5.11</version>
</dependency>
複製代碼
或Grade依賴 :react
implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'
implementation 'com.itextpdf:itextpdf:5.5.11'
implementation 'com.itextpdf:itext-asian:5.2.0'
implementation 'com.itextpdf.tool:xmlworker:5.5.11'
複製代碼
html = html模板 + 參數
git
thymleaf的模板語法與大多數模板引擎的語法都比較相似, 對於前端框架如vue、angular、react的開發者只須要一點點查閱手冊就能夠迅速上手, 此處是 thymleaf模板語法文檔.spring
spring boot
引入thymleaf的方式很是簡單, 後文會提供必要的配置, 下方是使用thymleaf模板引擎生成html的代碼片斷:前端框架
public String generateHtml(String templateName, Map<String, Object> data) {
Context ctx = new Context();
ctx.setVariables(data);
return templateEngine.process(templateName, ctx);
}
複製代碼
其中: templateEngine
是org.thymeleaf.TemplateEngine
實例, 經過依賴注入取得.框架
templateName
是模板的名字, 與模板文件的名稱對應, 模板文件的路徑經過下面這段代碼片斷來配置, 這也是thymleaf所須要的全部配置.maven
@Configuration
public class ThymeleafConfiguration {
@Bean
public ClassLoaderTemplateResolver emailTemplateResolver() {
ClassLoaderTemplateResolver templateResolver = new ClassLoaderTemplateResolver();
templateResolver.setPrefix("templates/");
templateResolver.setTemplateMode("HTML5");
templateResolver.setSuffix(".html");
templateResolver.setCharacterEncoding("UTF-8");
templateResolver.setOrder(1);
return templateResolver;
}
}
複製代碼
這段配置會掃描 /src/main/resources/templates
路徑下, 後綴爲.html
的文件做爲html模板.
例如: String html = generateHtml("pdf/report", context);
則會將模板文件/src/main/resources/templates/pdf/reprot.html
與context
中的數據, 合成html內容.
pdf = render(html)
利用com.itextpdf.text.pdf.PdfWriter
將html 渲染成pdf, 下方的代碼片斷已是所有代碼.
private File render(String html) throws IOException, DocumentException {
FileOutputStream os = null;
try {
File outputFile = File.createTempFile("temp.pdf");
os = new FileOutputStream(outputFile);
Document document = new Document(PageSize.A4);
PdfWriter pdfWriter = PdfWriter.getInstance(document, os);
pdfWriter.setPageEvent(new Header());
document.setMargins(30, 30, 40, 50);
document.open();
InputStream htmlStream = new ByteArrayInputStream(html.getBytes("UTF-8"));
XMLWorkerHelper.getInstance().parseXHtml(
pdfWriter,
document,
htmlStream,
Charset.forName("UTF-8"),
fontProvider);
document.close();
return outputFile;
} finally {
if (os != null) {
try {
os.close();
} catch (IOException e) { /*ignore*/ }
}
}
}
private FontProvider fontProvider = new FontProvider() {
@Override
public boolean isRegistered(String s) {
return false;
}
@Override
public Font getFont(String fontFamily, String charset, boolean arg2, float size, int style, BaseColor color) {
BaseFont chinese = null;
try {
chinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
} catch (DocumentException | IOException e) {
e.printStackTrace();
}
return new Font(chinese, size, style, color);
}
};
複製代碼
值得注意的地方是, itextpdf對於中文支持不是十分友好, 若是不手動設置中文字體, 渲染出來的中文會變成空白.
上面的代碼片斷中fontProvider
展現了自定義字體的方法.
水印PDF = paint(原PDF)
public static File paint(File file) throws IOException, DocumentException {
PdfReader reader = new PdfReader(file.getPath());
File dest = File.createTempFile("withMask", "pdf", file.getParentFile());
PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(dest));
int n = reader.getNumberOfPages();
stamper.setRotateContents(false);
// text watermark
BaseFont chinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.EMBEDDED);
Font f = new Font(chinese, 80);
Font fontHeader = new Font(chinese, 10);
// transparency
PdfGState gs1 = new PdfGState();
gs1.setFillOpacity(0.1f);
// properties
PdfContentByte over;
Rectangle pagesize;
float x, y;
// loop over every page
for (int i = 1; i <= n; i++) {
pagesize = reader.getPageSize(i);
over = stamper.getOverContent(i);
over.saveState();
over.setGState(gs1);
x = (pagesize.getLeft() + pagesize.getRight()) / 2 - 25;
y = (pagesize.getTop() + pagesize.getBottom()) / 2 + 60;
Phrase p = new Phrase("水印文字", f);
ColumnText.showTextAligned(over, Element.ALIGN_CENTER, p, x, y, 45);
over.restoreState();
}
stamper.close();
reader.close();
return dest;
}
複製代碼