1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2024-12-28 22:51:59 +00:00

feat(mobile): new video slider ui (#14126)

This commit is contained in:
Mert 2024-11-13 19:49:25 -05:00 committed by GitHub
parent 5a2af558fb
commit 11403abfbc
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 120 additions and 122 deletions

View file

@ -20,7 +20,7 @@ const String defaultColorPresetName = "indigo";
const Color immichBrandColorLight = Color(0xFF4150AF); const Color immichBrandColorLight = Color(0xFF4150AF);
const Color immichBrandColorDark = Color(0xFFACCBFA); const Color immichBrandColorDark = Color(0xFFACCBFA);
const Color whiteOpacity75 = Color.fromARGB((0.75 * 255) ~/ 1, 255, 255, 255); const Color whiteOpacity75 = Color.fromARGB((0.75 * 255) ~/ 1, 255, 255, 255);
const Color blackOpacity40 = Color.fromARGB((0.40 * 255) ~/ 1, 0, 0, 0); const Color blackOpacity90 = Color.fromARGB((0.90 * 255) ~/ 1, 0, 0, 0);
final Map<ImmichColorPreset, ImmichTheme> _themePresetsMap = { final Map<ImmichColorPreset, ImmichTheme> _themePresetsMap = {
ImmichColorPreset.indigo: ImmichTheme( ImmichColorPreset.indigo: ImmichTheme(

View file

@ -5,6 +5,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:fluttertoast/fluttertoast.dart'; import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/constants/immich_colors.dart';
import 'package:immich_mobile/extensions/build_context_extensions.dart'; import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/providers/album/album.provider.dart'; import 'package:immich_mobile/providers/album/album.provider.dart';
import 'package:immich_mobile/providers/album/current_album.provider.dart'; import 'package:immich_mobile/providers/album/current_album.provider.dart';
@ -327,39 +328,51 @@ class BottomGalleryBar extends ConsumerWidget {
child: AnimatedOpacity( child: AnimatedOpacity(
duration: const Duration(milliseconds: 100), duration: const Duration(milliseconds: 100),
opacity: ref.watch(showControlsProvider) ? 1.0 : 0.0, opacity: ref.watch(showControlsProvider) ? 1.0 : 0.0,
child: Column( child: DecoratedBox(
children: [ decoration: const BoxDecoration(
Visibility( gradient: LinearGradient(
visible: showVideoPlayerControls, begin: Alignment.bottomCenter,
child: const VideoControls(), end: Alignment.topCenter,
colors: [blackOpacity90, Colors.transparent],
), ),
BottomNavigationBar( ),
backgroundColor: Colors.black.withOpacity(0.4), position: DecorationPosition.background,
unselectedIconTheme: const IconThemeData(color: Colors.white), child: Padding(
selectedIconTheme: const IconThemeData(color: Colors.white), padding: EdgeInsets.only(top: 40.0),
unselectedLabelStyle: const TextStyle( child: Column(
color: Colors.white, children: [
fontWeight: FontWeight.w500, if (showVideoPlayerControls) const VideoControls(),
height: 2.3, BottomNavigationBar(
), elevation: 0.0,
selectedLabelStyle: const TextStyle( backgroundColor: Colors.transparent,
color: Colors.white, unselectedIconTheme: const IconThemeData(color: Colors.white),
fontWeight: FontWeight.w500, selectedIconTheme: const IconThemeData(color: Colors.white),
height: 2.3, unselectedLabelStyle: const TextStyle(
), color: Colors.white,
unselectedFontSize: 14, fontWeight: FontWeight.w500,
selectedFontSize: 14, height: 2.3,
selectedItemColor: Colors.white, ),
unselectedItemColor: Colors.white, selectedLabelStyle: const TextStyle(
showSelectedLabels: true, color: Colors.white,
showUnselectedLabels: true, fontWeight: FontWeight.w500,
items: height: 2.3,
albumActions.map((e) => e.keys.first).toList(growable: false), ),
onTap: (index) { unselectedFontSize: 14,
albumActions[index].values.first.call(index); selectedFontSize: 14,
}, selectedItemColor: Colors.white,
unselectedItemColor: Colors.white,
showSelectedLabels: true,
showUnselectedLabels: true,
items: albumActions
.map((e) => e.keys.first)
.toList(growable: false),
onTap: (index) {
albumActions[index].values.first.call(index);
},
),
],
), ),
], ),
), ),
), ),
); );

View file

@ -1,5 +1,4 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:immich_mobile/constants/immich_colors.dart';
@pragma('vm:prefer-inline') @pragma('vm:prefer-inline')
String _formatDuration(Duration position) { String _formatDuration(Duration position) {
@ -24,8 +23,8 @@ class FormattedDuration extends StatelessWidget {
_formatDuration(data), _formatDuration(data),
style: const TextStyle( style: const TextStyle(
fontSize: 14.0, fontSize: 14.0,
color: whiteOpacity75, color: Colors.white,
fontWeight: FontWeight.normal, fontWeight: FontWeight.w500,
), ),
textAlign: TextAlign.center, textAlign: TextAlign.center,
), ),

View file

@ -1,7 +1,5 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/constants/immich_colors.dart';
import 'package:immich_mobile/providers/asset_viewer/show_controls.provider.dart';
import 'package:immich_mobile/widgets/asset_viewer/video_position.dart'; import 'package:immich_mobile/widgets/asset_viewer/video_position.dart';
/// The video controls for the [videoPlayerControlsProvider] /// The video controls for the [videoPlayerControlsProvider]
@ -12,24 +10,11 @@ class VideoControls extends ConsumerWidget {
Widget build(BuildContext context, WidgetRef ref) { Widget build(BuildContext context, WidgetRef ref) {
final isPortrait = final isPortrait =
MediaQuery.orientationOf(context) == Orientation.portrait; MediaQuery.orientationOf(context) == Orientation.portrait;
return AnimatedOpacity( return isPortrait
opacity: ref.watch(showControlsProvider) ? 1.0 : 0.0, ? const VideoPosition()
duration: const Duration(milliseconds: 100), : const Padding(
child: isPortrait padding: EdgeInsets.symmetric(horizontal: 60.0),
? const ColoredBox( child: VideoPosition(),
color: blackOpacity40, );
child: Padding(
padding: EdgeInsets.symmetric(horizontal: 24.0),
child: VideoPosition(),
),
)
: const ColoredBox(
color: blackOpacity40,
child: Padding(
padding: EdgeInsets.symmetric(horizontal: 128.0),
child: VideoPosition(),
),
),
);
} }
} }

View file

@ -1,23 +0,0 @@
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/providers/asset_viewer/video_player_controls_provider.dart';
class VideoMuteButton extends ConsumerWidget {
const VideoMuteButton({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
return IconButton(
icon: ref.watch(
videoPlayerControlsProvider.select((value) => value.mute),
)
? const Icon(Icons.volume_off)
: const Icon(Icons.volume_up),
onPressed: () =>
ref.read(videoPlayerControlsProvider.notifier).toggleMute(),
color: Colors.white,
padding: const EdgeInsets.all(0),
alignment: Alignment.centerRight,
);
}
}

View file

@ -7,7 +7,6 @@ import 'package:immich_mobile/constants/immich_colors.dart';
import 'package:immich_mobile/providers/asset_viewer/video_player_controls_provider.dart'; import 'package:immich_mobile/providers/asset_viewer/video_player_controls_provider.dart';
import 'package:immich_mobile/providers/asset_viewer/video_player_value_provider.dart'; import 'package:immich_mobile/providers/asset_viewer/video_player_value_provider.dart';
import 'package:immich_mobile/widgets/asset_viewer/formatted_duration.dart'; import 'package:immich_mobile/widgets/asset_viewer/formatted_duration.dart';
import 'package:immich_mobile/widgets/asset_viewer/video_mute_button.dart';
class VideoPosition extends HookConsumerWidget { class VideoPosition extends HookConsumerWidget {
const VideoPosition({super.key}); const VideoPosition({super.key});
@ -20,38 +19,52 @@ class VideoPosition extends HookConsumerWidget {
final wasPlaying = useRef<bool>(true); final wasPlaying = useRef<bool>(true);
return duration == Duration.zero return duration == Duration.zero
? const _VideoPositionPlaceholder() ? const _VideoPositionPlaceholder()
: Row( : Column(
children: [ children: [
FormattedDuration(position), Padding(
Expanded( // align with slider's inherent padding
child: Slider( padding: const EdgeInsets.symmetric(horizontal: 12.0),
value: min( child: Row(
position.inMicroseconds / duration.inMicroseconds * 100, mainAxisAlignment: MainAxisAlignment.spaceBetween,
100, children: [
), FormattedDuration(position),
min: 0, FormattedDuration(duration),
max: 100, ],
thumbColor: Colors.white,
activeColor: Colors.white,
inactiveColor: whiteOpacity75,
onChangeStart: (value) {
final state = ref.read(videoPlaybackValueProvider).state;
wasPlaying.value = state != VideoPlaybackState.paused;
ref.read(videoPlayerControlsProvider.notifier).pause();
},
onChangeEnd: (value) {
if (wasPlaying.value) {
ref.read(videoPlayerControlsProvider.notifier).play();
}
},
onChanged: (position) {
ref.read(videoPlayerControlsProvider.notifier).position =
position;
},
), ),
), ),
FormattedDuration(duration), Row(
const VideoMuteButton(), children: [
Expanded(
child: Slider(
value: min(
position.inMicroseconds / duration.inMicroseconds * 100,
100,
),
min: 0,
max: 100,
thumbColor: Colors.white,
activeColor: Colors.white,
inactiveColor: whiteOpacity75,
onChangeStart: (value) {
final state =
ref.read(videoPlaybackValueProvider).state;
wasPlaying.value = state != VideoPlaybackState.paused;
ref.read(videoPlayerControlsProvider.notifier).pause();
},
onChangeEnd: (value) {
if (wasPlaying.value) {
ref.read(videoPlayerControlsProvider.notifier).play();
}
},
onChanged: (position) {
ref
.read(videoPlayerControlsProvider.notifier)
.position = position;
},
),
),
],
),
], ],
); );
} }
@ -64,22 +77,33 @@ class _VideoPositionPlaceholder extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return const Row( return const Column(
children: [ children: [
FormattedDuration(Duration.zero), Padding(
Expanded( padding: EdgeInsets.symmetric(horizontal: 12.0),
child: Slider( child: Row(
value: 0.0, mainAxisAlignment: MainAxisAlignment.spaceBetween,
min: 0, children: [
max: 100, FormattedDuration(Duration.zero),
thumbColor: Colors.white, FormattedDuration(Duration.zero),
activeColor: Colors.white, ],
inactiveColor: whiteOpacity75,
onChanged: _onChangedDummy,
), ),
), ),
FormattedDuration(Duration.zero), Row(
VideoMuteButton(), children: [
Expanded(
child: Slider(
value: 0.0,
min: 0,
max: 100,
thumbColor: Colors.white,
activeColor: Colors.white,
inactiveColor: whiteOpacity75,
onChanged: _onChangedDummy,
),
),
],
),
], ],
); );
} }