java实现word中插入附件(支持所有文件格式)

前言

通过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. 展示效果

相关推荐
Donald_brian2 小时前
线程同步
java·开发语言·jvm
全栈陈序员2 小时前
【Python】基础语法入门(十五)——标准库精选:提升效率的内置工具箱
开发语言·人工智能·python·学习
全靠bug跑2 小时前
Nacos 入门实战:部署、服务注册与发现全指南
java·spring cloud·docker·nacos
郑州光合科技余经理2 小时前
技术视角:海外版一站式同城生活服务平台源码解析
java·开发语言·uni-app·php·排序算法·objective-c·生活
喵了meme2 小时前
Linux学习日记19:线程同步与互斥锁
java·jvm·学习
郑州光合科技余经理2 小时前
海外版生活服务系统源码 | 外卖+跑腿一站式平台技术解析
java·开发语言·javascript·git·spring cloud·php·生活
小小Fred2 小时前
Cortex-M3 LR寄存器的特殊值EXC_RETURN
java·开发语言·jvm
小小心愿家2 小时前
线程——对于锁的进一步认识
java·开发语言
曹牧2 小时前
Java: FATAL ERROR: processing of -javaagent failed
java·开发语言