2012-04-03 62 views
1

目標是能夠將PDF文件添加到應用程序的「資產」文件夾中,並讓用戶打開該PDF PDF應用程序(ezPDFReader,Adobe Reader等)。這並非完全簡單,因爲PDF首先需要將資產文件夾複製到設備存儲上。如果設備上沒有PDF閱讀器應用程序,我們需要顯示一條警告消息並指示他們到Play商店下載。如何從Android應用程序打開PDF(在單獨的PDF查看器應用程序中)

回答

6

這是爲我工作的解決方案。這是一個名爲PdfHandler的類,下面介紹如何使用它 - 只需將pdf文件放入資源文件夾。如果可用,它將在PDF查看器應用程序中打開它,並且如果不顯示警告消息並將用戶引導至Adobe Reader Play商店條目。

PdfHandler pdf = new PdfHandler(this); 
pdf.openPdf("document.pdf"); 

PdfHandler類。您應該將文本資源移動到Android字符串資源中。

import android.app.AlertDialog; 
import android.content.Context; 
import android.content.DialogInterface; 
import android.content.Intent; 
import android.content.pm.PackageManager; 
import android.content.res.AssetManager; 
import android.net.Uri; 
import android.util.Log; 

import java.io.File; 
import java.io.FileOutputStream; 
import java.io.IOException; 
import java.io.InputStream; 
import java.util.List; 

/** 
* Handles the opening of PDF files, using whatever app the user has installed to view PDFs 
* If no PDF app installed, shows a warning and directs them to appropriate Store listing 
*/ 
public class PdfHandler { 

    public PdfHandler(Context context) { 
     this.context = context; 
    } 

    public void openPdf(String filename) { 
     if (isPdfAppAvailable()) { 
      copyPdfAndOpenIt(filename); 
     } else { 
      showPdfWarning(); 
     } 
    } 

    private void copyPdfAndOpenIt(String filename) { 
     try { 
      File file = copyPdfFromAssetsToStorage(filename); 
      startPdfIntent(file); 
     } catch (Exception e) { 
      Log.e("PdfHandler", "Error handling the PDF file", e); 
     } 
    } 

    private File copyPdfFromAssetsToStorage(String filename) throws Exception { 
     String tempFilename = "temp.pdf"; 
     AssetManager is = context.getAssets(); 
     InputStream inputStream = is.open(filename); 
     String outFilename = context.getFilesDir() + "/" + tempFilename; 
     FileOutputStream outputStream = context.openFileOutput(tempFilename, Context.MODE_WORLD_READABLE); 
     copy(inputStream, outputStream); 
     outputStream.flush(); 
     outputStream.close(); 
     inputStream.close(); 
     return new File(outFilename); 
    } 

    private void copy(InputStream fis, FileOutputStream fos) throws IOException { 
     byte[] b = new byte[8]; 
     int i; 
     while ((i = fis.read(b)) != -1) { 
      fos.write(b, 0, i); 
     } 
    } 

    private boolean isPdfAppAvailable() { 
     PackageManager packageManager = context.getPackageManager(); 
     Intent testIntent = new Intent(Intent.ACTION_VIEW); 
     testIntent.setType("application/pdf"); 
     List list = packageManager.queryIntentActivities(testIntent, PackageManager.MATCH_DEFAULT_ONLY); 
     return list.size() > 0; 
    } 

    private void startPdfIntent(File file) { 
     Intent intent = new Intent(); 
     intent.setAction(Intent.ACTION_VIEW); 
     Uri uri = Uri.fromFile(file); 
     intent.setDataAndType(uri, "application/pdf"); 
     context.startActivity(intent); 
    } 

    private void showPdfWarning() { 
     AlertDialog.Builder builder = new AlertDialog.Builder(context); 
     builder.setMessage("Please install an app to view PDF file"); 
     builder.setCancelable(false); 
     builder.setPositiveButton("Install", getButtonListener()); 
     builder.setNegativeButton("Cancel", null); 
     builder.setTitle("PDF Viewer"); 
     AlertDialog alert = builder.create(); 
     alert.show(); 
    } 

    private DialogInterface.OnClickListener getButtonListener() { 
     return new DialogInterface.OnClickListener() { 
       public void onClick(DialogInterface dialogInterface, int i) { 
        goToGooglePlayStoreEntry(); 
       } 
      }; 
    } 

    private void goToGooglePlayStoreEntry() { 
     context.startActivity(getAppListingIntent()); 
    } 

    private Intent getAppListingIntent() { 
     Intent intent = new Intent(Intent.ACTION_VIEW); 
     String pdfAppPackageName = "com.adobe.reader"; 
     intent.setData(Uri.parse("market://details?id=" + pdfAppPackageName)); 
     intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 
     return intent; 
    } 

    private Context context; 

} 
+0

+1。很有用。一個很好的增強將是緩存文件的選項,以便它們不需要每次都被複制,在需要更快加載和更少電量消耗的情況下,並且不需要擔心持久性存儲使用情況。 – Nate 2013-02-16 23:30:14

+0

我很欣賞你的努力。它真的幫助其他人打開pdf外部應用程序。你能否給我建議我如何在沒有行動意圖的情況下在我的應用程序中打開pdf。 – 2017-04-03 09:35:44

0
LoginProgress task = new LoginProgress(); 
task.execute(Pdffile); 




public class LoginProgress extends AsyncTask<Void, Void, Void> { 
     private ProgressDialog dialog; 
     protected Context applicationContext; 
     @Override 
     protected void onPostExecute(Void result) { 
      super.onPostExecute(result); 
      this.dialog.dismiss(); 
     } 
     @Override 
     protected void onPreExecute() { 
      this.dialog = ProgressDialog.show(applicationContext, 
        "Data Loading", "Loading .....", true); 
     } 
     @Override 
     protected Void doInBackground(Void... params) { 
      // video(); 
      LoadUrl(Pdffile); 
      return null; 
     } 
     private void LoadUrl(File Pdffile) { 
      try { 
       u r stuff here 
      } catch (Exception e) { 
       e.printStackTrace(); 
      } 
     } 
    } 
1

發表奧利下的PdfHandler完美的作品,但它是一個有點慢,因爲它複製PDF首先從資產存儲它才能打開。如果使用ContentProvider,打開PDF將會更快。

請注意,如果您的文件名是MyPdf.pdf,則該文件應該是assets/public_pdfs/MyPdf.pdf.mp3。路徑public_pdfs僅用於導出您真正想要導出的pdf,並且.mp3擴展名用於防止壓縮。

AndroidManifest.xml中

<provider android:authorities="my.app.PdfContentProvider" android:enabled="true" android:exported="true" android:name="my.app.PdfContentProvider"> 
</provider> 

打開PDF

Intent intent = new Intent(); 
intent.setAction(Intent.ACTION_VIEW); 

Uri uri = Uri.parse("content://my.app.PdfContentProvider/" + filename); 
intent.setDataAndType(uri, "application/pdf"); 

startActivity(intent); 

PdfContentProvider.java

public class PdfContentProvider extends ContentProvider 
{ 
    private static final String PDFPATH = "public_pdfs/"; 

    @Override 
    public String getType(Uri uri) 
    { 
    return "application/pdf"; 
    } 

    @Override 
    public AssetFileDescriptor openAssetFile(Uri uri, String mode) throws FileNotFoundException 
    { 
    AssetManager am = getContext().getAssets(); 
    String file_name = uri.getLastPathSegment(); 

    if (file_name == null) throw new FileNotFoundException(); 
    AssetFileDescriptor afd = null; 
    try 
    { 
     afd = am.openFd(PDFPATH + file_name + ".mp3"); 
    } 
    catch (IOException e) 
    { 
     e.printStackTrace(); 
    } 
    return afd; 
    } 

    private final static String[] COLUMNS = {OpenableColumns.DISPLAY_NAME, OpenableColumns.SIZE}; 

    @Override 
    /** 
    * This function is required for it to work on the Quickoffice at Android 4.4 (KitKat) 
    */ 
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) 
    { 
    if (projection == null) 
    { 
     projection = COLUMNS; 
    } 

    String[] cols = new String[projection.length]; 
    Object[] values = new Object[projection.length]; 
    int i = 0; 
    for (String col : projection) 
    { 
     if (OpenableColumns.DISPLAY_NAME.equals(col)) 
     { 
     cols[i] = OpenableColumns.DISPLAY_NAME; 
     values[i++] = uri.getLastPathSegment(); 
     } 
     else if (OpenableColumns.SIZE.equals(col)) 
     { 
     cols[i] = OpenableColumns.SIZE; 
     values[i++] = AssetFileDescriptor.UNKNOWN_LENGTH; 
     } 
    } 

    cols = copyOf(cols, i); 
    values = copyOf(values, i); 

    final MatrixCursor cursor = new MatrixCursor(cols, 1); 
    cursor.addRow(values); 
    return cursor; 
    } 

    private static String[] copyOf(String[] original, int newLength) 
    { 
    final String[] result = new String[newLength]; 
    System.arraycopy(original, 0, result, 0, newLength); 
    return result; 
    } 

    private static Object[] copyOf(Object[] original, int newLength) 
    { 
    final Object[] result = new Object[newLength]; 
    System.arraycopy(original, 0, result, 0, newLength); 
    return result; 
    } 

    @Override 
    public boolean onCreate() 
    { 
    return true; 
    } 

    @Override 
    public Uri insert(Uri uri, ContentValues values) 
    { 
    return null; 
    } 

    @Override 
    public int delete(Uri uri, String selection, String[] selectionArgs) 
    { 
    return 0; 
    } 

    @Override 
    public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) 
    { 
    return 0; 
    } 
} 
相關問題