在安卓Java中将Pdf页面转换为位图

45

我需要在Android中将PDF文件(PDF页面)转换为位图(或图像文件)。

1. 使用Apache的Pdfbox jar,但它使用了一些在Android中不支持的Java类。 2. 尝试过将图像转换为PDF的Itext jar(我需要相反的操作)。 我已经尝试了许多jar包,但没有任何积极的结果。

byte[] bytes;
    try {

        File file = new File(this.getFilesDir().getAbsolutePath()+"/2010Q2_SDK_Overview.pdf");
        FileInputStream is = new FileInputStream(file);

        // Get the size of the file
        long length = file.length();
        bytes = new byte[(int) length];
        int offset = 0;
        int numRead = 0;
        while (offset < bytes.length && (numRead=is.read(bytes, offset, bytes.length-offset)) >= 0) {
            offset += numRead;
        }


        ByteBuffer buffer = ByteBuffer.NEW(bytes);
        String data = Base64.encodeToString(bytes, Base64.DEFAULT);
        PDFFile pdf_file = new PDFFile(buffer);
        PDFPage page = pdf_file.getPage(2);

        RectF rect = new RectF(0, 0, (int) page.getBBox().width(),
                (int) page.getBBox().height());
      //  Bitmap bufferedImage = Bitmap.createBitmap((int)rect.width(), (int)rect.height(),
         //        Bitmap.Config.ARGB_8888);

        Bitmap image = page.getImage((int)rect.width(), (int)rect.height(), rect);
        FileOutputStream os = new FileOutputStream(this.getFilesDir().getAbsolutePath()+"/pdf.jpg");
        image.compress(Bitmap.CompressFormat.JPEG, 80, os);

       // ((ImageView) findViewById(R.id.testView)).setImageBitmap(image);
我已经获取了图片文件, 这里输入图片描述 而不是这个, 这里输入图片描述
package com.test123;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;

import com.sun.pdfview.PDFFile;
import com.sun.pdfview.PDFPage;
import net.sf.andpdf.nio.ByteBuffer;

import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.RectF;
import android.os.Bundle;
import android.util.Base64;

public class Test123Activity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        byte[] bytes;
        try {

            File file = new File(this.getFilesDir().getAbsolutePath()+"/2010Q2_SDK_Overview.pdf");
            FileInputStream is = new FileInputStream(file);

            // Get the size of the file
            long length = file.length();
            bytes = new byte[(int) length];
            int offset = 0;
            int numRead = 0;
            while (offset < bytes.length && (numRead=is.read(bytes, offset, bytes.length-offset)) >= 0) {
                offset += numRead;
            }


            ByteBuffer buffer = ByteBuffer.NEW(bytes);
            String data = Base64.encodeToString(bytes, Base64.DEFAULT);
            PDFFile pdf_file = new PDFFile(buffer);
            PDFPage page = pdf_file.getPage(2);

            RectF rect = new RectF(0, 0, (int) page.getBBox().width(),
                    (int) page.getBBox().height());
          //  Bitmap bufferedImage = Bitmap.createBitmap((int)rect.width(), (int)rect.height(),
             //        Bitmap.Config.ARGB_8888);

            Bitmap image = page.getImage((int)rect.width(), (int)rect.height(), rect);
            FileOutputStream os = new FileOutputStream(this.getFilesDir().getAbsolutePath()+"/pdf.jpg");
            image.compress(Bitmap.CompressFormat.JPEG, 80, os);

            //((ImageView) findViewById(R.id.testView)).setImageBitmap(image);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

还有其他内置于Android应用程序的方法来显示PDF文件吗?


我不会Python... 有没有用Java的方法可以做到这一点? - Neela
当我在Eclipse中导入android-pdf-viewer-library项目时,我遇到了一些错误。 - user1129139
请分享源代码。你只提供了 try 块。 - user1129139
我在我的项目中只有上述的活动类和链接到其构建路径的PDFViewer.jar。 - Neela
Base64 仅存在于 API 级别 8(2.2),但在这里并没有使用,忘记注释了。你可以删除/注释它。 - Neela
显示剩余11条评论
7个回答

29

我解决了这个问题。方法很简单,只需要让设备有时间来渲染每一页就可以了。

要解决这个问题,你所要做的就是修改

PDFPage page = pdf_file.getPage(2);

PDFPage page = pdf_file.getPage(2, true);

在Android中查看PDF文件,首先需要将PDF文件转换为图片,然后将图片显示给用户。(我将使用WebView)

因此,我们需要使用这个。这是我编辑过的版本,基于这个Git

在将库导入项目后,需要创建活动页面。

XML代码:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <WebView
            android:id="@+id/webView1"
            android:layout_width="match_parent"
            android:layout_height="match_parent"/>

</LinearLayout>

Java:

//Imports:
import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Intent;
import android.graphics.Bitmap;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Environment;
import android.util.Base64;
import android.util.Log;
import android.view.View;
import android.view.ViewTreeObserver;
import android.webkit.WebView;
import com.sun.pdfview.PDFFile;
import com.sun.pdfview.PDFImage;
import com.sun.pdfview.PDFPage;
import com.sun.pdfview.PDFPaint;
import net.sf.andpdf.nio.ByteBuffer;
import net.sf.andpdf.refs.HardReference;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;

//Globals:
private WebView wv;
private int ViewSize = 0;

//OnCreate Method:
@Override
protected void onCreate(Bundle savedInstanceState)
{
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //Settings
    PDFImage.sShowImages = true; // show images
    PDFPaint.s_doAntiAlias = true; // make text smooth
    HardReference.sKeepCaches = true; // save images in cache

    //Setup webview
    wv = (WebView)findViewById(R.id.webView1);
    wv.getSettings().setBuiltInZoomControls(true);//show zoom buttons
    wv.getSettings().setSupportZoom(true);//allow zoom
    //get the width of the webview
    wv.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener()
    {
        @Override
        public void onGlobalLayout()
        {
            ViewSize = wv.getWidth();
            wv.getViewTreeObserver().removeGlobalOnLayoutListener(this);
        }
    });

    try
    {
        File file = new File(Environment.getExternalStorageDirectory().getPath() + "/randompdf.pdf");
        RandomAccessFile f = new RandomAccessFile(file, "r");
        byte[] data = new byte[(int)f.length()];
        f.readFully(data);
        pdfLoadImages(data);
    }
    catch(Exception ignored)
    {
    }
}

//Load Images:
private void pdfLoadImages(final byte[] data)
{
    try
    {
        // run async
        new AsyncTask<Void, Void, String>()
        {
            // create and show a progress dialog
            ProgressDialog progressDialog = ProgressDialog.show(MainActivity.this, "", "Opening...");

            @Override
            protected void onPostExecute(String html)
            {
                //after async close progress dialog
                progressDialog.dismiss();
                //load the html in the webview
                wv.loadDataWithBaseURL("", html, "text/html","UTF-8", "");
            }

            @Override
            protected String doInBackground(Void... params)
            {
                try
                {
                    //create pdf document object from bytes
                    ByteBuffer bb = ByteBuffer.NEW(data);
                    PDFFile pdf = new PDFFile(bb);
                    //Get the first page from the pdf doc
                    PDFPage PDFpage = pdf.getPage(1, true);
                    //create a scaling value according to the WebView Width
                    final float scale = ViewSize / PDFpage.getWidth() * 0.95f;
                    //convert the page into a bitmap with a scaling value
                    Bitmap page = PDFpage.getImage((int)(PDFpage.getWidth() * scale), (int)(PDFpage.getHeight() * scale), null, true, true);
                    //save the bitmap to a byte array
                    ByteArrayOutputStream stream = new ByteArrayOutputStream();
                    page.compress(Bitmap.CompressFormat.PNG, 100, stream);
                    byte[] byteArray = stream.toByteArray();
                    stream.reset();
                    //convert the byte array to a base64 string
                    String base64 = Base64.encodeToString(byteArray, Base64.NO_WRAP);
                    //create the html + add the first image to the html
                    String html = "<!DOCTYPE html><html><body bgcolor=\"#b4b4b4\"><img src=\"data:image/png;base64,"+base64+"\" hspace=10 vspace=10><br>";
                    //loop though the rest of the pages and repeat the above
                    for(int i = 2; i <= pdf.getNumPages(); i++)
                    {
                        PDFpage = pdf.getPage(i, true);
                        page = PDFpage.getImage((int)(PDFpage.getWidth() * scale), (int)(PDFpage.getHeight() * scale), null, true, true);
                        page.compress(Bitmap.CompressFormat.PNG, 100, stream);
                        byteArray = stream.toByteArray();
                        stream.reset();
                        base64 = Base64.encodeToString(byteArray, Base64.NO_WRAP);
                        html += "<img src=\"data:image/png;base64,"+base64+"\" hspace=10 vspace=10><br>";
                    }
                    stream.close();
                    html += "</body></html>";
                    return html;
                }
                catch (Exception e)
                {
                    Log.d("error", e.toString());
                }
                return null;
            }
        }.execute();
        System.gc();// run GC
    }
    catch (Exception e)
    {
        Log.d("error", e.toString());
    }
}

我已经想明白了,但编辑不够快,这个库是否也可以用于将位图转换为PDF? - Asiimwe
PDF页面中的文本图像不够清晰?有什么方法可以提高图像质量吗? - Asiimwe
我不知道。你可以在GitHub上询问,但我怀疑你会得到回应,因为该库的创建者已经不活跃了,他最后一次评论是10个月前。这个主题是最活跃的:https://github.com/jblough/Android-Pdf-Viewer-Library/issues/14 - string.Empty
1
PDFFile pdf = new PDFFile(bb); 在E/CounterA行出现了java.lang.NullPointerException错误。如何修复? - Jamshid
1
@NicolasTyler 我知道在上面的例子中图像没有被存储,请告诉我在哪里展示转换后的位图,因为在 Activity 中它只显示空白布局。我必须把 PDF 文件放在 assets 文件夹或其他地方吗?请建议。 - Abhishek Tamta
显示剩余23条评论

20

我知道这个问题很旧,但上周我遇到了这个问题,而且没有一个答案真正适用于我。

以下是我如何在不使用第三方库的情况下解决此问题的方法。

基于android开发者网站上的代码,使用android的PDFRenderer类(API 21+),我编写了以下方法:

private  ArrayList<Bitmap> pdfToBitmap(File pdfFile) {
    ArrayList<Bitmap> bitmaps = new ArrayList<>();

    try {
        PdfRenderer renderer = new PdfRenderer(ParcelFileDescriptor.open(pdfFile, ParcelFileDescriptor.MODE_READ_ONLY));

        Bitmap bitmap;
        final int pageCount = renderer.getPageCount();
        for (int i = 0; i < pageCount; i++) {
            PdfRenderer.Page page = renderer.openPage(i);

            int width = getResources().getDisplayMetrics().densityDpi / 72 * page.getWidth();
            int height = getResources().getDisplayMetrics().densityDpi / 72 * page.getHeight();
            bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);

            page.render(bitmap, null, null, PdfRenderer.Page.RENDER_MODE_FOR_DISPLAY);

            bitmaps.add(bitmap);

            // close the page
            page.close();

        }

        // close the renderer
        renderer.close();
    } catch (Exception ex) {
        ex.printStackTrace();
    }

    return bitmaps;

}

该方法返回一个位图数组,每个页面在PDF文件中对应一个位图。

根据此答案计算出高度和宽度,以创建更高质量的图像。


2
这会导致一个OOM异常。 - Ali Kazi
你可以尝试使用ARGB_4444,它占用的内存要少得多。 - Alberto Dallaporta
2
运作得很完美!!! 我浪费了一整天的时间来实施其他方案,最终这个方法运作得非常完美 :) - Krunal Panchal
要为位图添加背景,请添加此代码 https://dev59.com/EKPia4cB1Zd3GeqP1qQ3#45297757 - Mohd Qasim

2

这个问题有点老了,但我今天也遇到了同样的问题,所以这是我的解决方案:

/**
 * Use this to load a pdf file from your assets and render it to a Bitmap.
 * 
 * @param context
 *            current context.
 * @param filePath
 *            of the pdf file in the assets.
 * @return a bitmap.
 */
@Nullable
public static Bitmap renderToBitmap(Context context, String filePath) {
    Bitmap bi = null;
    InputStream inStream = null;
    try {
        AssetManager assetManager = context.getAssets();
        Log.d(TAG, "Attempting to copy this file: " + filePath);
        inStream = assetManager.open(filePath);
        bi = renderToBitmap(context, inStream);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            inStream.close();
        } catch (IOException e) {
            // do nothing because the stream has already been closed
        }
    }
    return bi;
}

/**
 * Use this to render a pdf file given as InputStream to a Bitmap.
 * 
 * @param context
 *            current context.
 * @param inStream
 *            the inputStream of the pdf file.
 * @return a bitmap.
 * @see https://github.com/jblough/Android-Pdf-Viewer-Library/
 */
@Nullable
public static Bitmap renderToBitmap(Context context, InputStream inStream) {
    Bitmap bi = null;
    try {
        byte[] decode = IOUtils.toByteArray(inStream);
        ByteBuffer buf = ByteBuffer.wrap(decode);
        PDFPage mPdfPage = new PDFFile(buf).getPage(0);
        float width = mPdfPage.getWidth();
        float height = mPdfPage.getHeight();
        RectF clip = null;
        bi = mPdfPage.getImage((int) (width), (int) (height), clip, true,
                true);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            inStream.close();
        } catch (IOException e) {
            // do nothing because the stream has already been closed
        }
    }
    return bi;
}

此外,我正在使用这个。我的pdf文件只包含一页图像,也许在其他情况下,您需要更新您的代码。
希望这能帮助到某个人。

你好,有没有类似这样的库,但是用户可以滚动浏览PDF页面,而不是点击上一页和下一页按钮?我正在使用同样的库。 - Kairi San
抱歉,我不知道,我只需要将PDF图像转换为位图。我建议您创建一个关于PDF分页的新问题...祝你好运。 - ahmed_khan_89
1
非常感谢您,代码与 https://github.com/jblough/Android-Pdf-Viewer-Library/ 库兼容。 - NickUnuchek
@ahmed_khan_89 为什么 PDF 中的输出位图比源图像小? - NickUnuchek
我不知道,我没有遇到这个问题。但是如果有帮助的话,你可以通过在getImage方法中传递不同的值来选择输出位图的宽度和高度。 - ahmed_khan_89

0

这是代码,希望能有所帮助。我成功地将所有页面渲染成了位图。干杯!

 try {
        pdfViewer = (ImageView) findViewById(R.id.pdfViewer);
        int x = pdfViewer.getWidth();
        int y = pdfViewer.getHeight();
        Bitmap bitmap = Bitmap.createBitmap(x, y, Bitmap.Config.ARGB_4444);
        String StoragePath= Environment.getExternalStorageDirectory()+ "/sample.pdf";
        File file = new File(StoragePath);
        PdfRenderer renderer = new PdfRenderer(ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_ONLY));
        if (currentPage < 0) {
            currentPage = 0;
        } else if (currentPage > renderer.getPageCount()) {
        }


        Matrix m = pdfViewer.getImageMatrix();
        Rect r = new Rect(0, 0, x, y);
        renderer.openPage(currentPage).render(bitmap, r, m, PdfRenderer.Page.RENDER_MODE_FOR_DISPLAY);
        pdfViewer.setImageMatrix(m);
        pdfViewer.setImageBitmap(bitmap);
        pdfViewer.invalidate();


    } catch (Exception ex) {
        Log.v(TAG, ex.getMessage());
    } finally {

    }

0

在尝试了所有答案之后,对于所有的PDF文件都没有起作用。自定义字体的PDF文件存在渲染问题。然后我尝试使用library

代码如下:

在你的应用程序build.gradle文件中添加以下依赖项:

implementation 'com.github.barteksc:android-pdf-viewer:3.2.0-beta.1'

将PDF页面转换为图像的代码:

      public static List<Bitmap> renderToBitmap(Context context, String filePath) {
            List<Bitmap> images = new ArrayList<>();
            PdfiumCore pdfiumCore = new PdfiumCore(context);
            try {
                File f = new File(pdfPath);
                ParcelFileDescriptor fd = ParcelFileDescriptor.open(f, ParcelFileDescriptor.MODE_READ_ONLY);
                PdfDocument pdfDocument = pdfiumCore.newDocument(fd);
                final int pageCount = pdfiumCore.getPageCount(pdfDocument);
                for (int i = 0; i < pageCount; i++) {
                    pdfiumCore.openPage(pdfDocument, i);
                    int width = pdfiumCore.getPageWidthPoint(pdfDocument, i);
                    int height = pdfiumCore.getPageHeightPoint(pdfDocument, i);
                    Bitmap bmp = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
                    pdfiumCore.renderPageBitmap(pdfDocument, bmp, i, 0, 0, width, height);
                    images.add(bmp);
                }
                pdfiumCore.closeDocument(pdfDocument);
            } catch(Exception e) {
                //todo with exception
            }
     return images;
   }

到目前为止,我尝试过的所有PDF文件都可以正常工作。


0

依赖项

dependencies {
    implementation 'com.tom_roush:pdfbox-android:1.8.10.1'
}

代码

    PDDocument pd = PDDocument.load (new File (in));
    PDFRenderer pr = new PDFRenderer (pd);
    Bitmap bitmap = pr.renderImageWithDPI(0, 300);

-1

我们可以使用Qoppa PDF工具包来完成。

使用Qoppa PDF工具包的步骤如下:

http://www.qoppa.com/android/pdfsdk/download下载它。

该文件包含四个项目:

  1. version_history.txt - 此文本文件将随着工具包的每个新版本进行更新。
  2. qoppapdf.jar - 这是工具包的主要jar文件。需要将此jar文件添加到项目的类路径中。
  3. assets文件夹 - 此文件夹包含工具包使用的资源,例如字体和图标。从zip文件中提取后,您需要将此文件夹中的文件复制到项目中的assets文件夹中。
  4. libs文件夹 - 此文件夹包含本机Android库,这些库用于解码某些JPEG图像以及Android不支持的JPEG 2000图像。需要将libs文件夹的内容复制到项目中的libs文件夹中。如果您的项目中没有libs文件夹,请创建一个并将此文件夹的内容复制到其中。

代码如下:

        //Converting PDF to Bitmap Image...
        StandardFontTF.mAssetMgr = getAssets();
        //Load document to get the first page
        try {
            PDFDocument pdf = new PDFDocument("/sdcard/PDFFilename.pdf", null);
            PDFPage page = pdf.getPage(0);

            //creating Bitmap and canvas to draw the page into
            int width = (int)Math.ceil(page.getDisplayWidth());
            int height = (int)Math.ceil(page.getDisplayHeight());

            Bitmap bm = Bitmap.createBitmap(width, height, Config.ARGB_8888);
            Canvas c = new Canvas(bm);
            page.paintPage(c);

            //Saving the Bitmap
            OutputStream os = new FileOutputStream("/sdcard/GeneratedImageByQoppa.jpg");
            bm.compress(CompressFormat.JPEG, 80, os);
            os.close();
        } catch (PDFException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (FileNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

2
不要尝试这个,浪费时间,谁想让他们的用户阅读带有水印的PDF文件呢? - DPP
2
请提供一种解决方案,可以在没有许可证的情况下防止水印。 - Jagadeesh
@NicolasTyler,你能告诉我们一种去除水印的方法吗? - Akhilesh Sk
2
很简单,你要么付费购买它,要么使用替代的库。 - string.Empty
2
qPDF - 如果你不愿意支付第一年6000美元和之后每年3000美元的费用,就不要费心去获取完整版。 - Rexb
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接