最近有一个ppt操作的需求,因此查了下相关的资料
ppt分类
(1)2007版之前的
是基于二进制的文件格式
细节没有完全公开,第三方厂商多是用单向工程方法猜测和分析出来的。WPS做得好一些,但开源的只有做得很差的LibreOffice(原OpenOffice)
(2)2007版以后的
是基于OOXML开放文档规范的,本质是一个ZIP包,压缩了XML文档和相关资源。
OOXML是是一种简洁、可靠的文件格式,这类格式可以更好地实现文档与后端系统之间的数据集成。
java工具
(1)Apache POI
是Apache软件基金会的开放源码函式库,POI提供API给Java程序
(2)Aspose.Slides
是一款处理pptx的商业软件
(3)Jacob
Java-COM Bridge
在Java与微软的COM组件之间构建一座桥梁。使用Jacob自带的DLL动态链接库,并通过JNI的方式实现了在Java平台上对COM程序的调用。
MSWordManager 类,是jacob官方发布的工具类,里面有大多数Java操作MS Office的工具。
在使用Jacob时,很重要的一点是,用户本地系统中必须安装有Word的应用程序。否则也就无法建立Java-COM桥,进而无法解析了。
因此决定试水 Apache POI
Apache POI 使用
1.下载http://poi.apache.org/download
2.修改CLASSPATH
解压下载的包,将下面5个jar的完整路径添加到CLASSPATH
4.介绍
在POI API中
PowerPoint PPT——格式为 HSLF 对应poi-scratchpad
PowerPoint PPTX 格式为 XSLF 对应 poi-ooxml
类型
格式
对应的包
Powerpoint '97(-2007)PPT
HSLF
poi-scratchpad-XXX.jar
PowerPoint 2007 PPTX
XSLF
poi-ooxml-XXX.jar
3使用
下面是基于pptx的
(1)生成空白文档
DealDocument.java
import org.apache.poi.xslf.usermodel.XMLSlideShow;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
public class DealDocument {
protected String path = "data";
/**
* 创建文档
*
* @param name
*/
public void createDocument(String name) {
XMLSlideShow ppt = new XMLSlideShow();
String fileName = this.path + File.separator + name;
File file = new File(fileName);
try {
FileOutputStream out = new FileOutputStream(file);
ppt.write(out);
System.out.println("Presentation created successfully");
out.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
Main.java
import java.io.*;
import java.net.URL;
import java.util.List;
import org.apache.poi.*;
public class Main {
public static void main(String[] args) throws IOException{
System.out.println("Hello World!");
DealDocument dd = new DealDocument();
dd.createDocument("example1.pptx");
}
}
输出
Hello World!
Presentation created successfully
并且在项目根目录的data下生成example1.pptx
说明:
如果出现错误
Exception in thread "main" java.lang.IncompatibleClassChangeError: Found class org.apache.poi.util.POILogger, but interface was expected
这个错误最可能的原因是poi的jar包使用了多个版本
使用下面的方法可以查看具体使用的哪里的jar包
(在网上找的,但是版本不同,第三段 org.apache.poi.hslf.HSLFSlideShow.class.getClassLoader()没有找到相关的类,将poi-scratchpad-4.1.0.jar反编译,找了个存在的类,版本是4.1.0)
ClassLoader classloader = org.apache.poi.poifs.filesystem.POIFSFileSystem.class.getClassLoader();
URL res = classloader.getResource("org/apache/poi/poifs/filesystem/POIFSFileSystem.class");
String path = res.getPath();
System.out.println("POI Core came from " + path);
classloader = org.apache.poi.POIDocument.class.getClassLoader();
res = classloader.getResource("org/apache/poi/POIDocument.class");
path = res.getPath();
System.out.println("POI OOXML came from " + path);
classloader=org.apache.poi.hdgf.HDGFDiagram.class.getClassLoader();
res = classloader.getResource("org/apache/poi/hdgf/HDGFDiagram.class");
path = res.getPath();
System.out.println("POI Scratchpad came from " + path);
输出
POI Core came from file:/C:/Program%20Files/Java/jdk1.8.0_151/jre/lib/ext/poi-3.17.jar!/org/apache/poi/poifs/filesystem/POIFSFileSystem.class
POI OOXML came from file:/C:/Program%20Files/Java/jdk1.8.0_151/jre/lib/ext/poi-3.17.jar!/org/apache/poi/POIDocument.class
POI Scratchpad came from file:/E:/java/project/ppt/ppttest/lib/poi-scratchpad-4.1.0.jar!/org/apache/poi/hdgf/HDGFDiagram.class
发现确实有一个class走了老包,想起了在对mpp文件进行读写时添加了poi-3.17.jar,因此把相关文件删掉,问题解决
(2)在已有文档添加空白页
在 DealDocument.java 里
在文件 data/tpl1.pptx,追加两个空白页
public void addNewSlide(String name){
String fileName = this.path + File.separator + name;
File file = new File(fileName);
try {
//opening an existing slide show
FileInputStream in = new FileInputStream(file);
XMLSlideShow ppt = new XMLSlideShow(in);
//adding slides to the slodeshow
XSLFSlide slide1= ppt.createSlide();
XSLFSlide slide2 = ppt.createSlide();
//saving the changes
FileOutputStream out = new FileOutputStream(file);
ppt.write(out);
System.out.println("Presentation edited successfully");
out.close();
}catch (IOException e) {
e.printStackTrace();
}
}
在mian里的使用
DealDocument dd = new DealDocument();
dd.addNewSlide("tpl1.pptx");
执行完后,输出
Hello World!
Presentation edited successfully
文件里多了两个空白页
(3)将PPT中幻灯片转成图片
import org.apache.poi.xslf.usermodel.*;
import java.awt.*;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
public class ImageConvert {
public void converter(String fileName) {
BackRS rs = new BackRS();
File file = new File(fileName);
String name = file.getName();
String filePath = file.getParent();
try {
FileInputStream in = new FileInputStream(file);
XMLSlideShow ppt = new XMLSlideShow(in);
Dimension pgsize = ppt.getPageSize();
String saveImagePathName = filePath + File.separator + getFileNameNoEx(name)+"_JPG";
File path = new File(saveImagePathName);
if (!path.exists()) {
path.mkdir();
}
Integer i = 0;
BufferedImage img =null;
for (XSLFSlide slide : ppt.getSlides()) {
//解决乱码问题
for(XSLFShape shape : slide.getShapes()){
if(shape instanceof XSLFTextShape) {
XSLFTextShape tsh = (XSLFTextShape)shape;
for(XSLFTextParagraph p : tsh){
for(XSLFTextRun r : p){
r.setFontFamily("宋体");
}
}
}
}
img = new BufferedImage(pgsize.width, pgsize.height, BufferedImage.TYPE_INT_RGB);
Graphics2D graphics = img.createGraphics();
graphics.setPaint(Color.white);
graphics.fill(new Rectangle2D.Float(0, 0, pgsize.width, pgsize.height));
slide.draw(graphics);
FileOutputStream out = new FileOutputStream(path + File.separator + (i+1) + ".JPG");
i++;
javax.imageio.ImageIO.write(img, "PNG", out);
out.close();
}
System.out.println("Image successfully converted.");
} catch (Exception e) {
System.out.println("error:"+e.getMessage());
}
}
/**
* 获取文件名,去除扩展名的
*/
public String getFileNameNoEx(String filename) {
if ((filename != null) && (filename.length() > 0)) {
int dot = filename.lastIndexOf('.');
if ((dot > -1) && (dot < (filename.length()))) {
return filename.substring(0, dot);
}
}
return filename;
}
}
fileName为完整的路径
图片生成的目录为,和ppt文件同一个目录下,目录名为:ppt文件名+_JPG图片名为,N.JPG (幻灯片从0开始编号,因此图片名为1.JPG……)
说明:
虽然对乱码问题做了处理,但是还会出现乱码,而且Apache POI是开源的,创始人离开后版本更新没有推进
可以改用Aspose.Slide for java 查看
精彩链接
发表评论