Использование BroadcastReceiver для обновления ProgressBar - PullRequest
1 голос
/ 03 апреля 2019

Я пытаюсь обновить ProgressBar из задачи Service.Я реализовал BroadcastReceiver, чтобы я мог взаимодействовать с потоком пользовательского интерфейса.Я обновляю ProgressBar в основной активности и получаю данные из операции MyService.Операция MyService выполняет задачу Async и обновляет намерение, которое должно быть отправлено обратно в методе OnProgressUpdate.

Вот мой код:

MainActivity:

package com.example.services;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.IntentFilter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.content.Intent;
import android.view.View;
import android.widget.ProgressBar;

import static android.content.Intent.ACTION_ATTACH_DATA;

public class MainActivity extends AppCompatActivity {

    private MyBroadRequestReceiver receiver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        IntentFilter filter = new IntentFilter(ACTION_ATTACH_DATA);
        receiver = new MyBroadRequestReceiver();
        registerReceiver( receiver, filter);
    }

    public void startService(View view) {
        startService(new Intent(getBaseContext(), MyService.class));
        //pb.setProgress();
    }
    public void stopService(View view) {
        stopService(new Intent(getBaseContext(), MyService.class));
    }
    public class MyBroadRequestReceiver extends BroadcastReceiver{

        @Override
        public void onReceive(Context context, Intent intent) {
            ProgressBar pb = (ProgressBar) findViewById(R.id.progressbar);
            int progress = intent.getFlags();
            pb.setProgress(progress);

        }

    }
}

MyService:

package com.example.services;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.widget.ProgressBar;
import android.widget.Toast;
import java.net.MalformedURLException;
import java.net.URL;
import android.os.AsyncTask;
import android.util.Log;
import java.util.Timer;
import java.util.TimerTask;

public class MyService extends Service {

    int counter = 0;
    static final int UPDATE_INTERVAL = 1000;
    private Timer timer = new Timer();

    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Toast.makeText(this, "Service Started", Toast.LENGTH_LONG).show();
        doSomethingRepeatedly();

        try {
            new DoBackgroundTask().execute(
                    new URL("http://www.amazon.com/somefiles.pdf"),
                    new URL("http://www.wrox.com/somefiles.pdf"),
                    new URL("http://www.google.com/somefiles.pdf"),
                    new URL("http://www.learn2develop.net/somefiles.pdf"));
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        if (timer != null){
            timer.cancel();
        }
        Toast.makeText(this, "Service Destroyed", Toast.LENGTH_LONG).show();
    }

    private void doSomethingRepeatedly() {
        timer.scheduleAtFixedRate(new TimerTask() {
            public void run() {
                Log.d("MyService", String.valueOf(++counter));

            }
        }, 0, UPDATE_INTERVAL);
    }

    private class DoBackgroundTask extends AsyncTask<URL, Integer, Long> {
        protected Long doInBackground(URL... urls) {
            int count = urls.length;
            long totalBytesDownloaded = 0;
            for (int i = 0; i < count; i++) {
                totalBytesDownloaded += DownloadFile(urls[i]);
                //Intent broadcastIntent = new Intent();
                //broadcastIntent.setAction(Intent.ACTION_ATTACH_DATA);
                //sendBroadcast(broadcastIntent);
                publishProgress((int) (((i + 1) / (float) count) * 100));
            }
            return totalBytesDownloaded;
        }

        protected void onProgressUpdate(Integer... progress) {
            Log.d("Downloading files", String.valueOf(progress[0]) + "% downloaded");
            Intent broadcastIntent = new Intent();
            broadcastIntent.setAction("com.example.services.MainActivity");
            //broadcastIntent.putExtra("progress",progress);
            broadcastIntent.setFlags(progress[0]);
            sendBroadcast(broadcastIntent);
            Toast.makeText(getBaseContext(),
                    String.valueOf(progress[0]) + "% downloaded-"+counter,
                    Toast.LENGTH_LONG).show();

        }

        protected void onPostExecute(Long result) {
            Toast.makeText(getBaseContext(), "Downloaded " + result + " bytes",
                    Toast.LENGTH_LONG).show();
            //stopSelf();
        }
    }

    private int DownloadFile(URL url) {
        try {
            //---simulate taking some time to download a file---
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //---return an arbitrary number representing
        // the size of the file downloaded---
        return 100;
    }
}

Пожалуйста, посмотрите на мой onProgressUpdate и скажите мне, если я делаю это исправление.Мой ProgressBar вообще не обновляется.

Ответы [ 4 ]

0 голосов
/ 03 апреля 2019

Есть некоторые проблемы с вашим кодом.

Во-первых, если вам нужно обновить индикатор выполнения операции из службы, эффективный способ - использовать LocalBroadcastManager вместо обычной трансляции (известной как глобальная).широковещание).

Во-вторых, поскольку в службе вы используете broadcastIntent.setAction("com.example.services.MainActivity");, а затем в упражнении вы должны использовать то же действие для получения широковещания.

IntentFilter filter = new IntentFilter("com.example.services.MainActivity");

вместо

IntentFilter filter = new IntentFilter(ACTION_ATTACH_DATA);

Наконец, поскольку вы используете AsyncTask в службе, чтобы избежать утечки контекста службы, рекомендуется объявлять asynctask как статический класс.

Соберите все вместе.

MyService.java

public class MyService extends Service {

    int counter = 0;
    static final int UPDATE_INTERVAL = 1000;
    private Timer timer = new Timer();

    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Toast.makeText(this, "Service Started", Toast.LENGTH_LONG).show();
        doSomethingRepeatedly();

        try {
            new DoBackgroundTask(this).execute(
                    new URL("http://www.amazon.com/somefiles.pdf"),
                    new URL("http://www.wrox.com/somefiles.pdf"),
                    new URL("http://www.google.com/somefiles.pdf"),
                    new URL("http://www.learn2develop.net/somefiles.pdf"));
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }

        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        if (timer != null) {
            timer.cancel();
        }
        Toast.makeText(this, "Service Destroyed", Toast.LENGTH_LONG).show();
    }

    private void doSomethingRepeatedly() {
        timer.scheduleAtFixedRate(new TimerTask() {
            public void run() {
                Log.d("MyService", String.valueOf(++counter));

            }
        }, 0, UPDATE_INTERVAL);
    }

    // Declare asynctask as static class.
    private static class DoBackgroundTask extends AsyncTask<URL, Integer, Long> {

        // Using WeakReference to keep the context of the service to avoid leaking.
        private WeakReference<Context> mContext;

        DoBackgroundTask(Context context) {
            mContext = new WeakReference<>(context);
        }

        protected Long doInBackground(URL... urls) {
            int count = urls.length;
            long totalBytesDownloaded = 0;
            for (int i = 0; i < count; i++) {
                totalBytesDownloaded += DownloadFile(urls[i]);
                //Intent broadcastIntent = new Intent();
                //broadcastIntent.setAction(Intent.ACTION_ATTACH_DATA);
                //sendBroadcast(broadcastIntent);
                publishProgress((int) (((i + 1) / (float) count) * 100));
            }
            return totalBytesDownloaded;
        }

        protected void onProgressUpdate(Integer... progress) {
            Log.d("Downloading files", String.valueOf(progress[0]) + "% downloaded");
            Intent broadcastIntent = new Intent();
            broadcastIntent.setAction("com.example.services.MainActivity");
            //broadcastIntent.putExtra("progress",progress);
            broadcastIntent.setFlags(progress[0]);

            Context context = mContext.get();
            if (context != null) {
                LocalBroadcastManager.getInstance(context).sendBroadcast(broadcastIntent);

                int counter = ((MyService)context).counter;
                Toast.makeText(context,
                        String.valueOf(progress[0]) + "% downloaded-" + counter,
                        Toast.LENGTH_LONG).show();
            }
        }

        protected void onPostExecute(Long result) {
            Context context = mContext.get();
            if (context != null) {
                Toast.makeText(context, "Downloaded " + result + " bytes",
                        Toast.LENGTH_LONG).show();
                //stopSelf();
            }
        }

        private int DownloadFile(URL url) {
            try {
                //---simulate taking some time to download a file---
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //---return an arbitrary number representing
            // the size of the file downloaded---
            return 100;
        }
    }
}

MainActivity.java

public class MainActivity extends AppCompatActivity {

    private MyBroadRequestReceiver receiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void onStart() {
        super.onStart();
        // Register broadcast receiver.
        IntentFilter filter = new IntentFilter("com.example.services.MainActivity");
        receiver = new MyBroadRequestReceiver();
        LocalBroadcastManager.getInstance(this).registerReceiver(receiver, filter);
    }

    @Override
    protected void onStop() {
        // Unregister broadcast receiver.
        LocalBroadcastManager.getInstance(this).unregisterReceiver(receiver);
        super.onStop();
    }

    public void startService(View view) {
        startService(new Intent(getBaseContext(), MyService.class));
    }

    public void stopService(View view) {
        stopService(new Intent(getBaseContext(), MyService.class));
    }

    public class MyBroadRequestReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
            ProgressBar pb = (ProgressBar) findViewById(R.id.progressbar);
            int progress = intent.getFlags();
            pb.setProgress(progress);
        }
    }
}
0 голосов
/ 03 апреля 2019

Во-первых, это не хорошее решение проблемы, которую вы решаете.Пожалуйста, просмотрите Google Android docs Руководство по Backgournd

Я предлагаю вам перейти на DownloadManager.

0 голосов
/ 03 апреля 2019

Ваш фильтр намерений определен как "ACTION_ATTACH_DATA"

IntentFilter filter = new IntentFilter(ACTION_ATTACH_DATA);

Итак, отправьте вашу трансляцию так:

Intent i = new Intent(ACTION_ATTACH_DATA);
sendBroadcast(i);

Также не забудьте отменить регистрацию трансляции на onDestroy

0 голосов
/ 03 апреля 2019

Поскольку вы не запускаете метод ononCreate (). Служба не будет работать.

...