简体   繁体   中英

How to update android fragment UI from service?

I am developing an android application which will play live radio audio stream. I have implemeted android MediaPlayer API for this purpose which constantly receives the audio data from a remote source.

On the click of play button, I have started a Service and controling the MediaPlayer instance from there. I am successful in doing all so and my working code is as follows:

ListenFragment.java

playbtn_flat.setOnClickListener(new View.OnClickListener()
    {
        public void onClick(View v)
        {
            // TODO Auto-generated method stub
            if(isNetworkConnected())
            {
                serviceIntent.putExtra("sentAudioLink",strAudioLink);
                try
                {
                    getActivity().startService(new Intent(getContext(),PlayService.class));
                }
                catch (Exception e)
                {
                    Toast.makeText(getContext(),e.getClass().getName()+" "+e.getMessage(),Toast.LENGTH_SHORT).show();
                }

            }

        }
    });

PlayService.java:

public class PlayService extends Service implements MediaPlayer.OnCompletionListener,MediaPlayer.OnPreparedListener,
        MediaPlayer.OnErrorListener,MediaPlayer.OnSeekCompleteListener,MediaPlayer.OnInfoListener,
        MediaPlayer.OnBufferingUpdateListener {

    private MediaPlayer mediaPlayer = new MediaPlayer();
    private String sentAudioLink;

    @Override
    public void onCreate() {
        mediaPlayer.setWakeMode(getApplicationContext(), PowerManager.PARTIAL_WAKE_LOCK);
        mediaPlayer.setOnCompletionListener(this);
        mediaPlayer.setOnPreparedListener(this);
        mediaPlayer.setOnErrorListener(this);
        mediaPlayer.setOnSeekCompleteListener(this);
        mediaPlayer.setOnInfoListener(this);
        mediaPlayer.setOnBufferingUpdateListener(this);
        mediaPlayer.reset();
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        sentAudioLink = intent.getExtras().getString("sentAudioLink");
        mediaPlayer.reset();
        //setup the media player data source using the strAudioLink value
        if (!mediaPlayer.isPlaying()) {
            try {
                mediaPlayer.setDataSource(sentAudioLink);
                //prepare media player
                mediaPlayer.prepareAsync();
            } catch (IllegalStateException e) {
                Log.e("workingerror1", e.toString());
            } catch (IllegalArgumentException e) {
                Log.e("workingerror2", e.toString());
            } catch (IOException e) {
                Log.e("workingerror3", e.toString());
            }
        }
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.stop();
        }
        mediaPlayer.release();
    }

    @Override
    public void onBufferingUpdate(MediaPlayer mp, int percent) {

    }

    @Override
    public void onCompletion(MediaPlayer mp) {
        stopMedia();
        stopSelf();
    }

    public void stopMedia() {
        if (mediaPlayer.isPlaying())
            mediaPlayer.stop();
    }

    @Override
    public boolean onError(MediaPlayer mp, int what, int extra) {
        Toast.makeText(this, "Error" + what, Toast.LENGTH_LONG).show();
        Log.e("workingmain_error", "error");
        Log.e("workingmain_error2", String.format("Error(%s%s)", what, extra));
        return false;
    }

    @Override
    public boolean onInfo(MediaPlayer mp, int what, int extra) {
        return false;
    }

    @Override
    public void onPrepared(MediaPlayer mp) {
        playMedia();
    }

    public void playMedia() {
        if (!mediaPlayer.isPlaying()) {
            Log.e("working4", "error");
            mediaPlayer.start();
        }
    }

    @Override
    public void onSeekComplete(MediaPlayer mp) {

    }

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

The only problem that i face is that I want to change the text in UI TextView from " Listen Live " to " Loading... " and then to " On Air " as the media player gets started.

my ListenFragment UI is: 在此处输入图片说明

You can use broadcast receiver for that purpose. In your service send a broadcast, and inside your fragment you can register to listen for that and the change your UI according to that.

For eg, use

 Intent intent = new Intent("intent_filter");
 intent.putString("messageToShow","On Air");
 LocalBroadcastManager localBroadcastManager = LocalBroadcastManager.getInstance(this);
 localBroadcastManager.sendBroadcast(intent);

And for receiving that in your fragment use:

 LocalBroadcastManager.getInstance(this).registerReceiver(new YourReceiver(),new IntentFilter("intent_filter"));

You can read more on https://developer.android.com/reference/android/support/v4/content/LocalBroadcastManager

You can also take a look at EventBus.

A possible solution will be to use a bound service https://developer.android.com/guide/components/bound-services to have a reference to an IBinder instance and let you communicate to your service. The communication (from service to fragment) can be done by providing a listener object ( IPlayServiceListener for instance).

This is how your service might look like:

public class PlayService extends Service implements MediaPlayer.OnCompletionListener,MediaPlayer.OnPreparedListener,
        MediaPlayer.OnErrorListener,MediaPlayer.OnSeekCompleteListener,MediaPlayer.OnInfoListener,
        MediaPlayer.OnBufferingUpdateListener {


    private final IBinder mBinder = new PlayServiceLocalBinder();
    private IPlayServiceListener listener = null;

    /** Binder allowing external components to interact with the service */
    public class PlayServiceLocalBinder extends Binder {
        public PlayService getService() {
            // Return this instance of LocalService so clients can call public methods
            return PlayService.this;
        }
    }

    public void setListener(IPlayServiceListener listener) {
        this.listener = listener;
    }

    ....

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }
}

then in order to start the service:

Intent intent = new Intent(getContext(), PlayService.class);
bindService(intent, mPlayServiceConnection, Context.BIND_AUTO_CREATE));

private final ServiceConnection mPlayServiceConnection = new ServiceConnection() {

        @Override
        public void onServiceConnected(ComponentName className, IBinder service) {
            PlayService.PlayServiceLocalBinder binder = (PlayService.PlayServiceLocalBinder) service;
            mPlayService = binder.getService();
            mPlayService.setListener(YourFragment.this);
        }

        @Override
        public void onServiceDisconnected(ComponentName className) {

        }

    };

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM