Android使用Itext生成pdf文件

对于生成pdf文件,上一篇文章介绍了使用android原生的方式生成pdf,使用android原生的方式对于将view上的内容生成pdf非常的简单,但有缺憾,就是生成的pdf文佳很多,对于在项目中需要将生成的pdf文件发送出去,这时就会发现发送的时间有点长了,这对于用户来说肯定是不可以接受的了,所以就有了这里接受的Itext了。

对于Itext,主要有两个版本,一个是5.x,另一个是7.x,这两个版本是完全是不兼容的,其区别可以参考官网:https://itextpdf.com/blog/itext-7-and-itext-5-roadmaps-differences-updates

5.x的文档:https://itextsupport.com/apidocs/itext5/latest/

7.x的文档:https://itextsupport.com/apidocs/itext7/latest/

在项目中使用的是5.x,对于7.x没怎么去研究,下面主要是对5.x版本的使用介绍:

首先是下载一个5.x版本的jar包,链接:https://pan.baidu.com/s/1Gno-jNI7L15KSKZZGFai-g,下载完后,至于怎么导包就不用多说了吧,接下来就是看看怎么使用了,先看看文字怎么生存pdf:

   public void textTransformPdf(String content,String pdf_save_address){
        Document doc = new Document();// 创建一个document对象
        FileOutputStream fos;
        try {
            fos = new FileOutputStream(pdf_save_address); // pdf_address为Pdf文件保存到sd卡的路径
            PdfWriter.getInstance(doc, fos);
            doc.open();
            doc.setPageCount(1);
            doc.add(new Paragraph(content, setChineseFont())); // result为保存的字符串
            // ,setChineseFont()为pdf字体
            // 一定要记得关闭document对象
            doc.close();
        } catch (FileNotFoundException e1) {
            e1.printStackTrace();
        } catch (DocumentException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    public Font setChineseFont() {
        BaseFont bf = null;
        Font fontChinese = null;
        try {
            // STSong-Light : Adobe的字体
            // UniGB-UCS2-H : pdf 字体
            bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
            fontChinese = new Font(bf, 12, Font.NORMAL);
        } catch (DocumentException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return fontChinese;
    }

这里设置字体是为了解决中文的问题,把生存pdf的内容和保存pdf的路径传进去就ok了,这是简单的使用,对于文字的设置也是可以的,后面讲到。(参考:https://blog.csdn.net/xuwenneng/article/details/52995392

在讲了将文字生存pdf后,接下来就该说说如何将图片生成pdf了,话不多说,上代码:

    public void imgTransformPdf(String[] imgPaths,String pdf_save_address){
        Document doc = new Document(PageSize.A4, 0, 0, 0, 0);
        try {
            //获取PDF书写器
            PdfWriter.getInstance(doc, new FileOutputStream(pdf_save_address));
            //打开文档
            doc.open();
            //图片对象
            Image img  = null;
            //遍历
            for (int i = 0; i < imgPaths.length; i++) {
                //获取图片
                img = Image.getInstance(new URL(imgPaths[i]));
                //使图片与A4纸张大小自适应
                img.scaleToFit(new Rectangle(PageSize.A4));
                //添加到PDF文档
                doc.add(img);
                //下一页,每张图片一页
                doc.newPage();
            }

        } catch (Exception e) {
            e.printStackTrace();
        }finally{
            //关闭文档
            doc.close();
        }

    }

这里就是根据传入的图片路径生成pdf,这里的设置是一张图片占用一页pdf。(参考:https://blog.csdn.net/HoKis/article/details/54136785

上面讲到的都是单一的类型生成pdf,在实际中肯定是需要生成文字和图片混合的pdf,而且在排版上肯定也是会有一定要求的,接下来就来看下我封装的一个简单实现:

public class PdfItextUtil {

    private Document document;

    // savePath:保存pdf的路径
    public PdfItextUtil(String savePath) throws FileNotFoundException, DocumentException {
        //创建新的PDF文档:A4大小,左右上下边框均为0
        document = new Document(PageSize.A4,50,50,30,30);
        //获取PDF书写器
        PdfWriter.getInstance(document, new FileOutputStream(savePath));
        //打开文档
        document.open();
    }

    public void close(){
        if (document.isOpen()) {
            document.close();
        }
    }

    // 添加图片到pdf中,这张图片在pdf中居中显示
    // imgPath:图片的路径,我使用的是sdcard中图片
    // imgWidth:图片在pdf中所占的宽
    // imgHeight:图片在pdf中所占的高
    public PdfItextUtil addImageToPdfCenterH(@NonNull String imgPath, float imgWidth, float imgHeight) throws IOException, DocumentException {
        //获取图片
        Image img = Image.getInstance(imgPath);
        img.setAlignment(Element.ALIGN_CENTER);
        img.scaleToFit(imgWidth,imgHeight);
        //添加到PDF文档
        document.add(img);

        return this;
    }

    public PdfItextUtil addPngToPdf(InputStream inputStream) throws DocumentException, IOException {
        Image img = PngImage.getImage(inputStream);
        img.setAlignment(Element.ALIGN_CENTER);
        //添加到PDF文档
        document.add(img);
        return this;
    }

    // 添加文本到pdf中
    public PdfItextUtil addTextToPdf(String content) throws DocumentException {
        Paragraph elements = new Paragraph(content, setChineseFont());
        elements.setAlignment(Element.ALIGN_BASELINE);
//        elements.setIndentationLeft(55);  //设置距离左边的距离
        document.add(elements); // result为保存的字符串
        return this;
    }

    // 给pdf添加个标题,居中黑体
    public PdfItextUtil addTitleToPdf(String title){
        try {
            Paragraph elements = new Paragraph(title, setChineseTiltleFont(18));
            elements.setAlignment(Element.ALIGN_CENTER);
            document.add(elements); // result为保存的字符串
        } catch (DocumentException e) {
            e.printStackTrace();
        }
        return this;
    }

    private Font setChineseFont() {
        return setChineseFont(12);
    }

    private Font setChineseFont(int size) {
        BaseFont bf;
        Font fontChinese = null;
        try {
            // STSong-Light : Adobe的字体
            // UniGB-UCS2-H : pdf 字体
            bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
            fontChinese = new Font(bf, size, Font.NORMAL);
        } catch (DocumentException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return fontChinese;
    }

    private Font setChineseTiltleFont(int size) {
        BaseFont bf;
        Font fontChinese = null;
        try {
            // STSong-Light : Adobe的字体
            // UniGB-UCS2-H : pdf 字体
            bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
            fontChinese = new Font(bf, size, Font.BOLD);
        } catch (DocumentException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return fontChinese;
    }
}

封装完后就该怎么使用了,直接上代码了:

    public void toPDF() {
        PdfItextUtil pdfItextUtil = null;
        try {
            pdfItextUtil = new PdfItextUtil(getSavePdfFilePath())
                    .addTitleToPdf(getTvString(tv_title))
                    .addTextToPdf(getTvString(tv_part1))
                    .addImageToPdfCenterH(getImageFilePath(),160,160)
                    .addTextToPdf(getTvString(tv_part2))
                    .addImageToPdfCenterH(getImageFilePath(),160,160)
                    .addTextToPdf(getTvString(tv_content));
        } catch (IOException e) {
            e.printStackTrace();
        } catch (DocumentException e) {
            e.printStackTrace();
        } finally {
            if (pdfItextUtil != null)
                pdfItextUtil.close();
        }
    }

getTvString(tv_title)这个方法是获取TextView中的内容,这里主要还是拼接一个pdf,按顺序从上往下拼接,最后就是看下生成pdf的效果了,直接上图:

猜你喜欢

转载自blog.csdn.net/tangedegushi/article/details/81539320
今日推荐