Estou tentando reproduzir o mesmo progresso que o DownloadManager mostra na barra de notificação dentro do meu aplicativo, mas meu progresso nunca é publicado. Estou tentando atualizá-lo usando runOnUiThread (), mas por algum motivo ele não foi atualizado.
meu download:
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();
meu método 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);
}
Meu log está funcionando perfeitamente, enquanto meu download está em execução diz "Download em andamento!" e quando estiver completo "Download concluído!", mas o mesmo não ocorre no meu progresso, por quê? Eu realmente preciso de alguma ajuda, outras lógicas para fazer que são muito apreciadas
android
progress-bar
Victor Laerte
fonte
fonte
Respostas:
Você está dividindo dois inteiros:
Se
bytes_downloaded
for menor quebytes_total
,(bytes_downloaded / bytes_total)
será 0 e seu progresso, portanto, sempre será 0.Altere o seu cálculo para
para obter o progresso em percentis inteiros (embora com base).
fonte
division by zero
erro fatal. É por isso que eu fiz assimfinal int dl_progress = ( bytes_total > 0 ? (int) ((bytes_downloaded * 100L) / bytes_total) : 0 );
A resposta de Paul está correta, mas com downloads maiores você atingirá o máximo int bem rápido e começará a ter um progresso negativo. Usei isso para resolver o problema:
fonte
Como paul disse, você está dividindo dois inteiros, com resultado sempre <1.
Sempre lance seu número antes de dividir, que calcula e retorna em ponto flutuante.
Não se esqueça de lidar com DivByZero.
fonte
Caso alguém precise da implementação do recuperador de progresso de download da pergunta de @Victor Laerte em Kotlin com RxJava, aqui vai:
DownloadStateRetriever.kt
Eu adicionei extensões ao cursor para mais clareza de código:
CursorExtensions.kt
fonte