แสดงความคืบหน้าการดาวน์โหลดภายในกิจกรรมโดยใช้ DownloadManager


90

ฉันพยายามทำซ้ำความคืบหน้าเดียวกับที่ 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);
}

บันทึกของฉันทำงานได้อย่างสมบูรณ์ในขณะที่การดาวน์โหลดของฉันกำลังทำงานอยู่แจ้งว่า "กำลังดาวน์โหลด!" และเมื่อเสร็จสิ้น "ดาวน์โหลดเสร็จสมบูรณ์!" แต่สิ่งเดียวกันนี้ไม่เกิดขึ้นกับความคืบหน้าของฉันทำไม? ฉันต้องการความช่วยเหลือจริงๆตรรกะอื่น ๆ ในการทำเช่นนั้นได้รับการชื่นชมจริงๆ


เป็นไปได้ไหมว่าไฟล์ของคุณมีขนาดเล็กเกินไปและการดาวน์โหลดจะเสร็จสิ้นก่อนที่จะเผยแพร่ความคืบหน้า แบบสอบถามการดาวน์โหลดส่งคืนอะไรในงานของคุณ? หากงานถูกดำเนินการหลังจากช่วงเวลาหนึ่งคุณอาจมีการดำเนินการอื่น ๆ ที่ใช้งานได้ยาวนานในเธรดหลักของคุณ
Paul Lammertsma

ฉันอัปเดตรหัสแล้วคุณช่วยดูตอนนี้ได้ไหม และเกี่ยวกับความยาวไฟล์ที่ไม่เล็กเกินไปฉันสามารถดูความคืบหน้าการดาวน์โหลดได้ที่แถบการแจ้งเตือน
Victor Laerte

คำตอบ:


62

คุณกำลังหารจำนวนเต็มสองจำนวน:

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);

เพื่อให้ได้ความคืบหน้าในเปอร์เซ็นต์ไทล์ทั้งหมด (แม้ว่าจะมีพื้น)


@AZ_ ขอบคุณสำหรับการสนับสนุนของคุณ ฉันขอแนะนำให้คุณเพิ่มคำตอบของคุณเองด้วยโซลูชันที่ละเอียดยิ่งขึ้น
Paul Lammertsma

ไม่เป็นไรฉันไม่ต้องการใส่คำตอบอื่นของคำตอบที่ยอมรับแล้วเพราะมันจะยากสำหรับผู้ใช้ คุณสามารถเลือกที่จะไม่ยอมรับการแก้ไขของฉันได้ :)
AZ_

1
หากกิจกรรมของคุณสิ้นสุดลงและคุณต้องการยกเลิกการดาวน์โหลดคุณจะได้รับdivision by zeroข้อผิดพลาดร้ายแรง นั่นเป็นเหตุผลว่าทำไมฉันถึงชอบ final int dl_progress = ( bytes_total > 0 ? (int) ((bytes_downloaded * 100L) / bytes_total) : 0 );
KaHa6uc

17

คำตอบของ Paul ถูกต้อง แต่ด้วยการดาวน์โหลดที่มากขึ้นคุณจะเข้าสู่ max int ได้อย่างรวดเร็วและเริ่มได้รับความคืบหน้าเชิงลบ ฉันใช้สิ่งนี้เพื่อแก้ปัญหา:

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

คุณพูดถูก; ฉันได้แก้ไขคำตอบของฉันเพื่อให้แน่ใจว่าคนอื่นจะไม่ทำผิดแบบเดียวกัน
Paul Lammertsma

5

ในกรณีที่มีคนต้องการใช้งานความคืบหน้าการดาวน์โหลดจากคำถามของ @Victor Laerte ใน Kotlin กับ RxJava ที่นี่คุณไป:

DownloadStateRetriever.kt

class DownloadStateRetriever(private val downloadManager: DownloadManager) {

    fun retrieve(id: Long) {
        var downloading = AtomicBoolean(true)

        val disposable = Observable.fromCallable {
            val query = DownloadManager.Query().setFilterById(id)
            val cursor = downloadManager.query(query)

            cursor.moveToFirst()

            val bytesDownloaded = cursor.intValue(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR)
            val bytesTotal = cursor.intValue(DownloadManager.COLUMN_TOTAL_SIZE_BYTES)

            if (isSuccessful(cursor)) downloading.set(false)
            cursor.close()

            if (bytesTotal == 0) 0.toInt() else ((bytesDownloaded * 100F) / bytesTotal).toInt()
        }
                .subscribeOn(Schedulers.newThread())
                .delay(1, TimeUnit.SECONDS)
                .repeatUntil { !downloading.get() }
                .subscribe {
                    Timber.i("Subscribed to $id. progress: $it")
                }
    }

    private fun isSuccessful(cursor: Cursor) = status(cursor) == DownloadManager.STATUS_SUCCESSFUL

    private fun status(cursor: Cursor) = cursor.intValue(DownloadManager.COLUMN_STATUS)
}

ฉันได้เพิ่มส่วนขยายลงในเคอร์เซอร์เพื่อความชัดเจนของโค้ดมากขึ้น:

CursorExtensions.kt

import android.database.Cursor

fun Cursor.column(which: String) = this.getColumnIndex(which)
fun Cursor.intValue(which: String): Int = this.getInt(column(which))
fun Cursor.floatValue(which: String): Float = this.getFloat(column(which))
fun Cursor.stringValue(which: String): String = this.getString(column(which))
fun Cursor.doubleValue(which: String): Double = this.getDouble(column(which))


4

ดังที่คุณพอลกล่าวคุณกำลังหารจำนวนเต็มสองจำนวนโดยให้ผลลัพธ์เสมอ <1

มักจะโยนหมายเลขของคุณก่อนที่จะแบ่งซึ่งคำนวณและผลตอบแทนในการลอยจุด

อย่าลืมจัดการกับ DivByZero

final int dl_progress = (int) ((double)bytes_downloaded / (double)bytes_total * 100f);
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.