• 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。

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

    文件(夹)名 作用
    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。

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

    使用步骤

    一、项目组织结构图

    注意事项:

    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

    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

    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

    混淆配置

    参考资料

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

    Poi官网

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

    poi替换模板标签为图片

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

    使用POI读写Word doc文件

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

    项目demo下载地址

    https://github.com/haiyuKing/PoiDemo

  • 相关阅读:
    中国移动全球通寻宝第四期攻略
    推荐一个奥运门票交流的网站!
    国务院发出通知明确2008年部分节假日放假安排
    [转]开发人员如何谈加薪,抛砖引玉,欢迎大家各抒己见。
    中级.NET开发人员应该知道些什么?
    linux系统管理 系统安全及应用
    Linux10 进程 定时任务 僵尸进程
    shell脚本练习
    虚拟机和liunx系统的装机
    Linux三剑客awk的应用对比案例详解
  • 原文地址:https://www.cnblogs.com/whycxb/p/9789760.html
Copyright © 2020-2023  润新知