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.
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.