PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

释放双眼,带上耳机,听听看~!

版权声明:本文为HaiyuKing原创文章,转载请注明出处!

前言

使用Poi实现android中根据模板文件生成Word文档的功能。这里的模板文件是doc文件。如果模板文件是docx文件的话,请阅读下一篇文章《PoiDocxDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0),目前只能java生成】

注意:

  • POI 4.0.0 is the first release to require Java 8 or newer.

前期准备

1、下载poi相关jar包

官网下载地址:http://poi.apache.org/download.html

如果windows系统,则下载zip文件;如果是linux系统则选择.tar.gz。

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

将下载后的压缩包解压,会得到以下文件。

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

文件(夹)名 作用
docs 文档(包括API文档和如何使用及版本信息)
lib doc功能实现依赖的包
ooxml-lib docx功能实现依赖的包
LICENSE  
NOTICE  
poi-4.0.0.jar 基础类
poi-examples-4.0.0.jar 不明确,不知道什么作用
poi-excelant-4.0.0.jar excel功能实现
poi-ooxml-4.0.0.jar docx功能实现
poi-ooxml-schemas-4.0.0.jar docx功能实现相关类
poi-scratchpad-4.0.0.jar doc功能实现

 

2、制作docx模板或者doc模板文件

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

代码分析

1、可以看到doc和docx文件分别对应着组件HWPF和XWPF,而HWPF和XWPF则对应着poi-scratchpad和poi-ooxml。

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

官网地址:http://poi.apache.org/components/index.html

使用步骤

一、项目组织结构图

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

注意事项:

1、  导入类文件后需要change包名以及重新import R文件路径

2、  Values目录下的文件(strings.xml、dimens.xml、colors.xml等),如果项目中存在,则复制里面的内容,不要整个覆盖

二、导入步骤

1、将poi相关jar文件导入项目中(Demo采用的是module方式)

引用jar文件参考《【Android Studio安装部署系列】十七、Android studio引用第三方库、jar、so、arr文件

注意:

解析doc文件,需要引用下面的jar文件:

  • poi-4.0.0.jar
  • poi-scratchpad-4.0.0.jar
  • libs目录下的commons-collections4-4.2.jar

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

 

2、在poilib和app的build.gradle文件中添加以下代码

poilib

apply plugin: \'com.android.library\'

android {
    compileSdkVersion 28



    defaultConfig {
        minSdkVersion 16
        targetSdkVersion 28
        versionCode 1
        versionName \"1.0\"

        testInstrumentationRunner \"android.support.test.runner.AndroidJUnitRunner\"
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile(\'proguard-android.txt\'), \'proguard-rules.pro\'
        }
    }

    //poi compileOptions { sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 }

}

dependencies {
    implementation fileTree(dir: \'libs\', include: [\'*.jar\'])

    implementation \'com.android.support:appcompat-v7:28.0.0\'
    testImplementation \'junit:junit:4.12\'
    androidTestImplementation \'com.android.support.test:runner:1.0.2\'
    androidTestImplementation \'com.android.support.test.espresso:espresso-core:3.0.2\'

    //Poi=doc api files(\'libs/poi-4.0.0.jar\') api files(\'libs/poi-scratchpad-4.0.0.jar\') api files(\'libs/commons-collections4-4.2.jar\')
}

app

apply plugin: \'com.android.application\'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId \"com.why.project.poidemo\"
        minSdkVersion 16
        targetSdkVersion 28
        versionCode 1
        versionName \"1.0\"
        testInstrumentationRunner \"android.support.test.runner.AndroidJUnitRunner\"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile(\'proguard-android.txt\'), \'proguard-rules.pro\'
        }
    }

    //poi compileOptions { sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 }
}

dependencies {
    implementation fileTree(include: [\'*.jar\'], dir: \'libs\')
    implementation \'com.android.support:appcompat-v7:28.0.0\'
    implementation \'com.android.support.constraint:constraint-layout:1.1.3\'
    testImplementation \'junit:junit:4.12\'
    androidTestImplementation \'com.android.support.test:runner:1.0.2\'
    androidTestImplementation \'com.android.support.test.espresso:espresso-core:3.0.2\'
    //poi implementation project(\':poilib\')
}

3、在poilib这个module中添加PoiUtils.java文件

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

4、将模板文件复制到项目的assets目录下

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

三、使用方法

1、根据doc模板生成doc文件的关键代码

MainActivity.java

package com.why.project.poidemo;

import android.content.Context;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;

import com.why.project.poilib.PoiUtils;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;

public class MainActivity extends AppCompatActivity {

    private Context mContext;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mContext = this;

        //利用doc模板生成doc文件
        findViewById(R.id.btn_poi_doc).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                try { InputStream templetDocStream = getAssets().open(\"请假单模板2.doc\"); String targetDocPath = mContext.getExternalFilesDir(\"poi\").getPath() + File.separator + \"请假单2.doc\";//这个目录,不需要申请存储权限
 Map<String, String> dataMap = new HashMap<String, String>(); dataMap.put(\"$writeDate$\", \"2018年10月14日\"); dataMap.put(\"$name$\", \"HaiyuKing\"); dataMap.put(\"$dept$\", \"移动开发组\"); dataMap.put(\"$leaveType$\", \"☑倒休 √年假 ✔事假 ☐病假 ☐婚假 ☐产假 ☐其他\"); dataMap.put(\"$leaveReason$\", \"倒休一天。\"); dataMap.put(\"$leaveStartDate$\", \"2018年10月14日上午\"); dataMap.put(\"$leaveEndDate$\", \"2018年10月14日下午\"); dataMap.put(\"$leaveDay$\", \"1\"); dataMap.put(\"$leaveLeader$\", \"同意\"); dataMap.put(\"$leaveDeptLeaderImg$\", \"同意!\"); PoiUtils.writeToDoc(templetDocStream,targetDocPath,dataMap); } catch (IOException e) { e.printStackTrace(); }

            }
        });
    }
}

 

 PoiUtils.java

package com.why.project.poidemo.poi;

import org.apache.poi.hwpf.HWPFDocument;
import org.apache.poi.hwpf.usermodel.Range;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;

/**
 * Created by HaiyuKing
 * Used poi工具类封装
 * 在使用POI写word doc文件的时候我们必须要先有一个doc文件才行,因为我们在写doc文件的时候是通过HWPFDocument来写的,
 * 而HWPFDocument是要依附于一个doc文件的。所以通常的做法是我们先在硬盘上准备好一个内容空白的doc文件,然后建立一个基于该空白文件的HWPFDocument。
 * 之后我们就可以往HWPFDocument里面新增内容了,然后再把它写入到另外一个doc文件中,这样就相当于我们使用POI生成了word doc文件。
 */
public class PoiUtils {

    /**
     * 生成一个doc文件
     * @param templetDocPath  模板文件的完整路径
     * @param targetDocPath 生成的目标文件的完整路径
     * @param dataMap 替换的数据*/
    public static void writeToDoc(String templetDocPath, String targetDocPath, Map<String,String> dataMap){
        try
        {
            //得到模板doc文件的HWPFDocument对象
            InputStream in = new FileInputStream(templetDocPath);
            writeToDoc(in,targetDocPath,dataMap);
        }
        catch(IOException e)
        {
            e.printStackTrace();
        }
    }

    /**
     * 生成一个doc文件,主要用于直接读取asset目录下的模板文件,不用先复制到sd卡中
     * @param templetDocInStream  模板文件的InputStream
     * @param targetDocPath 生成的目标文件的完整路径
     * @param dataMap 替换的数据*/
    public static void writeToDoc(InputStream templetDocInStream, String targetDocPath, Map<String,String> dataMap){
        try
        {
            //得到模板doc文件的HWPFDocument对象
            HWPFDocument HDoc = new HWPFDocument(templetDocInStream); // 获取word文本内容,整个文本
            Range range = HDoc.getRange(); // 替换文本内容,将自定义的$xxx$替换成实际文本
            for(Map.Entry<String, String> entry : dataMap.entrySet()) { range.replaceText(entry.getKey(), entry.getValue()); } //写到另一个文件中
            FileOutputStream out = new FileOutputStream(targetDocPath, true); //把doc输出到输出流中
 HDoc.write(out); out.close(); templetDocInStream.close();
        }
        catch(IOException e)
        {
            e.printStackTrace();
        }
        catch(Exception e)
        {
            e.printStackTrace();
        }
    }

}

2、效果

生成的文件路径:内部存储——Android——data——com.why.project.poidemo——files——poi——请假单2.doc

PoiDemo【Android将表单数据生成Word文档的方案之二(基于Poi4.0.0)】

混淆配置

参考资料

Android使用ApachePOI组件读写Worddoc和docx文件

Poi官网

Android中使用POI加载与显示word文档

poi替换模板标签为图片

POI用addPicture插入图片到word里面无法显示

使用POI读写Word doc文件

Android中使用POI加载与显示word文档

项目demo下载地址

https://github.com/haiyuKing/PoiDemo

给TA打赏
共{{data.count}}人
人已打赏
随笔日记

逻辑思维驱动 (测试) 工作管理

2020-11-9 4:37:18

随笔日记

习近平主席“一带一路”论坛开幕式主旨演讲 透露出哪些新信号?

2020-11-9 4:37:20

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索