2024-12-18 20:12:14 +01:00
|
|
|
import 'dart:math' show max;
|
|
|
|
import 'dart:developer' show log;
|
|
|
|
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
|
|
|
|
|
|
import 'package:gergle/api/commands.dart';
|
|
|
|
import 'package:gergle/state/connection_state_bloc.dart';
|
|
|
|
import 'package:gergle/player_ui/main.dart';
|
|
|
|
|
|
|
|
class PlayerUIBottomBar extends StatelessWidget {
|
|
|
|
const PlayerUIBottomBar({super.key});
|
|
|
|
|
|
|
|
static String formatTime(Duration duration) {
|
|
|
|
final hours = duration.inHours;
|
|
|
|
final minutes = duration.inMinutes.remainder(60).toString().padLeft(2, '0');
|
|
|
|
final seconds = duration.inSeconds.remainder(60).toString().padLeft(2, '0');
|
|
|
|
|
|
|
|
if (hours > 0) {
|
|
|
|
return '$hours:$minutes:$seconds';
|
|
|
|
} else {
|
|
|
|
return '$minutes:$seconds';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return BottomAppBar(
|
2024-12-19 13:17:31 +01:00
|
|
|
color: Theme.of(context).colorScheme.surfaceBright,
|
2024-12-18 20:12:14 +01:00
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
IconButton(
|
|
|
|
icon: const Icon(Icons.skip_previous),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Skip to previous track',
|
2024-12-18 20:12:14 +01:00
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.playlistPrevious());
|
|
|
|
},
|
|
|
|
),
|
|
|
|
playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.isPlaying],
|
|
|
|
builder: (context, playerState) {
|
|
|
|
return IconButton(
|
|
|
|
icon: (playerState.isPlaying)
|
|
|
|
? const Icon(Icons.pause)
|
|
|
|
: const Icon(Icons.play_arrow),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: (playerState.isPlaying) ? 'Pause' : 'Play',
|
2024-12-18 20:12:14 +01:00
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.togglePlayback());
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
icon: const Icon(Icons.skip_next),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Skip to next track',
|
2024-12-18 20:12:14 +01:00
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.playlistNext());
|
|
|
|
},
|
|
|
|
),
|
|
|
|
Expanded(
|
2024-12-19 15:09:15 +01:00
|
|
|
child: LayoutBuilder(
|
|
|
|
builder: (context, constraint) {
|
|
|
|
return Row(
|
2024-12-18 20:12:14 +01:00
|
|
|
children: [
|
2024-12-19 15:09:15 +01:00
|
|
|
playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.currentPercentPosition, p.duration],
|
|
|
|
builder: (context, playerState) {
|
|
|
|
final milliseconds =
|
|
|
|
(playerState.currentPercentPosition! *
|
|
|
|
playerState.duration.inMilliseconds *
|
|
|
|
0.01)
|
|
|
|
.round();
|
|
|
|
return Text(
|
|
|
|
formatTime(
|
|
|
|
Duration(milliseconds: milliseconds),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
2024-12-18 20:12:14 +01:00
|
|
|
),
|
2024-12-19 15:09:15 +01:00
|
|
|
Flexible(
|
2024-12-18 20:12:14 +01:00
|
|
|
flex: 5,
|
2024-12-19 15:09:15 +01:00
|
|
|
child: playerBlocBuilder(
|
|
|
|
buildProps: (p) => [
|
|
|
|
p.cachedTimestamp,
|
|
|
|
p.currentPercentPosition,
|
|
|
|
p.duration,
|
|
|
|
],
|
|
|
|
builder: (context, playerState) {
|
|
|
|
// NOTE: slider throws if the value is over 100, so 99.999 is used to avoid
|
|
|
|
// hitting the limit with floating point errors.
|
|
|
|
double cachedPercent = playerState.cachedTimestamp !=
|
|
|
|
null
|
|
|
|
? ((playerState.cachedTimestamp! /
|
|
|
|
max(playerState.duration.inMilliseconds,
|
|
|
|
0.00000000000001)) *
|
|
|
|
1000 *
|
|
|
|
99.999)
|
|
|
|
: 0.0;
|
|
|
|
if (0 < cachedPercent || cachedPercent > 100) {
|
|
|
|
cachedPercent = 0;
|
|
|
|
}
|
|
|
|
return Slider(
|
|
|
|
value: playerState.currentPercentPosition ?? 0,
|
|
|
|
max: 100.0,
|
|
|
|
secondaryTrackValue: cachedPercent,
|
|
|
|
onChanged: (value) {
|
|
|
|
log('Setting time to $value');
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.time(value));
|
|
|
|
},
|
|
|
|
);
|
2024-12-18 20:12:14 +01:00
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
2024-12-19 15:09:15 +01:00
|
|
|
playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.duration],
|
|
|
|
builder: (context, playerState) =>
|
|
|
|
Text(formatTime(playerState.duration)),
|
2024-12-18 20:12:14 +01:00
|
|
|
),
|
2024-12-19 15:09:15 +01:00
|
|
|
SizedBox(
|
|
|
|
width: max(constraint.maxWidth / 6, 200),
|
|
|
|
child: playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.volume],
|
|
|
|
builder: (context, playerState) => Slider(
|
|
|
|
value: playerState.volume,
|
|
|
|
max: 130.0,
|
|
|
|
secondaryTrackValue: 100.0,
|
|
|
|
onChanged: (value) {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.volume(value));
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
2024-12-18 20:12:14 +01:00
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.subtitleTracks],
|
|
|
|
builder: (context, playerState) => PopupMenuButton(
|
|
|
|
icon: const Icon(Icons.subtitles),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Select subtitle track',
|
2024-12-18 20:12:14 +01:00
|
|
|
enabled: playerState.subtitleTracks.isNotEmpty,
|
|
|
|
itemBuilder: (context) {
|
|
|
|
return playerState.subtitleTracks
|
|
|
|
.map((track) => PopupMenuItem(
|
|
|
|
value: track.id,
|
|
|
|
child: Text(track.title),
|
|
|
|
))
|
|
|
|
.toList();
|
|
|
|
},
|
|
|
|
onSelected: (value) {
|
|
|
|
// TODO: add command for changing subtitle track
|
|
|
|
throw UnimplementedError();
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
playerBlocBuilder(
|
|
|
|
buildProps: (p) => [p.isLooping],
|
|
|
|
builder: (context, playerState) => IconButton(
|
|
|
|
icon: const Icon(Icons.repeat),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Toggle playlist looping',
|
2024-12-18 20:12:14 +01:00
|
|
|
isSelected: playerState.isLooping,
|
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.setLooping(!playerState.isLooping));
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
icon: const Icon(Icons.shuffle),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Shuffle playlist',
|
2024-12-18 20:12:14 +01:00
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.shuffle());
|
|
|
|
},
|
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
icon: const Icon(Icons.delete_forever),
|
2024-12-23 00:12:32 +01:00
|
|
|
tooltip: 'Clear playlist',
|
|
|
|
color: Colors.redAccent,
|
2024-12-18 20:12:14 +01:00
|
|
|
onPressed: () {
|
|
|
|
BlocProvider.of<ConnectionStateBloc>(context)
|
|
|
|
.add(Command.playlistClear());
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|