package com.ruoyi.common.utils; import com.itextpdf.text.*; import com.itextpdf.text.pdf.BaseFont; import com.itextpdf.text.pdf.PdfPCell; import com.itextpdf.text.pdf.PdfPTable; import com.itextpdf.text.pdf.PdfWriter; import org.apache.commons.codec.binary.Base64; import org.apache.commons.collections4.CollectionUtils; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletResponse; import java.lang.reflect.Method; import java.net.URL; import java.net.URLEncoder; import java.util.ArrayList; import java.util.List; /** * @Author: tjf * @Date: 2024/3/18 15:34 * @Describe: */ public class PDFUtil { private static final Logger logger = LogManager.getLogger(PDFUtil.class); /** * fontSize_normal : (正文字体大小11号). */ public static final float FONTSIZE_NORMAL = 11f; /** * fontSize_titile : (标题字体大小14号). */ public static final float FONTSIZE_TITILE = 14f; /** * FONTSIZE_COVER : (封面字体大小32号). */ public static final float FONTSIZE_COVER = 32f; /** * normalFont : (通用字体样式:宋体、11号). */ private static Font normalFont = null; /** * titleFont : (通用标题字体样式:宋体、14号、加粗). */ private static Font titleFont = null; /** * coverFont : (通用封面字体样式:宋体、28号、加粗). */ private static Font coverFont = null; /** * 标题是否居中,true-居中、false-默认居左 */ private static final Boolean titleCenter = true; /** * getBaseFont : (获取可以解析中文的字体:使用宋体).
* * @author * @return * @since JDK 1.8 */ public static BaseFont getBaseFontChinese() { try { // 宋体资源文件路径,可以从C://Windows//Fonts//simsun.ttc拷贝到相应目录下 URL path = PDFUtil.class.getResource("/config/fonts/simsun.ttc"); return BaseFont.createFont(path + ",0", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED); // 本地测试:使用windows自带的宋体文件 //return BaseFont.createFont("C://Windows//Fonts//simsun.ttc,0", BaseFont.IDENTITY_H, false); } catch (Exception e) { logger.error("设置字体样式失败", e); return null; } } /** * getNormalFont : (获取普通字体样式).
* * @author * @return * @since JDK 1.8 */ public static Font getNormalFont() { if (normalFont == null) { BaseFont bfChinese = getBaseFontChinese(); normalFont = new Font(bfChinese, FONTSIZE_NORMAL, Font.NORMAL); } return normalFont; } /** * getTitleFont : (获取标题通用字体).
* * @author * @return * @since JDK 1.8 */ public static Font getTitleFont() { if (titleFont == null) { BaseFont bfChinese = getBaseFontChinese(); titleFont = new Font(bfChinese, FONTSIZE_TITILE, Font.BOLD); } return titleFont; } /** * getTitleFont : (获取封面通用字体).
* * @author * @return * @since JDK 1.8 */ public static Font getCoverFontFont() { if (coverFont == null) { BaseFont bfChinese = getBaseFontChinese(); coverFont = new Font(bfChinese, FONTSIZE_COVER, Font.BOLD); } return coverFont; } /** * getfieldValue : (通过反射,根据方法名,执行方法,最终返回行结果的toString值).
* * @author * @param 方法执行的入参 * @param object 对象 * @param methodName 方法名 * @param args 方法执行参数 * @since JDK 1.8 */ private static String getfieldValue(T object, String methodName, Object... args) { try { Method method = object.getClass().getMethod(methodName); Object value = method.invoke(object, args); return value == null ? "" : value.toString(); } catch (Exception e) { logger.error("getfieldValue error", e); return ""; } } /** * analysisPicBase64Info : (解析base64图片信息).
* * @author * @param picBase64Info 图片base64信息,或前台echart通过调用getDataURL()方法获取的图片信息 * @return 图片经过base64解码后的信息 * @since JDK 1.8 */ public static Element analysisPicBase64Info(String picBase64Info) { if (StringUtils.isEmpty(picBase64Info)) { // 空段落 return new Paragraph(); } // 1.获取图片base64字符串信息:若入参是通过前台echarts调用getDataURL()方法获取的,则该字符串包含逗号,且则逗号后面的内容才是图片的信息 String pictureInfoStr = picBase64Info.indexOf(",") == -1 ? picBase64Info : picBase64Info.split(",")[1]; // 2.将图片信息进行base64解密 byte[] imgByte = Base64.decodeBase64(pictureInfoStr); // 对异常的数据进行处理 /** * .图片的原始表达ascii码范围是0-255, * .这里面有一些不可见的编码。然后为了图片正确传输才转成编码base64的0-63, * .当从base64转成byte时,byte的范围是[-128,127],那么此时就会可能产生负数,而负数不是在ascii的范围里,所以需要转换一下 */ for (int i = 0; i < imgByte.length; i++) { if (imgByte[i] < 0) { imgByte[i] += 256; } } try { return Image.getInstance(imgByte); } catch (Exception e) { logger.error("analysisPicBase64Info error", e); return new Paragraph(); } } /** * analysisPicBase64Info_batch : (批量解析base64加密的图片信息,生成Image对象).
* * @author * @param picBase64Infos 经过base64加密的图片信息 * @return * @since JDK 1.8 */ public static List analysisPicBase64Info_batch(List picBase64Infos) { List images = new ArrayList(); for (String li : picBase64Infos) { Element image = analysisPicBase64Info(li); images.add(image); } return images; } /** * createImage : (根据图片的base64加密文件创建pdf图片).
* * @author * @param picBase64Info base64加密后的图片信息(支持台echart通过调用getDataURL()方法获取的图片信息) * @param title 段落标题 * @param percentX 图片缩放比例X轴 * @param percentY 图片缩放比例Y轴 * @param titleCenter 标题是否居中,true-居中、false-默认居左 * @return 返回图片段落 * @since JDK 1.8 */ public static Paragraph createImageFromEncodeBase64(String picBase64Info, String title, float percentX, float percentY, boolean titleCenter) { // 1.获取图片 Element element = analysisPicBase64Info(picBase64Info); // 2.创建段落,并添加标题 Paragraph paragraph = new Paragraph(title, getTitleFont()); // 空行 paragraph.add(Chunk.NEWLINE); paragraph.add(Chunk.NEWLINE); if (!(element instanceof Image)) { // 图片解析失败 return paragraph; } Image image = (Image) element; // 3.设置图片缩放比例 image.scalePercent(percentX, percentY); // 4.图片放入该段落 paragraph.add(image); return paragraph; } /** * createImageFromEncodeBase64_batch : (批量创建).
* * @author * @param picBase64Infos 图片base64加密后的信息(支持台echart通过调用getDataURL()方法获取的图片信息) * @param titles 段落标题 * @param percentXs X轴缩放比例 * @param percentYs Y轴缩放比例 * @return * @since JDK 1.8 */ public static Paragraph createImageFromEncodeBase64_batch(List picBase64Infos, List titles, List percentXs, List percentYs) { Paragraph paragraphs = new Paragraph(); for (int i = 0; i <= picBase64Infos.size(); i++) { Paragraph imagePara = createImageFromEncodeBase64(picBase64Infos.get(i), titles.get(i), percentXs.get(i), percentYs.get(i), titleCenter); if (!imagePara.isEmpty()) { paragraphs.add(imagePara); // 空行 paragraphs.add(Chunk.NEWLINE); paragraphs.add(Chunk.NEWLINE); } } return paragraphs; } /** * createTable : (创建table段落).
* * @author * @param * @param list 构建table的数据 * @param title 该段落取的名字 * @param methodNames 需要调用的方法名,用来获取单元格数据。通常是某个属性的get方法 * @return * @since JDK 1.8 */ public static Paragraph createTable(List list, String title, String[] tableHead, List methodNames) { return createTable(list, FONTSIZE_NORMAL, FONTSIZE_TITILE, title, tableHead, methodNames, false); } /** * createTableByList : (创建table段落).
* * @author * @param * @param listData * @param normalFontSize 正文字体大小 * @param titleFontSize 标题字体大小 * @param title 段落名称 * @param methodNames 获取表格属性的方法名 * @param b * @return * @since JDK 1.8 */ public static Paragraph createTable(List listData, float normalFontSize, float titleFontSize, String title, String[] tableHead, List methodNames, boolean b) { // 1.创建一个段落 Paragraph paragraph = new Paragraph(title, getTitleFont()); // 空行 paragraph.add(Chunk.NEWLINE); paragraph.add(Chunk.NEWLINE); // 3.创建一个表格 PdfPTable table = new PdfPTable(methodNames.size());// 列数 paragraph.add(table); // 4.构造表头 for (String head : tableHead) { head = StringUtils.isEmpty(head) ? "" : head; PdfPCell cell = new PdfPCell(new Paragraph(head, getNormalFont())); cell.setBackgroundColor( new BaseColor(Integer.parseInt("124"), Integer.parseInt("185"), Integer.parseInt("252")));// 背景色 cell.setMinimumHeight(Float.parseFloat("15"));// 单元格最小高度 cell.setHorizontalAlignment(Element.ALIGN_CENTER);// 水平居中 table.addCell(cell); } if (CollectionUtils.isEmpty(listData)) { // 没有数据,添加一行空单元格,并返回 for (int i = 0; i < methodNames.size(); i++) { table.addCell(new Paragraph(" "));// 有一个空格,否则添加不了 } return paragraph; } // 5.构造table数据 for (T li : listData) { for(String name : methodNames) { String valueStr = getfieldValue(li, name); PdfPCell cell = new PdfPCell(new Paragraph(valueStr, getNormalFont())); cell.setHorizontalAlignment(Element.ALIGN_CENTER);// 水平居中 table.addCell(cell); } } // 5.返回 return paragraph; } /** * addToTable : (从段落中找到第一个table,向该table中追加数据).
* ().
* * @author * @param * @param paragraph * @param listData * @param methodNames * @since JDK 1.8 */ public static void addToTable(Paragraph paragraph, List listData, List methodNames) { for (Element ele : paragraph) { if (!(ele instanceof PdfPTable)) { // 不是table元素,直接跳过 continue; } // 找到第一个table元素 PdfPTable table = (PdfPTable) ele; for (T data : listData) { for (String name : methodNames) { String valueStr = getfieldValue(data, name); PdfPCell cell = new PdfPCell(new Paragraph(valueStr, getNormalFont())); cell.setHorizontalAlignment(Element.ALIGN_CENTER);// 水平居中 table.addCell(cell); } } break; } } /** * exportDocument : (生成并下载PDF文档).
* ().
* * @author * @param document 文档对象 * @param cover 封面:若不是null,则会先添加封面,并另起新页面添加段落 * @param paragraphs 需要组成PDF文件的段落 * @param response 请求的响应对象 * @param fileName 生成的文件名称,不需要加pdf后缀 * @since JDK 1.8 */ public static void exportDocument(Document document, Paragraph cover, List paragraphs, HttpServletResponse response, String fileName) { try (ServletOutputStream out = response.getOutputStream()) { response.setContentType("application/binary;charset=UTF-8"); response.setHeader("Content-Disposition", "attachment;fileName=" + URLEncoder.encode(fileName + ".pdf", "UTF-8")); PdfWriter.getInstance(document, out); // 打开文档 document.open(); if (cover != null) { document.add(cover); // 起新页面 document.newPage(); } StringBuilder errorMsg = new StringBuilder(); for (int i = 0; i < paragraphs.size(); i++) { try { // 将段落添加到文档 document.add(paragraphs.get(i)); // 空行 document.add(Chunk.NEWLINE); document.add(Chunk.NEWLINE); } catch (DocumentException e) { errorMsg.append("PDF文件生成出错,请检查第:").append(i).append("个段落"); } } if (!StringUtils.isEmpty(errorMsg.toString())) { logger.error(errorMsg); } // 关闭文档 document.close(); out.flush(); out.close(); } catch (Exception e) { logger.error("生成PDF文档并下载,出错:", e); } } /** * setDefaultIndentationLeft : (设置段落默认左边距).
* * @author * @param paragraph * @since JDK 1.8 */ public static void setDefaultIndentationLeft(Paragraph paragraph) { paragraph.setIndentationLeft(Float.parseFloat("30")); } /** * addBlankLine : (添加空行).
* * @author * @param paragraph 需要添加空行的段落 * @param lineNum 需要添加空行的个数 * @since JDK 1.8 */ public static void addBlankLine(Paragraph paragraph, int lineNum) { if (paragraph == null) { return; } for (int i = 0; i < lineNum; i++) { paragraph.add(Chunk.NEWLINE); } } }