前言
通过java在word文档中嵌入附件,需使用ole方式嵌入,包含图标生成,编写demo快速实现。
实现demo代码如下:
1. 引入依赖
xml
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>5.2.2</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>5.2.2</version>
</dependency>
2. 组装附件图标(图标+附件名称)
java
package com.example.demo.example;
import com.example.demo.AttachmentIconEnum;
import org.apache.commons.codec.binary.Base64;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;
public class AttachmentIconUtils {
/**
* @Description //TODO 获取附件图标
* @Date 2025/12/12 16:35
* @Param [fileName]
* @return byte[]
**/
public static byte[] getAttachmentIconByte(String fileName) {
String suffix = "";
if(fileName.indexOf(".") != -1){
suffix = fileName.split("\\.")[1];
}
String base64Icon;
switch(suffix){
case "doc":
case "docx":
base64Icon = AttachmentIconEnum.DOCX.getIcon();
break;
case "xls":
case "xlsx":
base64Icon = AttachmentIconEnum.XLSX.getIcon();
break;
default:
base64Icon = AttachmentIconEnum.DEFAULT.getIcon();
break;
}
String encodingPrefix = "base64,";
if(base64Icon.contains(encodingPrefix)) {
int contentStartIndex = base64Icon.indexOf(encodingPrefix) + encodingPrefix.length();
base64Icon = base64Icon.substring(contentStartIndex);
}
byte[] templateIconBytes = Base64.decodeBase64(base64Icon);
try {
return makeAttachmentIcon(templateIconBytes,fileName);
}catch (Exception e){
return templateIconBytes;
}
}
/**
* @Description //TODO 制作附件图标
* @Date 2025/12/12 16:35
* @Param [templateIconBytes, fileName]
* @return byte[]
**/
public static byte[] makeAttachmentIcon(byte[] templateIconBytes,String fileName) throws Exception {
// 读取要插入的图片
try(InputStream inputStream = new ByteArrayInputStream(templateIconBytes);
ByteArrayOutputStream baos = new ByteArrayOutputStream()){
BufferedImage overlayImage = ImageIO.read(inputStream);
// 计算新图片尺寸:宽度 = 插入图宽度,高度 = 插入图高度 + 文字区域(60px)
int width = overlayImage.getWidth() + 60;
int textHeight = 60;
int height = overlayImage.getHeight() + textHeight;
// 创建带白色背景的新图片(TYPE_INT_RGB 不含透明通道,背景为白)
BufferedImage resultImage = new BufferedImage(120, 120, BufferedImage.TYPE_INT_RGB);
Graphics2D g2d = resultImage.createGraphics();
// 设置白色背景
g2d.setColor(Color.WHITE);
g2d.fillRect(0, 0, width, height);
// 绘制插入的图片(顶部对齐)
g2d.drawImage(overlayImage, 30, 10, null);
// 设置文字样式
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g2d.setColor(Color.BLACK);
g2d.setFont(new Font("Microsoft YaHei", Font.PLAIN, 18)); // 支持中文
// 计算文字居中位置
FontMetrics fm = g2d.getFontMetrics();
int textWidth = fm.stringWidth(fileName);
int x = (width - textWidth) / 2;
int y = overlayImage.getHeight() + ((textHeight - fm.getHeight()) / 2) + fm.getAscent();
// 绘制文字
g2d.drawString(fileName, x, y);
// 释放资源
g2d.dispose();
ImageIO.write(resultImage, "png", baos);
return baos.toByteArray();
}
}
}
3. 默认附件图标
java
public enum AttachmentIconEnum {
DOCX(""),
XLSX(""),
DEFAULT("");
private String icon;
AttachmentIconEnum(String icon) {
this.icon = icon;
}
public String getIcon() {
return icon;
}
}
4. 组装Ole对象,插入附件到word文档中
java
import com.example.demo.example.AttachmentIconUtils;
import org.apache.poi.ooxml.POIXMLTypeLoader;
import org.apache.poi.ooxml.util.DocumentHelper;
import org.apache.poi.openxml4j.opc.*;
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
import org.apache.poi.xwpf.usermodel.*;
import org.apache.xmlbeans.XmlObject;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.CTR;
import org.w3c.dom.Document;
import org.xml.sax.InputSource;
import java.io.*;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.charset.StandardCharsets;
import java.util.UUID;
import java.util.concurrent.ThreadLocalRandom;
public class AttachmentUtil {
public static String attachmentFilePath = "C:\\Users\\PC\\Desktop\\test.xlsx";
public static void main(String[] args) throws Exception {
XWPFDocument doc = new XWPFDocument();
// 创建段落
XWPFParagraph p1 = doc.createParagraph();
XWPFRun run1 = p1.createRun();
// 设置超链接文本
run1.setText("插入附件演示:");
run1.setColor("0000FF");
XWPFParagraph p2 = doc.createParagraph();
XWPFRun run2 = p2.createRun();
CTR ctr = run2.getCTR();
File attachmentFile = new File(attachmentFilePath);
//组装附件显示的图标
byte[] attachmentIconByte = AttachmentIconUtils.getAttachmentIconByte(attachmentFile.getName());
String imageRId = doc.addPictureData(attachmentIconByte,XWPFDocument.PICTURE_TYPE_PNG);
//插入并组装附件结构
String uuidRandom = UUID.randomUUID().toString().replace("-", "") + ThreadLocalRandom.current().nextInt(1024);
String shapeId = "_x0000_i20" + uuidRandom;
byte[] attachmentData = getAttachmentData(attachmentFile);
String programId = "Package";
String partNamePath = "/word/embeddings/oleObject1.bin";
PackagePartName partName = PackagingURIHelper.createPartName(partNamePath);
PackagePart packagePart = doc.getPackage().createPart(partName, "application/vnd.openxmlformats-officedocument.oleObject");
try(OutputStream out = packagePart.getOutputStream()){
out.write(attachmentData);
}
PackageRelationship ole = doc.getPackagePart().addRelationship(
partName,
TargetMode.INTERNAL,
"http://schemas.openxmlformats.org/officeDocument/2006/relationships/oleObject");
String embeddId = ole.getId();
String wObjectXml = "<w:object xmlns:w=\"http://schemas.openxmlformats.org/wordprocessingml/2006/main\" xmlns:v=\"urn:schemas-microsoft-com:vml\" xmlns:o=\"urn:schemas-microsoft-com:office:office\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\" w:dxaOrig=\"1520\" w:dyaOrig=\"960\"><v:shape id=\"" + shapeId + "\" o:spt=\"75\" type=\"#_x0000_t75\" style=\"height:48.0pt;width:48.0pt;\" o:ole=\"t\" filled=\"f\" o:preferrelative=\"t\" stroked=\"f\" coordsize=\"21600,21600\"><v:fill on=\"f\" focussize=\"0,0\"/><v:stroke on=\"f\"/><v:imagedata r:id=\"" + imageRId + "\" o:title=\"\"/><o:lock v:ext=\"edit\" aspectratio=\"t\"/></v:shape><o:OLEObject Type=\"Embed\" ProgID=\"" + programId + "\" ShapeID=\"" + shapeId + "\" DrawAspect=\"Icon\" ObjectID=\"" + shapeId + "\" r:id=\"" + embeddId + "\" /></w:object>";
Document document = DocumentHelper.readDocument(new InputSource(new StringReader(wObjectXml)));
ctr.set((XmlObject)XmlObject.Factory.parse(document.getDocumentElement(), POIXMLTypeLoader.DEFAULT_XML_OPTIONS));
//写入文件(输出到桌面)
String desktopPath = System.getProperty("user.home") + File.separator + "Desktop/test";
String outputPath = desktopPath + File.separator + "word_test_" + System.currentTimeMillis() + ".docx";
try(FileOutputStream fileOutputStream = new FileOutputStream(outputPath)){
doc.write(fileOutputStream);
}
System.out.println("文档已生成:" + outputPath);
doc.close();
}
private static byte[] getAttachmentData(File attachmentFile) throws Exception {
String templatePath = System.getProperty("user.dir") + "\\oleObject1.bin";
String Ole10NativeKey = "\u0001Ole10Native";
POIFSFileSystem fs = new POIFSFileSystem(new FileInputStream(templatePath));
byte[] newData = createOle10Native(attachmentFile);
fs.getRoot().getEntry(Ole10NativeKey).delete();
try(ByteArrayInputStream bis = new ByteArrayInputStream(newData)) {
fs.createOrUpdateDocument(bis,Ole10NativeKey);
}
try(ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
fs.writeFilesystem(baos);
return baos.toByteArray();
}
}
private static byte[] createOle10Native(File attachmentFile) throws IOException {
String filePath = attachmentFile.getPath();
String fileName = attachmentFile.getName();
String tempFilePath = System.getProperty("java.io.tmpdir") + "\\" + fileName;
FileInputStream fileInputStream = new FileInputStream(filePath);
byte[] fileByte = fileInputStream.readAllBytes();
int size = 33 + fileName.length() + filePath.length() + tempFilePath.length() + fileByte.length + 2 * fileName.length() + 2 * filePath.length() + 2 * tempFilePath.length();
ByteBuffer buffer = ByteBuffer.allocate(size).order(ByteOrder.LITTLE_ENDIAN);
// Version
buffer.putInt(260); // Set to 0x00000101
// Flags (usually 0)
buffer.putShort((short)2);
// fileName
byte[] fileNameBytes = fileName.getBytes(StandardCharsets.ISO_8859_1);
buffer.put(fileNameBytes);
buffer.put((byte)0);
// filePath
byte[] filePathBytes = filePath.getBytes(StandardCharsets.ISO_8859_1);
buffer.put(filePathBytes);
byte[] bytes = new byte[]{0,0,0,3,0};
buffer.put(bytes);
// tempFilePath
buffer.putInt(tempFilePath.length() + 1);
byte[] temPathBytes = tempFilePath.getBytes(StandardCharsets.ISO_8859_1);
buffer.put(temPathBytes);
buffer.put((byte) 0);
// file length
buffer.putInt(fileByte.length);
// File content
buffer.put(fileByte);
// 文件临时路径2长度
buffer.putInt(tempFilePath.length());
// 文件临时路径2
for(int i = 0; i < tempFilePath.length(); i++) {
buffer.put(temPathBytes[i]);
buffer.put((byte)0);
}
// 文件名称2长度
buffer.putInt(fileName.length());
// 文件名称2
for(int i = 0; i < fileName.length(); i++) {
buffer.put(fileNameBytes[i]);
buffer.put((byte)0);
}
// 文件路径2长度
buffer.putInt(filePath.length());
// 文件路径2
for(int i = 0; i < filePath.length(); i++) {
buffer.put(filePathBytes[i]);
buffer.put((byte)0);
}
return buffer.array();
}
}
5. 指定ole对象模板,执行main代码时,需要获取ole模板对象组装新的ole对象,模板下载,可在文章顶部下载即可
6. 展示效果
