feat: add accessibility setting and conditionally disable animations

This commit is contained in:
Anton Stubenbord
2023-11-16 23:44:02 +01:00
parent a17f658df5
commit 12be81d93b
21 changed files with 304 additions and 77 deletions

View File

@@ -0,0 +1 @@
* Neue Einstellung um Animationen zu deaktivieren

View File

@@ -0,0 +1 @@
* Add setting to disable animations

View File

@@ -0,0 +1,89 @@
import 'dart:ui';
import 'package:flutter/widgets.dart';
import 'package:flutter_animate/flutter_animate.dart';
import 'package:hive/hive.dart';
import 'package:paperless_mobile/core/database/hive/hive_config.dart';
import 'package:paperless_mobile/core/database/hive/hive_extensions.dart';
import 'package:paperless_mobile/features/settings/view/widgets/global_settings_builder.dart';
import 'package:paperless_mobile/routing/navigation_keys.dart';
extension AccessibilityAwareAnimationDurationExtension on Duration {
Duration accessible() {
bool shouldDisableAnimations = WidgetsBinding.instance.disableAnimations ||
Hive.globalSettingsBox.getValue()!.disableAnimations;
// print(shouldDisableAnimations);
if (shouldDisableAnimations) {
return 0.seconds;
}
return this;
}
}
extension AccessibleHero on Hero {
Widget accessible() {
return GlobalSettingsBuilder(
builder: (context, settings) {
return HeroMode(
enabled: WidgetsBinding.instance.disableAnimations ||
!settings.disableAnimations,
child: this,
);
},
);
// bool shouldDisableAnimations = WidgetsBinding.instance.disableAnimations ||
// Hive.globalSettingsBox.getValue()!.disableAnimations;
// return _AccessibilityAwareObserverWidget(
// accessibilityAwareBuilder: (context, accessibilityFeatures) {
// return HeroMode(
// enabled: !accessibilityFeatures.disableAnimations,
// child: this,
// );
// },
// );
}
}
class _AccessibilityAwareObserverWidget extends StatefulWidget {
final Widget Function(
BuildContext context,
AccessibilityFeatures accessibilityFeatures,
) accessibilityAwareBuilder;
const _AccessibilityAwareObserverWidget({
super.key,
required this.accessibilityAwareBuilder,
});
@override
State<_AccessibilityAwareObserverWidget> createState() =>
_AccessibilityAwareObserverWidgetState();
}
class _AccessibilityAwareObserverWidgetState
extends State<_AccessibilityAwareObserverWidget>
with WidgetsBindingObserver {
late final AccessibilityFeatures _accessibilityFeatures;
@override
void initState() {
super.initState();
_accessibilityFeatures = WidgetsBinding.instance.accessibilityFeatures;
}
@override
void didChangeAccessibilityFeatures() {
super.didChangeAccessibilityFeatures();
setState(() {
_accessibilityFeatures = WidgetsBinding.instance.accessibilityFeatures;
});
print("Accessibility features changed");
}
@override
Widget build(BuildContext context) {
return widget.accessibilityAwareBuilder(
context,
_accessibilityFeatures,
);
}
}

View File

@@ -0,0 +1,58 @@
import 'dart:io';
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:flutter/widgets.dart';
import 'package:go_router/go_router.dart';
import 'package:hive/hive.dart';
import 'package:paperless_mobile/core/database/hive/hive_config.dart';
import 'package:paperless_mobile/core/database/hive/hive_extensions.dart';
Page<T> accessiblePlatformPage<T>({
required Widget child,
String? name,
Object? arguments,
String? restorationId,
LocalKey? key,
bool allowSnapshotting = true,
bool fullscreenDialog = false,
bool maintainState = true,
String? title,
}) {
final shouldDisableAnimations = WidgetsBinding.instance.disableAnimations ||
Hive.globalSettingsBox.getValue()!.disableAnimations;
if (shouldDisableAnimations) {
return NoTransitionPage(
key: key,
name: name,
arguments: arguments,
restorationId: restorationId,
child: child,
);
}
if (Platform.isAndroid) {
return MaterialPage(
child: child,
name: name,
restorationId: restorationId,
arguments: arguments,
allowSnapshotting: allowSnapshotting,
fullscreenDialog: fullscreenDialog,
key: key,
maintainState: maintainState,
);
} else if (Platform.isIOS) {
return CupertinoPage(
child: child,
allowSnapshotting: allowSnapshotting,
arguments: arguments,
fullscreenDialog: fullscreenDialog,
key: key,
maintainState: maintainState,
name: name,
restorationId: restorationId,
title: title,
);
}
throw UnsupportedError("The current platform is not supported.");
}

View File

@@ -35,6 +35,9 @@ class GlobalSettings with HiveObjectMixin {
@HiveField(8, defaultValue: false)
bool skipDocumentPreprarationOnUpload;
@HiveField(9, defaultValue: false)
bool disableAnimations;
GlobalSettings({
required this.preferredLocaleSubtag,
this.preferredThemeMode = ThemeMode.system,
@@ -45,5 +48,6 @@ class GlobalSettings with HiveObjectMixin {
this.defaultShareType = FileDownloadType.alwaysAsk,
this.enforceSinglePagePdfUpload = false,
this.skipDocumentPreprarationOnUpload = false,
this.disableAnimations = false,
});
}

View File

@@ -0,0 +1,20 @@
import 'dart:async';
import 'package:flutter/widgets.dart';
class GoRouterRefreshStream extends ChangeNotifier {
GoRouterRefreshStream(Stream<dynamic> stream) {
notifyListeners();
_subscription = stream.asBroadcastStream().listen(
(dynamic _) => notifyListeners(),
);
}
late final StreamSubscription<dynamic> _subscription;
@override
void dispose() {
_subscription.cancel();
super.dispose();
}
}

View File

@@ -1,4 +1,6 @@
import 'package:flutter/material.dart';
import 'package:flutter_animate/flutter_animate.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
@@ -24,7 +26,7 @@ class HintCard extends StatelessWidget {
crossFadeState:
show ? CrossFadeState.showFirst : CrossFadeState.showSecond,
secondChild: const SizedBox.shrink(),
duration: const Duration(milliseconds: 500),
duration: 500.milliseconds.accessible(),
firstChild: Card(
elevation: elevation,
child: Column(

View File

@@ -164,7 +164,10 @@ class AppDrawer extends StatelessWidget {
return child;
}
return child
.animate(onPlay: (c) => c.repeat(reverse: true))
.animate(
onPlay: (c) => c.repeat(reverse: true),
autoPlay: !MediaQuery.disableAnimationsOf(context),
)
.fade(duration: 1.seconds, begin: 1, end: 0.3);
},
),

View File

@@ -63,6 +63,7 @@ class ChangelogDialog extends StatelessWidget {
}
const _versionNumbers = {
"58": "3.1.4",
"57": "3.1.3",
"56": "3.1.2",
"55": "3.1.1",

View File

@@ -4,6 +4,7 @@ import 'package:go_router/go_router.dart';
import 'package:intl/date_symbol_data_local.dart';
import 'package:open_filex/open_filex.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/core/bloc/connectivity_cubit.dart';
import 'package:paperless_mobile/core/database/tables/local_user_account.dart';
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
@@ -60,6 +61,8 @@ class _DocumentDetailsPageState extends State<DocumentDetailsPage> {
@override
Widget build(BuildContext context) {
final disableAnimations = MediaQuery.disableAnimationsOf(context);
debugPrint(disableAnimations.toString());
final hasMultiUserSupport =
context.watch<LocalUserAccount>().hasMultiUserSupport;
final tabLength = 4 + (hasMultiUserSupport && false ? 1 : 0);
@@ -150,7 +153,7 @@ class _DocumentDetailsPageState extends State<DocumentDetailsPage> {
],
),
),
);
).accessible();
},
),
),

View File

@@ -50,8 +50,8 @@ class _DocumentEditPageState extends State<DocumentEditPage>
late final Animation<double> _animation;
@override
void initState() {
super.initState();
void didChangeDependencies() {
super.didChangeDependencies();
_animationController = AnimationController(
duration: const Duration(milliseconds: 150),
vsync: this,

View File

@@ -4,6 +4,7 @@ import 'dart:math' as math;
import 'package:collection/collection.dart';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/core/extensions/document_extensions.dart';
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
import 'package:paperless_mobile/features/document_search/cubit/document_search_cubit.dart';
@@ -67,7 +68,7 @@ class _DocumentSearchPageState extends State<DocumentSearchPage> {
context.read<DocumentSearchCubit>().search(query);
},
),
),
).accessible(),
actions: [
IconButton(
color: theme.colorScheme.onSurfaceVariant,

View File

@@ -2,6 +2,7 @@ import 'package:cached_network_image/cached_network_image.dart';
import 'package:flutter/material.dart';
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/helpers/connectivity_aware_action_wrapper.dart';
import 'package:paperless_mobile/routing/routes/documents_route.dart';
import 'package:paperless_mobile/routing/routes/shells/authenticated_route.dart';
@@ -46,7 +47,7 @@ class DocumentPreview extends StatelessWidget {
return Hero(
tag: "thumb_$documentId",
child: _buildPreview(context),
);
).accessible();
}
return _buildPreview(context);
}),

View File

@@ -1,7 +1,9 @@
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
import 'package:paperless_mobile/routing/routes/saved_views_route.dart';
import 'package:paperless_mobile/routing/routes/shells/authenticated_route.dart';
@@ -38,7 +40,7 @@ class _SavedViewChipState extends State<SavedViewChip>
super.initState();
_animationController = AnimationController(
vsync: this,
duration: const Duration(milliseconds: 200),
duration: const Duration(milliseconds: 200).accessible(),
);
_animation = _animationController.drive(Tween(begin: 0, end: 1));
}

View File

@@ -1,8 +1,9 @@
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/core/widgets/shimmer_placeholder.dart';
import 'package:paperless_mobile/accessibility/accessibility_utils.dart';
import 'package:paperless_mobile/core/extensions/flutter_extensions.dart';
import 'package:paperless_mobile/core/widgets/shimmer_placeholder.dart';
import 'package:paperless_mobile/features/documents/view/widgets/saved_views/saved_view_chip.dart';
import 'package:paperless_mobile/features/saved_view/cubit/saved_view_cubit.dart';
import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
@@ -41,7 +42,7 @@ class _SavedViewsWidgetState extends State<SavedViewsWidget>
super.initState();
_animationController = AnimationController(
vsync: this,
duration: const Duration(milliseconds: 200),
duration: const Duration(milliseconds: 200).accessible(),
);
_animation = _animationController.drive(Tween(begin: 0, end: 0.5));
}

View File

@@ -7,6 +7,7 @@ import 'package:paperless_mobile/features/settings/view/widgets/clear_storage_se
import 'package:paperless_mobile/features/settings/view/widgets/color_scheme_option_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/default_download_file_type_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/default_share_file_type_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/disable_animations_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/enforce_pdf_upload_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/language_selection_setting.dart';
import 'package:paperless_mobile/features/settings/view/widgets/skip_document_prepraration_on_share_setting.dart';
@@ -39,6 +40,8 @@ class SettingsPage extends StatelessWidget {
const SkipDocumentPreprationOnShareSetting(),
_buildSectionHeader(context, S.of(context)!.storage),
const ClearCacheSetting(),
_buildSectionHeader(context, 'Accessibility'),
const DisableAnimationsSetting(),
_buildSectionHeader(context, S.of(context)!.misc),
const AppLogsTile(),
const ChangelogsTile(),

View File

@@ -0,0 +1,23 @@
import 'package:flutter/material.dart';
import 'package:paperless_mobile/features/settings/view/widgets/global_settings_builder.dart';
import 'package:paperless_mobile/generated/l10n/app_localizations.dart';
class DisableAnimationsSetting extends StatelessWidget {
const DisableAnimationsSetting({super.key});
@override
Widget build(BuildContext context) {
return GlobalSettingsBuilder(builder: (context, settings) {
return SwitchListTile(
value: settings.disableAnimations,
title: Text('Disable animations'),
subtitle: Text('Disables page transitions and most animations.'
' Temporary workaround until system accessibility settings can be used.'),
onChanged: (val) async {
settings.disableAnimations = val;
settings.save();
},
);
});
}
}

View File

@@ -19,6 +19,7 @@ import 'package:local_auth/local_auth.dart';
import 'package:logger/logger.dart' as l;
import 'package:package_info_plus/package_info_plus.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/accessibility/accessible_page.dart';
import 'package:paperless_mobile/constants.dart';
import 'package:paperless_mobile/core/bloc/connectivity_cubit.dart';
import 'package:paperless_mobile/core/database/hive/hive_config.dart';
@@ -29,6 +30,7 @@ import 'package:paperless_mobile/core/exception/server_message_exception.dart';
import 'package:paperless_mobile/core/factory/paperless_api_factory.dart';
import 'package:paperless_mobile/core/factory/paperless_api_factory_impl.dart';
import 'package:paperless_mobile/core/interceptor/language_header.interceptor.dart';
import 'package:paperless_mobile/core/notifier/go_router_refresh_stream.dart';
import 'package:paperless_mobile/features/logging/data/formatted_printer.dart';
import 'package:paperless_mobile/features/logging/data/logger.dart';
import 'package:paperless_mobile/features/logging/data/mirrored_file_output.dart';
@@ -301,8 +303,9 @@ class _GoRouterShellState extends State<GoRouterShell> {
initialLocation: "/login",
routes: [
ShellRoute(
builder: (context, state, child) {
return Provider.value(
pageBuilder: (context, state, child) {
return accessiblePlatformPage(
child: Provider.value(
value: widget.apiFactory,
child: BlocListener<AuthenticationCubit, AuthenticationState>(
listener: (context, state) {
@@ -329,7 +332,8 @@ class _GoRouterShellState extends State<GoRouterShell> {
const LandingRoute().go(context);
break;
case AuthenticatingState state:
AuthenticatingRoute(state.currentStage.name).push(context);
AuthenticatingRoute(state.currentStage.name)
.push(context);
break;
case LoggingOutState():
const LoggingOutRoute().go(context);
@@ -343,6 +347,7 @@ class _GoRouterShellState extends State<GoRouterShell> {
},
child: child,
),
),
);
},
navigatorKey: rootNavigatorKey,

View File

@@ -5,6 +5,7 @@ import 'package:flutter/widgets.dart';
import 'package:go_router/go_router.dart';
import 'package:hive_flutter/adapters.dart';
import 'package:paperless_api/paperless_api.dart';
import 'package:paperless_mobile/accessibility/accessible_page.dart';
import 'package:paperless_mobile/core/database/hive/hive_config.dart';
import 'package:paperless_mobile/core/database/tables/global_settings.dart';
import 'package:paperless_mobile/core/database/tables/local_user_account.dart';
@@ -141,12 +142,16 @@ class AuthenticatedRoute extends ShellRouteData {
const AuthenticatedRoute();
@override
Widget builder(
Page<void> pageBuilder(
BuildContext context,
GoRouterState state,
Widget navigator,
) {
final currentUserId = Hive.box<GlobalSettings>(HiveBoxes.globalSettings)
return accessiblePlatformPage(
child: Builder(
builder: (context) {
final currentUserId =
Hive.box<GlobalSettings>(HiveBoxes.globalSettings)
.getValue()!
.loggedInUserId;
if (currentUserId == null) {
@@ -169,5 +174,8 @@ class AuthenticatedRoute extends ShellRouteData {
),
),
);
},
),
);
}
}

View File

@@ -1,7 +1,8 @@
import 'package:flutter/material.dart';
class AnimatedTouchBubblePart extends StatefulWidget {
const AnimatedTouchBubblePart({super.key,
const AnimatedTouchBubblePart({
super.key,
required this.dragging,
required this.size,
});
@@ -22,6 +23,7 @@ class _AnimatedTouchBubblePartState extends State<AnimatedTouchBubblePart>
@override
void didChangeDependencies() {
super.didChangeDependencies();
_controller = AnimationController(
duration: const Duration(milliseconds: 1000),
vsync: this,
@@ -40,7 +42,6 @@ class _AnimatedTouchBubblePartState extends State<AnimatedTouchBubblePart>
);
_controller.repeat();
super.didChangeDependencies();
}
@override

View File

@@ -15,7 +15,7 @@ publish_to: "none" # Remove this line if you wish to publish to pub.dev
# In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion.
# Read more about iOS versioning at
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
version: 3.1.3+57
version: 3.1.4+58
environment:
sdk: ">=3.0.0 <4.0.0"