mirror of
https://github.com/immich-app/immich.git
synced 2025-10-17 12:00:44 -04:00
* feat(mobile): Add Kid (Readonly) Mode toggle This commit introduces a "Kid (Readonly) Mode" feature. - Adds a `KidModeProvider` to manage the state of Kid Mode. - Implements a `KidModeCheckbox` widget in the app bar dialog to toggle Kid Mode. - When Kid Mode is enabled, - Disables selecting the multigrid & the bottom bar - Removes the top bar from view Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Reverts the changes to devtools_options.yaml file Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> refactor: replace Kid Mode with Readonly Mode This commit replaces the "Kid Mode" feature with a more generic "Readonly Mode". - Renamed `KidModeProvider` to `ReadonlyModeProvider`. - Readonly Mode state is now persisted in app settings. - Added a new app setting `allowUserAvatarOverride` to toggle read-only mode. - Updated translations. - Added a message in the app bar dialog indicating when read-only mode is active. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Address comments - - Removes the `allowUserAvatarOverride` setting. - Hides the bottom gallery bar when read-only mode is enabled. - Adds an icon on the main app bar when read-only mode is enabled with a snackbar. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Update to snackbar - When toggling readonly mode from either the settings or the app bar, a snackbar notification will now appear. - The readonly mode message in the profile drawer has been restyled. - The upload button in the app bar is now hidden when readonly mode is enabled. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Removes clearing of snackbar Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Address Comments - Consolidated snackbar messages for enabling/disabling readonly mode. - Ensured the "Select All" icon in asset group titles is hidden in readonly mode. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Adds in the missing translation keys for readonly_mode Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Fix translation Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Fix check failure for BorderRadius Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Changes: - Adjusted AppBar background color in readonly mode. - Removes cross-out pencil icon button in favor of above. - Hides the "Edit" icon next to date/time, disable description and onTap for people and location when readonly mode is enabled. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> Address comments from Alex - Moved readonly mode check to `GalleryAppBar` to hide the entire `TopControlAppBar` when readonly mode is enabled. - Changed `toggleReadonlyMode` in `ImmichAppBar` to directly toggle the state. Signed-off-by: Sudheer Puthana <Sud-Puth@users.noreply.github.com> migrate readonly mode to new beta timeline remove readonly mode from legacy UI only show readonly functionality when on beta timeline simplify selection icon update generated provider chore: more formatting * fix: bad merge * chore: use Notifier for readonlyModeProvider * fix: drag select now honors readonly mode * fix: disable asset bottom sheet in readonly * fix: disable editing user icon when in readonly * chore: remove generated file * fix: disable tabs instead entire tab bar This solves the issues with the scrubber * chore: remove unneeded import * chore: lint * remove unused condition in bottomsheet --------- Co-authored-by: Brandon Wees <brandonwees@gmail.com> Co-authored-by: Alex <alex.tran1502@gmail.com>
153 lines
6.0 KiB
Dart
153 lines
6.0 KiB
Dart
import 'package:auto_route/auto_route.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/constants/enums.dart';
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|
import 'package:immich_mobile/domain/models/timeline.model.dart';
|
|
import 'package:immich_mobile/domain/utils/event_stream.dart';
|
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
|
import 'package:immich_mobile/extensions/translate_extensions.dart';
|
|
import 'package:immich_mobile/presentation/widgets/action_buttons/cast_action_button.widget.dart';
|
|
import 'package:immich_mobile/presentation/widgets/action_buttons/favorite_action_button.widget.dart';
|
|
import 'package:immich_mobile/presentation/widgets/action_buttons/motion_photo_action_button.widget.dart';
|
|
import 'package:immich_mobile/presentation/widgets/action_buttons/unfavorite_action_button.widget.dart';
|
|
import 'package:immich_mobile/presentation/widgets/asset_viewer/asset_viewer.state.dart';
|
|
import 'package:immich_mobile/providers/cast.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/asset_viewer/current_asset.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/readonly_mode.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/current_album.provider.dart';
|
|
import 'package:immich_mobile/providers/routes.provider.dart';
|
|
import 'package:immich_mobile/providers/user.provider.dart';
|
|
import 'package:immich_mobile/routing/router.dart';
|
|
|
|
class ViewerTopAppBar extends ConsumerWidget implements PreferredSizeWidget {
|
|
const ViewerTopAppBar({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
final asset = ref.watch(currentAssetNotifier);
|
|
if (asset == null) {
|
|
return const SizedBox.shrink();
|
|
}
|
|
|
|
final album = ref.watch(currentRemoteAlbumProvider);
|
|
|
|
final user = ref.watch(currentUserProvider);
|
|
final isOwner = asset is RemoteAsset && asset.ownerId == user?.id;
|
|
final isInLockedView = ref.watch(inLockedViewProvider);
|
|
final isReadonlyModeEnabled = ref.watch(readonlyModeProvider);
|
|
|
|
final previousRouteName = ref.watch(previousRouteNameProvider);
|
|
final showViewInTimelineButton =
|
|
previousRouteName != TabShellRoute.name &&
|
|
previousRouteName != AssetViewerRoute.name &&
|
|
previousRouteName != null;
|
|
|
|
final isShowingSheet = ref.watch(assetViewerProvider.select((state) => state.showingBottomSheet));
|
|
int opacity = ref.watch(assetViewerProvider.select((state) => state.backgroundOpacity));
|
|
final showControls = ref.watch(assetViewerProvider.select((s) => s.showingControls));
|
|
|
|
if (!showControls) {
|
|
opacity = 0;
|
|
}
|
|
|
|
final isCasting = ref.watch(castProvider.select((c) => c.isCasting));
|
|
|
|
final actions = <Widget>[
|
|
if (isCasting || (asset.hasRemote)) const CastActionButton(menuItem: true),
|
|
if (album != null && album.isActivityEnabled && album.isShared)
|
|
IconButton(
|
|
icon: const Icon(Icons.chat_outlined),
|
|
onPressed: () {
|
|
context.navigateTo(const DriftActivitiesRoute());
|
|
},
|
|
),
|
|
if (showViewInTimelineButton)
|
|
IconButton(
|
|
onPressed: () async {
|
|
await context.maybePop();
|
|
await context.navigateTo(const TabShellRoute(children: [MainTimelineRoute()]));
|
|
EventStream.shared.emit(ScrollToDateEvent(asset.createdAt));
|
|
},
|
|
icon: const Icon(Icons.image_search),
|
|
tooltip: 'view_in_timeline'.t(context: context),
|
|
),
|
|
if (asset.hasRemote && isOwner && !asset.isFavorite)
|
|
const FavoriteActionButton(source: ActionSource.viewer, menuItem: true),
|
|
if (asset.hasRemote && isOwner && asset.isFavorite)
|
|
const UnFavoriteActionButton(source: ActionSource.viewer, menuItem: true),
|
|
if (asset.isMotionPhoto) const MotionPhotoActionButton(menuItem: true),
|
|
const _KebabMenu(),
|
|
];
|
|
|
|
final lockedViewActions = <Widget>[
|
|
if (isCasting || (asset.hasRemote)) const CastActionButton(menuItem: true),
|
|
const _KebabMenu(),
|
|
];
|
|
|
|
return IgnorePointer(
|
|
ignoring: opacity < 255,
|
|
child: AnimatedOpacity(
|
|
opacity: opacity / 255,
|
|
duration: Durations.short2,
|
|
child: AppBar(
|
|
backgroundColor: isShowingSheet ? Colors.transparent : Colors.black.withAlpha(125),
|
|
leading: const _AppBarBackButton(),
|
|
iconTheme: const IconThemeData(size: 22, color: Colors.white),
|
|
actionsIconTheme: const IconThemeData(size: 22, color: Colors.white),
|
|
shape: const Border(),
|
|
actions: isShowingSheet || isReadonlyModeEnabled
|
|
? null
|
|
: isInLockedView
|
|
? lockedViewActions
|
|
: actions,
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
@override
|
|
Size get preferredSize => const Size.fromHeight(60.0);
|
|
}
|
|
|
|
class _KebabMenu extends ConsumerWidget {
|
|
const _KebabMenu();
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
return IconButton(
|
|
onPressed: () {
|
|
EventStream.shared.emit(const ViewerOpenBottomSheetEvent());
|
|
},
|
|
icon: const Icon(Icons.more_vert_rounded),
|
|
);
|
|
}
|
|
}
|
|
|
|
class _AppBarBackButton extends ConsumerWidget {
|
|
const _AppBarBackButton();
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
final isShowingSheet = ref.watch(assetViewerProvider.select((state) => state.showingBottomSheet));
|
|
final backgroundColor = isShowingSheet && !context.isDarkTheme ? Colors.white : Colors.black;
|
|
final foregroundColor = isShowingSheet && !context.isDarkTheme ? Colors.black : Colors.white;
|
|
|
|
return Padding(
|
|
padding: const EdgeInsets.only(left: 12.0),
|
|
child: ElevatedButton(
|
|
style: ElevatedButton.styleFrom(
|
|
backgroundColor: backgroundColor,
|
|
shape: const CircleBorder(),
|
|
iconSize: 22,
|
|
iconColor: foregroundColor,
|
|
padding: EdgeInsets.zero,
|
|
elevation: isShowingSheet ? 4 : 0,
|
|
),
|
|
onPressed: context.maybePop,
|
|
child: const Icon(Icons.arrow_back_rounded),
|
|
),
|
|
);
|
|
}
|
|
}
|