New FAB and track number fix

This commit is contained in:
Karim Abou Zeid 2015-04-02 13:41:21 +02:00
commit 3153299592
16 changed files with 60 additions and 40 deletions

View file

@ -23,8 +23,8 @@ android {
applicationId "com.kabouzeid.gramophone" applicationId "com.kabouzeid.gramophone"
minSdkVersion 16 minSdkVersion 16
targetSdkVersion 22 targetSdkVersion 22
versionCode 8 versionCode 9
versionName "0.9.4.1b" versionName "0.9.5b"
} }
compileOptions { compileOptions {
@ -57,7 +57,7 @@ dependencies {
compile 'com.nhaarman.listviewanimations:lib-manipulation:3.1.0@aar' compile 'com.nhaarman.listviewanimations:lib-manipulation:3.1.0@aar'
compile 'com.nhaarman.listviewanimations:lib-core-slh:3.1.0@aar' compile 'com.nhaarman.listviewanimations:lib-core-slh:3.1.0@aar'
compile 'com.nineoldandroids:library:2.4.0' compile 'com.nineoldandroids:library:2.4.0'
compile 'com.melnykov:floatingactionbutton:1.2.0' compile 'com.github.clans:fab:1.2.0'
compile 'com.github.ksoichiro:android-observablescrollview:1.5.0' compile 'com.github.ksoichiro:android-observablescrollview:1.5.0'
compile 'com.mcxiaoke.volley:library:1.0.15' compile 'com.mcxiaoke.volley:library:1.0.15'
compile 'com.squareup.picasso:picasso:2.5.0' compile 'com.squareup.picasso:picasso:2.5.0'

View file

@ -52,8 +52,10 @@ public class AlbumSongAdapter extends RecyclerView.Adapter<AlbumSongAdapter.View
public void onBindViewHolder(ViewHolder holder, int position) { public void onBindViewHolder(ViewHolder holder, int position) {
final Song song = dataSet.get(position); final Song song = dataSet.get(position);
final int trackNumber = MusicUtil.getFixedTrackNumber(song.trackNumber);
final String trackNumberString = trackNumber > 0 ? String.valueOf(trackNumber) : "-";
holder.trackNumber.setText(trackNumberString);
holder.songTitle.setText(song.title); holder.songTitle.setText(song.title);
holder.trackNumber.setText(String.valueOf(MusicUtil.getFixedTrackNumber(song.trackNumber)));
holder.artistName.setText(MusicUtil.getReadableDurationString(song.duration)); holder.artistName.setText(MusicUtil.getReadableDurationString(song.duration));
} }

View file

@ -6,51 +6,58 @@ import android.appwidget.AppWidgetProvider;
import android.content.ComponentName; import android.content.ComponentName;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.support.annotation.NonNull;
import android.widget.RemoteViews; import android.widget.RemoteViews;
import com.kabouzeid.gramophone.R; import com.kabouzeid.gramophone.R;
import com.kabouzeid.gramophone.helper.MusicPlayerRemote;
import com.kabouzeid.gramophone.model.Song;
import com.kabouzeid.gramophone.service.MusicService; import com.kabouzeid.gramophone.service.MusicService;
import com.kabouzeid.gramophone.ui.activities.MusicControllerActivity; import com.kabouzeid.gramophone.ui.activities.MusicControllerActivity;
import com.kabouzeid.gramophone.util.MusicUtil;
import com.kabouzeid.gramophone.util.Util;
import com.squareup.picasso.Picasso;
/** /**
* Implementation of App Widget functionality. * Implementation of App Widget functionality.
*/ */
public class MusicPlayerWidget extends AppWidgetProvider { public class MusicPlayerWidget extends AppWidgetProvider {
private static MusicPlayerWidget instance;
public static synchronized MusicPlayerWidget getInstance() {
if (instance == null) {
instance = new MusicPlayerWidget();
}
return instance;
}
@Override @Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) { public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.music_player_widget); update(context, appWidgetManager, appWidgetIds);
appWidgetManager.updateAppWidget(appWidgetIds, views);
} }
public static void update(Context context, AppWidgetManager manager, int[] ids) {
final RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.music_player_widget);
linkButtons(context, views);
final Song song = MusicPlayerRemote.getCurrentSong();
@Override if (song.id != -1) {
public void onEnabled(Context context) { views.setTextViewText(R.id.song_title, song.title);
// Enter relevant functionality for when the first widget is created }
Picasso.with(context)
.load(MusicUtil.getAlbumArtUri(song.albumId))
.error(R.drawable.default_album_art)
.into(views, R.id.album_art, ids);
int playPauseRes = MusicPlayerRemote.isPlaying() ? R.drawable.ic_pause_black_36dp : R.drawable.ic_play_arrow_black_36dp;
views.setImageViewResource(R.id.button_toggle_play_pause, playPauseRes);
for (int widgetId : ids) {
manager.updateAppWidget(widgetId, views);
}
} }
@Override private static void linkButtons(final Context context, final RemoteViews views) {
public void onDisabled(Context context) {
// Enter relevant functionality for when the last widget is disabled
}
private void linkButtons(final Context context, final RemoteViews views) {
views.setOnClickPendingIntent(R.id.album_art, retrievePlaybackActions(context, 0)); views.setOnClickPendingIntent(R.id.album_art, retrievePlaybackActions(context, 0));
views.setOnClickPendingIntent(R.id.button_toggle_play_pause, retrievePlaybackActions(context, 1)); views.setOnClickPendingIntent(R.id.button_toggle_play_pause, retrievePlaybackActions(context, 1));
views.setOnClickPendingIntent(R.id.button_next, retrievePlaybackActions(context, 2)); views.setOnClickPendingIntent(R.id.button_next, retrievePlaybackActions(context, 2));
views.setOnClickPendingIntent(R.id.button_prev, retrievePlaybackActions(context, 3)); views.setOnClickPendingIntent(R.id.button_prev, retrievePlaybackActions(context, 3));
} }
private PendingIntent retrievePlaybackActions(final Context context, final int which) { private static PendingIntent retrievePlaybackActions(final Context context, final int which) {
Intent action; Intent action;
PendingIntent pendingIntent; PendingIntent pendingIntent;
final ComponentName serviceName = new ComponentName(context, MusicService.class); final ComponentName serviceName = new ComponentName(context, MusicService.class);
@ -77,6 +84,13 @@ public class MusicPlayerWidget extends AppWidgetProvider {
} }
return null; return null;
} }
public static void updateWidgets(Context context) {
AppWidgetManager man = AppWidgetManager.getInstance(context);
int[] ids = man.getAppWidgetIds(
new ComponentName(context, MusicPlayerWidget.class));
update(context, man, ids);
}
} }

View file

@ -2,6 +2,7 @@ package com.kabouzeid.gramophone.service;
import android.app.PendingIntent; import android.app.PendingIntent;
import android.app.Service; import android.app.Service;
import android.appwidget.AppWidgetManager;
import android.content.BroadcastReceiver; import android.content.BroadcastReceiver;
import android.content.ComponentName; import android.content.ComponentName;
import android.content.ContentUris; import android.content.ContentUris;
@ -24,6 +25,7 @@ import android.widget.Toast;
import com.kabouzeid.gramophone.App; import com.kabouzeid.gramophone.App;
import com.kabouzeid.gramophone.R; import com.kabouzeid.gramophone.R;
import com.kabouzeid.gramophone.appwidget.MusicPlayerWidget;
import com.kabouzeid.gramophone.helper.PlayingNotificationHelper; import com.kabouzeid.gramophone.helper.PlayingNotificationHelper;
import com.kabouzeid.gramophone.helper.ShuffleHelper; import com.kabouzeid.gramophone.helper.ShuffleHelper;
import com.kabouzeid.gramophone.misc.AppKeys; import com.kabouzeid.gramophone.misc.AppKeys;
@ -211,6 +213,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
player = null; player = null;
} }
playingNotificationHelper.updatePlayState(isPlaying()); playingNotificationHelper.updatePlayState(isPlaying());
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED);
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP);
} }
@ -260,6 +263,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
if (isLastTrack() && getRepeatMode() == REPEAT_MODE_NONE) { if (isLastTrack() && getRepeatMode() == REPEAT_MODE_NONE) {
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.QUEUE_COMPLETED); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.QUEUE_COMPLETED);
playingNotificationHelper.updatePlayState(isPlaying()); playingNotificationHelper.updatePlayState(isPlaying());
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED);
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP);
} else { } else {
@ -288,6 +292,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
player.setDataSource(getApplicationContext(), trackUri); player.setDataSource(getApplicationContext(), trackUri);
currentSongId = getPlayingQueue().get(getPosition()).id; currentSongId = getPlayingQueue().get(getPosition()).id;
updateNotification(); updateNotification();
MusicPlayerWidget.updateWidgets(this);
updateRemoteControlClient(); updateRemoteControlClient();
player.prepareAsync(); player.prepareAsync();
} catch (Exception e) { } catch (Exception e) {
@ -296,6 +301,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
Toast.makeText(getApplicationContext(), getResources().getString(R.string.unplayable_file), Toast.LENGTH_SHORT).show(); Toast.makeText(getApplicationContext(), getResources().getString(R.string.unplayable_file), Toast.LENGTH_SHORT).show();
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.STOP);
playingNotificationHelper.updatePlayState(false); playingNotificationHelper.updatePlayState(false);
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_STOPPED);
try { try {
updateNotification(); updateNotification();
@ -434,6 +440,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
player.start(); player.start();
isPlayerPrepared = true; isPlayerPrepared = true;
playingNotificationHelper.updatePlayState(isPlaying()); playingNotificationHelper.updatePlayState(isPlaying());
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PLAYING); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PLAYING);
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.PLAY); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.PLAY);
savePosition(); savePosition();
@ -596,6 +603,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
if (isPlaying()) { if (isPlaying()) {
player.pause(); player.pause();
playingNotificationHelper.updatePlayState(isPlaying()); playingNotificationHelper.updatePlayState(isPlaying());
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PAUSED); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PAUSED);
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.PAUSE); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.PAUSE);
} }
@ -607,6 +615,7 @@ public class MusicService extends Service implements MediaPlayer.OnPreparedListe
if (isPlayerPrepared) { if (isPlayerPrepared) {
player.start(); player.start();
playingNotificationHelper.updatePlayState(isPlaying()); playingNotificationHelper.updatePlayState(isPlaying());
MusicPlayerWidget.updateWidgets(this);
remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PLAYING); remoteControlClient.setPlaybackState(RemoteControlClient.PLAYSTATE_PLAYING);
notifyOnMusicRemoteEventListeners(MusicRemoteEvent.RESUME); notifyOnMusicRemoteEventListeners(MusicRemoteEvent.RESUME);
} else { } else {

View file

@ -8,13 +8,13 @@ import android.view.MotionEvent;
import android.view.View; import android.view.View;
import android.widget.Toast; import android.widget.Toast;
import com.github.clans.fab.FloatingActionButton;
import com.kabouzeid.gramophone.App; import com.kabouzeid.gramophone.App;
import com.kabouzeid.gramophone.R; import com.kabouzeid.gramophone.R;
import com.kabouzeid.gramophone.helper.MusicPlayerRemote; import com.kabouzeid.gramophone.helper.MusicPlayerRemote;
import com.kabouzeid.gramophone.misc.SmallOnGestureListener; import com.kabouzeid.gramophone.misc.SmallOnGestureListener;
import com.kabouzeid.gramophone.model.MusicRemoteEvent; import com.kabouzeid.gramophone.model.MusicRemoteEvent;
import com.kabouzeid.gramophone.util.NavigationUtil; import com.kabouzeid.gramophone.util.NavigationUtil;
import com.melnykov.fab.FloatingActionButton;
import com.squareup.otto.Subscribe; import com.squareup.otto.Subscribe;
/** /**

View file

@ -19,6 +19,7 @@ import android.view.animation.OvershootInterpolator;
import android.widget.ImageView; import android.widget.ImageView;
import com.afollestad.materialdialogs.MaterialDialog; import com.afollestad.materialdialogs.MaterialDialog;
import com.github.clans.fab.FloatingActionButton;
import com.github.ksoichiro.android.observablescrollview.ObservableScrollView; import com.github.ksoichiro.android.observablescrollview.ObservableScrollView;
import com.kabouzeid.gramophone.App; import com.kabouzeid.gramophone.App;
import com.kabouzeid.gramophone.R; import com.kabouzeid.gramophone.R;
@ -29,7 +30,6 @@ import com.kabouzeid.gramophone.ui.activities.base.AbsBaseActivity;
import com.kabouzeid.gramophone.util.MusicUtil; import com.kabouzeid.gramophone.util.MusicUtil;
import com.kabouzeid.gramophone.util.Util; import com.kabouzeid.gramophone.util.Util;
import com.kabouzeid.gramophone.util.ViewUtil; import com.kabouzeid.gramophone.util.ViewUtil;
import com.melnykov.fab.FloatingActionButton;
import com.nineoldandroids.view.ViewHelper; import com.nineoldandroids.view.ViewHelper;
import com.nineoldandroids.view.ViewPropertyAnimator; import com.nineoldandroids.view.ViewPropertyAnimator;
@ -56,7 +56,6 @@ public abstract class AbsTagEditorActivity extends AbsBaseActivity {
public static final String TAG = AbsTagEditorActivity.class.getSimpleName(); public static final String TAG = AbsTagEditorActivity.class.getSimpleName();
private static final int REQUEST_CODE_SELECT_IMAGE = 1337; private static final int REQUEST_CODE_SELECT_IMAGE = 1337;
private App app;
private int id; private int id;
private int headerVariableSpace; private int headerVariableSpace;
private int paletteColorPrimary; private int paletteColorPrimary;
@ -398,10 +397,6 @@ public abstract class AbsTagEditorActivity extends AbsBaseActivity {
protected abstract void loadImageFromFile(Uri selectedFile); protected abstract void loadImageFromFile(Uri selectedFile);
protected App getApp() {
return app;
}
protected String getSongTitle() { protected String getSongTitle() {
try { try {
return getAudioFile(songPaths.get(0)).getTagOrCreateAndSetDefault().getFirst(FieldKey.TITLE); return getAudioFile(songPaths.get(0)).getTagOrCreateAndSetDefault().getFirst(FieldKey.TITLE);

View file

@ -130,7 +130,7 @@
android:layout_margin="16dp" android:layout_margin="16dp"
android:background="?attr/music_controller_container_color"> android:background="?attr/music_controller_container_color">
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_centerInParent="true" android:layout_centerInParent="true"

View file

@ -70,7 +70,7 @@
android:background="@android:color/transparent"/> android:background="@android:color/transparent"/>
</LinearLayout> </LinearLayout>
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_gravity="right" android:layout_gravity="right"

View file

@ -150,7 +150,7 @@
android:layout_alignParentTop="true" android:layout_alignParentTop="true"
android:background="#00000000"/> android:background="#00000000"/>
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_gravity="bottom|right" android:layout_gravity="bottom|right"

View file

@ -70,7 +70,7 @@
android:background="@android:color/transparent"/> android:background="@android:color/transparent"/>
</LinearLayout> </LinearLayout>
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_gravity="right" android:layout_gravity="right"

View file

@ -79,7 +79,7 @@
android:layout_gravity="right|bottom" android:layout_gravity="right|bottom"
android:fitsSystemWindows="true"> android:fitsSystemWindows="true">
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_margin="16dp" android:layout_margin="16dp"

View file

@ -166,7 +166,7 @@
android:scaleType="fitCenter" android:scaleType="fitCenter"
android:src="@drawable/ic_shuffle_grey600_48dp"/> android:src="@drawable/ic_shuffle_grey600_48dp"/>
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_centerInParent="true"/> android:layout_centerInParent="true"/>

View file

@ -29,7 +29,7 @@
android:layout_gravity="right|bottom" android:layout_gravity="right|bottom"
android:fitsSystemWindows="true"> android:fitsSystemWindows="true">
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_margin="16dp" android:layout_margin="16dp"

View file

@ -191,7 +191,7 @@
android:layout_alignParentTop="true" android:layout_alignParentTop="true"
android:background="#00000000"/> android:background="#00000000"/>
<com.melnykov.fab.FloatingActionButton <com.github.clans.fab.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
style="@style/PlayPauseFab" style="@style/PlayPauseFab"
android:layout_gravity="bottom|right" android:layout_gravity="bottom|right"

View file

@ -4,6 +4,7 @@
<dimen name="activity_vertical_margin">16dp</dimen> <dimen name="activity_vertical_margin">16dp</dimen>
<dimen name="default_item_margin">16dp</dimen> <dimen name="default_item_margin">16dp</dimen>
<dimen name="list_top_margin">8dp</dimen>
<!-- Per the design guidelines, navigation drawers should be between 240dp and 320dp: <!-- Per the design guidelines, navigation drawers should be between 240dp and 320dp:
https://developer.android.com/design/patterns/navigation-drawer.html --> https://developer.android.com/design/patterns/navigation-drawer.html -->

View file

@ -70,7 +70,6 @@
<item name="elevation">8dp</item> <item name="elevation">8dp</item>
<item name="android:layout_width">wrap_content</item> <item name="android:layout_width">wrap_content</item>
<item name="android:layout_height">wrap_content</item> <item name="android:layout_height">wrap_content</item>
<item name="fab_shadow">true</item>
</style> </style>
<style name="NotificationButtonParent"> <style name="NotificationButtonParent">