`
步青龙
  • 浏览: 297956 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
博客专栏
72ba33fb-eefe-3de1-bd65-82a6e579265d
Java面试
浏览量:0
社区版块
存档分类
最新评论

POI向word添加图片,表格

    博客分类:
  • POI
 
阅读更多
package com.xxx.yyy.commons;
 
import java.io.IOException;
import java.io.InputStream;

import org.apache.poi.openxml4j.opc.OPCPackage;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.apache.poi.xwpf.usermodel.XWPFParagraph;
import org.apache.xmlbeans.XmlException;
import org.apache.xmlbeans.XmlToken;
import org.openxmlformats.schemas.drawingml.x2006.main.CTNonVisualDrawingProps;
import org.openxmlformats.schemas.drawingml.x2006.main.CTPositiveSize2D;
import org.openxmlformats.schemas.drawingml.x2006.wordprocessingDrawing.CTInline;
  
/** 
 * @author POI 导出图片bug修复
 *  
 */  
public class CustomXWPFDocument extends XWPFDocument {  
    public CustomXWPFDocument(InputStream in) throws IOException {  
        super(in);  
    }  
  
    /** 
     *  
     */  
    public CustomXWPFDocument() {  
        super();  
        // TODO Auto-generated constructor stub   
    }  
  
    /** 
     * @param pkg 
     * @throws IOException 
     */  
    public CustomXWPFDocument(OPCPackage pkg) throws IOException {  
        super(pkg);  
        // TODO Auto-generated constructor stub   
    }  // picAttch 图片后面追加的字符串 可以是空格
    public void createPicture(XWPFParagraph paragraph,int id, int width, int height,String picAttch) {  
        final int EMU = 9525;  
        width *= EMU;  
        height *= EMU;  
        String blipId = getAllPictures().get(id).getPackageRelationship()  
                .getId();  
  
        CTInline inline = paragraph.createRun().getCTR()  
                .addNewDrawing().addNewInline();  
        paragraph.createRun().setText(picAttch);
        String picXml = ""  
                + "<a:graphic xmlns:a=\"http://schemas.openxmlformats.org/drawingml/2006/main\">"  
                + "   <a:graphicData uri=\"http://schemas.openxmlformats.org/drawingml/2006/picture\">"  
                + "      <pic:pic xmlns:pic=\"http://schemas.openxmlformats.org/drawingml/2006/picture\">"  
                + "         <pic:nvPicPr>" + "            <pic:cNvPr id=\""  
                + id  
                + "\" name=\"Generated\"/>"  
                + "            <pic:cNvPicPr/>"  
                + "         </pic:nvPicPr>"  
                + "         <pic:blipFill>"  
                + "            <a:blip r:embed=\""  
                + blipId  
                + "\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\"/>"  
                + "            <a:stretch>"  
                + "               <a:fillRect/>"  
                + "            </a:stretch>"  
                + "         </pic:blipFill>"  
                + "         <pic:spPr>"  
                + "            <a:xfrm>"  
                + "               <a:off x=\"0\" y=\"0\"/>"  
                + "               <a:ext cx=\""  
                + width  
                + "\" cy=\""  
                + height  
                + "\"/>"  
                + "            </a:xfrm>"  
                + "            <a:prstGeom prst=\"rect\">"  
                + "               <a:avLst/>"  
                + "            </a:prstGeom>"  
                + "         </pic:spPr>"  
                + "      </pic:pic>"  
                + "   </a:graphicData>" + "</a:graphic>";  
  
        // CTGraphicalObjectData graphicData =   
        inline.addNewGraphic().addNewGraphicData();  
        XmlToken xmlToken = null;  
        try {  
            xmlToken = XmlToken.Factory.parse(picXml);  
        } catch (XmlException xe) {  
            xe.printStackTrace();  
        }  
        inline.set(xmlToken);  
        // graphicData.set(xmlToken);   
  
        inline.setDistT(0);  
        inline.setDistB(0);  
        inline.setDistL(0);  
        inline.setDistR(0);  
  
        CTPositiveSize2D extent = inline.addNewExtent();  
        extent.setCx(width);  
        extent.setCy(height);  
  
        CTNonVisualDrawingProps docPr = inline.addNewDocPr();  
        docPr.setId(id);  
        docPr.setName("图片" + id);  
        docPr.setDescr("");  
    }  
}  

POI或者JXL、docx4j,都是基于公开的标准做的。Office自2007版本开始使用ooxml,压缩的xml文档,2007之前都是二进制的B+树。但是2007+版本也只是把文件通用性结构部分公开了,微软的半开放导致一残废则都残废。所以设计到真正的图表,Flash 都没法子了。上面的类是代替POI中的XWPFDocument。因为继承了XWPFDocument,所以addPictureData不用重写了,图片显示的方法新增一下。这是因为POI 在拼装xml的时候把图片数据放入了,但是没有给提供显示的方法。用了OOXML就要用2007+版本,之所以选择它是因为2007+已经成为了世界标准。2007更是一个里程碑(不谈对微软情感看法)。(忘记说了POI 对word处理相当不给力,上面的代码是添加图片的bug处理,即使因为官方少了一段显示图片的xml)

 action:

  

	public void exportToWord2(){
		try {
	        int picWidth = 240;
	        int picHeight = 150;
	        int picType = XWPFDocument.PICTURE_TYPE_JPEG;
	        String filePath = "简报.docx";
	        response.reset(); // 清空buffer
	        response.setContentType("application/vnd.openxmlformats-officedocument.wordprocessingml.document");
			filePath = java.net.URLEncoder.encode(filePath, "UTF-8");
			response.setHeader("Content-Disposition", "attachment; filename="+ new String(filePath.getBytes("UTF-8"), "GBK"));// 定义文件名
			response.setHeader("Pragma", "no-cache");
			response.setHeader("Expires", " 0");
			ServletOutputStream fOut = response.getOutputStream();
			XWPFParagraph paragraph = null;
			XWPFTable table = null;
			CustomXWPFDocument doc = new CustomXWPFDocument();
			paragraph = doc.createParagraph();
			doc.addPictureData(picInputStream,picType);
			doc.createPicture(paragraph,doc.getAllPictures().size()-1, picWidth, picHeight,"");
			doc.write(fOut);
			fOut.flush();
			fOut.close();
			response.getOutputStream().flush();
			response.getOutputStream().close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

 其中picInputStream 省略,这个图可以是又Jfreechart得到。

  

JFreeChart chart = null; 
//.......省略
ByteArrayOutputStream out = new ByteArrayOutputStream();
			ChartUtilities.writeChartAsPNG(out, chart, 400, 300);
	ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray());
return in;

  

这样通过Jfreechart就可以得到ByteArrayInputStream.

因为addPictureData方法接收的参数是InputStream 看到网上很多人喜欢先写入操作系统,然后读取到流,我不喜。在生成图片的时候直接放到ByteArrayInputStream(继承了抽象类InputStream),直接获取就少了存储相关的操作。JFreechart获取inputStream也就这2行代码。

createPicture方法的picAttch是图片之间放的东西 可以是N个空格做间隔,也可以是文字

弄一个word文件写入磁盘中测试例子,包含POI生成表格到word,jfreechart和POI生成图插入到word.需要jfreechart和poi ooxml的jar

import java.awt.Color;
import java.awt.Font;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.FileOutputStream;
import java.text.DecimalFormat;
import java.text.NumberFormat;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.apache.poi.xwpf.usermodel.XWPFTable;
import org.apache.poi.xwpf.usermodel.XWPFTableRow;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartUtilities;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.labels.StandardPieSectionLabelGenerator;
import org.jfree.chart.plot.PiePlot;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.data.category.DefaultCategoryDataset;
import org.jfree.data.general.DefaultPieDataset;
 

public class CreateTablesWithPOI {
	public static void main(String[] args) {
		String outputFile = "D:\\test.docx";
		CustomXWPFDocument document = new CustomXWPFDocument();
		XWPFTable tableOne = document.createTable();
		XWPFTableRow tableOneRowOne = tableOne.getRow(0);
		tableOneRowOne.getCell(0).setText("第1行第1列");
		tableOneRowOne.addNewTableCell().setText("第1行第2列");
		tableOneRowOne.addNewTableCell().setText("第1行第3列");
		tableOneRowOne.addNewTableCell().setText("第1行第4列");
		XWPFTableRow tableOneRowTwo = tableOne.createRow();
		tableOneRowTwo.getCell(0).setText("第2行第1列");
		tableOneRowTwo.getCell(1).setText("第2行第2列");
		tableOneRowTwo.getCell(2).setText("第2行第3列");
		FileOutputStream fOut;
		try {
			fOut = new FileOutputStream(outputFile);
			ByteArrayInputStream  in = getPieChartImage();
			String ind = document.addPictureData(in, XWPFDocument.PICTURE_TYPE_JPEG); 
			System.out.println("pic ID=" + ind);
			document.createPicture(document.getAllPictures().size()-1, 200, 200,"    "); 
			// 放第二张图
			ind = document.addPictureData(getBarChartImage(), XWPFDocument.PICTURE_TYPE_JPEG); 
			System.out.println("pic ID=" + ind);
			document.createPicture(document.getAllPictures().size()-1, 200, 200,"    "); 
			document.write(fOut); 
			fOut.flush();
			// 操作结束,关闭文件
			fOut.close();
		} catch (Exception e) {
			e.printStackTrace();
		} 
	}
	public static ByteArrayInputStream getPieChartImage() {
		ByteArrayInputStream in = null;
		DefaultPieDataset pieDataset = new DefaultPieDataset();
		pieDataset.setValue(" 北京局 ", 20);
		pieDataset.setValue(" 上海局 ", 18);
		pieDataset.setValue(" 天津局 ", 16);
		pieDataset.setValue(" 重庆局 ", 15);
		pieDataset.setValue(" 山东局 ", 45);
		JFreeChart chart = ChartFactory.createPieChart3D(" 企业备案图 ", pieDataset,
				true, false, false);
		// 设置标题字体样式
		chart.getTitle().setFont(new Font(" 黑体 ", Font.BOLD, 20));
		// 设置饼状图里描述字体样式
		PiePlot piePlot = (PiePlot) chart.getPlot();
		piePlot.setLabelFont(new Font(" 黑体 ", Font.BOLD, 10));
		// 设置显示百分比样式
		piePlot.setLabelGenerator(new StandardPieSectionLabelGenerator(
				(" {0}({2}) "), NumberFormat.getNumberInstance(),
				new DecimalFormat(" 0.00% ")));
		// 设置统计图背景
		piePlot.setBackgroundPaint(Color.white);
		// 设置图片最底部字体样式
		chart.getLegend().setItemFont(new Font(" 黑体 ", Font.BOLD, 10));
		try {
			ByteArrayOutputStream out = new ByteArrayOutputStream();
			ChartUtilities.writeChartAsPNG(out, chart, 400, 300);
		    in  = new ByteArrayInputStream(out.toByteArray());
		} catch (Exception e) {
			e.printStackTrace();
		} 
		return in;
	}
	
	public static ByteArrayInputStream getBarChartImage() {
		ByteArrayInputStream in = null;
		DefaultCategoryDataset dataset =new DefaultCategoryDataset(); 
		dataset.addValue(100,"Spring Security","Jan");
		dataset.addValue(200,"jBPM 4","Jan");
		dataset.addValue(300,"Ext JS","Jan");
		dataset.addValue(400,"JFreeChart","Jan");
		JFreeChart chart = ChartFactory.createBarChart("chart","num","type",dataset, PlotOrientation.VERTICAL, true, false, false); 
		// 设置标题字体样式
		chart.getTitle().setFont(new Font(" 黑体 ", Font.BOLD, 20));
		// 设置饼状图里描述字体样式
		// 设置图片最底部字体样式
		chart.getLegend().setItemFont(new Font(" 黑体 ", Font.BOLD, 10));
		try {
			ByteArrayOutputStream out = new ByteArrayOutputStream();
			ChartUtilities.writeChartAsPNG(out, chart, 400, 300);
		    in  = new ByteArrayInputStream(out.toByteArray());
		} catch (Exception e) {
			e.printStackTrace();
		} 
		return in;
	}
}

 

 POI中设置一段文字我们可以写一个静态通用方法createParagraphContent,大致内容如下POI.java
 

	public static void createParagraphContent(CustomXWPFDocument doc,String content) {
		XWPFRun title= doc.createParagraph().createRun();// 设置一个新的段落
		title.setText(content); 
		title.setFontFamily("宋体"); 
		title.setBold(true); 
	}

 itext的例子请参考  http://oneinit.iteye.com/blog/1529917

图表  http://www.dotblogs.com.tw/angus/archive/2010/05/19/15332.aspx

 

分享到:
评论
7 楼 liushimiao0104 2017-12-14  
请问下怎么删除word里面的图片呢
6 楼 步青龙 2015-12-07  
army520 写道
写的很详细,帮我解决了问题,非常感谢楼主分享。poi版本需3.8或以上。以前版本不支持导出图片。

还是希望能通过fusioncharts 的图表能插入进去,哎
5 楼 Breather.杨 2015-02-23  
OK 看错了
4 楼 Breather.杨 2015-02-23  
List<XWPFPictureData> XwpfPictureDataList = doc.getAllPictures();
        String blipId = doc.getAllPictures().get(id).getPackageRelationship().getId(); 文档里没图片明显会报异常。
3 楼 步青龙 2014-07-01  
andafriend 写道
请问 要如何将 表格创建在指定的地方

你看里面的例子,运行一下试试,布局原理跟HTML的table原理是一样的
2 楼 andafriend 2014-06-25  
请问 要如何将 表格创建在指定的地方
1 楼 army520 2014-05-12  
写的很详细,帮我解决了问题,非常感谢楼主分享。poi版本需3.8或以上。以前版本不支持导出图片。

相关推荐

    Java利用poi对word插入文字图片

    本项目“Java利用poi对word插入文字图片”是一个具体的示例,旨在教给你如何使用Apache POI API在Word文档中插入文字和图片。下面将详细阐述相关的知识点。 首先,Apache POI提供了HWPF(Horrible Word Processor ...

    使用poi替换word中的图片

    本篇将深入探讨如何使用Apache POI库来替换Word文档中的图片,无论文档是.doc还是.docx格式。 首先,我们需要理解Apache POI的工作原理。POI提供了一个高级API,允许开发者读取、创建和修改Office文档。对于Word...

    基于poi导出word以及图片

    在本教程中,我们将重点讨论如何利用Apache POI 3.13版本来导出Word文档,并结合图片操作。 首先,Apache POI提供了一个叫做HWPF(Horrible Word Processor Format)的API来处理老版的Word(.doc)文件,而XWPF...

    Apache poi 根据word模板生成word报表 替换 循环列表 图片

    在本项目中,我们将关注如何使用Apache POI来根据预设的Word模板生成包含替换内容、循环列表和图片的动态Word报表。 首先,我们需要理解Apache POI的核心概念。在处理Word文档时,POI提供了HWPF(Horrible Word ...

    Java Poi流根据Word模板插入相应的文本、表格和图片,并生成新的Word报告。

    在这个场景中,我们关注的是如何利用Java POI库通过Word模板生成包含特定文本、表格和图片的报告。 首先,我们需要了解Java POI中的XWPFDocument类,它是用来处理.docx文件的。XWPFDocument可以读取、修改和创建...

    poi操作word表格

    在本主题中,我们将聚焦于如何使用Apache POI来操作Word文档中的表格。 在Word文档中,表格是一种常用的数据展示和组织方式。使用Apache POI,我们可以创建、修改、读取和格式化Word文档中的表格。以下是一些关键...

    java POI-lib,word中写入图片

    2. **添加图片数据**: 将图片数据添加到Word文档的包中,通过调用`XWPFDocument.addPictureData`方法,返回一个唯一的图片ID。 3. **创建图片对象**: 使用`XWPFDocument.createPicture`方法,传入之前获取的图片ID...

    java使用poi操作.doc word模板替换,循环插入表格

    本篇文章将深入探讨如何利用POI库在Word文档中进行模板替换以及循环插入表格的操作。 首先,我们需要理解Apache POI的工作原理。POI提供了HWPFOI(用于处理老版的.doc文件)和XWPF(用于处理新版的.docx文件)两个...

    java poi 填充word(合并单元格,添加图片,设置字号)(csdn)————程序.pdf

    在本示例中,我们看到如何使用Java POI来填充Word文档,包括合并单元格、添加图片以及设置字号。以下是对代码的详细解释: 1. **初始化Document对象**: 首先,`Document`类是Apache POI库中的核心类,用于表示...

    利用poi+word模版书签,向word中插入数据

    在IT行业中,尤其是在文档自动化处理领域,Apache POI是一个非常重要的工具,它允许开发者使用Java来操作Microsoft Office格式的文件,如Word、Excel和PowerPoint。本教程将深入讲解如何利用Apache POI的XWPF模块...

    Java Poi流 根据Word模板插入相应的文本、表格和图片,生成新的Word报告

    在本主题中,我们将深入探讨如何使用Java POI流处理Word模板,插入文本、表格和图片,以及生成新的Word报告。 1. **Java POI流处理Word模板**: - POI API提供了`XWPFDocument`类来处理`.docx`文件,这是Word 2007...

    poi导出word例子(图片)

    在这个例子中,我们将聚焦于使用POI库在Java中导出包含图片的Word文档。这个过程涉及到几个关键步骤,包括创建Word文档对象、添加文本内容、以及插入图片。 首先,你需要在项目中引入Apache POI的依赖。通常,这...

    使用java Apache poi 根据word模板生成word报表例子

    使用java Apache poi 根据word模板生成word报表 仅支持docx格式的word文件,大概是word2010及以后版本,doc格式不支持。 使用说明:https://blog.csdn.net/u012775558/article/details/79678701

    java生成word,包括word表格,表格插入图片,jar包都在了

    在生成Word表格和插入图片时,Apache POI提供了丰富的API接口。 2. **生成Word表格**:使用Apache POI,你可以创建表格,设定行数、列数,填充数据。表格的样式,如边框、颜色、字体等,也可以进行定制。例如,你...

    POI生成word文档

    Apache POI是一个强大的Java库,专门用于处理Microsoft Office格式的文件,如Word、Excel和PowerPoint。在这个场景中,我们关注的是使用POI来生成Word文档。POI库提供了丰富的API,使得开发者能够轻松地创建、修改和...

    POI DOCX 完美文本、表格模板文字替换并实现在指定位置插入图片浮于文字上方

    在本文中,我们将深入...以上就是关于"POI DOCX 完美文本、表格模板文字替换并实现在指定位置插入图片浮于文字上方"的知识点概述。通过熟练掌握Apache POI库,开发者可以灵活地处理DOCX文档,实现各种定制化的需求。

    poi实现world文档动态插入表格

    应公司业务要求,需要在world文档中生成表格,最开始同事使用的freemarker操作xml标签可以实现world中插入表格, 但是每次修改模板需要改xml文件,几万行中去找对应的标签然后修改,容易找错位置并且电脑非常卡,相当麻烦;...

    poi 将echar报表生成到word table表格中

    在这个场景中,我们关注的是如何使用POI将ECharts图表生成到Word文档的表格中。ECharts是一个强大的、基于JavaScript的数据可视化库,常用于生成各种图表,如折线图、柱状图、饼图等。 首先,你需要了解如何使用...

Global site tag (gtag.js) - Google Analytics