Merge pull request #33 from gskinnerTeam/feature-improve-a11y

Feature improve a11y
This commit is contained in:
Shawn 2022-09-20 19:39:14 -06:00 committed by GitHub
commit 16bf6715cd
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 512 additions and 452 deletions

View File

@ -3,15 +3,15 @@ import 'package:wonders/common_libs.dart';
import 'package:wonders/logic/common/string_utils.dart'; import 'package:wonders/logic/common/string_utils.dart';
class AppPageIndicator extends StatefulWidget { class AppPageIndicator extends StatefulWidget {
AppPageIndicator( AppPageIndicator({
{Key? key, Key? key,
required this.count, required this.count,
required this.controller, required this.controller,
this.onDotPressed, this.onDotPressed,
this.color, this.color,
this.dotSize, this.dotSize,
String? semanticPageTitle}) String? semanticPageTitle,
: semanticPageTitle = semanticPageTitle ?? $strings.appPageDefaultTitlePage, }) : semanticPageTitle = semanticPageTitle ?? $strings.appPageDefaultTitlePage,
super(key: key); super(key: key);
final int count; final int count;
final PageController controller; final PageController controller;
@ -33,9 +33,11 @@ class _AppPageIndicatorState extends State<AppPageIndicator> {
widget.controller.addListener(_handlePageChanged); widget.controller.addListener(_handlePageChanged);
} }
int get _controllerPage => widget.controller.page?.round() ?? 0; int get _controllerPage => _currentPage.value;
void _handlePageChanged() => _currentPage.value = _controllerPage; void _handlePageChanged() {
_currentPage.value = widget.controller.page!.round();
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -48,11 +50,12 @@ class _AppPageIndicatorState extends State<AppPageIndicator> {
valueListenable: _currentPage, valueListenable: _currentPage,
builder: (_, value, child) { builder: (_, value, child) {
return Semantics( return Semantics(
container: true,
liveRegion: true, liveRegion: true,
focusable: false,
readOnly: true,
label: StringUtils.supplant($strings.appPageSemanticSwipe, { label: StringUtils.supplant($strings.appPageSemanticSwipe, {
'{pageTitle}': widget.semanticPageTitle, '{pageTitle}': widget.semanticPageTitle,
'{count}': (value % (widget.count) + 1).toString(), '{count}': (_controllerPage % (widget.count) + 1).toString(),
'{total}': widget.count.toString(), '{total}': widget.count.toString(),
}), }),
child: Container()); child: Container());

View File

@ -138,19 +138,14 @@ class AppBtn extends StatelessWidget {
// add press effect: // add press effect:
if (pressEffect) button = _ButtonPressEffect(button); if (pressEffect) button = _ButtonPressEffect(button);
// add semantics: // add semantics?
if (semanticLabel.isEmpty) { if (semanticLabel.isEmpty) return button;
button = ExcludeSemantics(child: button); return Semantics(
} else { label: semanticLabel,
button = Semantics( button: true,
label: semanticLabel, container: true,
button: true, child: ExcludeSemantics(child: button),
container: true, );
child: button,
);
}
return button;
} }
} }

View File

@ -0,0 +1,50 @@
import 'package:wonders/common_libs.dart';
class PopRouterOnOverScroll extends StatefulWidget {
const PopRouterOnOverScroll({Key? key, required this.child, required this.controller}) : super(key: key);
final ScrollController controller;
final Widget child;
@override
State<PopRouterOnOverScroll> createState() => _PopRouterOnOverScrollState();
}
class _PopRouterOnOverScrollState extends State<PopRouterOnOverScroll> {
final _scrollToPopThreshold = 70;
bool _isPointerDown = false;
@override
void initState() {
super.initState();
widget.controller.addListener(_handleScrollChanged);
}
@override
void didUpdateWidget(covariant PopRouterOnOverScroll oldWidget) {
if (widget.controller != oldWidget.controller) {
widget.controller.addListener(_handleScrollChanged);
}
super.didUpdateWidget(oldWidget);
}
bool _checkPointerIsDown(d) => _isPointerDown = d.dragDetails != null;
@override
Widget build(BuildContext context) {
return NotificationListener<ScrollUpdateNotification>(
onNotification: _checkPointerIsDown,
child: widget.child,
);
}
void _handleScrollChanged() {
// If user pulls far down on the elastic list, pop back to
final px = widget.controller.position.pixels;
if (px < -_scrollToPopThreshold) {
if (_isPointerDown) {
context.pop();
widget.controller.removeListener(_handleScrollChanged);
}
}
}
}

View File

@ -6,6 +6,7 @@ import 'package:wonders/logic/data/highlight_data.dart';
import 'package:wonders/ui/common/controls/app_page_indicator.dart'; import 'package:wonders/ui/common/controls/app_page_indicator.dart';
import 'package:wonders/ui/common/controls/simple_header.dart'; import 'package:wonders/ui/common/controls/simple_header.dart';
import 'package:wonders/ui/common/static_text_scale.dart'; import 'package:wonders/ui/common/static_text_scale.dart';
part 'widgets/_blurred_image_bg.dart'; part 'widgets/_blurred_image_bg.dart';
part 'widgets/_carousel_item.dart'; part 'widgets/_carousel_item.dart';
@ -25,14 +26,24 @@ class _ArtifactScreenState extends State<ArtifactCarouselScreen> {
// Locally store loaded artifacts. // Locally store loaded artifacts.
late List<HighlightData> _artifacts; late List<HighlightData> _artifacts;
late PageController _controller; late PageController _controller;
final _currentIndex = ValueNotifier(0);
double get _currentOffset { double get _currentOffset {
bool inited = _controller.hasClients && _controller.position.haveDimensions; bool hasOffset = _controller.hasClients && _controller.position.haveDimensions;
return inited ? _controller.page! : _controller.initialPage * 1.0; return hasOffset ? _controller.page! : _controller.initialPage * 1.0;
} }
int get _currentIndex => _currentOffset.round() % _artifacts.length; HighlightData get _currentArtifact => _artifacts[_currentIndex.value];
double get _backdropWidth {
final w = context.widthPx;
return w <= _maxElementWidth ? w : min(w * _partialElementWidth, _maxElementWidth);
}
double get _backdropHeight => math.min(context.heightPx * 0.65, _maxElementHeight);
bool get _small => _backdropHeight / _maxElementHeight < 0.7;
@override @override
void initState() { void initState() {
@ -43,7 +54,7 @@ class _ArtifactScreenState extends State<ArtifactCarouselScreen> {
// start at a high offset so we can scroll backwards: // start at a high offset so we can scroll backwards:
initialPage: _artifacts.length * 9999, initialPage: _artifacts.length * 9999,
viewportFraction: 0.5, viewportFraction: 0.5,
); )..addListener(_handleCarouselScroll);
} }
@override @override
@ -52,6 +63,8 @@ class _ArtifactScreenState extends State<ArtifactCarouselScreen> {
super.dispose(); super.dispose();
} }
void _handleCarouselScroll() => _currentIndex.value = _currentOffset.round() % _artifacts.length;
void _handleArtifactTap(int index) { void _handleArtifactTap(int index) {
int delta = index - _currentOffset.round(); int delta = index - _currentOffset.round();
if (delta == 0) { if (delta == 0) {
@ -72,146 +85,155 @@ class _ArtifactScreenState extends State<ArtifactCarouselScreen> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Container( return ValueListenableBuilder<int>(
color: $styles.colors.greyStrong, valueListenable: _currentIndex,
child: AnimatedBuilder(animation: _controller, builder: _buildScreen), builder: (context, index, _) {
return Container(
color: $styles.colors.greyStrong,
child: Stack(
children: [
/// Background image
Positioned.fill(
child: _BlurredImageBg(url: _currentArtifact.imageUrl),
),
/// Content
Column(
children: [
SimpleHeader($strings.artifactsTitleArtifacts, showBackBtn: false, isTransparent: true),
Gap($styles.insets.xs),
Expanded(
child: Stack(children: [
// White arch, covering bottom half:
_buildWhiteArch(),
// Carousel
_buildCarouselPageView(),
// Text content
_buildBottomTextContent(),
]),
),
],
),
],
),
);
},
); );
} }
Widget _buildScreen(BuildContext context, _) { Widget _buildWhiteArch() {
final double w = context.widthPx; return BottomCenter(
final double backdropWidth = w <= _maxElementWidth ? w : min(w * _partialElementWidth, _maxElementWidth); child: Container(
final double backdropHeight = math.min(context.heightPx * 0.65, _maxElementHeight); width: _backdropWidth,
final bool small = backdropHeight / _maxElementHeight < 0.7; height: _backdropHeight,
final HighlightData artifact = _artifacts[_currentIndex]; decoration: BoxDecoration(
color: $styles.colors.offWhite.withOpacity(0.8),
borderRadius: BorderRadius.vertical(top: Radius.circular(999)),
),
),
);
}
return Stack( Widget _buildBottomTextContent() {
children: [ return BottomCenter(
Positioned.fill(child: _BlurredImageBg(url: artifact.imageUrl)), child: Container(
Column( width: _backdropWidth,
padding: EdgeInsets.symmetric(horizontal: $styles.insets.md),
child: Column(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
SimpleHeader($strings.artifactsTitleArtifacts, showBackBtn: false, isTransparent: true), Gap($styles.insets.md),
Gap($styles.insets.xs), Container(
Expanded( width: _backdropWidth,
child: Stack(children: [ padding: EdgeInsets.symmetric(horizontal: $styles.insets.sm),
// White arch, covering bottom half: child: Column(
BottomCenter( mainAxisSize: MainAxisSize.min,
child: Container( mainAxisAlignment: MainAxisAlignment.center,
width: backdropWidth, children: [
height: backdropHeight, IgnorePointer(
decoration: BoxDecoration( ignoringSemantics: false,
color: $styles.colors.offWhite.withOpacity(0.8), child: Semantics(
borderRadius: BorderRadius.vertical(top: Radius.circular(999)), button: true,
), onIncrease: () => _handleArtifactTap(_currentOffset.round() + 1),
), onDecrease: () => _handleArtifactTap(_currentOffset.round() - 1),
), onTap: () => _handleArtifactTap(_currentOffset.round()),
liveRegion: true,
// Carousel: child: Column(
Center( mainAxisSize: MainAxisSize.min,
child: SizedBox( children: [
width: backdropWidth, Container(
child: PageView.builder( height: _small ? 90 : 110,
controller: _controller, alignment: Alignment.center,
clipBehavior: Clip.none, child: StaticTextScale(
itemBuilder: (context, index) { child: Text(
bool isCurrentIndex = index % _artifacts.length == _currentIndex; _currentArtifact.title,
return ExcludeSemantics( overflow: TextOverflow.ellipsis,
excluding: isCurrentIndex == false, style: $styles.text.h2.copyWith(color: $styles.colors.black, height: 1.2),
child: MergeSemantics( textAlign: TextAlign.center,
child: Semantics( maxLines: 2,
onIncrease: () => _handleArtifactTap(_currentIndex + 1),
onDecrease: () => _handleArtifactTap(_currentIndex - 1),
child: _CarouselItem(
index: index,
currentPage: _currentOffset,
artifact: _artifacts[index % _artifacts.length],
bottomPadding: backdropHeight,
maxWidth: backdropWidth,
maxHeight: backdropHeight,
onPressed: () => _handleArtifactTap(index),
), ),
), ),
), ),
); if (!_small) Gap($styles.insets.xxs),
}, Text(
_currentArtifact.date.isEmpty ? '--' : _currentArtifact.date,
style: $styles.text.body,
textAlign: TextAlign.center,
),
],
).animate(key: ValueKey(_currentArtifact.artifactId)).fadeIn(),
), ),
), ),
), Gap(_small ? $styles.insets.xs : $styles.insets.sm),
AppPageIndicator(
// Text content count: _artifacts.length,
BottomCenter( controller: _controller,
child: Container( semanticPageTitle: $strings.artifactsSemanticArtifact,
width: backdropWidth,
padding: EdgeInsets.symmetric(horizontal: $styles.insets.md),
child: Column(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Gap($styles.insets.md),
_buildContent(context, artifact, backdropWidth, small),
Gap(small ? $styles.insets.sm : $styles.insets.md),
AppBtn.from(
text: $strings.artifactsButtonBrowse,
icon: Icons.search,
expand: true,
onPressed: _handleSearchTap,
),
Gap(small ? $styles.insets.md : $styles.insets.lg),
],
),
), ),
), ],
]), ),
), ),
Gap(_small ? $styles.insets.sm : $styles.insets.md),
AppBtn.from(
text: $strings.artifactsButtonBrowse,
icon: Icons.search,
expand: true,
onPressed: _handleSearchTap,
),
Gap(_small ? $styles.insets.md : $styles.insets.lg),
], ],
), ),
], ),
); );
} }
Widget _buildContent(BuildContext context, HighlightData artifact, double width, bool small) { Widget _buildCarouselPageView() {
return Container( return Center(
width: width, child: SizedBox(
padding: EdgeInsets.symmetric(horizontal: $styles.insets.sm), width: _backdropWidth,
child: Column( child: ExcludeSemantics(
mainAxisSize: MainAxisSize.min, child: PageView.builder(
mainAxisAlignment: MainAxisAlignment.center, controller: _controller,
children: [ clipBehavior: Clip.none,
IgnorePointer( itemBuilder: (context, index) => AnimatedBuilder(
child: ExcludeSemantics( animation: _controller,
child: Column( builder: (_, __) {
mainAxisSize: MainAxisSize.min, return _CarouselItem(
children: [ index: index,
Container( currentPage: _currentOffset,
height: small ? 90 : 110, artifact: _artifacts[index % _artifacts.length],
alignment: Alignment.center, bottomPadding: _backdropHeight,
child: StaticTextScale( maxWidth: _backdropWidth,
child: Text( maxHeight: _backdropHeight,
artifact.title, onPressed: () => _handleArtifactTap(index),
overflow: TextOverflow.ellipsis, );
style: $styles.text.h2.copyWith(color: $styles.colors.black, height: 1.2), },
textAlign: TextAlign.center,
maxLines: 2,
),
),
),
if (!small) Gap($styles.insets.xxs),
Text(
artifact.date.isEmpty ? '--' : artifact.date,
style: $styles.text.body,
textAlign: TextAlign.center,
),
],
).animate(key: ValueKey(artifact.artifactId)).fadeIn(),
), ),
), ),
Gap(small ? $styles.insets.xs : $styles.insets.sm), ),
AppPageIndicator(
count: _artifacts.length,
controller: _controller,
semanticPageTitle: $strings.artifactsSemanticArtifact,
),
],
), ),
); );
} }

View File

@ -22,7 +22,7 @@ class _CarouselItem extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return AppBtn.basic( return AppBtn.basic(
semanticLabel: '${artifact.title} ${artifact.date}', semanticLabel: 'Explore artifact details',
onPressed: onPressed, onPressed: onPressed,
pressEffect: false, pressEffect: false,
child: _ImagePreview( child: _ImagePreview(

View File

@ -1,4 +1,3 @@
import 'package:flutter/cupertino.dart';
import 'package:wonders/common_libs.dart'; import 'package:wonders/common_libs.dart';
import 'package:wonders/logic/common/string_utils.dart'; import 'package:wonders/logic/common/string_utils.dart';
import 'package:wonders/logic/data/artifact_data.dart'; import 'package:wonders/logic/data/artifact_data.dart';
@ -7,8 +6,8 @@ import 'package:wonders/ui/common/controls/app_loading_indicator.dart';
import 'package:wonders/ui/common/gradient_container.dart'; import 'package:wonders/ui/common/gradient_container.dart';
import 'package:wonders/ui/common/modals/fullscreen_url_img_viewer.dart'; import 'package:wonders/ui/common/modals/fullscreen_url_img_viewer.dart';
part 'widgets/_header.dart';
part 'widgets/_content.dart'; part 'widgets/_content.dart';
part 'widgets/_header.dart';
class ArtifactDetailsScreen extends StatefulWidget { class ArtifactDetailsScreen extends StatefulWidget {
const ArtifactDetailsScreen({Key? key, required this.artifactId}) : super(key: key); const ArtifactDetailsScreen({Key? key, required this.artifactId}) : super(key: key);
@ -35,13 +34,21 @@ class _ArtifactDetailsScreenState extends State<ArtifactDetailsScreen> {
crossAxisAlignment: CrossAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
Center(child: Icon(Icons.warning_amber_outlined, color: $styles.colors.accent1, size: $styles.insets.lg,)), Center(
child: Icon(
Icons.warning_amber_outlined,
color: $styles.colors.accent1,
size: $styles.insets.lg,
)),
Gap($styles.insets.xs), Gap($styles.insets.xs),
SizedBox(width: $styles.insets.xxl * 3, child: Text( SizedBox(
StringUtils.supplant($strings.artifactDetailsErrorNotFound, {'{artifactId}': widget.artifactId}), width: $styles.insets.xxl * 3,
style: $styles.text.body.copyWith(color: $styles.colors.offWhite), child: Text(
textAlign: TextAlign.center, StringUtils.supplant($strings.artifactDetailsErrorNotFound, {'{artifactId}': widget.artifactId}),
),), style: $styles.text.body.copyWith(color: $styles.colors.offWhite),
textAlign: TextAlign.center,
),
),
], ],
).animate().fadeIn(); ).animate().fadeIn();
} else if (!snapshot.hasData) { } else if (!snapshot.hasData) {

View File

@ -14,6 +14,7 @@ import 'package:wonders/ui/common/curved_clippers.dart';
import 'package:wonders/ui/common/google_maps_marker.dart'; import 'package:wonders/ui/common/google_maps_marker.dart';
import 'package:wonders/ui/common/gradient_container.dart'; import 'package:wonders/ui/common/gradient_container.dart';
import 'package:wonders/ui/common/hidden_collectible.dart'; import 'package:wonders/ui/common/hidden_collectible.dart';
import 'package:wonders/ui/common/pop_router_on_over_scroll.dart';
import 'package:wonders/ui/common/scaling_list_item.dart'; import 'package:wonders/ui/common/scaling_list_item.dart';
import 'package:wonders/ui/common/static_text_scale.dart'; import 'package:wonders/ui/common/static_text_scale.dart';
import 'package:wonders/ui/common/themed_text.dart'; import 'package:wonders/ui/common/themed_text.dart';
@ -47,8 +48,6 @@ class _WonderEditorialScreenState extends State<WonderEditorialScreen> {
late final ScrollController _scroller = ScrollController()..addListener(_handleScrollChanged); late final ScrollController _scroller = ScrollController()..addListener(_handleScrollChanged);
final _scrollPos = ValueNotifier(0.0); final _scrollPos = ValueNotifier(0.0);
final _sectionIndex = ValueNotifier(0); final _sectionIndex = ValueNotifier(0);
final _scrollToPopThreshold = 50;
bool _isPointerDown = false;
@override @override
void dispose() { void dispose() {
@ -60,17 +59,8 @@ class _WonderEditorialScreenState extends State<WonderEditorialScreen> {
void _handleScrollChanged() { void _handleScrollChanged() {
_scrollPos.value = _scroller.position.pixels; _scrollPos.value = _scroller.position.pixels;
widget.onScroll.call(_scrollPos.value); widget.onScroll.call(_scrollPos.value);
// If user pulls far down on the elastic list, pop back to
if (_scrollPos.value < -_scrollToPopThreshold) {
if (_isPointerDown) {
context.pop();
_scroller.removeListener(_handleScrollChanged);
}
}
} }
bool _checkPointerIsDown(d) => _isPointerDown = d.dragDetails != null;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return LayoutBuilder(builder: (_, constraints) { return LayoutBuilder(builder: (_, constraints) {
@ -79,8 +69,8 @@ class _WonderEditorialScreenState extends State<WonderEditorialScreen> {
double minAppBarHeight = shortMode ? 80 : 120; double minAppBarHeight = shortMode ? 80 : 120;
double maxAppBarHeight = shortMode ? 400 : 500; double maxAppBarHeight = shortMode ? 400 : 500;
return NotificationListener<ScrollUpdateNotification>( return PopRouterOnOverScroll(
onNotification: _checkPointerIsDown, controller: _scroller,
child: ColoredBox( child: ColoredBox(
color: $styles.colors.offWhite, color: $styles.colors.offWhite,
child: Stack( child: Stack(
@ -116,7 +106,7 @@ class _WonderEditorialScreenState extends State<WonderEditorialScreen> {
CustomScrollView( CustomScrollView(
primary: false, primary: false,
controller: _scroller, controller: _scroller,
cacheExtent: 500, cacheExtent: 1000,
slivers: [ slivers: [
/// Invisible padding at the top of the list, so the illustration shows through the btm /// Invisible padding at the top of the list, so the illustration shows through the btm
SliverToBoxAdapter( SliverToBoxAdapter(

View File

@ -28,35 +28,37 @@ class _ScrollingContent extends StatelessWidget {
final String dropChar = value.substring(0, 1); final String dropChar = value.substring(0, 1);
final textScale = MediaQuery.of(context).textScaleFactor; final textScale = MediaQuery.of(context).textScaleFactor;
final double dropCapWidth = StringUtils.measure(dropChar, dropStyle).width * textScale; final double dropCapWidth = StringUtils.measure(dropChar, dropStyle).width * textScale;
final bool skipCaps = !localeLogic.isEnglish || MediaQuery.of(context).accessibleNavigation; final bool skipCaps = !localeLogic.isEnglish;
return Semantics( return Semantics(
label: value, label: value,
child: !skipCaps child: ExcludeSemantics(
? DropCapText( child: !skipCaps
_fixNewlines(value).substring(1), ? DropCapText(
dropCap: DropCap( _fixNewlines(value).substring(1),
width: dropCapWidth, dropCap: DropCap(
height: $styles.text.body.fontSize! * $styles.text.body.height! * 2, width: dropCapWidth,
child: Transform.translate( height: $styles.text.body.fontSize! * $styles.text.body.height! * 2,
offset: Offset(0, bodyStyle.fontSize! * (bodyStyle.height! - 1) - 2), child: Transform.translate(
child: Text( offset: Offset(0, bodyStyle.fontSize! * (bodyStyle.height! - 1) - 2),
dropChar, child: Text(
overflow: TextOverflow.visible, dropChar,
style: $styles.text.dropCase.copyWith( overflow: TextOverflow.visible,
color: $styles.colors.accent1, style: $styles.text.dropCase.copyWith(
height: 1, color: $styles.colors.accent1,
height: 1,
),
), ),
), ),
), ),
), style: $styles.text.body,
style: $styles.text.body, dropCapPadding: EdgeInsets.only(right: 6),
dropCapPadding: EdgeInsets.only(right: 6), dropCapStyle: $styles.text.dropCase.copyWith(
dropCapStyle: $styles.text.dropCase.copyWith( color: $styles.colors.accent1,
color: $styles.colors.accent1, height: 1,
height: 1, ),
), )
) : Text(value, style: bodyStyle),
: Text(value, style: bodyStyle), ),
); );
} }

View File

@ -22,18 +22,16 @@ class _VerticalSwipeController {
void handleVerticalSwipeUpdate(DragUpdateDetails details) { void handleVerticalSwipeUpdate(DragUpdateDetails details) {
if (swipeReleaseAnim.isAnimating) swipeReleaseAnim.stop(); if (swipeReleaseAnim.isAnimating) swipeReleaseAnim.stop();
if (details.delta.dy > 0) {
swipeAmt.value = 0; isPointerDown.value = true;
} else { double value = (swipeAmt.value - details.delta.dy / _pullToViewDetailsThreshold).clamp(0, 1);
isPointerDown.value = true; if (value != swipeAmt.value) {
double value = (swipeAmt.value - details.delta.dy / _pullToViewDetailsThreshold).clamp(0, 1); swipeAmt.value = value;
if (value != swipeAmt.value) { if (swipeAmt.value == 1) {
swipeAmt.value = value; onSwipeComplete();
if (swipeAmt.value == 1) {
onSwipeComplete();
}
} }
} }
//print(_swipeUpAmt.value); //print(_swipeUpAmt.value);
} }
@ -53,18 +51,14 @@ class _VerticalSwipeController {
} }
/// Utility method to wrap a gesture detector and wire up the required handlers. /// Utility method to wrap a gesture detector and wire up the required handlers.
Widget wrapGestureDetector(Widget child, {Key? key}) => Semantics( Widget wrapGestureDetector(Widget child, {Key? key}) => GestureDetector(
button: false, key: key,
child: GestureDetector( excludeFromSemantics: true,
key: key, onTapDown: (_) => handleTapDown(),
onTapDown: (_) { onTapUp: (_) => handleTapCancelled(),
handleTapDown(); onVerticalDragUpdate: handleVerticalSwipeUpdate,
}, onVerticalDragEnd: (_) => handleVerticalSwipeCancelled(),
onTapUp: (_) => handleTapCancelled(), onVerticalDragCancel: handleVerticalSwipeCancelled,
onVerticalDragUpdate: handleVerticalSwipeUpdate, behavior: HitTestBehavior.translucent,
onVerticalDragEnd: (_) => handleVerticalSwipeCancelled(), child: child);
onVerticalDragCancel: handleVerticalSwipeCancelled,
behavior: HitTestBehavior.translucent,
child: child),
);
} }

View File

@ -1,4 +1,3 @@
import 'package:flutter/rendering.dart';
import 'package:wonders/common_libs.dart'; import 'package:wonders/common_libs.dart';
import 'package:wonders/logic/common/string_utils.dart'; import 'package:wonders/logic/common/string_utils.dart';
import 'package:wonders/logic/data/wonder_data.dart'; import 'package:wonders/logic/data/wonder_data.dart';
@ -94,12 +93,16 @@ class _HomeScreenState extends State<HomeScreen> with SingleTickerProviderStateM
} }
void _startDelayedFgFade() async { void _startDelayedFgFade() async {
for (var a in _fadeAnims) { try {
a.value = 0; for (var a in _fadeAnims) {
} a.value = 0;
await Future.delayed(300.ms); }
for (var a in _fadeAnims) { await Future.delayed(300.ms);
a.forward(); for (var a in _fadeAnims) {
a.forward();
}
} on Exception catch (e) {
debugPrint(e.toString());
} }
} }
@ -109,227 +112,227 @@ class _HomeScreenState extends State<HomeScreen> with SingleTickerProviderStateM
_startDelayedFgFade(); _startDelayedFgFade();
_fadeInOnNextBuild = false; _fadeInOnNextBuild = false;
} }
return _swipeController.wrapGestureDetector(
Container(
color: $styles.colors.black,
child: Stack(
children: [
/// Background
..._buildBgChildren(),
/// Clouds return _swipeController.wrapGestureDetector(Container(
FractionallySizedBox( color: $styles.colors.black,
widthFactor: 1, child: Stack(
heightFactor: .5, children: [
child: AnimatedClouds(wonderType: currentWonder.type, opacity: 1), Stack(
), children: [
/// Background
..._buildBgAndClouds(),
/// Wonders Illustrations /// Wonders Illustrations (main content)
MergeSemantics( _buildMgPageView(),
child: Semantics(
header: true,
image: true,
liveRegion: true,
onIncrease: () => _setPageIndex(_wonderIndex + 1),
onDecrease: () => _setPageIndex(_wonderIndex - 1),
child: PageView.builder(
controller: _pageController,
onPageChanged: _handlePageViewChanged,
itemBuilder: _buildMgChild,
),
),
),
Stack(children: [ /// Foreground illustrations and gradients
/// Foreground gradient-1, gets darker when swiping up _buildFgAndGradients(),
BottomCenter(
child: _buildSwipeableBgGradient(currentWonder.type.bgColor.withOpacity(.65)),
),
/// Foreground decorators /// Controls that float on top of the various illustrations
..._buildFgChildren(), _buildFloatingUi(),
],
).animate().fadeIn(),
],
),
));
}
/// Foreground gradient-2, gets darker when swiping up Widget _buildMgPageView() {
BottomCenter( return ExcludeSemantics(
child: _buildSwipeableBgGradient(currentWonder.type.bgColor.withOpacity(1)), child: PageView.builder(
), controller: _pageController,
onPageChanged: _handlePageViewChanged,
/// Floating controls / UI itemBuilder: (_, index) {
AnimatedSwitcher( final wonder = _wonders[index % _wonders.length];
duration: $styles.times.fast, final wonderType = wonder.type;
child: RepaintBoundary( bool isShowing = _isSelected(wonderType);
key: ObjectKey(currentWonder), return _swipeController.buildListener(
child: OverflowBox( builder: (swipeAmt, _, child) {
child: Column( final config = WonderIllustrationConfig.mg(
mainAxisSize: MainAxisSize.min, isShowing: isShowing,
children: [ zoom: .05 * swipeAmt,
SizedBox(width: double.infinity), );
const Spacer(), return WonderIllustration(wonderType, config: config);
},
/// Title Content );
LightText( },
child: IgnorePointer(
ignoringSemantics: false,
child: MergeSemantics(
child: Transform.translate(
offset: Offset(0, 30),
child: Column(
children: [
// Hide the title when the menu is open for visual polish
AnimatedOpacity(
opacity: _isMenuOpen ? 0 : 1,
duration: $styles.times.fast,
child: WonderTitleText(currentWonder, enableShadows: true),
),
Gap($styles.insets.md),
AppPageIndicator(
count: _numWonders,
controller: _pageController,
color: $styles.colors.white,
dotSize: 8,
onDotPressed: _handlePageIndicatorDotPressed,
semanticPageTitle: $strings.homeSemanticWonder,
),
Gap($styles.insets.md),
],
),
),
),
),
),
/// Animated arrow and background
/// Wrap in a container that is full-width to make it easier to find for screen readers
MergeSemantics(
child: Container(
width: double.infinity,
alignment: Alignment.center,
/// Lose state of child objects when index changes, this will re-run all the animated switcher and the arrow anim
key: ValueKey(_wonderIndex),
child: Stack(
children: [
/// Expanding rounded rect that grows in height as user swipes up
Positioned.fill(
child: _buildSwipeableArrowBg(),
),
/// Arrow Btn that fades in and out
_AnimatedArrowButton(onTap: _showDetailsPage, semanticTitle: currentWonder.title),
],
),
),
),
Gap($styles.insets.md),
],
),
),
),
),
/// Menu Btn
TopLeft(
child: AnimatedOpacity(
duration: $styles.times.fast,
opacity: _isMenuOpen ? 0 : 1,
child: MergeSemantics(
child: Semantics(
sortKey: OrdinalSortKey(0),
child: CircleIconBtn(
icon: AppIcons.menu,
onPressed: _handleOpenMenuPressed,
semanticLabel: $strings.homeSemanticOpenMain,
).safe(),
),
),
),
),
]),
],
).animate().fadeIn(),
), ),
); );
} }
Widget _buildMgChild(_, index) { List<Widget> _buildBgAndClouds() {
final wonder = _wonders[index % _wonders.length]; return [
final wonderType = wonder.type; // Background
bool isShowing = _isSelected(wonderType); ..._wonders.map((e) {
return _swipeController.buildListener(builder: (swipeAmt, _, child) { final config = WonderIllustrationConfig.bg(isShowing: _isSelected(e.type));
final config = WonderIllustrationConfig.mg( return WonderIllustration(e.type, config: config);
isShowing: isShowing, }).toList(),
zoom: .05 * swipeAmt, // Clouds
); FractionallySizedBox(
return ExcludeSemantics( widthFactor: 1,
excluding: !isShowing, heightFactor: .5,
child: Semantics( child: AnimatedClouds(wonderType: currentWonder.type, opacity: 1),
label: wonder.title, )
child: WonderIllustration(wonderType, config: config), ];
),
);
});
} }
List<Widget> _buildBgChildren() { Widget _buildFgAndGradients() {
return _wonders.map((e) { Widget buildSwipeableBgGradient(Color fgColor) {
final config = WonderIllustrationConfig.bg(isShowing: _isSelected(e.type)); return _swipeController.buildListener(builder: (swipeAmt, isPointerDown, _) {
return WonderIllustration(e.type, config: config); return IgnorePointer(
}).toList(); child: FractionallySizedBox(
} heightFactor: .6,
child: Container(
List<Widget> _buildFgChildren() { decoration: BoxDecoration(
return _wonders.map((e) { gradient: LinearGradient(
return _swipeController.buildListener(builder: (swipeAmt, _, child) { begin: Alignment.topCenter,
final config = WonderIllustrationConfig.fg( end: Alignment.bottomCenter,
isShowing: _isSelected(e.type), colors: [
zoom: .4 * (_swipeOverride ?? swipeAmt), fgColor.withOpacity(0),
); fgColor.withOpacity(.5 + fgColor.opacity * .25 + (isPointerDown ? .05 : 0) + swipeAmt * .20),
return Animate( ],
effects: const [FadeEffect()], stops: const [0, 1],
onPlay: _handleFadeAnimInit, ),
child: IgnorePointer(child: WonderIllustration(e.type, config: config)));
});
}).toList();
}
Widget _buildSwipeableArrowBg() {
return _swipeController.buildListener(
builder: (swipeAmt, _, child) {
double heightFactor = .5 + .5 * (1 + swipeAmt * 4);
return FractionallySizedBox(
alignment: Alignment.bottomCenter,
heightFactor: heightFactor,
child: Opacity(opacity: swipeAmt * .5, child: child),
);
},
child: VtGradient(
[$styles.colors.white.withOpacity(0), $styles.colors.white.withOpacity(1)],
const [.3, 1],
borderRadius: BorderRadius.circular(99),
),
);
}
Widget _buildSwipeableBgGradient(Color fgColor) {
return _swipeController.buildListener(builder: (swipeAmt, isPointerDown, _) {
return IgnorePointer(
child: FractionallySizedBox(
heightFactor: .6,
child: Container(
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
fgColor.withOpacity(0),
fgColor.withOpacity(.5 + fgColor.opacity * .25 + (isPointerDown ? .05 : 0) + swipeAmt * .20),
],
stops: const [0, 1],
), ),
), ),
), ),
);
});
}
final gradientColor = currentWonder.type.bgColor;
return Stack(children: [
/// Foreground gradient-1, gets darker when swiping up
BottomCenter(
child: buildSwipeableBgGradient(gradientColor.withOpacity(.65)),
),
/// Foreground decorators
..._wonders.map((e) {
return _swipeController.buildListener(builder: (swipeAmt, _, child) {
final config = WonderIllustrationConfig.fg(
isShowing: _isSelected(e.type),
zoom: .4 * (_swipeOverride ?? swipeAmt),
);
return Animate(
effects: const [FadeEffect()],
onPlay: _handleFadeAnimInit,
child: IgnorePointer(child: WonderIllustration(e.type, config: config)));
});
}).toList(),
/// Foreground gradient-2, gets darker when swiping up
BottomCenter(
child: buildSwipeableBgGradient(gradientColor),
),
]);
}
Widget _buildFloatingUi() {
return Stack(children: [
/// Floating controls / UI
AnimatedSwitcher(
duration: $styles.times.fast,
child: RepaintBoundary(
child: OverflowBox(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
SizedBox(width: double.infinity),
const Spacer(),
/// Title Content
LightText(
child: IgnorePointer(
ignoringSemantics: false,
child: Transform.translate(
offset: Offset(0, 30),
child: Column(
children: [
Semantics(
liveRegion: true,
button: true,
header: true,
onIncrease: () => _setPageIndex(_wonderIndex + 1),
onDecrease: () => _setPageIndex(_wonderIndex - 1),
onTap: () => _showDetailsPage(),
// Hide the title when the menu is open for visual polish
child: AnimatedOpacity(
opacity: _isMenuOpen ? 0 : 1,
duration: $styles.times.fast,
child: WonderTitleText(currentWonder, enableShadows: true),
),
),
Gap($styles.insets.md),
AppPageIndicator(
count: _numWonders,
controller: _pageController,
color: $styles.colors.white,
dotSize: 8,
onDotPressed: _handlePageIndicatorDotPressed,
semanticPageTitle: $strings.homeSemanticWonder,
),
Gap($styles.insets.md),
],
),
),
),
),
/// Animated arrow and background
/// Wrap in a container that is full-width to make it easier to find for screen readers
Container(
width: double.infinity,
alignment: Alignment.center,
/// Lose state of child objects when index changes, this will re-run all the animated switcher and the arrow anim
key: ValueKey(_wonderIndex),
child: Stack(
children: [
/// Expanding rounded rect that grows in height as user swipes up
Positioned.fill(
child: _swipeController.buildListener(
builder: (swipeAmt, _, child) {
double heightFactor = .5 + .5 * (1 + swipeAmt * 4);
return FractionallySizedBox(
alignment: Alignment.bottomCenter,
heightFactor: heightFactor,
child: Opacity(opacity: swipeAmt * .5, child: child),
);
},
child: VtGradient(
[$styles.colors.white.withOpacity(0), $styles.colors.white.withOpacity(1)],
const [.3, 1],
borderRadius: BorderRadius.circular(99),
),
)),
/// Arrow Btn that fades in and out
_AnimatedArrowButton(onTap: _showDetailsPage, semanticTitle: currentWonder.title),
],
),
),
Gap($styles.insets.md),
],
),
),
), ),
); ),
});
/// Menu Btn
TopLeft(
child: AnimatedOpacity(
duration: $styles.times.fast,
opacity: _isMenuOpen ? 0 : 1,
child: MergeSemantics(
child: CircleIconBtn(
icon: AppIcons.menu,
onPressed: _handleOpenMenuPressed,
semanticLabel: $strings.homeSemanticOpenMain,
).safe(),
),
),
),
]);
} }
} }

View File

@ -9,31 +9,19 @@ class _EventsList extends StatefulWidget {
} }
class _EventsListState extends State<_EventsList> { class _EventsListState extends State<_EventsList> {
late final ScrollController _scroller = ScrollController()..addListener(_handleScrollChanged); final ScrollController _scroller = ScrollController();
bool _hasPopped = false;
bool _isPointerDown = false;
@override @override
void dispose() { void dispose() {
_scroller.dispose(); _scroller.dispose();
super.dispose(); super.dispose();
} }
void _handleScrollChanged() {
if (!_isPointerDown) return;
if (_scroller.position.pixels < -100 && !_hasPopped) {
_hasPopped = true;
context.pop();
}
}
bool _checkPointerIsDown(d) => _isPointerDown = d.dragDetails != null;
void _handleGlobalTimelinePressed() => context.push(ScreenPaths.timeline(widget.data.type)); void _handleGlobalTimelinePressed() => context.push(ScreenPaths.timeline(widget.data.type));
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return NotificationListener<ScrollUpdateNotification>( return PopRouterOnOverScroll(
onNotification: _checkPointerIsDown, controller: _scroller,
child: LayoutBuilder(builder: (_, constraints) { child: LayoutBuilder(builder: (_, constraints) {
return Stack( return Stack(
children: [ children: [

View File

@ -6,6 +6,7 @@ import 'package:wonders/ui/common/compass_divider.dart';
import 'package:wonders/ui/common/curved_clippers.dart'; import 'package:wonders/ui/common/curved_clippers.dart';
import 'package:wonders/ui/common/hidden_collectible.dart'; import 'package:wonders/ui/common/hidden_collectible.dart';
import 'package:wonders/ui/common/list_gradient.dart'; import 'package:wonders/ui/common/list_gradient.dart';
import 'package:wonders/ui/common/pop_router_on_over_scroll.dart';
import 'package:wonders/ui/common/themed_text.dart'; import 'package:wonders/ui/common/themed_text.dart';
import 'package:wonders/ui/common/timeline_event_card.dart'; import 'package:wonders/ui/common/timeline_event_card.dart';
import 'package:wonders/ui/common/wonders_timeline_builder.dart'; import 'package:wonders/ui/common/wonders_timeline_builder.dart';

View File

@ -1,7 +1,7 @@
name: wonders name: wonders
description: Explore the famous wonders of the world. description: Explore the famous wonders of the world.
publish_to: "none" publish_to: "none"
version: 1.9.4 version: 1.9.5
environment: environment:
sdk: ">=2.16.0 <3.0.0" sdk: ">=2.16.0 <3.0.0"

View File

@ -1,3 +1,8 @@
# 1.9.5
- Improved a11y for Home Screen and Artifact Carousel
- Polished collection icons
- Added hero support to fullscreen Artifact Details
# 1.9.4 # 1.9.4
- Fix tappable arrow on home page - Fix tappable arrow on home page
- Reduce text size for collapsing pull quote - Reduce text size for collapsing pull quote