studio downloadmanager android download-manager

downloadmanager - download file android studio



Android DownloadManager API: ¿abrir archivo después de descargar? (2)

Estoy enfrentando el problema de abrir el archivo descargado después de la descarga exitosa a través de la API DownloadManager. En mi código:

Uri uri=Uri.parse("http://www.nasa.gov/images/content/206402main_jsc2007e113280_hires.jpg"); Environment .getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS) .mkdirs(); lastDownload = mgr.enqueue(new DownloadManager.Request(uri) .setAllowedNetworkTypes(DownloadManager.Request.NETWORK_WIFI | DownloadManager.Request.NETWORK_MOBILE) .setAllowedOverRoaming(false) .setTitle("app update") .setDescription("New version 1.1") .setShowRunningNotification(true) .setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, "a.apk")); Cursor c=mgr.query(new DownloadManager.Query().setFilterById(lastDownload)); if(c.getInt(c.getColumnIndex(DownloadManager.COLUMN_STATUS)) == 8) { try { mgr.openDownloadedFile(c.getLong(c.getColumnIndex(DownloadManager.COLUMN_ID))); } catch (NumberFormatException e) { // TODO Auto-generated catch block e.printStackTrace(); Log.d("MGR", "Error"); } catch (FileNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); Log.d("MGR", "Error"); } }

El problema es cuándo se if(c.getInt(c.getColumnIndex(DownloadManager.COLUMN_STATUS))==8) . Tengo el estado -1 y una excepción. ¿Hay alguna forma mejor de abrir archivos descargados con la DownloadManager API ? En mi ejemplo, estoy descargando una imagen grande, en una situación real, estaría descargando un archivo APK y necesito mostrar un cuadro de diálogo de instalación inmediatamente después de la actualización.

Edición: Me di cuenta de que el estado = 8 es después de la descarga exitosa. Es posible que tenga un enfoque diferente de "comprobar la descarga exitosa"

Gracias


Debe registrar un receptor para cuando se complete la descarga:

registerReceiver(onComplete, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));

y un controlador BroadcastReciever

BroadcastReceiver onComplete=new BroadcastReceiver() { public void onReceive(Context ctxt, Intent intent) { // Do Something } };

Compre en lugar de que yo lo robe todo, le sugiero que lo compruebe.

EDITAR:

Solo como una sugerencia, no recomendaría el uso de API 9 todavía: http://developer.android.com/resources/dashboard/platform-versions.html

Hay maneras de evitar esto, creando su propio controlador de descargas, como lo hice yo, porque no queríamos alejar a la mayoría de la base de usuarios de nuestro Android, para eso necesitará: Crear AsyncTask que maneje la descarga del archivo.

y recomendaré crear un cuadro de diálogo de descarga de algún tipo (si dices que es un archivo grande, lo haría aparecer en el área de notificación).

y que necesitarás para manejar la apertura del archivo:

protected void openFile(String fileName) { Intent install = new Intent(Intent.ACTION_VIEW); install.setDataAndType(Uri.fromFile(new File(fileName)), "MIME-TYPE"); startActivity(install); }


Problema

Android DownloadManager API: ¿abrir archivo después de descargar?

Solución

/** * Used to download the file from url. * <p/> * 1. Download the file using Download Manager. * * @param url Url. * @param fileName File Name. */ public void downloadFile(final Activity activity, final String url, final String fileName) { try { if (url != null && !url.isEmpty()) { Uri uri = Uri.parse(url); activity.registerReceiver(attachmentDownloadCompleteReceive, new IntentFilter( DownloadManager.ACTION_DOWNLOAD_COMPLETE)); DownloadManager.Request request = new DownloadManager.Request(uri); request.setMimeType(getMimeType(uri.toString())); request.setTitle(fileName); request.setDescription("Downloading attachment.."); request.allowScanningByMediaScanner(); request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName); DownloadManager dm = (DownloadManager) activity.getSystemService(Context.DOWNLOAD_SERVICE); dm.enqueue(request); } } catch (IllegalStateException e) { Toast.makeText(activity, "Please insert an SD card to download file", Toast.LENGTH_SHORT).show(); } } /** * Used to get MimeType from url. * * @param url Url. * @return Mime Type for the given url. */ private String getMimeType(String url) { String type = null; String extension = MimeTypeMap.getFileExtensionFromUrl(url); if (extension != null) { MimeTypeMap mime = MimeTypeMap.getSingleton(); type = mime.getMimeTypeFromExtension(extension); } return type; } /** * Attachment download complete receiver. * <p/> * 1. Receiver gets called once attachment download completed. * 2. Open the downloaded file. */ BroadcastReceiver attachmentDownloadCompleteReceive = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (DownloadManager.ACTION_DOWNLOAD_COMPLETE.equals(action)) { long downloadId = intent.getLongExtra( DownloadManager.EXTRA_DOWNLOAD_ID, 0); openDownloadedAttachment(context, downloadId); } } }; /** * Used to open the downloaded attachment. * * @param context Content. * @param downloadId Id of the downloaded file to open. */ private void openDownloadedAttachment(final Context context, final long downloadId) { DownloadManager downloadManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE); DownloadManager.Query query = new DownloadManager.Query(); query.setFilterById(downloadId); Cursor cursor = downloadManager.query(query); if (cursor.moveToFirst()) { int downloadStatus = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS)); String downloadLocalUri = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI)); String downloadMimeType = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_MEDIA_TYPE)); if ((downloadStatus == DownloadManager.STATUS_SUCCESSFUL) && downloadLocalUri != null) { openDownloadedAttachment(context, Uri.parse(downloadLocalUri), downloadMimeType); } } cursor.close(); } /** * Used to open the downloaded attachment. * <p/> * 1. Fire intent to open download file using external application. * * 2. Note: * 2.a. We can''t share fileUri directly to other application (because we will get FileUriExposedException from Android7.0). * 2.b. Hence we can only share content uri with other application. * 2.c. We must have declared FileProvider in manifest. * 2.c. Refer - https://developer.android.com/reference/android/support/v4/content/FileProvider.html * * @param context Context. * @param attachmentUri Uri of the downloaded attachment to be opened. * @param attachmentMimeType MimeType of the downloaded attachment. */ private void openDownloadedAttachment(final Context context, Uri attachmentUri, final String attachmentMimeType) { if(attachmentUri!=null) { // Get Content Uri. if (ContentResolver.SCHEME_FILE.equals(attachmentUri.getScheme())) { // FileUri - Convert it to contentUri. File file = new File(attachmentUri.getPath()); attachmentUri = FileProvider.getUriForFile(activity, "com.freshdesk.helpdesk.provider", file);; } Intent openAttachmentIntent = new Intent(Intent.ACTION_VIEW); openAttachmentIntent.setDataAndType(attachmentUri, attachmentMimeType); openAttachmentIntent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); try { context.startActivity(openAttachmentIntent); } catch (ActivityNotFoundException e) { Toast.makeText(context, context.getString(R.string.unable_to_open_file), Toast.LENGTH_LONG).show(); } } }

Inicializar detalles de FileProvider

Decleare FileProvider en AndroidManifest

<provider android:name="android.support.v4.content.FileProvider" android:authorities="com.freshdesk.helpdesk.provider" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_path"/> </provider>

Agregue el siguiente archivo "res -> xml -> file_path.xml"

<?xml version="1.0" encoding="utf-8"?> <paths xmlns:android="http://schemas.android.com/apk/res/android"> <external-path name="attachment_file" path="."/> </paths>

Nota

¿Por qué usar FileProvider?

  1. Desde Android 7.0 no podemos compartir FileUri con otra aplicación.
  2. Usando "DownloadManager.COLUMN_LOCAL_URI" obtendremos solo FileUri, por lo tanto, debemos convertirlo en ContentUri y compartir con otra aplicación.

Problema con el uso de "DownloadManager.getUriForDownloadedFile (id largo)"

  1. No use "DownloadManager.getUriForDownloadedFile (long id)" - Para obtener Uri desde downloadId para abrir el archivo usando una aplicación externa.
  2. Debido a que el método "getUriForDownloadedFile" de Android 6.0 y 7.0 devuelve uri local (al que solo puede acceder nuestra aplicación), no podemos compartir ese Uri con otra aplicación porque no pueden acceder a ese uri (pero está corregido en Android 7.1 ver Android confirmar aquí ).
  3. Refere el código fuente de Android DownloadManager.java y Downloads.java
  4. Por lo tanto, siempre use la Columna "DownloadManager.COLUMN_LOCAL_URI" para obtener Uri.

Referencia

  1. https://developer.android.com/reference/android/app/DownloadManager.html
  2. https://developer.android.com/reference/android/support/v4/content/FileProvider.html