Multiple changes:
- Split up UI into several files - Handle connection state better - Add images - Granular rebuilding of state dependent widgets - Fix usage of alert dialogs - Add some basic theming - Add dialog for adding multiple links at once
This commit is contained in:
181
lib/player_ui/bottom_bar.dart
Normal file
181
lib/player_ui/bottom_bar.dart
Normal file
@@ -0,0 +1,181 @@
|
||||
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(
|
||||
child: Row(
|
||||
children: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.skip_previous),
|
||||
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),
|
||||
onPressed: () {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.togglePlayback());
|
||||
},
|
||||
);
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.skip_next),
|
||||
onPressed: () {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.playlistNext());
|
||||
},
|
||||
),
|
||||
Expanded(
|
||||
child: playerBlocBuilder(
|
||||
buildProps: (p) => [
|
||||
p.cachedTimestamp,
|
||||
p.duration,
|
||||
p.currentPercentPosition,
|
||||
p.volume,
|
||||
],
|
||||
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 (cachedPercent > 100) {
|
||||
cachedPercent = 0;
|
||||
}
|
||||
return Flex(
|
||||
direction: Axis.horizontal,
|
||||
children: [
|
||||
Text(
|
||||
formatTime(
|
||||
Duration(
|
||||
milliseconds:
|
||||
playerState.currentPercentPosition != null
|
||||
? (playerState.currentPercentPosition! *
|
||||
playerState.duration.inMilliseconds *
|
||||
0.01)
|
||||
.round()
|
||||
: 0,
|
||||
),
|
||||
),
|
||||
),
|
||||
// Text(((playerState.currentPercentPosition ?? 0.0) *
|
||||
// 0.01 *
|
||||
// playerState.duration)
|
||||
// .toString()),
|
||||
Expanded(
|
||||
flex: 5,
|
||||
child: 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));
|
||||
},
|
||||
),
|
||||
),
|
||||
Text(formatTime(playerState.duration)),
|
||||
// TODO: set minimum width for this slider
|
||||
Expanded(
|
||||
flex: 1,
|
||||
child: Slider(
|
||||
value: playerState.volume,
|
||||
max: 130.0,
|
||||
secondaryTrackValue: 100.0,
|
||||
onChanged: (value) {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.volume(value));
|
||||
},
|
||||
),
|
||||
),
|
||||
Text('${playerState.volume.round()}%'),
|
||||
],
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
playerBlocBuilder(
|
||||
buildProps: (p) => [p.subtitleTracks],
|
||||
builder: (context, playerState) => PopupMenuButton(
|
||||
icon: const Icon(Icons.subtitles),
|
||||
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),
|
||||
isSelected: playerState.isLooping,
|
||||
onPressed: () {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.setLooping(!playerState.isLooping));
|
||||
},
|
||||
),
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.shuffle),
|
||||
onPressed: () {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.shuffle());
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.delete_forever),
|
||||
onPressed: () {
|
||||
BlocProvider.of<ConnectionStateBloc>(context)
|
||||
.add(Command.playlistClear());
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user