释放双眼,带上耳机,听听看~!
版权声明:本文为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包
如果windows系统,则下载zip文件;如果是linux系统则选择.tar.gz。
将下载后的压缩包解压,会得到以下文件。
文件(夹)名
作用
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模板文件
代码分析
1、可以看到doc和docx文件分别对应着组件HWPF和XWPF,而HWPF和XWPF则对应着poi-scratchpad和poi-ooxml。
使用步骤
一、项目组织结构图
注意事项:
1、 导入类文件后需要change包名以及重新import R文件路径
2、 Values目录下的文件(strings.xml、dimens.xml、colors.xml等),如果项目中存在,则复制里面的内容,不要整个覆盖
二、导入步骤
1、将poi相关jar文件导入项目中(Demo采用的是module方式)
注意:
解析doc文件,需要引用下面的jar文件:
poi-4.0.0.jar
poi-scratchpad-4.0.0.jar
libs目录下的commons-collections4-4.2.jar
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文件
4、将模板文件复制到项目的assets目录下
三、使用方法
1、根据doc模板生成doc文件的关键代码
MainActivity.java
packagecom.why.project.poidemo;importandroid.content.Context;importandroid.os.Bundle;importandroid.support.v7.app.AppCompatActivity;importandroid.view.View;importcom.why.project.poilib.PoiUtils;importjava.io.File;importjava.io.IOException;importjava.io.InputStream;importjava.util.HashMap;importjava.util.Map;public class MainActivity extendsAppCompatActivity {privateContext mContext;
@Overrideprotected voidonCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mContext= this;//利用doc模板生成doc文件
findViewById(R.id.btn_poi_doc).setOnClickListener(newView.OnClickListener() {
@Overridepublic voidonClick(View view) {try{ InputStream templetDocStream= getAssets().open(\"请假单模板2.doc\"); String targetDocPath= mContext.getExternalFilesDir(\"poi\").getPath() + File.separator + \"请假单2.doc\";//这个目录,不需要申请存储权限
Map dataMap = new HashMap(); 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
packagecom.why.project.poidemo.poi;importorg.apache.poi.hwpf.HWPFDocument;importorg.apache.poi.hwpf.usermodel.Range;importjava.io.FileInputStream;importjava.io.FileOutputStream;importjava.io.IOException;importjava.io.InputStream;importjava.util.Map;/*** Created by HaiyuKing
* Used poi工具类封装
* 在使用POI写word doc文件的时候我们必须要先有一个doc文件才行,因为我们在写doc文件的时候是通过HWPFDocument来写的,
* 而HWPFDocument是要依附于一个doc文件的。所以通常的做法是我们先在硬盘上准备好一个内容空白的doc文件,然后建立一个基于该空白文件的HWPFDocument。
* 之后我们就可以往HWPFDocument里面新增内容了,然后再把它写入到另外一个doc文件中,这样就相当于我们使用POI生成了word doc文件。*/
public classPoiUtils {/*** 生成一个doc文件
*@paramtempletDocPath 模板文件的完整路径
*@paramtargetDocPath 生成的目标文件的完整路径
*@paramdataMap 替换的数据*/
public static void writeToDoc(String templetDocPath, String targetDocPath, MapdataMap){try{//得到模板doc文件的HWPFDocument对象
InputStream in = newFileInputStream(templetDocPath);
writeToDoc(in,targetDocPath,dataMap);
}catch(IOException e)
{
e.printStackTrace();
}
}/*** 生成一个doc文件,主要用于直接读取asset目录下的模板文件,不用先复制到sd卡中
*@paramtempletDocInStream 模板文件的InputStream
*@paramtargetDocPath 生成的目标文件的完整路径
*@paramdataMap 替换的数据*/
public static void writeToDoc(InputStream templetDocInStream, String targetDocPath, MapdataMap){try{//得到模板doc文件的HWPFDocument对象
HWPFDocument HDoc = newHWPFDocument(templetDocInStream);//获取word文本内容,整个文本
Range range =HDoc.getRange();//替换文本内容,将自定义的$xxx$替换成实际文本
for(Map.Entryentry : 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
混淆配置
无
参考资料
项目demo下载地址