mirror of
https://github.com/immich-app/immich.git
synced 2026-06-04 13:15:22 -04:00
9d4a6614b1
* feat(mobile): handle Android ACTION_VIEW intent - add ViewIntent Pigeon API and generated bindings - implement Android ViewIntentPlugin + iOS no-op host - route ExternalMediaViewer by ViewIntentAttachment - buffer pending view intents and flush on user ready/resume * feat(mobile): fallback to computed checksum for timeline match - hash local asset on-demand when checksum missing - search main timeline by localId or checksum before standalone viewer - persist computed hash into local_asset_entity * fix(mobile): proper handling is user authenticated * feat(mobile): open ACTION_VIEW fallback in AssetViewer drop ExternalMediaViewer route * feat(mobile): add logger * test(mobile): add unit tests for view intent pending/flush flow * fix(mobile): fix format * fix(mobile): remove redundant iOS code update code related to LocalAsset model and asset viewer * refactor(mobile): simplify view intent flow and support file-backed ACTION_VIEW assets remove redundant view intent model/repository layer handle transient ACTION_VIEW files in viewer/upload flow clean up managed temp files for fallback assets * refactor(mobile): extract MediaStore utils and resolve view intents via merged assets * refactor(mobile): move deferred view intents into providers, split view-intent providers, and clean up ACTION_VIEW handling * refactor(mobile): resolve merge conflicts use NativeSyncApi for hash files instead method from removed BackgroundServicePlugin.kt * style(mobile): format files * style(mobile): format files #2 * refactor(mobile): lazily materialize view-intent files and clean up temp-file handling * fix(mobile): flush pending view intents after login navigation * refactor(mobile): split view intent handler by platform and trigger it from app events * refactor(mobile): move view intent handling behind platform-specific factories * refactor(mobile): simplify code * fix(mobile): hand off deep-link viewer to main timeline after upload Add MainTimelineHandoffCoordinator to switch the asset viewer to the main timeline once a view-intent asset is uploaded and becomes available, and guard viewer reload/navigation transitions to avoid race conditions and crashes. * refactor(mobile): use remote asset ids for view intent handoff and simplify resolver * refactor(mobile): resolve merge conflicts * style(mobile): reformat code * style(mobile): reformat code #2 * fix(mobile): stabilize Android view intent asset resolution and fallback viewer * refactor(mobile): share AssetViewer pre-navigation state preparation * fix(mobile): wait for main timeline before deferred view intent handoff * refactor(mobile): decouple view intent asset resolver from providers * fix(mobile): avoid double pop when canceling upload dialog * fix(mobile): resolve view intent MIME type with fallbacks * docs(mobile): clarify view intent fallback asset TODO * fix(mobile): resolve merge conflicts * cleanup * lint --------- Co-authored-by: Peter Ombodi <peter.ombodi@gmail.com> Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
104 lines
3.8 KiB
Dart
104 lines
3.8 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|
import 'package:immich_mobile/domain/services/timeline.service.dart';
|
|
import 'package:immich_mobile/platform/view_intent_api.g.dart';
|
|
import 'package:immich_mobile/providers/asset_viewer/asset_viewer.provider.dart';
|
|
import 'package:immich_mobile/providers/auth.provider.dart';
|
|
import 'package:immich_mobile/providers/view_intent/view_intent_file_path.provider.dart';
|
|
import 'package:immich_mobile/providers/view_intent/view_intent_handler.provider.dart';
|
|
import 'package:immich_mobile/providers/view_intent/view_intent_pending.provider.dart';
|
|
import 'package:immich_mobile/routing/router.dart';
|
|
import 'package:immich_mobile/services/view_intent.service.dart';
|
|
import 'package:immich_mobile/services/view_intent_asset_resolver.service.dart';
|
|
import 'package:logging/logging.dart';
|
|
|
|
class AndroidViewIntentHandler implements ViewIntentHandler {
|
|
final Ref _ref;
|
|
final ViewIntentService _viewIntentService;
|
|
final ViewIntentAssetResolver _viewIntentAssetResolver;
|
|
final AppRouter _router;
|
|
static final Logger _logger = Logger('ViewIntentHandler');
|
|
|
|
AndroidViewIntentHandler(Ref ref)
|
|
: _ref = ref,
|
|
_viewIntentService = ref.read(viewIntentServiceProvider),
|
|
_viewIntentAssetResolver = ref.read(viewIntentAssetResolverProvider),
|
|
_router = ref.watch(appRouterProvider);
|
|
|
|
@override
|
|
void init() {
|
|
// Covers cold start from a view intent before the first lifecycle "resumed".
|
|
unawaited(onAppResumed());
|
|
}
|
|
|
|
@override
|
|
Future<void> onAppResumed() => _checkForViewIntent();
|
|
|
|
@override
|
|
Future<void> flushDeferredViewIntent() => _flushPending();
|
|
|
|
Future<void> _checkForViewIntent() async {
|
|
final attachment = await _viewIntentService.consumeViewIntent();
|
|
if (attachment != null) {
|
|
await handle(attachment);
|
|
return;
|
|
}
|
|
|
|
if (_ref.read(viewIntentPendingProvider) == null) {
|
|
await _viewIntentService.cleanupStaleTempFiles();
|
|
}
|
|
}
|
|
|
|
Future<void> _flushPending() async {
|
|
final pendingAttachment = _ref.read(viewIntentPendingProvider.notifier).takeIfFresh();
|
|
_logger.info('flushPending, pendingAttachment:$pendingAttachment');
|
|
if (pendingAttachment != null) {
|
|
await handle(pendingAttachment);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> handle(ViewIntentPayload attachment) async {
|
|
_logger.info(
|
|
'handle attachment, mimeType:${attachment.mimeType}, localAssetId=${attachment.localAssetId}, path=${attachment.path}, isAuthenticated:${_ref.read(authProvider).isAuthenticated}',
|
|
);
|
|
|
|
if (!_ref.read(authProvider).isAuthenticated) {
|
|
_ref.read(viewIntentPendingProvider.notifier).defer(attachment);
|
|
return;
|
|
}
|
|
|
|
final resolvedAsset = await _viewIntentAssetResolver.resolve(attachment);
|
|
_logger.fine('resolved view intent asset: ${resolvedAsset.asset}');
|
|
await _openAssetViewer(
|
|
resolvedAsset.asset,
|
|
resolvedAsset.timelineService,
|
|
viewIntentFilePath: resolvedAsset.viewIntentFilePath,
|
|
);
|
|
}
|
|
|
|
Future<void> _openAssetViewer(BaseAsset asset, TimelineService timelineService, {String? viewIntentFilePath}) async {
|
|
final notifier = _ref.read(assetViewerProvider.notifier);
|
|
notifier.reset();
|
|
if (asset.isVideo) {
|
|
notifier.setControls(false);
|
|
}
|
|
notifier.setAsset(asset);
|
|
|
|
if (viewIntentFilePath != null) {
|
|
_ref.read(viewIntentFilePathProvider.notifier).setPath(viewIntentFilePath);
|
|
unawaited(_viewIntentService.setManagedTempFilePath(viewIntentFilePath));
|
|
} else {
|
|
_ref.read(viewIntentFilePathProvider.notifier).clear();
|
|
unawaited(_viewIntentService.cleanupManagedTempFile());
|
|
}
|
|
|
|
await _router.replaceAll([
|
|
const TabShellRoute(),
|
|
AssetViewerRoute(initialIndex: 0, timelineService: timelineService),
|
|
]);
|
|
}
|
|
}
|