一、 背景说明

在Eclipse环境下,开发JAVA代码操作PPT,支持对PPT模板的修改。包括修改文本标签、图表、表格。满足大多数软件生成PPT报告的要求,即先收工创建好模板,在程序中修改模板数据。

二、 开发环境搭建

下载jacob开源组件,解压后把jacob.jar添加到项目中。拷贝对应的dll文件到Path路径下。32位系统采用X86的dll, 64位系统采用X64的dll。

三、 实现合并PPT公共函数

import com.jacob.activeX.ActiveXComponent;
import com.jacob.com.Dispatch;
import com.jacob.com.Variant;

/**
 * 通过jacob组件调用COM接口完成PPT文件的合并。合并后图表数据不丢失,用户可正常手工修改。
 * 调用函数前将jacob.jar添加到项目中,同时将jcaob.dll拷贝到path路径下。
 * @author Elon
 *
 */
public class MergePPT
{
    /**
     * 合并多个PPT文件。要求输出文件和合并文件均已存在,不创建新文件。
     * @param outPutPPTPath 合并后输出的文件路径。
     * @param mergePPTPathList 依次追加合并的文件。
     */
    public synchronized static void merge(String outPutPPTPath, List<String> mergePPTPathList)
    {
        // 启动 office PowerPoint程序
        ActiveXComponent pptApp = new ActiveXComponent("PowerPoint.Application");
        Dispatch.put(pptApp, "Visible", new Variant(true));   

        Dispatch presentations = pptApp.getProperty("Presentations").toDispatch();  

        // 打开输出文件
        Dispatch outputPresentation = Dispatch.call(presentations, "Open", outPutPPTPath, false,
                false, true).toDispatch();

        // 循环添加合并文件
        for (String mergeFile : mergePPTPathList)
        {
            Dispatch mergePresentation = Dispatch.call(presentations, "Open", mergeFile, false,
                false, true).toDispatch();

            Dispatch mergeSildes = Dispatch.get(mergePresentation, "Slides").toDispatch();
            @SuppressWarnings("deprecation")
            int mergePageNum = Dispatch.get(mergeSildes, "Count").toInt();

            // 关闭合并文件
            Dispatch.call(mergePresentation, "Close");

            Dispatch outputSlides = Dispatch.call(outputPresentation, "Slides").toDispatch();
            @SuppressWarnings("deprecation")
            int outputPageNum = Dispatch.get(outputSlides, "Count").toInt();

            // 追加待合并文件内容到输出文件末尾
            Dispatch.call(outputSlides, "InsertFromFile", mergeFile, outputPageNum, 1, mergePageNum);
        }

        // 保存输出文件,关闭退出PowerPonit.
        Dispatch.call(outputPresentation, "Save");
        Dispatch.call(outputPresentation, "Close");
        Dispatch.call(pptApp, "Quit");
    }
}

四、 修改PPT中的文本、图表公共函数

/**
 * 柱状图、饼图值类型
 * @author Elon
 *
 */
public class GraphValue implements Serializable
{
    /**
     *
     */
    private static final long serialVersionUID = 4055958383254660746L;

    /**
     * 系列名称
     */
    private String serialName = "";

    /**
     * 系列的值
     */
    private ArrayList<String> valueList = new ArrayList<String>();

    public String getSerialName()
    {
        return serialName;
    }

    public void setSerialName(String serialName)
    {
        this.serialName = serialName;
    }

    public ArrayList<String> getValueList()
    {
        return valueList;
    }

    public void setValueList(ArrayList<String> valueList)
    {
        this.valueList = valueList;
    }

    @Override
    public String toString()
    {
        String str = "serialName:" + serialName + " valueList:";

        for (String value : valueList)
        {
            str += value + ";";
        }

        return str;
    }
}

/**
 *
 * 修改PPT图表、文本信息公共类定义。
 *
 * @author Elon
 * @version 1.0, 2015年12月16日
 */
public class WritePPTUtil
{
    /**
     * PPT文件路径
     */
    public String pptPath = "";

    /**
     * ppt对象
     */
    private XMLSlideShow ppt = null;

    /**
     * 当前页签
     */
    private XSLFSlide curSlide = null;

    /**
     * 打开ppt文件
     * @param pptPath ppt文件路径
     * @throws XmlException
     * @throws OpenXML4JException
     * @throws IOException 异常
     */
    public void open(String pptPath) throws OpenXML4JException, IOException, XmlException
    {
        this.pptPath = pptPath;
        XSLFSlideShow slideShow = new XSLFSlideShow(pptPath);
        ppt = new XMLSlideShow(slideShow.getPackage());
    }

    /**
     * 保存关闭文件
     * @throws IOException
     */
    public void close() throws IOException
    {
        FileOutputStream out = new FileOutputStream(pptPath + "_new.pptx");
        ppt.write(out);
        out.close();

    }

    /**
     * 选中指定的PPT页签
     * @param slideIndex 页签索引
     * @return true:操作成功;false:操作失败
     */
    public boolean selectSlide(int slideIndex)
    {
        if(ppt == null)
        {
            return false;
        }

        XSLFSlide[] slideArray = ppt.getSlides();
        if(slideArray.length < slideIndex)
        {
            return false;
        }

        curSlide = slideArray[slideIndex];
        return true;
    }

    /**
     * 修改文本框内容
     * @param shapeValue
     * @param value
     * @return
     */
    public boolean modifyTextBox(String shapeValue, String value)
    {
        XSLFShape shape = getShape(shapeValue, ShapeType.TEXT_BOX_TYPE);
        if (shape == null)
        {
            return false;
        }

        XSLFTextBox textBox = (XSLFTextBox)shape;
        textBox.setText(value);
        return true;
    }

    /**
     *
     * 修改图表内容。
     *
     * @param ShapeName 图表名称
     * @param valueList
     * @return
     */
    public boolean modifyBargraph(String ShapeName, ArrayList<GraphValue> valueList)
    {
        XSLFShape shape = getShape("", ShapeType.GRAPHIC_TYPE);
        XSLFGraphicFrame graphicShape = (XSLFGraphicFrame)shape;

        XSLFSheet sheet = graphicShape.getSheet();

        // 修改PPT图表内嵌的sheet的内容(待续)。

        return true;
    }

    /**
     * 获取PPT页签中指定的控件。
     * @param shapeValue
     * @param shapeType
     * @return
     */
    private XSLFShape getShape(String shapeValue, ShapeType shapeType)
    {
        if(curSlide == null)
        {
            return null;
        }

        String shapeName = "";

        XSLFShape[] shapeArray = curSlide.getShapes();
        for (int i = 0; i < shapeArray.length; ++i)
        {
            shapeName = shapeArray[i].getShapeName();
            XSLFShape shape = shapeArray[i];

            if ((shapeType == ShapeType.TEXT_BOX_TYPE)
                    && (shape instanceof XSLFTextBox))
            {
                XSLFTextBox textBox = (XSLFTextBox)shape;
                String value = textBox.getText();
                if(value.equals(shapeValue))
                {
                    return shapeArray[i];
                }
            }
            else if ((shapeType == ShapeType.GRAPHIC_TYPE)
                    && (shape instanceof XSLFGraphicFrame))
            {
                return shape;
            }
        }

        return null;
    }
}

五、 调用PPT合并函数测试代码

public class TestPPTMain
{
    public static void main(String[] args) throws OpenXML4JException, IOException, XmlException
    {
        String outPutPPTPath = "D:\\TEMP\\template\\1.pptx";

        List<String> mergePPTPathList = new ArrayList<String>();
        mergePPTPathList.add("D:\\TEMP\\template\\2.pptx");
        mergePPTPathList.add("D:\\TEMP\\template\\3.pptx");

        MergePPT.merge(outPutPPTPath, mergePPTPathList);;
    }
}

JAVA通过COM接口操作PPT的更多相关文章

  1. C++通过COM接口操作PPT

    一. 背景 在VS环境下,开发C++代码操作PPT,支持对PPT模板的修改.包括修改文本标签.图表.表格.满足大多数软件生成PPT报告的要求,先手工创建好PPT模板,在程序中修改模板数据. 二. 开发 ...

  2. Java使用Unsafe接口操作数组Demo

    public class unSafeArrayDemo { private static final sun.misc.Unsafe UNSAFE; private static final lon ...

  3. Java NIO Path接口和Files类配合操作文件

    Java NIO Path接口和Files类配合操作文件 @author ixenos Path接口 1.Path表示的是一个目录名序列,其后还可以跟着一个文件名,路径中第一个部件是根部件时就是绝对路 ...

  4. java poi 操作ppt

    java poi 操作ppt 可以参考: https://www.w3cschool.cn/apache_poi_ppt/apache_poi_ppt_installation.html http:/ ...

  5. java抽象类和接口区别

    深入理解Java的接口和抽象类 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太多不同的地方.很多人在初学的 ...

  6. 如何让Java和C++接口互相调用:JNI使用指南

    如何让Java和C++接口互相调用:JNI使用指南 转自:http://cn.cocos2d-x.org/article/index?type=cocos2d-x&url=/doc/cocos ...

  7. Java中的队列:java.util.Queue接口

    队列是一种特殊的线性表,它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作. Queue接口与List.Set同一级别,都是继承了Collection接口.Linked ...

  8. Java集合Map接口与Map.Entry学习

    Java集合Map接口与Map.Entry学习 Map接口不是Collection接口的继承.Map接口用于维护键/值对(key/value pairs).该接口描述了从不重复的键到值的映射. (1) ...

  9. java的List接口的实现类 ArrayList,LinkedList,Vector 的区别

    Java的List接口有3个实现类,分别是ArrayList.LinkedList.Vector,他们用于存放多个元素,维护元素的次序,而且允许元素重复. 3个具体实现类的区别如下: 1. Array ...

随机推荐

  1. PHP系统左侧菜单栏的管理与实现

    在日常的开发工作中,面对后台的日益增长的业务,以及后期业务的迭代开发,通常会选择添加菜单栏的形式来扩充业务功能,同样日益增长的后台菜单选项也为我们后期的维护,产生了一定的困难性.为此我总结出自己关于左 ...

  2. python进阶学习笔记(二)

    1.模块和包的概念 python的解决方案是把同名的模块放到不同的包中 1.1,导入模块 要使用一个模块,我们必须首先导入该模块.Python使用import语句导入一个模块.例如,导入系统自带的模块 ...

  3. ubuntu16.04编译安装mysql-boost-5.7.21并编译成php扩展测试与使用

    我之前的文章已经改造了自定义MVC框架中的工具类(验证码,图片上传,图像处理,分类)4个类,接下来,就要改造模型类,模型类肯定要连接数据库,由于我的Ubuntu Linux是裸装的php(目前只编译了 ...

  4. MongoDB 搭建可复制群集

    一.概述 MongoDB复制群集支持节点故障自动切换,最小配置应包含3个节点,正常情况下应该至少包含两个数据节点,第三个节点可以是数据节点也可以是仲裁节点.仲裁节点的作用是当出现偶数节点导致无法仲裁的 ...

  5. 端到端测试工具--testcafe

    写在前面 随着业务的增加,复杂性的增加,我们更需要保证页面不能出错,之前需要每次上线之前需要每次人工测试,如果有好多改动,为保证业务不出错,需要耗费更多的时间来测试,所以我们需要写一些测试来保证业务的 ...

  6. iOS程序闪退的原因以及处理办法

    iOS程序闪退是一种比较常见的现象.闪退的情况很多,造成程序闪退的原因也很多. ================================启动时闪退======================= ...

  7. 来了解一下Mysql索引的相关知识:基础概念、性能影响、索引类型、创建原则、注意事项

    索引的基础概念索引类似于书籍的目录,要想找到一本书的某个特定主题,需要先查找书的目录,定位对应的页码:存储引擎使用类似的方式进行数据查询,先去索引当中找到对应的值,然后根据匹配的索引找到对应的数据行 ...

  8. JS在线生成二维码

    Js代码 百度云公开下载地址:http://pan.baidu.com/s/1nvjTXB7 Html+Php代码 <volist name="huodong_list" i ...

  9. python数据可视化学习1

    import matplotlib.pyplot as plt input_values = [1,2,3,4,5] #输入值 squares = [1,4,9,16,25] #输出值 plt.plo ...

  10. lua 操作数据库

    操作mysql主要用到了lua-resty-mysql库,代码可以在github上找得到 而且上面也有实例代码 由于官网给出的例子比较基本,代码也比较多,所以我这里主要介绍一些怎么封装一下,简化我们调 ...