2024-05-23 20:36:42 +02:00
|
|
|
import 'package:auto_size_text/auto_size_text.dart';
|
|
|
|
import 'package:fast_cached_network_image/fast_cached_network_image.dart';
|
2024-05-23 19:14:08 +02:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:ocarina/api/audio/audioplayer_service.dart';
|
|
|
|
import 'package:ocarina/util/util.dart';
|
2024-05-23 20:36:42 +02:00
|
|
|
import 'package:responsive_sizer/responsive_sizer.dart';
|
|
|
|
import 'package:shimmer/shimmer.dart';
|
2024-05-23 19:14:08 +02:00
|
|
|
|
|
|
|
/// The player widget
|
|
|
|
///
|
|
|
|
/// Showcases the playing song's details and features playback controls
|
|
|
|
class Player extends StatefulWidget {
|
|
|
|
/// The player widget
|
|
|
|
///
|
|
|
|
/// Showcases the playing song's details and features playback controls
|
|
|
|
const Player({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<Player> createState() => PlayerState();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// State of [Player]
|
|
|
|
class PlayerState extends State<Player> {
|
|
|
|
void update() {
|
|
|
|
logger.d(AudioPlayerService().song?.title);
|
|
|
|
setState(() {});
|
|
|
|
}
|
|
|
|
|
2024-05-23 20:36:42 +02:00
|
|
|
var _showFullControls = false;
|
|
|
|
final _sheetController = DraggableScrollableController();
|
2024-05-23 19:14:08 +02:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-05-23 20:36:42 +02:00
|
|
|
return NotificationListener<DraggableScrollableNotification>(
|
|
|
|
onNotification: (n) {
|
|
|
|
if (n.extent > 0.3) {
|
|
|
|
_showFullControls = true;
|
|
|
|
} else {
|
|
|
|
_showFullControls = false;
|
|
|
|
}
|
|
|
|
setState(() {});
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
child: DraggableScrollableSheet(
|
|
|
|
controller: _sheetController,
|
|
|
|
initialChildSize: 0.1,
|
|
|
|
snap: true,
|
|
|
|
snapSizes: const [0.1, 1],
|
|
|
|
minChildSize: 0.1,
|
|
|
|
builder: (c, s) => SingleChildScrollView(
|
|
|
|
controller: s,
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
AnimatedOpacity(
|
|
|
|
opacity: _showFullControls ? 0 : 1,
|
|
|
|
duration: const Duration(milliseconds: 300),
|
|
|
|
child: ClipRRect(
|
|
|
|
borderRadius: BorderRadius.circular(8),
|
|
|
|
child: GestureDetector(
|
|
|
|
onTap: () {
|
|
|
|
_sheetController.animateTo(
|
|
|
|
(_sheetController.size == 1) ? 0.1 : 1,
|
|
|
|
duration: const Duration(milliseconds: 300),
|
|
|
|
curve: Curves.easeIn,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
child: Container(
|
|
|
|
color: Theme.of(context).colorScheme.primaryContainer,
|
|
|
|
height: 10.h,
|
|
|
|
width: 100.w,
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(8),
|
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
SizedBox(
|
|
|
|
height: 10.h,
|
|
|
|
width: 10.h,
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(8),
|
|
|
|
child: ClipRRect(
|
|
|
|
child: (AudioPlayerService().song == null)
|
|
|
|
? ColoredBox(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.primaryContainer,
|
|
|
|
child: Center(
|
|
|
|
child: Icon(
|
|
|
|
Icons.music_note,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
: (FastCachedImage(
|
|
|
|
url: AudioPlayerService()
|
|
|
|
.song!
|
|
|
|
.coverArtUrl,
|
|
|
|
loadingBuilder: (c, d) =>
|
|
|
|
Shimmer.fromColors(
|
|
|
|
baseColor: Colors.grey.shade300,
|
|
|
|
highlightColor:
|
|
|
|
Colors.grey.shade100,
|
|
|
|
child: Container(
|
|
|
|
color: Colors.grey,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
errorBuilder: (c, _, __) {
|
|
|
|
logger
|
|
|
|
..e(_)
|
|
|
|
..e(__);
|
|
|
|
return ColoredBox(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.primaryContainer,
|
|
|
|
child: Center(
|
|
|
|
child: Icon(
|
|
|
|
Icons.music_note,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
width: 5,
|
|
|
|
),
|
|
|
|
AutoSizeText(
|
|
|
|
AudioPlayerService().song == null
|
|
|
|
? "Nothing"
|
|
|
|
: AudioPlayerService().song!.title,
|
|
|
|
style: TextStyle(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
fontSize: 14,
|
|
|
|
decoration: TextDecoration.none,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
SizedBox(
|
|
|
|
height: 90.h,
|
|
|
|
width: 100.w,
|
|
|
|
child: Center(
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
ClipRRect(
|
|
|
|
child: (AudioPlayerService().song == null)
|
|
|
|
? ColoredBox(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.primaryContainer,
|
|
|
|
child: Center(
|
|
|
|
child: Icon(
|
|
|
|
Icons.music_note,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
: (FastCachedImage(
|
|
|
|
url: AudioPlayerService().song!.coverArtUrl,
|
|
|
|
loadingBuilder: (c, d) => Shimmer.fromColors(
|
|
|
|
baseColor: Colors.grey.shade300,
|
|
|
|
highlightColor: Colors.grey.shade100,
|
|
|
|
child: Container(
|
|
|
|
color: Colors.grey,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
errorBuilder: (c, _, __) {
|
|
|
|
logger
|
|
|
|
..e(_)
|
|
|
|
..e(__);
|
|
|
|
return ColoredBox(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.primaryContainer,
|
|
|
|
child: Center(
|
|
|
|
child: Icon(
|
|
|
|
Icons.music_note,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
)),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
2024-05-23 19:14:08 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|