본문으로 건너뛰기

최고의 Android PDF 라이브러리

올해 최고의 Android PDF 라이브러리.

PDF는 Portable Document Format의 약자입니다. 1990년대에 개발된 파일 형식으로 텍스트 형식 및 이미지를 포함한 문서를 응용 소프트웨어, 하드웨어 및 운영 체제와 독립적인 방식으로 제공합니다.

PDF는 Adobe에서 24년 전 '1993년 6월 15일'에 처음 출시했지만 현재는 국제표준화기구(ISO)에서 관리하는 공개 표준입니다.

안드로이드 장치용 무료 및 상업용 PDF 리더 응용 프로그램이 많이 있습니다. 그러나 개발자로서 우리는 우리와 우리 친구들이 사용할 수 있는 작은 PDF 리더를 구축해야 합니다.

생각만큼 어렵지 않으며 이를 지원하기 위해 여러 오픈 소스 라이브러리가 존재합니다.

이 기사에서 우리는 이러한 라이브러리 중 일부와 이를 사용하는 방법에 대한 스니펫을 살펴볼 것입니다.

시작하자.

1. AndroidPdf 뷰어

이것은 PDF 문서를 표시하기 위한 오픈 소스 라이브러리입니다. 이 PDF는 PdfiumAndroid로 렌더링됩니다.

AndroidPdfViewer는 현재 가장 인기 있는 Android PDF 보기 라이브러리입니다. bartesk에서 관리하며 다양한 버전의 라이브러리를 독립적으로 출시했습니다.

예를 들어 많은 사람들이 여전히 AndroidPdfView를 사용하고 있지만 AndroidPdfViewV1[AndroidPdfViewV2]가 있습니다. (https://github.com/barteksc/AndroidPdfViewerV2).

이 라이브러리는 animations, gestures, zoomdouble tap 지원을 지원합니다.

이 라이브러리를 사용하는 것은 매우 쉽습니다.

먼저 앱 수준 build.gradle에 포함하기만 하면 됩니다.

implementation 'com.github.barteksc:android-pdf-viewer:2.8.2'

그런 다음 레이아웃에서:

<com.github.barteksc.pdfviewer.PDFView
android_id="@+id/pdfView"
android_layout_width="match_parent"
android_layout_height="match_parent"/>

그런 다음 다양한 소스에서 PDF를 로드할 수 있습니다.

pdfView.fromUri(Uri)
or
pdfView.fromFile(File)
or
pdfView.fromBytes(byte[])
or
pdfView.fromStream(InputStream) // stream is written to bytearray - native code cannot use Java Streams
or
pdfView.fromSource(DocumentSource)
or
pdfView.fromAsset(String)
.pages(0, 2, 1, 3, 3, 3) // all pages are displayed by default
.enableSwipe(true) // allows to block changing pages using swipe
.swipeHorizontal(false)
.enableDoubletap(true)
.defaultPage(0)
// allows to draw something on the current page, usually visible in the middle of the screen
.onDraw(onDrawListener)
// allows to draw something on all pages, separately for every page. Called only for visible pages
.onDrawAll(onDrawListener)
.onLoad(onLoadCompleteListener) // called after document is loaded and starts to be rendered
.onPageChange(onPageChangeListener)
.onPageScroll(onPageScrollListener)
.onError(onErrorListener)
.onPageError(onPageErrorListener)
.onRender(onRenderListener) // called after document is rendered for the first time
// called on single tap, return true if handled, false to toggle scroll handle visibility
.onTap(onTapListener)
.enableAnnotationRendering(false) // render annotations (such as comments, colors or forms)
.password(null)
.scrollHandle(null)
.enableAntialiasing(true) // improve rendering a little bit on low-res screens
// spacing between pages in dp. To define spacing color, set view background
.spacing(0)
.invalidPageColor(Color.WHITE) // color of page that is invalid and cannot be loaded
.load();

다음은 예입니다.

@EActivity(R.layout.activity_main)
@OptionsMenu(R.menu.options)
public class PDFViewActivity extends AppCompatActivity implements OnPageChangeListener, OnLoadCompleteListener,
OnPageErrorListener {

private static final String TAG = PDFViewActivity.class.getSimpleName();

private final static int REQUEST_CODE = 42;
public static final int PERMISSION_CODE = 42042;

public static final String SAMPLE_FILE = "sample.pdf";
public static final String READ_EXTERNAL_STORAGE = "android.permission.READ_EXTERNAL_STORAGE";

@ViewById
PDFView pdfView;

@NonConfigurationInstance
Uri uri;

@NonConfigurationInstance
Integer pageNumber = 0;

String pdfFileName;

@OptionsItem(R.id.pickFile)
void pickFile() {
int permissionCheck = ContextCompat.checkSelfPermission(this,
READ_EXTERNAL_STORAGE);

if (permissionCheck != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(
this,
new String[]{READ_EXTERNAL_STORAGE},
PERMISSION_CODE
);

return;
}

launchPicker();
}

void launchPicker() {
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("application/pdf");
try {
startActivityForResult(intent, REQUEST_CODE);
} catch (ActivityNotFoundException e) {
//alert user that file manager not working
Toast.makeText(this, R.string.toast_pick_file_error, Toast.LENGTH_SHORT).show();
}
}

@AfterViews
void afterViews() {
pdfView.setBackgroundColor(Color.LTGRAY);
if (uri != null) {
displayFromUri(uri);
} else {
displayFromAsset(SAMPLE_FILE);
}
setTitle(pdfFileName);
}

private void displayFromAsset(String assetFileName) {
pdfFileName = assetFileName;

pdfView.fromAsset(SAMPLE_FILE)
.defaultPage(pageNumber)
.onPageChange(this)
.enableAnnotationRendering(true)
.onLoad(this)
.scrollHandle(new DefaultScrollHandle(this))
.spacing(10) // in dp
.onPageError(this)
.pageFitPolicy(FitPolicy.BOTH)
.load();
}

private void displayFromUri(Uri uri) {
pdfFileName = getFileName(uri);

pdfView.fromUri(uri)
.defaultPage(pageNumber)
.onPageChange(this)
.enableAnnotationRendering(true)
.onLoad(this)
.scrollHandle(new DefaultScrollHandle(this))
.spacing(10) // in dp
.onPageError(this)
.load();
}

@OnActivityResult(REQUEST_CODE)
public void onResult(int resultCode, Intent intent) {
if (resultCode == RESULT_OK) {
uri = intent.getData();
displayFromUri(uri);
}
}

@Override
public void onPageChanged(int page, int pageCount) {
pageNumber = page;
setTitle(String.format("%s %s / %s", pdfFileName, page + 1, pageCount));
}

public String getFileName(Uri uri) {
String result = null;
if (uri.getScheme().equals("content")) {
Cursor cursor = getContentResolver().query(uri, null, null, null, null);
try {
if (cursor != null && cursor.moveToFirst()) {
result = cursor.getString(cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME));
}
} finally {
if (cursor != null) {
cursor.close();
}
}
}
if (result == null) {
result = uri.getLastPathSegment();
}
return result;
}

@Override
public void loadComplete(int nbPages) {
PdfDocument.Meta meta = pdfView.getDocumentMeta();
Log.e(TAG, "title = " + meta.getTitle());
Log.e(TAG, "author = " + meta.getAuthor());
Log.e(TAG, "subject = " + meta.getSubject());
Log.e(TAG, "keywords = " + meta.getKeywords());
Log.e(TAG, "creator = " + meta.getCreator());
Log.e(TAG, "producer = " + meta.getProducer());
Log.e(TAG, "creationDate = " + meta.getCreationDate());
Log.e(TAG, "modDate = " + meta.getModDate());

printBookmarksTree(pdfView.getTableOfContents(), "-");

}

public void printBookmarksTree(List<PdfDocument.Bookmark> tree, String sep) {
for (PdfDocument.Bookmark b : tree) {

Log.e(TAG, String.format("%s %s, p %d", sep, b.getTitle(), b.getPageIdx()));

if (b.hasChildren()) {
printBookmarksTree(b.getChildren(), sep + "-");
}
}
}

/**
* Listener for response to user permission request
*
* @param requestCode Check that permission request code matches
* @param permissions Permissions that requested
* @param grantResults Whether permissions granted
*/
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[],
@NonNull int[] grantResults) {
if (requestCode == PERMISSION_CODE) {
if (grantResults.length > 0
&& grantResults[0] == PackageManager.PERMISSION_GRANTED) {
launchPicker();
}
}
}

@Override
public void onPageError(int page, Throwable t) {
Log.e(TAG, "Cannot load page " + page);
}
}

참조

아니요
1.찾아보기 예시
2.자세히 보기
3.직접 다운로드

2. PdfiumAndroid

AndroidPdfViewer의 동일한 작성자인 bartesk원본 저장소에서 PdfiumAndroid를 포크하고 몇 가지를 유지 관리하고 추가했습니다. 문서도.

원본 PdfiumAndroid는 유지 관리되지 않았습니다.

따라서 포크된 버전에는 몇 가지 문서가 있으며 적극적으로 관리되고 있습니다.

그는 인기 있는 AndroidPdfViewer와 함께 사용하기 위해 포크했습니다.

그러나 독립적으로 사용할 수도 있습니다.

먼저 종속성으로 추가해야 합니다.

implementation 'com.github.barteksc:pdfium-android:1.8.2'

다음은 간단한 예입니다.

void openPdf() {
ImageView iv = (ImageView) findViewById(R.id.imageView);
ParcelFileDescriptor fd = ...;
int pageNum = 0;
PdfiumCore pdfiumCore = new PdfiumCore(context);
try {
PdfDocument pdfDocument = pdfiumCore.newDocument(fd);

pdfiumCore.openPage(pdfDocument, pageNum);

int width = pdfiumCore.getPageWidthPoint(pdfDocument, pageNum);
int height = pdfiumCore.getPageHeightPoint(pdfDocument, pageNum);

// ARGB_8888 - best quality, high memory usage, higher possibility of OutOfMemoryError
// RGB_565 - little worse quality, twice less memory usage
Bitmap bitmap = Bitmap.createBitmap(width, height,
Bitmap.Config.RGB_565);
pdfiumCore.renderPageBitmap(pdfDocument, bitmap, pageNum, 0, 0,
width, height);
//if you need to render annotations and form fields, you can use
//the same method above adding 'true' as last param

iv.setImageBitmap(bitmap);

printInfo(pdfiumCore, pdfDocument);

pdfiumCore.closeDocument(pdfDocument); // important!
} catch(IOException ex) {
ex.printStackTrace();
}
}

public void printInfo(PdfiumCore core, PdfDocument doc) {
PdfDocument.Meta meta = core.getDocumentMeta(doc);
Log.e(TAG, "title = " + meta.getTitle());
Log.e(TAG, "author = " + meta.getAuthor());
Log.e(TAG, "subject = " + meta.getSubject());
Log.e(TAG, "keywords = " + meta.getKeywords());
Log.e(TAG, "creator = " + meta.getCreator());
Log.e(TAG, "producer = " + meta.getProducer());
Log.e(TAG, "creationDate = " + meta.getCreationDate());
Log.e(TAG, "modDate = " + meta.getModDate());

printBookmarksTree(core.getTableOfContents(doc), "-");

}

public void printBookmarksTree(List<PdfDocument.Bookmark> tree, String sep) {
for (PdfDocument.Bookmark b : tree) {

Log.e(TAG, String.format("%s %s, p %d", sep, b.getTitle(), b.getPageIdx()));

if (b.hasChildren()) {
printBookmarksTree(b.getChildren(), sep + "-");
}
}
}
보기다운로드
보기직접 다운로드

3. PdfBox-안드로이드

PdfBox-Android는 Android에서 사용할 수 있는 Apache의 PdfBox 라이브러리의 포트입니다. 상위 라이브러리에 있어야 하는 대부분의 기능은 PdfBox-Android에 이미 구현되어 있습니다.

PdfBox-Android는 전체 기능을 위해 Android API 19 이상이 필요합니다.

PdfBox-Android는 PDF 문서를 렌더링할 수 있는 또 다른 라이브러리입니다. Tom Roush가 작성했으며 여러 기여자가 있습니다.

PdfBox-Android 프로젝트의 기본 코드는 여기에 있는 Apache 2.0 라이선스에 따라 라이선스가 부여됩니다.

이 라이브러리는 4년 이상 존재했지만 여전히 정기적으로 업데이트됩니다.

PdfBox-Android 설치

PdfBox-Android를 설치하는 방법은 다음과 같습니다.

앱 수준 build.gradle로 이동하여 구현 statememt를 추가합니다.

dependencies {
implementation 'com.tom-roush:pdfbox-android:1.8.10.3'
}

최신 버전은 여기에서 확인할 수 있습니다.

Maven을 사용하는 경우:

<dependency>
<groupId>com.tom_roush</groupId>
<artifactId>pdfbox-android</artifactId>
<version>1.8.10.0</version>
<type>pom</type>
</dependency>

PDFBox를 호출하기 전에 라이브러리의 리소스 로더를 초기화하는 것이 좋습니다. PDFBox 메서드를 호출하기 전에 다음 줄을 추가합니다.

PDFBoxResourceLoader.init(getApplicationContext());

다음은 예입니다.

메인액티비티.자바

public class MainActivity extends Activity {
File root;
AssetManager assetManager;
Bitmap pageImage;
TextView tv;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}

@Override
protected void onStart() {
super.onStart();
setup();
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}

/**
* Initializes variables used for convenience
*/
private void setup() {
// Enable Android asset loading
PDFBoxResourceLoader.init(getApplicationContext());
// Find the root of the external storage.

root = getApplicationContext().getCacheDir();
assetManager = getAssets();
tv = (TextView) findViewById(R.id.statusTextView);
}

/**
* Creates a new PDF from scratch and saves it to a file
*/
public void createPdf(View v) {
PDDocument document = new PDDocument();
PDPage page = new PDPage();
document.addPage(page);

// Create a new font object selecting one of the PDF base fonts
PDFont font = PDType1Font.HELVETICA;
// Or a custom font
// try
// {
// // Replace MyFontFile with the path to the asset font you'd like to use.
// // Or use LiberationSans "com/tom_roush/pdfbox/resources/ttf/LiberationSans-Regular.ttf"
// font = PDType0Font.load(document, assetManager.open("MyFontFile.TTF"));
// }
// catch (IOException e)
// {
// Log.e("PdfBox-Android-Sample", "Could not load font", e);
// }

PDPageContentStream contentStream;

try {
// Define a content stream for adding to the PDF
contentStream = new PDPageContentStream(document, page);

// Write Hello World in blue text
contentStream.beginText();
contentStream.setNonStrokingColor(15, 38, 192);
contentStream.setFont(font, 12);
contentStream.newLineAtOffset(100, 700);
contentStream.showText("Hello World");
contentStream.endText();

// Load in the images
InputStream in = assetManager.open("falcon.jpg");
InputStream alpha = assetManager.open("trans.png");

// Draw a green rectangle
contentStream.addRect(5, 500, 100, 100);
contentStream.setNonStrokingColor(0, 255, 125);
contentStream.fill();

// Draw the falcon base image
PDImageXObject ximage = JPEGFactory.createFromStream(document, in);
contentStream.drawImage(ximage, 20, 20);

// Draw the red overlay image
Bitmap alphaImage = BitmapFactory.decodeStream(alpha);
PDImageXObject alphaXimage = LosslessFactory.createFromImage(document, alphaImage);
contentStream.drawImage(alphaXimage, 20, 20 );

// Make sure that the content stream is closed:
contentStream.close();

// Save the final pdf document to a file
String path = root.getAbsolutePath() + "/Created.pdf";
document.save(path);
document.close();
tv.setText("Successfully wrote PDF to " + path);

} catch (IOException e) {
Log.e("PdfBox-Android-Sample", "Exception thrown while creating PDF", e);
}
}

/**
* Loads an existing PDF and renders it to a Bitmap
*/
public void renderFile(View v) {
// Render the page and save it to an image file
try {
// Load in an already created PDF
PDDocument document = PDDocument.load(assetManager.open("Created.pdf"));
// Create a renderer for the document
PDFRenderer renderer = new PDFRenderer(document);
// Render the image to an RGB Bitmap
pageImage = renderer.renderImage(0, 1, ImageType.RGB);

// Save the render result to an image
String path = root.getAbsolutePath() + "/render.jpg";
File renderFile = new File(path);
FileOutputStream fileOut = new FileOutputStream(renderFile);
pageImage.compress(Bitmap.CompressFormat.JPEG, 100, fileOut);
fileOut.close();
tv.setText("Successfully rendered image to " + path);
// Optional: display the render result on screen
displayRenderedImage();
}
catch (IOException e)
{
Log.e("PdfBox-Android-Sample", "Exception thrown while rendering file", e);
}
}

/**
* Fills in a PDF form and saves the result
*/
public void fillForm(View v) {
try {
// Load the document and get the AcroForm
PDDocument document = PDDocument.load(assetManager.open("FormTest.pdf"));
PDDocumentCatalog docCatalog = document.getDocumentCatalog();
PDAcroForm acroForm = docCatalog.getAcroForm();

// Fill the text field
PDTextField field = (PDTextField) acroForm.getField("TextField");
field.setValue("Filled Text Field");
// Optional: don't allow this field to be edited
field.setReadOnly(true);

PDField checkbox = acroForm.getField("Checkbox");
((PDCheckBox) checkbox).check();

PDField radio = acroForm.getField("Radio");
((PDRadioButton)radio).setValue("Second");

PDField listbox = acroForm.getField("ListBox");
List<Integer> listValues = new ArrayList<>();
listValues.add(1);
listValues.add(2);
((PDListBox) listbox).setSelectedOptionsIndex(listValues);

PDField dropdown = acroForm.getField("Dropdown");
((PDComboBox) dropdown).setValue("Hello");

String path = root.getAbsolutePath() + "/FilledForm.pdf";
tv.setText("Saved filled form to " + path);
document.save(path);
document.close();
} catch (IOException e) {
Log.e("PdfBox-Android-Sample", "Exception thrown while filling form fields", e);
}
}

/**
* Strips the text from a PDF and displays the text on screen
*/
public void stripText(View v) {
String parsedText = null;
PDDocument document = null;
try {
document = PDDocument.load(assetManager.open("Hello.pdf"));
} catch(IOException e) {
Log.e("PdfBox-Android-Sample", "Exception thrown while loading document to strip", e);
}

try {
PDFTextStripper pdfStripper = new PDFTextStripper();
pdfStripper.setStartPage(0);
pdfStripper.setEndPage(1);
parsedText = "Parsed text: " + pdfStripper.getText(document);
}
catch (IOException e)
{
Log.e("PdfBox-Android-Sample", "Exception thrown while stripping text", e);
} finally {
try {
if (document != null) document.close();
}
catch (IOException e)
{
Log.e("PdfBox-Android-Sample", "Exception thrown while closing document", e);
}
}
tv.setText(parsedText);
}

/**
* Creates a simple pdf and encrypts it
*/
public void createEncryptedPdf(View v)
{
String path = root.getAbsolutePath() + "/crypt.pdf";

int keyLength = 128; // 128 bit is the highest currently supported

// Limit permissions of those without the password
AccessPermission ap = new AccessPermission();
ap.setCanPrint(false);

// Sets the owner password and user password
StandardProtectionPolicy spp = new StandardProtectionPolicy("12345", "hi", ap);

// Setups up the encryption parameters
spp.setEncryptionKeyLength(keyLength);
spp.setPermissions(ap);
BouncyCastleProvider provider = new BouncyCastleProvider();
Security.addProvider(provider);

PDFont font = PDType1Font.HELVETICA;
PDDocument document = new PDDocument();
PDPage page = new PDPage();

document.addPage(page);

try
{
PDPageContentStream contentStream = new PDPageContentStream(document, page);

// Write Hello World in blue text
contentStream.beginText();
contentStream.setNonStrokingColor(15, 38, 192);
contentStream.setFont(font, 12);
contentStream.newLineAtOffset(100, 700);
contentStream.showText("Hello World");
contentStream.endText();
contentStream.close();

// Save the final pdf document to a file
document.protect(spp); // Apply the protections to the PDF
document.save(path);
document.close();
tv.setText("Successfully wrote PDF to " + path);

}
catch (IOException e)
{
Log.e("PdfBox-Android-Sample", "Exception thrown while creating PDF for encryption", e);
}
}

/**
* Helper method for drawing the result of renderFile() on screen
*/
private void displayRenderedImage() {
new Thread() {
public void run() {
runOnUiThread(new Runnable() {
@Override
public void run() {
ImageView imageView = (ImageView) findViewById(R.id.renderedImageView);
imageView.setImageBitmap(pageImage);
}
});
}
}.start();
}
}

참조

아래에서 PdfBox-Android를 받으십시오.

번호위치링크
1.GitHub예제
2.GitHub직접 다운로드
3.GitHub찾아보기

4. PdfViewPager

SD 카드에 저장되거나 자산으로 연결되거나 원격 URL에서 다운로드된 PDF 문서를 렌더링할 수 있는 Android 위젯.

이 위젯은 활동 또는 조각에 PDF 문서를 표시할 수 있습니다.

중요 사항: PDFViewPagerPdfRenderer 클래스를 사용합니다.API 21 이상에서만 작동합니다. 자세한 내용은 공식 문서를 참조하세요.

데모는 다음과 같습니다.

PDFViewpager

1단계: 설치

앱 수준 build.gradle 파일에 다음 구현 문을 추가하여 설치합니다.

implementation 'es.voghdev.pdfviewpager:library:1.1.2'

2단계: 단계 레이아웃에 추가

PDFViewPager는 선언적으로 또는 명령적으로 페이지에 추가할 수 있습니다. 선언적으로 추가하려면 레이아웃에 다음을 추가하십시오.

<es.voghdev.pdfviewpager.library.PDFViewPager
android:id="@+id/pdfViewPager"
android:layout_width="match_parent"
android:layout_height="match_parent"/>

3단계: 코드 작성

자산 폴더에서 PDF 로드

PDF 파일이 자산 폴더에 있는 경우 캐시 디렉토리에 복사하십시오.

CopyAsset copyAsset = new CopyAssetThreadImpl(context, new Handler());
copyAsset.copy(asset, new File(getCacheDir(), "sample.pdf").getAbsolutePath(

그런 다음 다음과 같이 로드할 수 있습니다.

<es.voghdev.pdfviewpager.library.PDFViewPager
android:id="@+id/pdfViewPager"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:assetFileName="sample.pdf"/>

또는 다음과 같이:

pdfViewPager = new PDFViewPager(this, "sample.pdf");

그런 다음 다음과 같이 close()를 닫아 리소스를 해제합니다.

@Override
protected void onDestroy() {
super.onDestroy();

((PDFPagerAdapter) pdfViewPager.getAdapter()).close();
}

SD 카드에서 PDF 로드

PDF가 SD 카드에 있는 경우 PDFViewPager 개체를 만들어 SD 카드의 파일 위치를 전달합니다.

PDFViewPager pdfViewPager = new PDFViewPager(context, getPdfPathOnSDCard());

protected String getPdfPathOnSDCard() {
File f = new File(getExternalFilesDir("pdf"), "adobe.pdf");
return f.getAbsolutePath();
}

그런 다음 점유된 리소스를 해제합니다.

    @Override
protected void onDestroy() {
super.onDestroy();

((PDFPagerAdapter) pdfViewPager.getAdapter()).close();
}

원격 소스에서 PDF를 로드하는 방법

Android 매니페스트에 다음 권한을 추가하여 시작하세요.

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

그런 다음 Activity 또는 Fragment가 DownloadFile.Listener를 구현하도록 합니다.

public class RemotePDFActivity extends AppCompatActivity implements DownloadFile.Listener {

그런 다음 RemotePDFViewPager 개체를 만듭니다.

String url = "http://www.cals.uidaho.edu/edComm/curricula/CustRel_curriculum/content/sample.pdf";

RemotePDFViewPager remotePDFViewPager =
new RemotePDFViewPager(context, url, this);

그런 다음 이벤트를 처리합니다.

@Override
public void onSuccess(String url, String destinationPath) {
// That's the positive case. PDF Download went fine

adapter = new PDFPagerAdapter(this, "AdobeXMLFormsSamples.pdf");
remotePDFViewPager.setAdapter(adapter);
setContentView(remotePDFViewPager);
}

@Override
public void onFailure(Exception e) {
// This will be called if download fails
}

@Override
public void onProgressUpdate(int progress, int total) {
// You will get download progress here
// Always on UI Thread so feel free to update your views here
}

물론 어댑터를 닫습니다.

@Override
protected void onDestroy() {
super.onDestroy();

adapter.close();
}

아래에서 전체 예를 찾으십시오.

참조

번호링크
1.찾아보기 예제
2.읽기 자세히
3.팔로우 라이브러리 작성자