2024-05-24 00:10:11 +02:00
|
|
|
import 'dart:convert';
|
|
|
|
|
2024-05-23 20:36:42 +02:00
|
|
|
import 'package:auto_size_text/auto_size_text.dart';
|
2024-05-24 00:10:11 +02:00
|
|
|
import 'package:crypto/crypto.dart';
|
2024-05-23 20:36:42 +02:00
|
|
|
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';
|
2024-05-24 00:10:11 +02:00
|
|
|
import 'package:ocarina/api/subsonic/song.dart';
|
|
|
|
import 'package:ocarina/main.dart';
|
2024-05-23 19:14:08 +02:00
|
|
|
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-24 00:10:11 +02:00
|
|
|
import 'package:text_scroll/text_scroll.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() {
|
2024-05-24 00:10:11 +02:00
|
|
|
logger.d(AudioPlayerService().song?.coverArtUrl);
|
2024-05-23 19:14:08 +02:00
|
|
|
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),
|
2024-05-24 00:10:11 +02:00
|
|
|
child: ValueListenableBuilder<Song?>(
|
|
|
|
valueListenable: songNotifier,
|
|
|
|
builder: (c, t, w) {
|
|
|
|
return Row(
|
|
|
|
children: [
|
|
|
|
SizedBox(
|
|
|
|
height: 10.h,
|
|
|
|
width: 10.h,
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.all(8),
|
|
|
|
child: ClipRRect(
|
|
|
|
child: (t == null)
|
|
|
|
? ColoredBox(
|
2024-05-23 20:36:42 +02:00
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.primaryContainer,
|
|
|
|
child: Center(
|
|
|
|
child: Icon(
|
|
|
|
Icons.music_note,
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
),
|
|
|
|
),
|
2024-05-24 00:10:11 +02:00
|
|
|
)
|
|
|
|
: FastCachedImage(
|
|
|
|
key: Key(
|
|
|
|
md5
|
|
|
|
.convert(
|
|
|
|
utf8.encode(
|
|
|
|
t.coverArtUrl,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toString(),
|
|
|
|
),
|
|
|
|
url: t.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 20:36:42 +02:00
|
|
|
),
|
2024-05-24 00:10:11 +02:00
|
|
|
const SizedBox(
|
|
|
|
width: 5,
|
|
|
|
),
|
|
|
|
Expanded(
|
|
|
|
child: AutoSizeText(
|
|
|
|
AudioPlayerService().song == null
|
|
|
|
? "Nothing"
|
|
|
|
: AudioPlayerService().song!.title,
|
|
|
|
style: TextStyle(
|
|
|
|
color: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.onPrimaryContainer,
|
|
|
|
fontSize: 14,
|
|
|
|
decoration: TextDecoration.none,
|
|
|
|
),
|
|
|
|
overflowReplacement: TextScroll(
|
|
|
|
AudioPlayerService().song == null
|
|
|
|
? "Nothing"
|
|
|
|
: AudioPlayerService().song!.title,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
SizedBox(
|
|
|
|
width: 30.w,
|
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
IconButton(
|
|
|
|
onPressed: () async {
|
|
|
|
if (AudioPlayerService().song ==
|
|
|
|
null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (AudioPlayerService().isPlaying) {
|
|
|
|
await AudioPlayerService().pause();
|
|
|
|
} else {
|
|
|
|
AudioPlayerService().resume();
|
|
|
|
}
|
|
|
|
setState(() {});
|
|
|
|
},
|
|
|
|
icon: AnimatedCrossFade(
|
|
|
|
firstChild:
|
|
|
|
const Icon(Icons.play_arrow),
|
|
|
|
secondChild: const Icon(Icons.pause),
|
|
|
|
crossFadeState:
|
|
|
|
(AudioPlayerService().isPlaying)
|
|
|
|
? CrossFadeState.showSecond
|
|
|
|
: CrossFadeState.showFirst,
|
|
|
|
duration:
|
|
|
|
const Duration(milliseconds: 300),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
2024-05-23 20:36:42 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
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
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|