web-dev-qa-db-ja.com

DownloadManagerを使用してアクティビティ内のダウンロードの進行状況を表示する

DownloadManagerがアプリ内の通知バーに表示するのと同じ進行状況を再現しようとしていますが、進行状況は公開されません。 runOnUiThread()を使用して更新しようとしていますが、何らかの理由で更新されていません。

私のダウンロード:

String urlDownload = "https://dl.dropbox.com/s/ex4clsfmiu142dy/test.zip?token_hash=AAGD-XcBL8C3flflkmxjbzdr7_2W_i6CZ_3rM5zQpUCYaw&dl=1";
DownloadManager.Request request = new DownloadManager.Request(Uri.parse(urlDownload));

request.setDescription("Testando");
request.setTitle("Download");
request.allowScanningByMediaScanner();
request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, "teste.Zip");

final DownloadManager manager = (DownloadManager) getSystemService(Context.DOWNLOAD_SERVICE);

final long downloadId = manager.enqueue(request);

final ProgressBar mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);

new Thread(new Runnable() {

    @Override
    public void run() {

        boolean downloading = true;

        while (downloading) {

            DownloadManager.Query q = new DownloadManager.Query();
            q.setFilterById(downloadId);

            Cursor cursor = manager.query(q);
            cursor.moveToFirst();
            int bytes_downloaded = cursor.getInt(cursor
                    .getColumnIndex(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR));
            int bytes_total = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_TOTAL_SIZE_BYTES));

            if (cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS)) == DownloadManager.STATUS_SUCCESSFUL) {
                downloading = false;
            }

            final double dl_progress = (bytes_downloaded / bytes_total) * 100;

            runOnUiThread(new Runnable() {

                @Override
                public void run() {

                    mProgressBar.setProgress((int) dl_progress);

                }
            });

            Log.d(Constants.MAIN_VIEW_ACTIVITY, statusMessage(cursor));
            cursor.close();
        }

    }
}).start();

私のstatusMessageメソッド:

private String statusMessage(Cursor c) {
    String msg = "???";

    switch (c.getInt(c.getColumnIndex(DownloadManager.COLUMN_STATUS))) {
    case DownloadManager.STATUS_FAILED:
        msg = "Download failed!";
        break;

    case DownloadManager.STATUS_PAUSED:
        msg = "Download paused!";
        break;

    case DownloadManager.STATUS_PENDING:
        msg = "Download pending!";
        break;

    case DownloadManager.STATUS_RUNNING:
        msg = "Download in progress!";
        break;

    case DownloadManager.STATUS_SUCCESSFUL:
        msg = "Download complete!";
        break;

    default:
        msg = "Download is nowhere in sight";
        break;
    }

    return (msg);
}

ログは完全に機能し、ダウンロードの実行中は「ダウンロード中!」と表示され、完了したら「ダウンロードが完了しました!」と表示されますが、進行中に同じことが発生しません。ヘルプ、それを行う他のロジックは本当に感謝しています

81
Victor Laerte

2つの整数を分割しています:

final double dl_progress = (bytes_downloaded / bytes_total) * 100;

なので bytes_downloaded よりも少ない bytes_total(bytes_downloaded / bytes_total)は0になるため、進行状況は常に0になります。

計算を変更します

final int dl_progress = (int) ((bytes_downloaded * 100l) / bytes_total);

パーセンタイル全体(フロアリングされていても)の進捗状況を取得します。

59
Paul Lammertsma

Paulの答えは正しいですが、ダウンロードが大きい場合は、max intを非常にすばやくヒットし、ネガティブな進行を始めます。私はこれを使用して問題を解決しました:

final int dl_progress = (int) ((bytes_downloaded * 100l) / bytes_total);
17
JustinMorris

Paulが言ったように、2つの整数を分割し、結果は常に1未満です。

常に数字をキャスト除算。浮動小数点で計算して戻ります。

DivByZeroを処理することを忘れないでください。

final int dl_progress = (int) ((double)bytes_downloaded / (double)bytes_total * 100f);
5
Dennis C