package de.podfetcher.activity; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.ServiceConnection; import android.content.SharedPreferences; import android.content.res.Configuration; import android.graphics.PixelFormat; import android.media.MediaPlayer; import android.os.AsyncTask; import android.os.Bundle; import android.os.IBinder; import android.util.Log; import android.view.SurfaceHolder; import android.view.SurfaceView; import android.view.View; import android.view.View.OnClickListener; import android.widget.ImageButton; import android.widget.ImageView; import android.widget.SeekBar; import android.widget.SeekBar.OnSeekBarChangeListener; import android.widget.TextView; import android.widget.VideoView; import android.widget.ViewSwitcher; import com.actionbarsherlock.app.SherlockActivity; import com.actionbarsherlock.view.Menu; import de.podfetcher.PodcastApp; import de.podfetcher.R; import de.podfetcher.feed.FeedManager; import de.podfetcher.feed.FeedMedia; import de.podfetcher.service.PlaybackService; import de.podfetcher.service.PlayerStatus; import de.podfetcher.util.Converter; public class MediaplayerActivity extends SherlockActivity implements SurfaceHolder.Callback { private final String TAG = "MediaplayerActivity"; private static final int DEFAULT_SEEK_DELTA = 30000; // Seek-Delta to use // when using FF or // Rev Buttons private int orientation; private PlaybackService playbackService; private MediaPositionObserver positionObserver; private FeedMedia media; private PlayerStatus status; private FeedManager manager; // Widgets private ViewSwitcher viewswitcher; private ImageView imgvCover; private VideoView videoview; private TextView txtvStatus; private TextView txtvPosition; private TextView txtvLength; private SeekBar sbPosition; private ImageButton butPlay; private ImageButton butRev; private ImageButton butFF; @Override protected void onStop() { super.onStop(); Log.d(TAG, "Activity stopped"); try { unregisterReceiver(statusUpdate); } catch (IllegalArgumentException e) { // ignore } try { unbindService(mConnection); } catch (IllegalArgumentException e) { // ignore } if (positionObserver != null) { positionObserver.cancel(true); } } @Override public boolean onCreateOptionsMenu(Menu menu) { // TODO Auto-generated method stub return super.onCreateOptionsMenu(menu); } @Override protected void onResume() { super.onResume(); Log.d(TAG, "Resuming Activity"); bindToService(); } @Override public void onConfigurationChanged(Configuration newConfig) { super.onConfigurationChanged(newConfig); Log.d(TAG, "Configuration changed"); orientation = newConfig.orientation; positionObserver.cancel(true); if (orientation == Configuration.ORIENTATION_LANDSCAPE) { setContentView(R.layout.mediaplayer_activity); } else { setContentView(R.layout.mediaplayer_activity); } setupGUI(); if (playbackService != null && playbackService.isPlayingVideo()) { playbackService.resetVideoSurface(); if (!videoview.isShown()) { viewswitcher.showNext(); } } } @Override protected void onPause() { super.onPause(); if (playbackService.isRunning && playbackService != null && playbackService.isPlayingVideo()) { playbackService.pause(); } } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.d(TAG, "Creating Activity"); orientation = getResources().getConfiguration().orientation; getWindow().setFormat(PixelFormat.TRANSPARENT); this.setContentView(R.layout.mediaplayer_activity); manager = FeedManager.getInstance(); setupGUI(); bindToService(); } private void bindToService() { Intent serviceIntent = new Intent(this, PlaybackService.class); boolean bound = false; if (!PlaybackService.isRunning) { Log.d(TAG, "Trying to restore last played media"); SharedPreferences prefs = getApplicationContext() .getSharedPreferences(PodcastApp.PREF_NAME, 0); long mediaId = prefs.getLong(PlaybackService.PREF_LAST_PLAYED_ID, -1); long feedId = prefs.getLong( PlaybackService.PREF_LAST_PLAYED_FEED_ID, -1); if (mediaId != -1 && feedId != -1) { serviceIntent.putExtra(PlaybackService.EXTRA_FEED_ID, feedId); serviceIntent.putExtra(PlaybackService.EXTRA_MEDIA_ID, mediaId); serviceIntent.putExtra( PlaybackService.EXTRA_START_WHEN_PREPARED, false); serviceIntent.putExtra(PlaybackService.EXTRA_SHOULD_STREAM, prefs.getBoolean(PlaybackService.PREF_LAST_IS_STREAM, true)); startService(serviceIntent); bound = bindService(serviceIntent, mConnection, Context.BIND_AUTO_CREATE); } else { Log.d(TAG, "No last played media found"); status = PlayerStatus.STOPPED; handleStatus(); } } else { bound = bindService(serviceIntent, mConnection, 0); } Log.d(TAG, "Result for service binding: " + bound); } private void handleStatus() { switch (status) { case ERROR: setStatusMsg(R.string.player_error_msg, View.VISIBLE); handleError(); break; case PAUSED: setStatusMsg(R.string.player_paused_msg, View.VISIBLE); loadMediaInfo(); if (positionObserver != null) { positionObserver.cancel(true); positionObserver = null; } if (orientation == Configuration.ORIENTATION_PORTRAIT) { butPlay.setImageResource(android.R.drawable.ic_media_play); } break; case PLAYING: setStatusMsg(R.string.player_playing_msg, View.INVISIBLE); loadMediaInfo(); setupPositionObserver(); if (orientation == Configuration.ORIENTATION_PORTRAIT) { butPlay.setImageResource(android.R.drawable.ic_media_pause); } break; case PREPARING: setStatusMsg(R.string.player_preparing_msg, View.VISIBLE); break; case STOPPED: setStatusMsg(R.string.player_stopped_msg, View.VISIBLE); imgvCover.setImageBitmap(null); break; case PREPARED: loadMediaInfo(); setStatusMsg(R.string.player_ready_msg, View.VISIBLE); if (orientation == Configuration.ORIENTATION_PORTRAIT) { butPlay.setImageResource(android.R.drawable.ic_media_play); } break; case SEEKING: setStatusMsg(R.string.player_seeking_msg, View.VISIBLE); break; case AWAITING_VIDEO_SURFACE: Log.d(TAG, "Preparing video playback"); SurfaceHolder holder = videoview.getHolder(); playbackService.setVideoSurface(holder); holder.addCallback(this); if (orientation == Configuration.ORIENTATION_PORTRAIT) { if (!videoview.isShown()) { viewswitcher.showNext(); } } } } private void setStatusMsg(int resId, int visibility) { if (orientation == Configuration.ORIENTATION_PORTRAIT) { if (visibility == View.VISIBLE) { txtvStatus.setText(resId); } txtvStatus.setVisibility(visibility); } } private void setupPositionObserver() { if (positionObserver == null || positionObserver.isCancelled()) { positionObserver = new MediaPositionObserver() { @Override protected void onProgressUpdate(Void... v) { super.onProgressUpdate(); txtvPosition.setText(Converter .getDurationStringLong(playbackService.getPlayer() .getCurrentPosition())); updateProgressbarPosition(); } }; positionObserver.execute(playbackService.getPlayer()); } } private void updateProgressbarPosition() { Log.d(TAG, "Updating progressbar info"); MediaPlayer player = playbackService.getPlayer(); float progress = ((float) player.getCurrentPosition()) / player.getDuration(); sbPosition.setProgress((int) (progress * sbPosition.getMax())); } private void loadMediaInfo() { Log.d(TAG, "Loading media info"); if (media != null) { MediaPlayer player = playbackService.getPlayer(); if (orientation == Configuration.ORIENTATION_PORTRAIT) { getSupportActionBar().setSubtitle(media.getItem().getTitle()); getSupportActionBar().setTitle( media.getItem().getFeed().getTitle()); imgvCover.setImageBitmap(media.getItem().getFeed().getImage() .getImageBitmap()); } txtvPosition.setText(Converter.getDurationStringLong((player .getCurrentPosition()))); txtvLength.setText(Converter.getDurationStringLong(player .getDuration())); if (playbackService != null) { updateProgressbarPosition(); } else { sbPosition.setProgress(0); } } } private void setupGUI() { viewswitcher = (ViewSwitcher) findViewById(R.id.viewswitcher); sbPosition = (SeekBar) findViewById(R.id.sbPosition); videoview = (VideoView) findViewById(R.id.videoview); txtvPosition = (TextView) findViewById(R.id.txtvPosition); txtvLength = (TextView) findViewById(R.id.txtvLength); sbPosition.setOnSeekBarChangeListener(new OnSeekBarChangeListener() { int duration; float prog; @Override public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { if (fromUser) { prog = progress / ((float) seekBar.getMax()); duration = playbackService.getPlayer().getDuration(); txtvPosition.setText(Converter .getDurationStringLong((int) (prog * duration))); } } @Override public void onStartTrackingTouch(SeekBar seekBar) { // interrupt position Observer, restart later if (positionObserver != null) { positionObserver.cancel(true); positionObserver = null; } } @Override public void onStopTrackingTouch(SeekBar seekBar) { playbackService.seek((int) (prog * duration)); setupPositionObserver(); } }); if (orientation == Configuration.ORIENTATION_PORTRAIT) { imgvCover = (ImageView) findViewById(R.id.imgvCover); txtvStatus = (TextView) findViewById(R.id.txtvStatus); butPlay = (ImageButton) findViewById(R.id.butPlay); butRev = (ImageButton) findViewById(R.id.butRev); butFF = (ImageButton) findViewById(R.id.butFF); butPlay.setOnClickListener(playbuttonListener); butFF.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if (status == PlayerStatus.PLAYING) { playbackService.seekDelta(DEFAULT_SEEK_DELTA); } } }); butRev.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if (status == PlayerStatus.PLAYING) { playbackService.seekDelta(-DEFAULT_SEEK_DELTA); } } }); } else { setTheme(R.style.Theme_Sherlock_Light_NoActionBar); videoview.setOnClickListener(playbuttonListener); } } private OnClickListener playbuttonListener = new OnClickListener() { @Override public void onClick(View v) { if (status == PlayerStatus.PLAYING) { playbackService.pause(); } else if (status == PlayerStatus.PAUSED || status == PlayerStatus.PREPARED) { playbackService.play(); } } }; private void handleError() { // TODO implement } private ServiceConnection mConnection = new ServiceConnection() { public void onServiceConnected(ComponentName className, IBinder service) { playbackService = ((PlaybackService.LocalBinder) service) .getService(); status = playbackService.getStatus(); media = playbackService.getMedia(); registerReceiver(statusUpdate, new IntentFilter( PlaybackService.ACTION_PLAYER_STATUS_CHANGED)); handleStatus(); Log.d(TAG, "Connection to Service established"); } @Override public void onServiceDisconnected(ComponentName name) { playbackService = null; Log.d(TAG, "Disconnected from Service"); } }; private BroadcastReceiver statusUpdate = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { Log.d(TAG, "Received statusUpdate Intent."); status = playbackService.getStatus(); handleStatus(); } }; /** Refreshes the current position of the media file that is playing. */ public class MediaPositionObserver extends AsyncTask { private static final int WAITING_INTERVALL = 1000; private MediaPlayer player; @Override protected void onCancelled() { Log.d(TAG, "Task was cancelled"); } @Override protected Void doInBackground(MediaPlayer... p) { Log.d(TAG, "Background Task started"); player = p[0]; while (player.isPlaying() && !isCancelled()) { try { Thread.sleep(WAITING_INTERVALL); } catch (InterruptedException e) { Log.d(TAG, "Thread was interrupted while waiting. Finishing now"); return null; } publishProgress(); } Log.d(TAG, "Background Task finished"); return null; } } private boolean holderCreated; @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { holder.setFixedSize(width, height); } @Override public void surfaceCreated(SurfaceHolder holder) { holderCreated = true; Log.d(TAG, "Videoview holder created"); if (status == PlayerStatus.AWAITING_VIDEO_SURFACE) { playbackService.setVideoSurface(holder); } } @Override public void surfaceDestroyed(SurfaceHolder holder) { holderCreated = false; } }