Removed legacy playlist methods and converted PlaylistsUtil to Kotlin
This commit is contained in:
parent
bb37692e46
commit
a3399e7aed
1 changed files with 12 additions and 317 deletions
|
@ -11,323 +11,18 @@
|
||||||
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
* See the GNU General Public License for more details.
|
* See the GNU General Public License for more details.
|
||||||
*/
|
*/
|
||||||
|
package code.name.monkey.retromusic.util
|
||||||
|
|
||||||
package code.name.monkey.retromusic.util;
|
import code.name.monkey.retromusic.db.PlaylistWithSongs
|
||||||
|
import code.name.monkey.retromusic.helper.M3UWriter.writeIO
|
||||||
|
import java.io.File
|
||||||
|
import java.io.IOException
|
||||||
|
|
||||||
import static android.provider.MediaStore.Audio.Playlists.EXTERNAL_CONTENT_URI;
|
object PlaylistsUtil {
|
||||||
import static code.name.monkey.retromusic.util.FileUtilsKt.getExternalStorageDirectory;
|
@Throws(IOException::class)
|
||||||
|
fun savePlaylistWithSongs(playlist: PlaylistWithSongs?): File {
|
||||||
import android.content.ContentResolver;
|
return writeIO(
|
||||||
import android.content.ContentValues;
|
File(getExternalStorageDirectory(), "Playlists"), playlist!!
|
||||||
import android.content.Context;
|
)
|
||||||
import android.database.Cursor;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.provider.BaseColumns;
|
|
||||||
import android.provider.MediaStore;
|
|
||||||
import android.widget.Toast;
|
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import code.name.monkey.retromusic.R;
|
|
||||||
import code.name.monkey.retromusic.db.PlaylistWithSongs;
|
|
||||||
import code.name.monkey.retromusic.helper.M3UWriter;
|
|
||||||
import code.name.monkey.retromusic.model.Playlist;
|
|
||||||
import code.name.monkey.retromusic.model.PlaylistSong;
|
|
||||||
import code.name.monkey.retromusic.model.Song;
|
|
||||||
|
|
||||||
public class PlaylistsUtil {
|
|
||||||
|
|
||||||
public static long createPlaylist(@NonNull final Context context, @Nullable final String name) {
|
|
||||||
int id = -1;
|
|
||||||
if (name != null && name.length() > 0) {
|
|
||||||
try {
|
|
||||||
Cursor cursor =
|
|
||||||
context
|
|
||||||
.getContentResolver()
|
|
||||||
.query(
|
|
||||||
EXTERNAL_CONTENT_URI,
|
|
||||||
new String[] {MediaStore.Audio.Playlists._ID},
|
|
||||||
MediaStore.Audio.PlaylistsColumns.NAME + "=?",
|
|
||||||
new String[] {name},
|
|
||||||
null);
|
|
||||||
if (cursor == null || cursor.getCount() < 1) {
|
|
||||||
final ContentValues values = new ContentValues(1);
|
|
||||||
values.put(MediaStore.Audio.PlaylistsColumns.NAME, name);
|
|
||||||
final Uri uri = context.getContentResolver().insert(EXTERNAL_CONTENT_URI, values);
|
|
||||||
if (uri != null) {
|
|
||||||
// Necessary because somehow the MediaStoreObserver is not notified when adding a
|
|
||||||
// playlist
|
|
||||||
context.getContentResolver().notifyChange(Uri.parse("content://media"), null);
|
|
||||||
Toast.makeText(
|
|
||||||
context,
|
|
||||||
context.getResources().getString(R.string.created_playlist_x, name),
|
|
||||||
Toast.LENGTH_SHORT)
|
|
||||||
.show();
|
|
||||||
id = Integer.parseInt(uri.getLastPathSegment());
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Playlist exists
|
|
||||||
if (cursor.moveToFirst()) {
|
|
||||||
id = cursor.getInt(cursor.getColumnIndex(MediaStore.Audio.Playlists._ID));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (cursor != null) {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
} catch (SecurityException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (id == -1) {
|
|
||||||
Toast.makeText(
|
|
||||||
context,
|
|
||||||
context.getResources().getString(R.string.could_not_create_playlist),
|
|
||||||
Toast.LENGTH_SHORT)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void deletePlaylists(
|
|
||||||
@NonNull final Context context, @NonNull final List<Playlist> playlists) {
|
|
||||||
final StringBuilder selection = new StringBuilder();
|
|
||||||
selection.append(MediaStore.Audio.Playlists._ID + " IN (");
|
|
||||||
for (int i = 0; i < playlists.size(); i++) {
|
|
||||||
selection.append(playlists.get(i).getId());
|
|
||||||
if (i < playlists.size() - 1) {
|
|
||||||
selection.append(",");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
selection.append(")");
|
|
||||||
try {
|
|
||||||
context.getContentResolver().delete(EXTERNAL_CONTENT_URI, selection.toString(), null);
|
|
||||||
context.getContentResolver().notifyChange(Uri.parse("content://media"), null);
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void addToPlaylist(
|
|
||||||
@NonNull final Context context,
|
|
||||||
final Song song,
|
|
||||||
final long playlistId,
|
|
||||||
final boolean showToastOnFinish) {
|
|
||||||
List<Song> helperList = new ArrayList<>();
|
|
||||||
helperList.add(song);
|
|
||||||
addToPlaylist(context, helperList, playlistId, showToastOnFinish);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void addToPlaylist(
|
|
||||||
@NonNull final Context context,
|
|
||||||
@NonNull final List<Song> songs,
|
|
||||||
final long playlistId,
|
|
||||||
final boolean showToastOnFinish) {
|
|
||||||
final int size = songs.size();
|
|
||||||
final ContentResolver resolver = context.getContentResolver();
|
|
||||||
final String[] projection =
|
|
||||||
new String[] {
|
|
||||||
"max(" + MediaStore.Audio.Playlists.Members.PLAY_ORDER + ")",
|
|
||||||
};
|
|
||||||
final Uri uri = MediaStore.Audio.Playlists.Members.getContentUri("external", playlistId);
|
|
||||||
Cursor cursor = null;
|
|
||||||
int base = 0;
|
|
||||||
|
|
||||||
try {
|
|
||||||
try {
|
|
||||||
cursor = resolver.query(uri, projection, null, null, null);
|
|
||||||
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
base = cursor.getInt(0) + 1;
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (cursor != null) {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int numInserted = 0;
|
|
||||||
for (int offSet = 0; offSet < size; offSet += 1000)
|
|
||||||
numInserted += resolver.bulkInsert(uri, makeInsertItems(songs, offSet, 1000, base));
|
|
||||||
|
|
||||||
if (showToastOnFinish) {
|
|
||||||
Toast.makeText(
|
|
||||||
context,
|
|
||||||
context
|
|
||||||
.getResources()
|
|
||||||
.getString(
|
|
||||||
R.string.inserted_x_songs_into_playlist_x,
|
|
||||||
numInserted,
|
|
||||||
getNameForPlaylist(context, playlistId)),
|
|
||||||
Toast.LENGTH_SHORT)
|
|
||||||
.show();
|
|
||||||
}
|
|
||||||
} catch (SecurityException exception) {
|
|
||||||
exception.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@NonNull
|
|
||||||
public static ContentValues[] makeInsertItems(
|
|
||||||
@NonNull final List<Song> songs, final int offset, int len, final int base) {
|
|
||||||
if (offset + len > songs.size()) {
|
|
||||||
len = songs.size() - offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
ContentValues[] contentValues = new ContentValues[len];
|
|
||||||
|
|
||||||
for (int i = 0; i < len; i++) {
|
|
||||||
contentValues[i] = new ContentValues();
|
|
||||||
contentValues[i].put(MediaStore.Audio.Playlists.Members.PLAY_ORDER, base + offset + i);
|
|
||||||
contentValues[i].put(
|
|
||||||
MediaStore.Audio.Playlists.Members.AUDIO_ID, songs.get(offset + i).getId());
|
|
||||||
}
|
|
||||||
return contentValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getNameForPlaylist(@NonNull final Context context, final long id) {
|
|
||||||
try {
|
|
||||||
Cursor cursor =
|
|
||||||
context
|
|
||||||
.getContentResolver()
|
|
||||||
.query(
|
|
||||||
EXTERNAL_CONTENT_URI,
|
|
||||||
new String[] {MediaStore.Audio.PlaylistsColumns.NAME},
|
|
||||||
BaseColumns._ID + "=?",
|
|
||||||
new String[] {String.valueOf(id)},
|
|
||||||
null);
|
|
||||||
if (cursor != null) {
|
|
||||||
try {
|
|
||||||
if (cursor.moveToFirst()) {
|
|
||||||
return cursor.getString(0);
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removeFromPlaylist(
|
|
||||||
@NonNull final Context context, @NonNull final Song song, long playlistId) {
|
|
||||||
Uri uri = MediaStore.Audio.Playlists.Members.getContentUri("external", playlistId);
|
|
||||||
String selection = MediaStore.Audio.Playlists.Members.AUDIO_ID + " =?";
|
|
||||||
String[] selectionArgs = new String[] {String.valueOf(song.getId())};
|
|
||||||
|
|
||||||
try {
|
|
||||||
context.getContentResolver().delete(uri, selection, selectionArgs);
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removeFromPlaylist(
|
|
||||||
@NonNull final Context context, @NonNull final List<PlaylistSong> songs) {
|
|
||||||
final long playlistId = songs.get(0).getPlaylistId();
|
|
||||||
Uri uri = MediaStore.Audio.Playlists.Members.getContentUri("external", playlistId);
|
|
||||||
String[] selectionArgs = new String[songs.size()];
|
|
||||||
for (int i = 0; i < selectionArgs.length; i++) {
|
|
||||||
selectionArgs[i] = String.valueOf(songs.get(i).getIdInPlayList());
|
|
||||||
}
|
|
||||||
StringBuilder selection = new StringBuilder(MediaStore.Audio.Playlists.Members._ID + " in (");
|
|
||||||
|
|
||||||
for (String selectionArg : selectionArgs) selection.append("?, ");
|
|
||||||
selection = new StringBuilder(selection.substring(0, selection.length() - 2) + ")");
|
|
||||||
|
|
||||||
try {
|
|
||||||
context.getContentResolver().delete(uri, selection.toString(), selectionArgs);
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean doPlaylistContains(
|
|
||||||
@NonNull final Context context, final long playlistId, final long songId) {
|
|
||||||
if (playlistId != -1) {
|
|
||||||
try {
|
|
||||||
Cursor c =
|
|
||||||
context
|
|
||||||
.getContentResolver()
|
|
||||||
.query(
|
|
||||||
MediaStore.Audio.Playlists.Members.getContentUri("external", playlistId),
|
|
||||||
new String[] {MediaStore.Audio.Playlists.Members.AUDIO_ID},
|
|
||||||
MediaStore.Audio.Playlists.Members.AUDIO_ID + "=?",
|
|
||||||
new String[] {String.valueOf(songId)},
|
|
||||||
null);
|
|
||||||
int count = 0;
|
|
||||||
if (c != null) {
|
|
||||||
count = c.getCount();
|
|
||||||
c.close();
|
|
||||||
}
|
|
||||||
return count > 0;
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean moveItem(
|
|
||||||
@NonNull final Context context, long playlistId, int from, int to) {
|
|
||||||
return MediaStore.Audio.Playlists.Members.moveItem(
|
|
||||||
context.getContentResolver(), playlistId, from, to);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void renamePlaylist(
|
|
||||||
@NonNull final Context context, final long id, final String newName) {
|
|
||||||
ContentValues contentValues = new ContentValues();
|
|
||||||
contentValues.put(MediaStore.Audio.PlaylistsColumns.NAME, newName);
|
|
||||||
try {
|
|
||||||
context
|
|
||||||
.getContentResolver()
|
|
||||||
.update(
|
|
||||||
EXTERNAL_CONTENT_URI,
|
|
||||||
contentValues,
|
|
||||||
MediaStore.Audio.Playlists._ID + "=?",
|
|
||||||
new String[] {String.valueOf(id)});
|
|
||||||
context.getContentResolver().notifyChange(Uri.parse("content://media"), null);
|
|
||||||
} catch (SecurityException ignored) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static File savePlaylist(Context context, Playlist playlist) throws IOException {
|
|
||||||
return M3UWriter.write(
|
|
||||||
new File(getExternalStorageDirectory(), "Playlists"), playlist);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static File savePlaylistWithSongs(PlaylistWithSongs playlist) throws IOException {
|
|
||||||
return M3UWriter.writeIO(
|
|
||||||
new File(getExternalStorageDirectory(), "Playlists"), playlist);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean doesPlaylistExist(@NonNull final Context context, final int playlistId) {
|
|
||||||
return playlistId != -1
|
|
||||||
&& doesPlaylistExist(
|
|
||||||
context,
|
|
||||||
MediaStore.Audio.Playlists._ID + "=?",
|
|
||||||
new String[] {String.valueOf(playlistId)});
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean doesPlaylistExist(@NonNull final Context context, final String name) {
|
|
||||||
return doesPlaylistExist(
|
|
||||||
context, MediaStore.Audio.PlaylistsColumns.NAME + "=?", new String[] {name});
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean doesPlaylistExist(
|
|
||||||
@NonNull Context context, @NonNull final String selection, @NonNull final String[] values) {
|
|
||||||
Cursor cursor =
|
|
||||||
context
|
|
||||||
.getContentResolver()
|
|
||||||
.query(EXTERNAL_CONTENT_URI, new String[]{}, selection, values, null);
|
|
||||||
|
|
||||||
boolean exists = false;
|
|
||||||
if (cursor != null) {
|
|
||||||
exists = cursor.getCount() != 0;
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
return exists;
|
|
||||||
}
|
|
||||||
}
|
}
|
Loading…
Add table
Add a link
Reference in a new issue