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的效果了,直接上图:

 

  • 3
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
好的,以下是一个使用iText PDF模板的Android例子: 1. 首先,添加iText依赖项到你的项目的build.gradle文件中: ``` implementation 'com.itextpdf:itextpdf:5.5.13.2' ``` 2. 在你的应用程序中创建一个PDF模板文件,例如template.pdf。可以使用Adobe Acrobat等PDF编辑器创建模板。 3. 创建一个新的Android Activity,在其中添加以下代码: ``` import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.os.Bundle; import android.os.Environment; import android.support.v7.app.AppCompatActivity; import android.view.View; import android.widget.Button; import android.widget.Toast; import com.itextpdf.text.Document; import com.itextpdf.text.Image; import com.itextpdf.text.pdf.PdfContentByte; import com.itextpdf.text.pdf.PdfReader; import com.itextpdf.text.pdf.PdfStamper; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; public class MainActivity extends AppCompatActivity { private Button btnGenerate; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); btnGenerate = findViewById(R.id.btn_generate); btnGenerate.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { generatePDF(); } }); } private void generatePDF() { //模板文件路径 String templatePath = Environment.getExternalStorageDirectory() + "/template.pdf"; //生成PDF文件路径 String pdfPath = Environment.getExternalStorageDirectory() + "/output.pdf"; try { //读取模板文件 PdfReader reader = new PdfReader(templatePath); //在PDF文件上添加内容 PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(pdfPath)); //获取PDF的页面 PdfContentByte canvas = stamper.getOverContent(1); //添加文本 canvas.beginText(); canvas.moveText(100, 500); canvas.setFontAndSize(BaseFont.createFont(), 12); canvas.showText("Hello World"); canvas.endText(); //添加图片 Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image); Image image = Image.getInstance(bitmap, null); image.setAbsolutePosition(100, 400); canvas.addImage(image); //关闭PDF文件 stamper.close(); reader.close(); Toast.makeText(this, "PDF generated successfully", Toast.LENGTH_SHORT).show(); } catch (IOException e) { e.printStackTrace(); Toast.makeText(this, "PDF generation failed", Toast.LENGTH_SHORT).show(); } catch (com.itextpdf.text.DocumentException e) { e.printStackTrace(); Toast.makeText(this, "PDF generation failed", Toast.LENGTH_SHORT).show(); } } } ``` 在这个例子中,我们首先定义了一个模板文件的路径和一个生成PDF文件的路径。然后使用iTextPdfReader和PdfStamper类分别读取模板文件生成PDF文件,并使用PdfContentByte类在PDF文件上添加文本和图片。 注意:需要在AndroidManifest.xml文件中添加存储权限。 希望对你有所帮助!
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值