Преобразование страницы PDF в растровое изображение в Android Java
Мне нужно конвертировать PDF-файл (PDF-страницу) в растровое изображение (или файл изображения) в Android.
1. Используется Pdfbox jar от Apache. Но он использует некоторые классы Java, которые не поддерживаются в Android. 2. Пробовал Itext jar, который конвертирует изображение в pdf (мне нужна его обратная операция). Подобным образом я пробовал много 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();
}
}
}
Иначе, есть ли другой способ отобразить PDF-файл в Android с помощью встроенной в приложение функции?
7 ответов
Я решил эту проблему. это так же просто, как дать устройству время на рендеринг каждой страницы.
Чтобы исправить это все, что вам нужно сделать, это изменить
PDFPage page = pdf_file.getPage(2);
в
PDFPage page = pdf_file.getPage(2, true);
Во-первых, для просмотра PDF в Android вы должны конвертировать PDF в изображения, а затем отображать их пользователю. (Я собираюсь использовать веб-просмотр)
Так что для этого нам нужна эта библиотека. Это моя отредактированная версия этого мерзавца.
После того, как вы импортировали библиотеку в свой проект, вам нужно создать свою деятельность.
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>
Ява:
//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());
}
}
Я знаю, что этот вопрос старый, но я столкнулся с этой проблемой на прошлой неделе, и ни один из ответов не помог мне.
Вот как я решаю эту проблему, не используя сторонние библиотеки.
Основываясь на коде с сайта разработчиков Android, использующего класс AndroidRdfRenderer (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.
Высота и ширина рассчитываются на основе этого ответа для создания изображений более высокого качества.
Этот вопрос немного устарел, но мне пришлось сделать то же самое сегодня, так что это мое решение:
/**
* 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-файл содержит только одну страницу и является изображением, может быть, в других случаях вам придется обновить свой код.
Надеюсь, это кому-нибудь поможет.
Зависимости
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);
Вот код, надеюсь, это поможет. Мне удалось отрендерить все страницы в Bitmap.Cheers
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 {
}
Пройдя и попробовав все ответы, ни один из них не работал для всех файлов PDF. Проблемы с рендерингом были в файлах PDF с пользовательскими шрифтами. Затем я попытался использовать библиотеку
Код выглядит следующим образом:
В файле 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-файлы, которые я пробовал.
Мы можем сделать это, используя Qoppa PDF toolkit.
Действия по использованию инструментария Qoppa PDF.
Загрузите его с http://www.qoppa.com/android/pdfsdk/download
Этот файл содержит четыре элемента:
- version_history.txt - этот текстовый файл будет обновляться с каждой новой версией инструментария.
- qoppapdf.jar - это основной файл jar для инструментария. Этот файл jar необходимо добавить в путь к классам для вашего проекта.
- папка assets - эта папка содержит ресурсы, используемые инструментарием, такие как шрифты и значки. После извлечения из zip-файлов вам нужно будет скопировать файлы из этой папки в папку активов в вашем проекте.
- Папка libs - эта папка содержит собственные библиотеки Android, эти библиотеки необходимы для декодирования определенных изображений JPEG, а также изображений JPEG 2000, которые не поддерживаются Android. Содержимое папки 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();
}