card shadow showing in the background of titlebar & searchbar fix

Create/Remove playlist support
Add/Remove Track to playlist(s) support
Favorite/Un-favorite from any PlaylistView's TrackTile support
This commit is contained in:
Kingkor Roy Tirtho 2022-04-17 18:07:49 +06:00
parent 14f51d4f4d
commit 4efd600d13
10 changed files with 532 additions and 452 deletions

View File

@ -67,6 +67,13 @@ class AlbumView extends HookConsumerWidget {
future: spotify.me.isSavedAlbums([album.id!]), future: spotify.me.isSavedAlbums([album.id!]),
builder: (context, snapshot) { builder: (context, snapshot) {
final isSaved = snapshot.data?.first == true; final isSaved = snapshot.data?.first == true;
if (!snapshot.hasData && !snapshot.hasError) {
return const SizedBox(
height: 25,
width: 25,
child: CircularProgressIndicator.adaptive(),
);
}
return HeartButton( return HeartButton(
isLiked: isSaved, isLiked: isSaved,
onPressed: () { onPressed: () {

View File

@ -8,7 +8,7 @@ import 'package:spotify/spotify.dart';
import 'package:spotube/components/Album/AlbumCard.dart'; import 'package:spotube/components/Album/AlbumCard.dart';
import 'package:spotube/components/Artist/ArtistCard.dart'; import 'package:spotube/components/Artist/ArtistCard.dart';
import 'package:spotube/components/Shared/PageWindowTitleBar.dart'; import 'package:spotube/components/Shared/PageWindowTitleBar.dart';
import 'package:spotube/components/Shared/TracksTableView.dart'; import 'package:spotube/components/Shared/TrackTile.dart';
import 'package:spotube/helpers/image-to-url-string.dart'; import 'package:spotube/helpers/image-to-url-string.dart';
import 'package:spotube/helpers/readable-number.dart'; import 'package:spotube/helpers/readable-number.dart';
import 'package:spotube/helpers/zero-pad-num-str.dart'; import 'package:spotube/helpers/zero-pad-num-str.dart';

View File

@ -169,7 +169,9 @@ class Home extends HookConsumerWidget {
}; };
}, [localStorage]); }, [localStorage]);
final titleBarContents = Row( final titleBarContents = Container(
color: Theme.of(context).backgroundColor,
child: Row(
children: [ children: [
Expanded( Expanded(
child: Row( child: Row(
@ -187,7 +189,7 @@ class Home extends HookConsumerWidget {
], ],
)) ))
], ],
); ));
return SafeArea( return SafeArea(
child: Scaffold( child: Scaffold(

View File

@ -1,3 +1,4 @@
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotube/components/Shared/HeartButton.dart'; import 'package:spotube/components/Shared/HeartButton.dart';
import 'package:spotube/components/Shared/PageWindowTitleBar.dart'; import 'package:spotube/components/Shared/PageWindowTitleBar.dart';
@ -45,6 +46,12 @@ class PlaylistView extends HookConsumerWidget {
final isPlaylistPlaying = playback.currentPlaylist?.id != null && final isPlaylistPlaying = playback.currentPlaylist?.id != null &&
playback.currentPlaylist?.id == playlist.id; playback.currentPlaylist?.id == playlist.id;
final update = useForceUpdate(); final update = useForceUpdate();
final getMe = useMemoized(() => spotify.me.get(), []);
final meSnapshot = useFuture<User>(getMe);
Future<List<bool>> isFollowing(User me) {
return spotify.playlists.followedBy(playlist.id!, [me.id!]);
}
return SafeArea( return SafeArea(
child: Scaffold( child: Scaffold(
@ -64,17 +71,27 @@ class PlaylistView extends HookConsumerWidget {
// nav back // nav back
const BackButton(), const BackButton(),
// heart playlist // heart playlist
if (auth.isLoggedIn) if (auth.isLoggedIn && meSnapshot.hasData)
FutureBuilder<List<bool>>( FutureBuilder<List<bool>>(
future: spotify.me.get().then( future: isFollowing(meSnapshot.data!),
(me) => spotify.playlists
.followedBy(playlist.id!, [me.id!]),
),
builder: (context, snapshot) { builder: (context, snapshot) {
final isFollowing = final isFollowing =
snapshot.data?.first ?? false; snapshot.data?.first ?? false;
if (!snapshot.hasData && !snapshot.hasError) {
return const SizedBox(
height: 25,
width: 25,
child: CircularProgressIndicator.adaptive(),
);
}
return HeartButton( return HeartButton(
isLiked: isFollowing, isLiked: isFollowing,
icon: playlist.owner?.id != null &&
meSnapshot.data?.id ==
playlist.owner?.id
? Icons.delete_outline_rounded
: null,
onPressed: () async { onPressed: () async {
try { try {
isFollowing isFollowing
@ -124,6 +141,9 @@ class PlaylistView extends HookConsumerWidget {
tracks, tracks,
currentTrack: currentTrack, currentTrack: currentTrack,
), ),
playlistId: playlist.id,
userPlaylist: playlist.owner?.id != null &&
playlist.owner!.id == meSnapshot.data?.id,
), ),
], ],
); );

View File

@ -6,7 +6,7 @@ import 'package:spotube/components/Album/AlbumCard.dart';
import 'package:spotube/components/Artist/ArtistCard.dart'; import 'package:spotube/components/Artist/ArtistCard.dart';
import 'package:spotube/components/Playlist/PlaylistCard.dart'; import 'package:spotube/components/Playlist/PlaylistCard.dart';
import 'package:spotube/components/Shared/AnonymousFallback.dart'; import 'package:spotube/components/Shared/AnonymousFallback.dart';
import 'package:spotube/components/Shared/TracksTableView.dart'; import 'package:spotube/components/Shared/TrackTile.dart';
import 'package:spotube/helpers/image-to-url-string.dart'; import 'package:spotube/helpers/image-to-url-string.dart';
import 'package:spotube/helpers/simple-album-to-album.dart'; import 'package:spotube/helpers/simple-album-to-album.dart';
import 'package:spotube/helpers/zero-pad-num-str.dart'; import 'package:spotube/helpers/zero-pad-num-str.dart';
@ -22,8 +22,8 @@ class Search extends HookConsumerWidget {
Widget build(BuildContext context, ref) { Widget build(BuildContext context, ref) {
final SpotifyApi spotify = ref.watch(spotifyProvider); final SpotifyApi spotify = ref.watch(spotifyProvider);
final Auth auth = ref.watch(authProvider); final Auth auth = ref.watch(authProvider);
var controller = useTextEditingController(); final controller = useTextEditingController();
var searchTerm = useState(""); final searchTerm = useState("");
final albumController = useScrollController(); final albumController = useScrollController();
final playlistController = useScrollController(); final playlistController = useScrollController();
final artistController = useScrollController(); final artistController = useScrollController();
@ -34,6 +34,8 @@ class Search extends HookConsumerWidget {
} }
return Expanded( return Expanded(
child: Container(
color: Theme.of(context).backgroundColor,
child: Column( child: Column(
children: [ children: [
Padding( Padding(
@ -98,8 +100,8 @@ class Search extends HookConsumerWidget {
return Expanded( return Expanded(
child: SingleChildScrollView( child: SingleChildScrollView(
child: Padding( child: Padding(
padding: padding: const EdgeInsets.symmetric(
const EdgeInsets.symmetric(vertical: 8, horizontal: 20), vertical: 8, horizontal: 20),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
@ -217,6 +219,7 @@ class Search extends HookConsumerWidget {
) )
], ],
), ),
),
); );
} }
} }

View File

@ -3,9 +3,11 @@ import 'package:flutter/material.dart';
class HeartButton extends StatelessWidget { class HeartButton extends StatelessWidget {
final bool isLiked; final bool isLiked;
final void Function() onPressed; final void Function() onPressed;
final IconData? icon;
const HeartButton({ const HeartButton({
required this.isLiked, required this.isLiked,
required this.onPressed, required this.onPressed,
this.icon,
Key? key, Key? key,
}) : super(key: key); }) : super(key: key);
@ -13,7 +15,10 @@ class HeartButton extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return IconButton( return IconButton(
icon: Icon( icon: Icon(
!isLiked ? Icons.favorite_outline_rounded : Icons.favorite_rounded, icon ??
(!isLiked
? Icons.favorite_outline_rounded
: Icons.favorite_rounded),
color: isLiked ? Colors.green : null, color: isLiked ? Colors.green : null,
), ),
onPressed: onPressed, onPressed: onPressed,

View File

@ -70,6 +70,8 @@ class PageWindowTitleBar extends StatelessWidget
); );
} }
return WindowTitleBarBox( return WindowTitleBarBox(
child: Container(
color: Theme.of(context).backgroundColor,
child: Row( child: Row(
children: [ children: [
if (Platform.isMacOS) if (Platform.isMacOS)
@ -82,6 +84,7 @@ class PageWindowTitleBar extends StatelessWidget
const TitleBarActionButtons() const TitleBarActionButtons()
], ],
), ),
),
); );
} }
} }

View File

@ -0,0 +1,276 @@
import 'package:cached_network_image/cached_network_image.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/components/Shared/LinkText.dart';
import 'package:spotube/helpers/artists-to-clickable-artists.dart';
import 'package:spotube/hooks/useBreakpoints.dart';
import 'package:spotube/hooks/useForceUpdate.dart';
import 'package:spotube/models/Logger.dart';
import 'package:spotube/provider/Auth.dart';
import 'package:spotube/provider/Playback.dart';
import 'package:spotube/provider/SpotifyDI.dart';
class TrackTile extends HookConsumerWidget {
final Playback playback;
final MapEntry<int, Track> track;
final String duration;
final String? thumbnailUrl;
final void Function(Track currentTrack)? onTrackPlayButtonPressed;
final logger = getLogger(TrackTile);
final bool userPlaylist;
// null playlistId indicates its not inside a playlist
final String? playlistId;
TrackTile(
this.playback, {
required this.track,
required this.duration,
this.playlistId,
this.userPlaylist = false,
this.thumbnailUrl,
this.onTrackPlayButtonPressed,
Key? key,
}) : super(key: key);
@override
Widget build(BuildContext context, ref) {
final breakpoint = useBreakpoints();
final auth = ref.watch(authProvider);
final spotify = ref.watch(spotifyProvider);
final update = useForceUpdate();
final actionFavorite = useCallback((bool isLiked) async {
try {
isLiked
? await spotify.tracks.me.removeOne(track.value.id!)
: await spotify.tracks.me.saveOne(track.value.id!);
} catch (e, stack) {
logger.e("FavoriteButton.onPressed", e, stack);
} finally {
update();
}
}, [track.value.id, spotify]);
final actionRemoveFromPlaylist = useCallback(() async {
if (playlistId == null) return;
return await spotify.playlists.removeTrack(track.value.uri!, playlistId!);
}, [playlistId, spotify, track.value.uri]);
actionAddToPlaylist() async {
showDialog(
context: context,
builder: (context) {
return FutureBuilder<Iterable<PlaylistSimple>>(
future: spotify.playlists.me.all().then((playlists) async {
final me = await spotify.me.get();
return playlists.where((playlist) =>
playlist.owner?.id != null &&
playlist.owner!.id == me.id);
}),
builder: (context, snapshot) {
return HookBuilder(builder: (context) {
final playlistsCheck = useState(<String, bool>{});
return AlertDialog(
title: Text(
"Add `${track.value.name}` to following Playlists"),
titleTextStyle:
Theme.of(context).textTheme.bodyText1?.copyWith(
fontSize: 18,
fontWeight: FontWeight.bold,
),
actions: [
TextButton(
child: const Text("Cancel"),
onPressed: () => Navigator.pop(context),
),
ElevatedButton(
child: const Text("Add"),
onPressed: () async {
final selectedPlaylists = playlistsCheck
.value.entries
.where((entry) => entry.value)
.map((entry) => entry.key);
await Future.wait(
selectedPlaylists.map(
(playlistId) => spotify.playlists
.addTrack(track.value.uri!, playlistId),
),
).then((_) => Navigator.pop(context));
},
)
],
content: SizedBox(
height: 300,
width: 300,
child: !snapshot.hasData
? const Center(
child: CircularProgressIndicator.adaptive())
: ListView.builder(
shrinkWrap: true,
itemCount: snapshot.data!.length,
itemBuilder: (context, index) {
final playlist =
snapshot.data!.elementAt(index);
return CheckboxListTile(
title: Text(playlist.name!),
controlAffinity:
ListTileControlAffinity.leading,
value: playlistsCheck.value[playlist.id] ??
false,
onChanged: (val) {
playlistsCheck.value = {
...playlistsCheck.value,
playlist.id!: val == true
};
},
);
},
),
),
);
});
});
});
}
return Row(
children: [
SizedBox(
height: 20,
width: 25,
child: Text(
(track.key + 1).toString(),
textAlign: TextAlign.center,
),
),
if (thumbnailUrl != null)
Padding(
padding: EdgeInsets.symmetric(
horizontal: breakpoint.isMoreThan(Breakpoints.md) ? 8.0 : 0,
vertical: 8.0,
),
child: ClipRRect(
borderRadius: const BorderRadius.all(Radius.circular(5)),
child: CachedNetworkImage(
placeholder: (context, url) {
return Container(
height: 40,
width: 40,
color: Colors.green[300],
);
},
imageUrl: thumbnailUrl!,
maxHeightDiskCache: 40,
maxWidthDiskCache: 40,
),
),
),
IconButton(
icon: Icon(
playback.currentTrack?.id != null &&
playback.currentTrack?.id == track.value.id
? Icons.pause_circle_rounded
: Icons.play_circle_rounded,
color: Theme.of(context).primaryColor,
),
onPressed: () => onTrackPlayButtonPressed?.call(
track.value,
),
),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
track.value.name ?? "",
style: TextStyle(
fontWeight: FontWeight.bold,
fontSize: breakpoint.isSm ? 14 : 17,
),
overflow: TextOverflow.ellipsis,
),
artistsToClickableArtists(track.value.artists ?? [],
textStyle: TextStyle(
fontSize:
breakpoint.isLessThan(Breakpoints.lg) ? 12 : 14)),
],
),
),
if (breakpoint.isMoreThan(Breakpoints.md))
Expanded(
child: LinkText(
track.value.album!.name!,
"/album/${track.value.album?.id}",
extra: track.value.album,
overflow: TextOverflow.ellipsis,
),
),
if (!breakpoint.isSm) ...[
const SizedBox(width: 10),
Text(duration),
],
const SizedBox(width: 10),
if (auth.isLoggedIn)
FutureBuilder<bool>(
future: spotify.tracks.me.containsOne(track.value.id!),
builder: (context, snapshot) {
return PopupMenuButton(
icon: const Icon(Icons.more_horiz_rounded),
itemBuilder: (context) {
return [
PopupMenuItem(
child: Row(
children: const [
Icon(Icons.add_box_rounded),
SizedBox(width: 10),
Text("Add to Playlist"),
],
),
value: "add-playlist",
),
if (userPlaylist)
PopupMenuItem(
child: Row(
children: const [
Icon(Icons.remove_circle_outline_rounded),
SizedBox(width: 10),
Text("Remove from Playlist"),
],
),
value: "remove-playlist",
),
PopupMenuItem(
child: Row(
children: [
Icon(snapshot.data == true
? Icons.favorite_rounded
: Icons.favorite_border_rounded),
const SizedBox(width: 10),
const Text("Favorite")
],
),
value: "favorite",
)
];
},
onSelected: (value) {
switch (value) {
case "favorite":
actionFavorite(snapshot.data == true);
break;
case "add-playlist":
actionAddToPlaylist();
break;
case "remove-playlist":
actionRemoveFromPlaylist();
break;
}
},
);
})
],
);
}
}

View File

@ -1,24 +1,24 @@
import 'package:cached_network_image/cached_network_image.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotify/spotify.dart'; import 'package:spotify/spotify.dart';
import 'package:spotube/components/Shared/LinkText.dart'; import 'package:spotube/components/Shared/TrackTile.dart';
import 'package:spotube/helpers/artists-to-clickable-artists.dart';
import 'package:spotube/helpers/image-to-url-string.dart'; import 'package:spotube/helpers/image-to-url-string.dart';
import 'package:spotube/helpers/zero-pad-num-str.dart'; import 'package:spotube/helpers/zero-pad-num-str.dart';
import 'package:spotube/hooks/useBreakpoints.dart'; import 'package:spotube/hooks/useBreakpoints.dart';
import 'package:spotube/hooks/useForceUpdate.dart';
import 'package:spotube/models/Logger.dart';
import 'package:spotube/provider/Auth.dart';
import 'package:spotube/provider/Playback.dart'; import 'package:spotube/provider/Playback.dart';
import 'package:spotube/provider/SpotifyDI.dart';
class TracksTableView extends HookConsumerWidget { class TracksTableView extends HookConsumerWidget {
final void Function(Track currentTrack)? onTrackPlayButtonPressed; final void Function(Track currentTrack)? onTrackPlayButtonPressed;
final List<Track> tracks; final List<Track> tracks;
const TracksTableView(this.tracks, {Key? key, this.onTrackPlayButtonPressed}) final bool userPlaylist;
: super(key: key); final String? playlistId;
const TracksTableView(
this.tracks, {
Key? key,
this.onTrackPlayButtonPressed,
this.userPlaylist = false,
this.playlistId,
}) : super(key: key);
@override @override
Widget build(context, ref) { Widget build(context, ref) {
@ -85,9 +85,11 @@ class TracksTableView extends HookConsumerWidget {
"${track.value.duration?.inMinutes.remainder(60)}:${zeroPadNumStr(track.value.duration?.inSeconds.remainder(60) ?? 0)}"; "${track.value.duration?.inMinutes.remainder(60)}:${zeroPadNumStr(track.value.duration?.inSeconds.remainder(60) ?? 0)}";
return TrackTile( return TrackTile(
playback, playback,
playlistId: playlistId,
track: track, track: track,
duration: duration, duration: duration,
thumbnailUrl: thumbnailUrl, thumbnailUrl: thumbnailUrl,
userPlaylist: userPlaylist,
onTrackPlayButtonPressed: onTrackPlayButtonPressed, onTrackPlayButtonPressed: onTrackPlayButtonPressed,
); );
}).toList() }).toList()
@ -97,242 +99,3 @@ class TracksTableView extends HookConsumerWidget {
); );
} }
} }
class TrackTile extends HookConsumerWidget {
final Playback playback;
final MapEntry<int, Track> track;
final String duration;
final String? thumbnailUrl;
final void Function(Track currentTrack)? onTrackPlayButtonPressed;
final logger = getLogger(TrackTile);
TrackTile(
this.playback, {
required this.track,
required this.duration,
this.thumbnailUrl,
this.onTrackPlayButtonPressed,
Key? key,
}) : super(key: key);
@override
Widget build(BuildContext context, ref) {
final breakpoint = useBreakpoints();
final auth = ref.watch(authProvider);
final spotify = ref.watch(spotifyProvider);
final update = useForceUpdate();
final actionFavorite = useCallback((bool isLiked) async {
try {
isLiked
? await spotify.tracks.me.removeOne(track.value.id!)
: await spotify.tracks.me.saveOne(track.value.id!);
} catch (e, stack) {
logger.e("FavoriteButton.onPressed", e, stack);
} finally {
update();
}
}, [track.value.id, spotify]);
actionPlaylist() async {
showDialog(
context: context,
builder: (context) {
return FutureBuilder<Iterable<PlaylistSimple>>(
future: spotify.playlists.me.all().then((playlists) async {
final me = await spotify.me.get();
return playlists.where((playlist) =>
playlist.owner?.id != null &&
playlist.owner!.id == me.id);
}),
builder: (context, snapshot) {
return HookBuilder(builder: (context) {
final playlistsCheck = useState(<String, bool>{});
return AlertDialog(
title: Text(
"Add `${track.value.name}` to following Playlists"),
titleTextStyle:
Theme.of(context).textTheme.bodyText1?.copyWith(
fontSize: 18,
fontWeight: FontWeight.bold,
),
actions: [
TextButton(
child: const Text("Cancel"),
onPressed: () => Navigator.pop(context),
),
ElevatedButton(
child: const Text("Add"),
onPressed: () async {
final selectedPlaylists = playlistsCheck
.value.entries
.where((entry) => entry.value)
.map((entry) => entry.key);
await Future.wait(
selectedPlaylists.map(
(playlistId) => spotify.playlists
.addTrack(track.value.uri!, playlistId),
),
).then((_) => Navigator.pop(context));
},
)
],
content: SizedBox(
height: 300,
width: 300,
child: !snapshot.hasData
? const Center(
child: CircularProgressIndicator.adaptive())
: ListView.builder(
shrinkWrap: true,
itemCount: snapshot.data!.length,
itemBuilder: (context, index) {
final playlist =
snapshot.data!.elementAt(index);
return CheckboxListTile(
title: Text(playlist.name!),
controlAffinity:
ListTileControlAffinity.leading,
value: playlistsCheck.value[playlist.id] ??
false,
onChanged: (val) {
playlistsCheck.value = {
...playlistsCheck.value,
playlist.id!: val == true
};
},
);
},
),
),
);
});
});
});
}
return Row(
children: [
SizedBox(
height: 20,
width: 25,
child: Text(
(track.key + 1).toString(),
textAlign: TextAlign.center,
),
),
if (thumbnailUrl != null)
Padding(
padding: EdgeInsets.symmetric(
horizontal: breakpoint.isMoreThan(Breakpoints.md) ? 8.0 : 0,
vertical: 8.0,
),
child: ClipRRect(
borderRadius: const BorderRadius.all(Radius.circular(5)),
child: CachedNetworkImage(
placeholder: (context, url) {
return Container(
height: 40,
width: 40,
color: Colors.green[300],
);
},
imageUrl: thumbnailUrl!,
maxHeightDiskCache: 40,
maxWidthDiskCache: 40,
),
),
),
IconButton(
icon: Icon(
playback.currentTrack?.id != null &&
playback.currentTrack?.id == track.value.id
? Icons.pause_circle_rounded
: Icons.play_circle_rounded,
color: Theme.of(context).primaryColor,
),
onPressed: () => onTrackPlayButtonPressed?.call(
track.value,
),
),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
track.value.name ?? "",
style: TextStyle(
fontWeight: FontWeight.bold,
fontSize: breakpoint.isSm ? 14 : 17,
),
overflow: TextOverflow.ellipsis,
),
artistsToClickableArtists(track.value.artists ?? [],
textStyle: TextStyle(
fontSize:
breakpoint.isLessThan(Breakpoints.lg) ? 12 : 14)),
],
),
),
if (breakpoint.isMoreThan(Breakpoints.md))
Expanded(
child: LinkText(
track.value.album!.name!,
"/album/${track.value.album?.id}",
extra: track.value.album,
overflow: TextOverflow.ellipsis,
),
),
if (!breakpoint.isSm) ...[
const SizedBox(width: 10),
Text(duration),
],
const SizedBox(width: 10),
if (auth.isLoggedIn)
FutureBuilder<bool>(
future: spotify.tracks.me.containsOne(track.value.id!),
builder: (context, snapshot) {
return PopupMenuButton(
icon: const Icon(Icons.more_horiz_rounded),
itemBuilder: (context) {
return [
PopupMenuItem(
child: Row(
children: const [
Icon(Icons.add_box_rounded),
SizedBox(width: 10),
Text("Add to Playlist"),
],
),
value: "playlist",
),
PopupMenuItem(
child: Row(
children: [
Icon(snapshot.data == true
? Icons.favorite_rounded
: Icons.favorite_border_rounded),
const SizedBox(width: 10),
const Text("Favorite")
],
),
value: "favorite",
)
];
},
onSelected: (value) {
switch (value) {
case "favorite":
actionFavorite(snapshot.data == true);
break;
case "playlist":
actionPlaylist();
break;
}
},
);
})
],
);
}
}

View File

@ -119,6 +119,7 @@ class MyApp extends HookConsumerWidget {
scaffoldBackgroundColor: Colors.blueGrey[900], scaffoldBackgroundColor: Colors.blueGrey[900],
dialogBackgroundColor: Colors.blueGrey[800], dialogBackgroundColor: Colors.blueGrey[800],
shadowColor: Colors.black26, shadowColor: Colors.black26,
popupMenuTheme: PopupMenuThemeData(color: Colors.blueGrey[800]),
buttonTheme: const ButtonThemeData( buttonTheme: const ButtonThemeData(
buttonColor: Colors.green, buttonColor: Colors.green,
), ),