Compare commits
1 Commits
feat/mobil
...
chore/acti
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bee1698bb1 |
4
.github/workflows/build-mobile.yml
vendored
4
.github/workflows/build-mobile.yml
vendored
@@ -73,7 +73,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Restore Gradle Cache
|
- name: Restore Gradle Cache
|
||||||
id: cache-gradle-restore
|
id: cache-gradle-restore
|
||||||
uses: actions/cache/restore@0400d5f644dc74513175e3cd8d07132dd4860809 # v4
|
uses: actions/cache/restore@0400d5f644dc74513175e3cd8d07132dd4860809 # v4.2.4
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
~/.gradle/caches
|
~/.gradle/caches
|
||||||
@@ -130,7 +130,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Save Gradle Cache
|
- name: Save Gradle Cache
|
||||||
id: cache-gradle-save
|
id: cache-gradle-save
|
||||||
uses: actions/cache/save@0400d5f644dc74513175e3cd8d07132dd4860809 # v4
|
uses: actions/cache/save@0400d5f644dc74513175e3cd8d07132dd4860809 # v4.2.4
|
||||||
if: github.ref == 'refs/heads/main'
|
if: github.ref == 'refs/heads/main'
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
|
|||||||
@@ -26,7 +26,6 @@ import 'package:wakelock_plus/wakelock_plus.dart';
|
|||||||
|
|
||||||
@RoutePage()
|
@RoutePage()
|
||||||
class NativeVideoViewerPage extends HookConsumerWidget {
|
class NativeVideoViewerPage extends HookConsumerWidget {
|
||||||
static final log = Logger('NativeVideoViewer');
|
|
||||||
final Asset asset;
|
final Asset asset;
|
||||||
final bool showControls;
|
final bool showControls;
|
||||||
final int playbackDelayFactor;
|
final int playbackDelayFactor;
|
||||||
@@ -60,6 +59,8 @@ class NativeVideoViewerPage extends HookConsumerWidget {
|
|||||||
// Used to show the placeholder during hero animations for remote videos to avoid a stutter
|
// Used to show the placeholder during hero animations for remote videos to avoid a stutter
|
||||||
final isVisible = useState(Platform.isIOS && asset.isLocal);
|
final isVisible = useState(Platform.isIOS && asset.isLocal);
|
||||||
|
|
||||||
|
final log = Logger('NativeVideoViewerPage');
|
||||||
|
|
||||||
final isCasting = ref.watch(castProvider.select((c) => c.isCasting));
|
final isCasting = ref.watch(castProvider.select((c) => c.isCasting));
|
||||||
|
|
||||||
final isVideoReady = useState(false);
|
final isVideoReady = useState(false);
|
||||||
@@ -141,7 +142,7 @@ class NativeVideoViewerPage extends HookConsumerWidget {
|
|||||||
interval: const Duration(milliseconds: 100),
|
interval: const Duration(milliseconds: 100),
|
||||||
maxWaitTime: const Duration(milliseconds: 200),
|
maxWaitTime: const Duration(milliseconds: 200),
|
||||||
);
|
);
|
||||||
ref.listen(videoPlayerControlsProvider, (oldControls, newControls) {
|
ref.listen(videoPlayerControlsProvider, (oldControls, newControls) async {
|
||||||
final playerController = controller.value;
|
final playerController = controller.value;
|
||||||
if (playerController == null) {
|
if (playerController == null) {
|
||||||
return;
|
return;
|
||||||
@@ -152,14 +153,28 @@ class NativeVideoViewerPage extends HookConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final oldSeek = oldControls?.position.inMilliseconds;
|
final oldSeek = (oldControls?.position ?? 0) ~/ 1;
|
||||||
final newSeek = newControls.position.inMilliseconds;
|
final newSeek = newControls.position ~/ 1;
|
||||||
if (oldSeek != newSeek || newControls.restarted) {
|
if (oldSeek != newSeek || newControls.restarted) {
|
||||||
seekDebouncer.run(() => playerController.seekTo(newSeek));
|
seekDebouncer.run(() => playerController.seekTo(newSeek));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldControls?.pause != newControls.pause || newControls.restarted) {
|
if (oldControls?.pause != newControls.pause || newControls.restarted) {
|
||||||
unawaited(_onPauseChange(context, playerController, seekDebouncer, newControls.pause));
|
// Make sure the last seek is complete before pausing or playing
|
||||||
|
// Otherwise, `onPlaybackPositionChanged` can receive outdated events
|
||||||
|
if (seekDebouncer.isActive) {
|
||||||
|
await seekDebouncer.drain();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (newControls.pause) {
|
||||||
|
await playerController.pause();
|
||||||
|
} else {
|
||||||
|
await playerController.play();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
log.severe('Error pausing or playing video: $error');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -216,7 +231,7 @@ class NativeVideoViewerPage extends HookConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ref.read(videoPlaybackValueProvider.notifier).position = Duration(milliseconds: playbackInfo.position);
|
ref.read(videoPlaybackValueProvider.notifier).position = Duration(seconds: playbackInfo.position);
|
||||||
|
|
||||||
// Check if the video is buffering
|
// Check if the video is buffering
|
||||||
if (playbackInfo.status == PlaybackStatus.playing) {
|
if (playbackInfo.status == PlaybackStatus.playing) {
|
||||||
@@ -373,35 +388,4 @@ class NativeVideoViewerPage extends HookConsumerWidget {
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _onPauseChange(
|
|
||||||
BuildContext context,
|
|
||||||
NativeVideoPlayerController controller,
|
|
||||||
Debouncer seekDebouncer,
|
|
||||||
bool isPaused,
|
|
||||||
) async {
|
|
||||||
if (!context.mounted) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the last seek is complete before pausing or playing
|
|
||||||
// Otherwise, `onPlaybackPositionChanged` can receive outdated events
|
|
||||||
if (seekDebouncer.isActive) {
|
|
||||||
await seekDebouncer.drain();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!context.mounted) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (isPaused) {
|
|
||||||
await controller.pause();
|
|
||||||
} else {
|
|
||||||
await controller.play();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
log.severe('Error pausing or playing video: $error');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,7 +45,6 @@ bool _isCurrentAsset(BaseAsset asset, BaseAsset? currentAsset) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class NativeVideoViewer extends HookConsumerWidget {
|
class NativeVideoViewer extends HookConsumerWidget {
|
||||||
static final log = Logger('NativeVideoViewer');
|
|
||||||
final BaseAsset asset;
|
final BaseAsset asset;
|
||||||
final bool showControls;
|
final bool showControls;
|
||||||
final int playbackDelayFactor;
|
final int playbackDelayFactor;
|
||||||
@@ -79,6 +78,8 @@ class NativeVideoViewer extends HookConsumerWidget {
|
|||||||
// Used to show the placeholder during hero animations for remote videos to avoid a stutter
|
// Used to show the placeholder during hero animations for remote videos to avoid a stutter
|
||||||
final isVisible = useState(Platform.isIOS && asset.hasLocal);
|
final isVisible = useState(Platform.isIOS && asset.hasLocal);
|
||||||
|
|
||||||
|
final log = Logger('NativeVideoViewerPage');
|
||||||
|
|
||||||
final isCasting = ref.watch(castProvider.select((c) => c.isCasting));
|
final isCasting = ref.watch(castProvider.select((c) => c.isCasting));
|
||||||
|
|
||||||
Future<VideoSource?> createSource() async {
|
Future<VideoSource?> createSource() async {
|
||||||
@@ -159,7 +160,7 @@ class NativeVideoViewer extends HookConsumerWidget {
|
|||||||
interval: const Duration(milliseconds: 100),
|
interval: const Duration(milliseconds: 100),
|
||||||
maxWaitTime: const Duration(milliseconds: 200),
|
maxWaitTime: const Duration(milliseconds: 200),
|
||||||
);
|
);
|
||||||
ref.listen(videoPlayerControlsProvider, (oldControls, newControls) {
|
ref.listen(videoPlayerControlsProvider, (oldControls, newControls) async {
|
||||||
final playerController = controller.value;
|
final playerController = controller.value;
|
||||||
if (playerController == null) {
|
if (playerController == null) {
|
||||||
return;
|
return;
|
||||||
@@ -170,14 +171,28 @@ class NativeVideoViewer extends HookConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final oldSeek = oldControls?.position.inMilliseconds;
|
final oldSeek = (oldControls?.position ?? 0) ~/ 1;
|
||||||
final newSeek = newControls.position.inMilliseconds;
|
final newSeek = newControls.position ~/ 1;
|
||||||
if (oldSeek != newSeek || newControls.restarted) {
|
if (oldSeek != newSeek || newControls.restarted) {
|
||||||
seekDebouncer.run(() => playerController.seekTo(newSeek));
|
seekDebouncer.run(() => playerController.seekTo(newSeek));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oldControls?.pause != newControls.pause || newControls.restarted) {
|
if (oldControls?.pause != newControls.pause || newControls.restarted) {
|
||||||
unawaited(_onPauseChange(context, playerController, seekDebouncer, newControls.pause));
|
// Make sure the last seek is complete before pausing or playing
|
||||||
|
// Otherwise, `onPlaybackPositionChanged` can receive outdated events
|
||||||
|
if (seekDebouncer.isActive) {
|
||||||
|
await seekDebouncer.drain();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (newControls.pause) {
|
||||||
|
await playerController.pause();
|
||||||
|
} else {
|
||||||
|
await playerController.play();
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
log.severe('Error pausing or playing video: $error');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -236,7 +251,7 @@ class NativeVideoViewer extends HookConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ref.read(videoPlaybackValueProvider.notifier).position = Duration(milliseconds: playbackInfo.position);
|
ref.read(videoPlaybackValueProvider.notifier).position = Duration(seconds: playbackInfo.position);
|
||||||
|
|
||||||
// Check if the video is buffering
|
// Check if the video is buffering
|
||||||
if (playbackInfo.status == PlaybackStatus.playing) {
|
if (playbackInfo.status == PlaybackStatus.playing) {
|
||||||
@@ -392,31 +407,4 @@ class NativeVideoViewer extends HookConsumerWidget {
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _onPauseChange(
|
|
||||||
BuildContext context,
|
|
||||||
NativeVideoPlayerController controller,
|
|
||||||
Debouncer seekDebouncer,
|
|
||||||
bool isPaused,
|
|
||||||
) async {
|
|
||||||
if (!context.mounted) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the last seek is complete before pausing or playing
|
|
||||||
// Otherwise, `onPlaybackPositionChanged` can receive outdated events
|
|
||||||
if (seekDebouncer.isActive) {
|
|
||||||
await seekDebouncer.drain();
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (isPaused) {
|
|
||||||
await controller.pause();
|
|
||||||
} else {
|
|
||||||
await controller.play();
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
log.severe('Error pausing or playing video: $error');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import 'package:immich_mobile/providers/asset_viewer/video_player_value_provider
|
|||||||
class VideoPlaybackControls {
|
class VideoPlaybackControls {
|
||||||
const VideoPlaybackControls({required this.position, required this.pause, this.restarted = false});
|
const VideoPlaybackControls({required this.position, required this.pause, this.restarted = false});
|
||||||
|
|
||||||
final Duration position;
|
final double position;
|
||||||
final bool pause;
|
final bool pause;
|
||||||
final bool restarted;
|
final bool restarted;
|
||||||
}
|
}
|
||||||
@@ -13,7 +13,7 @@ final videoPlayerControlsProvider = StateNotifierProvider<VideoPlayerControls, V
|
|||||||
return VideoPlayerControls(ref);
|
return VideoPlayerControls(ref);
|
||||||
});
|
});
|
||||||
|
|
||||||
const videoPlayerControlsDefault = VideoPlaybackControls(position: Duration.zero, pause: false);
|
const videoPlayerControlsDefault = VideoPlaybackControls(position: 0, pause: false);
|
||||||
|
|
||||||
class VideoPlayerControls extends StateNotifier<VideoPlaybackControls> {
|
class VideoPlayerControls extends StateNotifier<VideoPlaybackControls> {
|
||||||
VideoPlayerControls(this.ref) : super(videoPlayerControlsDefault);
|
VideoPlayerControls(this.ref) : super(videoPlayerControlsDefault);
|
||||||
@@ -30,10 +30,10 @@ class VideoPlayerControls extends StateNotifier<VideoPlaybackControls> {
|
|||||||
state = videoPlayerControlsDefault;
|
state = videoPlayerControlsDefault;
|
||||||
}
|
}
|
||||||
|
|
||||||
Duration get position => state.position;
|
double get position => state.position;
|
||||||
bool get paused => state.pause;
|
bool get paused => state.pause;
|
||||||
|
|
||||||
set position(Duration value) {
|
set position(double value) {
|
||||||
if (state.position == value) {
|
if (state.position == value) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -62,7 +62,7 @@ class VideoPlayerControls extends StateNotifier<VideoPlaybackControls> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void restart() {
|
void restart() {
|
||||||
state = const VideoPlaybackControls(position: Duration.zero, pause: false, restarted: true);
|
state = const VideoPlaybackControls(position: 0, pause: false, restarted: true);
|
||||||
ref.read(videoPlaybackValueProvider.notifier).value = ref
|
ref.read(videoPlaybackValueProvider.notifier).value = ref
|
||||||
.read(videoPlaybackValueProvider.notifier)
|
.read(videoPlaybackValueProvider.notifier)
|
||||||
.value
|
.value
|
||||||
|
|||||||
@@ -33,8 +33,8 @@ class VideoPlaybackValue {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return VideoPlaybackValue(
|
return VideoPlaybackValue(
|
||||||
position: Duration(milliseconds: playbackInfo.position),
|
position: Duration(seconds: playbackInfo.position),
|
||||||
duration: Duration(milliseconds: videoInfo.duration),
|
duration: Duration(seconds: videoInfo.duration),
|
||||||
state: status,
|
state: status,
|
||||||
volume: playbackInfo.volume,
|
volume: playbackInfo.volume,
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ class VideoPosition extends HookConsumerWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ref.read(videoPlayerControlsProvider.notifier).position = seekToDuration;
|
ref.read(videoPlayerControlsProvider.notifier).position = seekToDuration.inSeconds.toDouble();
|
||||||
|
|
||||||
// This immediately updates the slider position without waiting for the video to update
|
// This immediately updates the slider position without waiting for the video to update
|
||||||
ref.read(videoPlaybackValueProvider.notifier).position = seekToDuration;
|
ref.read(videoPlaybackValueProvider.notifier).position = seekToDuration;
|
||||||
|
|||||||
@@ -1208,8 +1208,8 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
path: "."
|
path: "."
|
||||||
ref: c4cdb27
|
ref: "893894b"
|
||||||
resolved-ref: c4cdb277b02d956c4effb186905329bcb9771adc
|
resolved-ref: "893894b98b832be8a995a8d5d4c2289d0ad2d246"
|
||||||
url: "https://github.com/immich-app/native_video_player"
|
url: "https://github.com/immich-app/native_video_player"
|
||||||
source: git
|
source: git
|
||||||
version: "1.3.1"
|
version: "1.3.1"
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ dependencies:
|
|||||||
native_video_player:
|
native_video_player:
|
||||||
git:
|
git:
|
||||||
url: https://github.com/immich-app/native_video_player
|
url: https://github.com/immich-app/native_video_player
|
||||||
ref: 'c4cdb27'
|
ref: '893894b'
|
||||||
openapi:
|
openapi:
|
||||||
path: openapi
|
path: openapi
|
||||||
isar:
|
isar:
|
||||||
|
|||||||
Reference in New Issue
Block a user