From 5634aca0d37dcd925afff0fc2431c5674cc17f98 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Wed, 25 Jan 2023 21:19:32 +0100 Subject: [PATCH 01/12] Bump go_router --- pubspec.lock | 2 +- pubspec.yaml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/pubspec.lock b/pubspec.lock index ba980a62d..e4f587520 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -480,7 +480,7 @@ packages: name: go_router url: "https://pub.dartlang.org" source: hosted - version: "5.2.0" + version: "6.0.1" google_fonts: dependency: "direct main" description: diff --git a/pubspec.yaml b/pubspec.yaml index a2d94bf1e..48669f2f7 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -110,7 +110,7 @@ dependencies: flutter_cache_manager: ^3.3.0 # Navigation. - go_router: ^5.1.10 + go_router: ^6.0.1 # Displaying QR codes. qr_flutter: ^4.0.0 From 0eccf4538f894525b25a7dcdb442924ff45e4864 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Thu, 26 Jan 2023 11:52:26 +0100 Subject: [PATCH 02/12] Add new list state --- lib/blocs/list_state.dart | 65 ++++++++++++++++++++++++++++++++ lib/blocs/member_list_cubit.dart | 38 ++++++++----------- 2 files changed, 81 insertions(+), 22 deletions(-) diff --git a/lib/blocs/list_state.dart b/lib/blocs/list_state.dart index 232328d75..e0bf5972f 100644 --- a/lib/blocs/list_state.dart +++ b/lib/blocs/list_state.dart @@ -80,3 +80,68 @@ class ListState extends Equatable { isLoadingMore = false, isDone = true; } + +/// Generic type for states with a paginated list of results. +/// +/// There are a number of subtypes: +/// * [ErrorListState] - indicates that there was an error. +/// * [LoadingListState] - indicates that we are loading. +/// * [ResultsListState] - indicates that there are results. +/// * [DoneListState] - indicates that there are no more results. +/// * [LoadingMoreListState] - indicates that we are loading more results. +abstract class XListState extends Equatable { + const XListState(); + + /// A convenience method to get the results if they are available. + /// + /// Returns `[]` if this state is not a (subtype of) [ResultsListState]. + List get results => + this is ResultsListState ? (this as ResultsListState).results : []; + + /// A convenience method to get the error message if there is one. + /// + /// Returns `null` iff this state is not a (subtype of) [ErrorListState]. + String? get message => + this is ErrorListState ? (this as ErrorListState).message : null; + + @override + List get props => []; +} + +class LoadingListState extends XListState { + const LoadingListState(); +} + +class ErrorListState extends XListState { + @override + final String message; + + const ErrorListState(this.message); + + @override + List get props => [message]; +} + +class ResultsListState extends XListState { + @override + final List results; + + const ResultsListState(this.results); + + factory ResultsListState.withDone(List results, bool isDone) => + isDone ? DoneListState(results) : ResultsListState(results); + + @override + List get props => [results]; +} + +class LoadingMoreListState extends ResultsListState { + const LoadingMoreListState(super.results); + + factory LoadingMoreListState.from(ResultsListState state) => + LoadingMoreListState(state.results); +} + +class DoneListState extends ResultsListState { + const DoneListState(super.results); +} diff --git a/lib/blocs/member_list_cubit.dart b/lib/blocs/member_list_cubit.dart index 3a2b26922..430f07db8 100644 --- a/lib/blocs/member_list_cubit.dart +++ b/lib/blocs/member_list_cubit.dart @@ -7,7 +7,7 @@ import 'package:reaxit/config.dart' as config; import 'package:reaxit/blocs.dart'; import 'package:reaxit/models.dart'; -typedef MemberListState = ListState; +typedef MemberListState = XListState; class MemberListCubit extends Cubit { static const int firstPageSize = 60; @@ -27,10 +27,9 @@ class MemberListCubit extends Cubit { /// The offset to be used for the next paginated request. int _nextOffset = 0; - MemberListCubit(this.api) : super(const MemberListState.loading(results: [])); + MemberListCubit(this.api) : super(const LoadingListState()); Future load() async { - emit(state.copyWith(isLoading: true)); try { final query = _searchQuery; final membersResponse = await api.getMembers( @@ -49,30 +48,28 @@ class MemberListCubit extends Cubit { if (membersResponse.results.isEmpty) { if (query?.isEmpty ?? true) { - emit(const MemberListState.failure(message: 'There are no members.')); + emit(const ErrorListState('There are no members.')); } else { - emit(MemberListState.failure( - message: 'There are no members found for "$query".', + emit(ErrorListState( + 'There are no members found for "$query".', )); } } else { - emit(MemberListState.success( - results: membersResponse.results, - isDone: isDone, - )); + emit(ResultsListState.withDone(membersResponse.results, isDone)); } } on ApiException catch (exception) { - emit(MemberListState.failure(message: exception.message)); + emit(ErrorListState(exception.message)); } } Future more() async { - final oldState = state; - // Ignore calls to `more()` if there is no data, or already more coming. - if (oldState.isDone || oldState.isLoading || oldState.isLoadingMore) return; + if (state is! ResultsListState || + state is LoadingMoreListState || + state is DoneListState) return; + final oldState = state as ResultsListState; - emit(oldState.copyWith(isLoadingMore: true)); + emit(LoadingMoreListState.from(oldState)); try { final query = _searchQuery; @@ -87,17 +84,14 @@ class MemberListCubit extends Cubit { // changed since the request was made. if (query != _searchQuery) return; - final members = state.results + membersResponse.results; + final members = oldState.results + membersResponse.results; final isDone = members.length == membersResponse.count; _nextOffset += pageSize; - emit(MemberListState.success( - results: members, - isDone: isDone, - )); + emit(ResultsListState.withDone(members, isDone)); } on ApiException catch (exception) { - emit(MemberListState.failure(message: exception.getMessage())); + emit(ErrorListState(exception.getMessage())); } } @@ -110,7 +104,7 @@ class MemberListCubit extends Cubit { _searchDebounceTimer?.cancel(); if (query?.isEmpty ?? false) { /// Don't get results when the query is empty. - emit(const MemberListState.loading(results: [])); + emit(const LoadingListState()); } else { _searchDebounceTimer = Timer(config.searchDebounceTime, load); } From e0c18128504a1da7d2842632647bb3647e4e254d Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Thu, 26 Jan 2023 11:52:48 +0100 Subject: [PATCH 03/12] Add generic widget for paginated lists --- lib/ui/screens/members_screen.dart | 183 ++++++---------------- lib/ui/widgets.dart | 1 + lib/ui/widgets/paginated_scroll_view.dart | 129 +++++++++++++++ 3 files changed, 181 insertions(+), 132 deletions(-) create mode 100644 lib/ui/widgets/paginated_scroll_view.dart diff --git a/lib/ui/screens/members_screen.dart b/lib/ui/screens/members_screen.dart index 867c1eb68..3fe73cb76 100644 --- a/lib/ui/screens/members_screen.dart +++ b/lib/ui/screens/members_screen.dart @@ -3,40 +3,10 @@ import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:google_fonts/google_fonts.dart'; import 'package:reaxit/api/api_repository.dart'; import 'package:reaxit/blocs.dart'; +import 'package:reaxit/models/member.dart'; import 'package:reaxit/ui/widgets.dart'; -class MembersScreen extends StatefulWidget { - @override - State createState() => _MembersScreenState(); -} - -class _MembersScreenState extends State { - late ScrollController _controller; - late MemberListCubit _cubit; - - @override - void initState() { - _cubit = BlocProvider.of(context); - _controller = ScrollController()..addListener(_scrollListener); - super.initState(); - } - - void _scrollListener() { - if (_controller.position.pixels >= - _controller.position.maxScrollExtent - 300) { - // Only request loading more if that's not already happening. - if (!_cubit.state.isLoadingMore) { - _cubit.more(); - } - } - } - - @override - void dispose() { - _controller.dispose(); - super.dispose(); - } - +class MembersScreen extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( @@ -63,20 +33,16 @@ class _MembersScreenState extends State { ), drawer: MenuDrawer(), body: RefreshIndicator( - onRefresh: () async { - await _cubit.load(); - }, + onRefresh: () => BlocProvider.of(context).load(), child: BlocBuilder( builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return MemberListScrollView( - key: const PageStorageKey('members'), - controller: _controller, - listState: listState, - ); - } + return PaginatedScrollView( + state: listState, + onLoadMore: (context) { + BlocProvider.of(context).more(); + }, + resultsBuilder: (_, results) => [_MembersGrid(results)], + ); }, ), ), @@ -84,23 +50,36 @@ class _MembersScreenState extends State { } } -class MembersSearchDelegate extends SearchDelegate { - final MemberListCubit _cubit; - late final ScrollController _controller; +class _MembersGrid extends StatelessWidget { + const _MembersGrid(this.results); - MembersSearchDelegate(this._cubit) { - _controller = ScrollController()..addListener(_scrollListener); - } + final List results; - void _scrollListener() { - if (_controller.position.pixels >= - _controller.position.maxScrollExtent - 300) { - // Only request loading more if that's not already happening. - if (!_cubit.state.isLoadingMore) { - _cubit.more(); - } - } + @override + Widget build(BuildContext context) { + return SliverPadding( + padding: const EdgeInsets.fromLTRB(8, 8, 8, 0), + sliver: SliverGrid( + gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( + crossAxisCount: 3, + mainAxisSpacing: 8, + crossAxisSpacing: 8, + ), + delegate: SliverChildBuilderDelegate( + (context, index) => MemberTile( + member: results[index], + ), + childCount: results.length, + ), + ), + ); } +} + +class MembersSearchDelegate extends SearchDelegate { + final MemberListCubit _cubit; + + MembersSearchDelegate(this._cubit); @override ThemeData appBarTheme(BuildContext context) { @@ -144,15 +123,13 @@ class MembersSearchDelegate extends SearchDelegate { return BlocBuilder( bloc: _cubit..search(query), builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return MemberListScrollView( - key: const PageStorageKey('members-search'), - controller: _controller, - listState: listState, - ); - } + return PaginatedScrollView( + state: listState, + onLoadMore: (context) { + _cubit.more(); + }, + resultsBuilder: (_, results) => [_MembersGrid(results)], + ); }, ); } @@ -162,72 +139,14 @@ class MembersSearchDelegate extends SearchDelegate { return BlocBuilder( bloc: _cubit..search(query), builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return MemberListScrollView( - key: const PageStorageKey('members-search'), - controller: _controller, - listState: listState, - ); - } + return PaginatedScrollView( + state: listState, + onLoadMore: (context) { + _cubit.more(); + }, + resultsBuilder: (_, results) => [_MembersGrid(results)], + ); }, ); } } - -/// A ScrollView that shows a grid of [MemberTile]s. -/// -/// This does not take care of communicating with a Cubit. The [controller] -/// should do that. The [listState] also must not have an exception. -class MemberListScrollView extends StatelessWidget { - final ScrollController controller; - final MemberListState listState; - - const MemberListScrollView({ - Key? key, - required this.controller, - required this.listState, - }) : super(key: key); - - @override - Widget build(BuildContext context) { - return Scrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - physics: const RangeMaintainingScrollPhysics( - parent: AlwaysScrollableScrollPhysics(), - ), - slivers: [ - SliverPadding( - padding: const EdgeInsets.all(8), - sliver: SliverGrid( - gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( - crossAxisCount: 3, - mainAxisSpacing: 8, - crossAxisSpacing: 8, - ), - delegate: SliverChildBuilderDelegate( - (context, index) => MemberTile( - member: listState.results[index], - ), - childCount: listState.results.length, - ), - ), - ), - if (listState.isLoadingMore) - const SliverPadding( - padding: EdgeInsets.all(8), - sliver: SliverList( - delegate: SliverChildListDelegate.fixed([ - Center( - child: CircularProgressIndicator(), - ) - ]), - ), - ), - ], - )); - } -} diff --git a/lib/ui/widgets.dart b/lib/ui/widgets.dart index e3ffbc54d..094485332 100644 --- a/lib/ui/widgets.dart +++ b/lib/ui/widgets.dart @@ -7,6 +7,7 @@ export 'widgets/event_detail_card.dart'; export 'widgets/mark_present_dialog.dart'; export 'widgets/member_tile.dart'; export 'widgets/menu_drawer.dart'; +export 'widgets/paginated_scroll_view.dart'; export 'widgets/push_notification_dialog.dart'; export 'widgets/push_notification_overlay.dart'; export 'widgets/sales_order_dialog.dart'; diff --git a/lib/ui/widgets/paginated_scroll_view.dart b/lib/ui/widgets/paginated_scroll_view.dart new file mode 100644 index 000000000..426ee36db --- /dev/null +++ b/lib/ui/widgets/paginated_scroll_view.dart @@ -0,0 +1,129 @@ +import 'package:flutter/material.dart'; +import 'package:reaxit/blocs/list_state.dart'; + +/// A widget that displays and triggers loading of a paginated list. +class PaginatedScrollView extends StatefulWidget { + const PaginatedScrollView({ + super.key, + required this.state, + required this.onLoadMore, + required this.resultsBuilder, + this.loadingBuilder, + }); + + final XListState state; + + /// A builder that creates a list of slivers from the results. + /// + /// For example, this could return a list with a single [SliverGrid]. + final List Function( + BuildContext context, + List results, + ) resultsBuilder; + + /// An optional builder for a list of slivers to be shown when loading. + /// + /// If this is not provided, nothing will be shown. + final List Function(BuildContext context)? loadingBuilder; + + /// A callback that is called when more results should be loaded. + /// + /// This should trigger the loading of another page of results. + /// This is only called when the current state is [ResultsListState], + /// not from [LoadingMoreListState] or [DoneListState]. + final void Function(BuildContext context) onLoadMore; + + @override + State> createState() => _PaginatedScrollViewState(); +} + +class _PaginatedScrollViewState extends State> { + late ScrollController controller; + + @override + void initState() { + controller = ScrollController()..addListener(_scrollListener); + super.initState(); + } + + @override + void dispose() { + controller.dispose(); + super.dispose(); + } + + void _scrollListener() { + final position = controller.position; + if (position.pixels >= position.maxScrollExtent - 300) { + if (widget.state is ResultsListState && + widget.state is! DoneListState && + widget.state is! LoadingMoreListState) { + widget.onLoadMore(context); + } + } + } + + @override + Widget build(BuildContext context) { + late final List slivers; + if (widget.state is ErrorListState) { + slivers = [ + SliverSafeArea( + minimum: const EdgeInsets.all(16), + sliver: SliverToBoxAdapter( + child: Column( + children: [ + Container( + height: 100, + margin: const EdgeInsets.all(12), + child: Image.asset( + 'assets/img/sad-cloud.png', + fit: BoxFit.fitHeight, + ), + ), + Text(widget.state.message!, textAlign: TextAlign.center), + ], + ), + ), + ), + ]; + } else if (widget.state is LoadingListState) { + if (widget.loadingBuilder != null) { + slivers = widget.loadingBuilder!(context); + } else { + slivers = []; + } + } else { + final resultsSlivers = widget.resultsBuilder( + context, + widget.state.results, + ); + + slivers = [ + ...resultsSlivers, + if (widget.state is LoadingMoreListState) + const SliverPadding( + padding: EdgeInsets.only(top: 16), + sliver: SliverToBoxAdapter( + child: Center(child: CircularProgressIndicator()), + ), + ), + const SliverSafeArea( + minimum: EdgeInsets.only(bottom: 8), + sliver: SliverPadding(padding: EdgeInsets.zero), + ), + ]; + } + + return Scrollbar( + controller: controller, + child: CustomScrollView( + controller: controller, + physics: const RangeMaintainingScrollPhysics( + parent: AlwaysScrollableScrollPhysics(), + ), + slivers: slivers, + ), + ); + } +} From 4229d4ce8e69c0a57c8bf19e207ca7b2801ec9b3 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Thu, 26 Jan 2023 21:06:00 +0100 Subject: [PATCH 04/12] Add PaginatedCubit interface --- lib/blocs/member_list_cubit.dart | 13 +- lib/ui/screens/members_screen.dart | 44 ++---- lib/ui/widgets/paginated_scroll_view.dart | 158 ++++++++++++---------- 3 files changed, 102 insertions(+), 113 deletions(-) diff --git a/lib/blocs/member_list_cubit.dart b/lib/blocs/member_list_cubit.dart index 430f07db8..456399b22 100644 --- a/lib/blocs/member_list_cubit.dart +++ b/lib/blocs/member_list_cubit.dart @@ -1,18 +1,13 @@ import 'dart:async'; -import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:reaxit/api/api_repository.dart'; import 'package:reaxit/api/exceptions.dart'; import 'package:reaxit/config.dart' as config; import 'package:reaxit/blocs.dart'; import 'package:reaxit/models.dart'; +import 'package:reaxit/ui/widgets.dart'; -typedef MemberListState = XListState; - -class MemberListCubit extends Cubit { - static const int firstPageSize = 60; - static const int pageSize = 30; - +class MemberListCubit extends PaginatedCubit { final ApiRepository api; /// The last used search query. Can be set through `this.search(query)`. @@ -27,8 +22,9 @@ class MemberListCubit extends Cubit { /// The offset to be used for the next paginated request. int _nextOffset = 0; - MemberListCubit(this.api) : super(const LoadingListState()); + MemberListCubit(this.api) : super(firstPageSize: 60, pageSize: 30); + @override Future load() async { try { final query = _searchQuery; @@ -62,6 +58,7 @@ class MemberListCubit extends Cubit { } } + @override Future more() async { // Ignore calls to `more()` if there is no data, or already more coming. if (state is! ResultsListState || diff --git a/lib/ui/screens/members_screen.dart b/lib/ui/screens/members_screen.dart index 3fe73cb76..db8d1dbb6 100644 --- a/lib/ui/screens/members_screen.dart +++ b/lib/ui/screens/members_screen.dart @@ -34,16 +34,8 @@ class MembersScreen extends StatelessWidget { drawer: MenuDrawer(), body: RefreshIndicator( onRefresh: () => BlocProvider.of(context).load(), - child: BlocBuilder( - builder: (context, listState) { - return PaginatedScrollView( - state: listState, - onLoadMore: (context) { - BlocProvider.of(context).more(); - }, - resultsBuilder: (_, results) => [_MembersGrid(results)], - ); - }, + child: PaginatedScrollView( + resultsBuilder: (_, results) => [_MembersGrid(results)], ), ), ); @@ -120,33 +112,21 @@ class MembersSearchDelegate extends SearchDelegate { @override Widget buildResults(BuildContext context) { - return BlocBuilder( - bloc: _cubit..search(query), - builder: (context, listState) { - return PaginatedScrollView( - state: listState, - onLoadMore: (context) { - _cubit.more(); - }, - resultsBuilder: (_, results) => [_MembersGrid(results)], - ); - }, + return BlocProvider.value( + value: _cubit..search(query), + child: PaginatedScrollView( + resultsBuilder: (_, results) => [_MembersGrid(results)], + ), ); } @override Widget buildSuggestions(BuildContext context) { - return BlocBuilder( - bloc: _cubit..search(query), - builder: (context, listState) { - return PaginatedScrollView( - state: listState, - onLoadMore: (context) { - _cubit.more(); - }, - resultsBuilder: (_, results) => [_MembersGrid(results)], - ); - }, + return BlocProvider.value( + value: _cubit..search(query), + child: PaginatedScrollView( + resultsBuilder: (_, results) => [_MembersGrid(results)], + ), ); } } diff --git a/lib/ui/widgets/paginated_scroll_view.dart b/lib/ui/widgets/paginated_scroll_view.dart index 426ee36db..be6a4d08b 100644 --- a/lib/ui/widgets/paginated_scroll_view.dart +++ b/lib/ui/widgets/paginated_scroll_view.dart @@ -1,24 +1,38 @@ import 'package:flutter/material.dart'; +import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:reaxit/blocs/list_state.dart'; +abstract class PaginatedCubit extends Cubit> { + final int firstPageSize; + final int pageSize; + + PaginatedCubit({ + required this.firstPageSize, + required this.pageSize, + }) : super(const LoadingListState()); + + /// Load the first page of results. + Future load(); + + /// Load another page of results. + Future more(); +} + /// A widget that displays and triggers loading of a paginated list. -class PaginatedScrollView extends StatefulWidget { +class PaginatedScrollView> + extends StatefulWidget { const PaginatedScrollView({ super.key, - required this.state, - required this.onLoadMore, required this.resultsBuilder, this.loadingBuilder, }); - final XListState state; - /// A builder that creates a list of slivers from the results. /// /// For example, this could return a list with a single [SliverGrid]. final List Function( BuildContext context, - List results, + List results, ) resultsBuilder; /// An optional builder for a list of slivers to be shown when loading. @@ -26,18 +40,13 @@ class PaginatedScrollView extends StatefulWidget { /// If this is not provided, nothing will be shown. final List Function(BuildContext context)? loadingBuilder; - /// A callback that is called when more results should be loaded. - /// - /// This should trigger the loading of another page of results. - /// This is only called when the current state is [ResultsListState], - /// not from [LoadingMoreListState] or [DoneListState]. - final void Function(BuildContext context) onLoadMore; - @override - State> createState() => _PaginatedScrollViewState(); + State> createState() => + _PaginatedScrollViewState(); } -class _PaginatedScrollViewState extends State> { +class _PaginatedScrollViewState> + extends State> { late ScrollController controller; @override @@ -55,75 +64,78 @@ class _PaginatedScrollViewState extends State> { void _scrollListener() { final position = controller.position; if (position.pixels >= position.maxScrollExtent - 300) { - if (widget.state is ResultsListState && - widget.state is! DoneListState && - widget.state is! LoadingMoreListState) { - widget.onLoadMore(context); + final cubit = BlocProvider.of(context); + final state = cubit.state; + if (state is ResultsListState && + state is! DoneListState && + state is! LoadingMoreListState) { + cubit.more(); } } } @override Widget build(BuildContext context) { - late final List slivers; - if (widget.state is ErrorListState) { - slivers = [ - SliverSafeArea( - minimum: const EdgeInsets.all(16), - sliver: SliverToBoxAdapter( - child: Column( - children: [ - Container( - height: 100, - margin: const EdgeInsets.all(12), - child: Image.asset( - 'assets/img/sad-cloud.png', - fit: BoxFit.fitHeight, - ), + return BlocBuilder>( + builder: (context, state) { + late final List slivers; + if (state is ErrorListState) { + slivers = [ + SliverSafeArea( + minimum: const EdgeInsets.all(16), + sliver: SliverToBoxAdapter( + child: Column( + children: [ + Container( + height: 100, + margin: const EdgeInsets.all(12), + child: Image.asset( + 'assets/img/sad-cloud.png', + fit: BoxFit.fitHeight, + ), + ), + Text(state.message!, textAlign: TextAlign.center), + ], ), - Text(widget.state.message!, textAlign: TextAlign.center), - ], + ), ), - ), - ), - ]; - } else if (widget.state is LoadingListState) { - if (widget.loadingBuilder != null) { - slivers = widget.loadingBuilder!(context); - } else { - slivers = []; - } - } else { - final resultsSlivers = widget.resultsBuilder( - context, - widget.state.results, - ); + ]; + } else if (state is LoadingListState) { + if (widget.loadingBuilder != null) { + slivers = widget.loadingBuilder!(context); + } else { + slivers = []; + } + } else { + final resultSlivers = widget.resultsBuilder(context, state.results); - slivers = [ - ...resultsSlivers, - if (widget.state is LoadingMoreListState) - const SliverPadding( - padding: EdgeInsets.only(top: 16), - sliver: SliverToBoxAdapter( - child: Center(child: CircularProgressIndicator()), + slivers = [ + ...resultSlivers, + if (state is LoadingMoreListState) + const SliverPadding( + padding: EdgeInsets.only(top: 16), + sliver: SliverToBoxAdapter( + child: Center(child: CircularProgressIndicator()), + ), + ), + const SliverSafeArea( + minimum: EdgeInsets.only(bottom: 8), + sliver: SliverPadding(padding: EdgeInsets.zero), ), - ), - const SliverSafeArea( - minimum: EdgeInsets.only(bottom: 8), - sliver: SliverPadding(padding: EdgeInsets.zero), - ), - ]; - } + ]; + } - return Scrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - physics: const RangeMaintainingScrollPhysics( - parent: AlwaysScrollableScrollPhysics(), - ), - slivers: slivers, - ), + return Scrollbar( + controller: controller, + child: CustomScrollView( + controller: controller, + physics: const RangeMaintainingScrollPhysics( + parent: AlwaysScrollableScrollPhysics(), + ), + slivers: slivers, + ), + ); + }, ); } } From 896a07623815f1606d2fb8316e5414ab8df6e592 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Thu, 26 Jan 2023 21:19:38 +0100 Subject: [PATCH 05/12] Use PaginatedScrollView for albums too --- lib/blocs/album_list_cubit.dart | 48 ++++---- lib/blocs/member_list_cubit.dart | 5 +- lib/ui/screens/albums_screen.dart | 169 ++++++----------------------- lib/ui/screens/members_screen.dart | 52 ++++----- 4 files changed, 81 insertions(+), 193 deletions(-) diff --git a/lib/blocs/album_list_cubit.dart b/lib/blocs/album_list_cubit.dart index d0c7b1de3..4d379a5dc 100644 --- a/lib/blocs/album_list_cubit.dart +++ b/lib/blocs/album_list_cubit.dart @@ -1,18 +1,13 @@ import 'dart:async'; -import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:reaxit/api/api_repository.dart'; import 'package:reaxit/api/exceptions.dart'; +import 'package:reaxit/blocs/list_state.dart'; import 'package:reaxit/config.dart' as config; -import 'package:reaxit/blocs.dart'; import 'package:reaxit/models.dart'; +import 'package:reaxit/ui/widgets.dart'; -typedef AlbumListState = ListState; - -class AlbumListCubit extends Cubit { - static const int firstPageSize = 60; - static const int pageSize = 30; - +class AlbumListCubit extends PaginatedCubit { final ApiRepository api; /// The last used search query. Can be set through `this.search(query)`. @@ -27,10 +22,10 @@ class AlbumListCubit extends Cubit { /// The offset to be used for the next paginated request. int _nextOffset = 0; - AlbumListCubit(this.api) : super(const AlbumListState.loading(results: [])); + AlbumListCubit(this.api) : super(firstPageSize: 60, pageSize: 30); + @override Future load() async { - emit(state.copyWith(isLoading: true)); try { final query = _searchQuery; final albumsResponse = await api.getAlbums( @@ -49,30 +44,28 @@ class AlbumListCubit extends Cubit { if (albumsResponse.results.isEmpty) { if (query?.isEmpty ?? true) { - emit(const AlbumListState.failure(message: 'There are no albums.')); + emit(const ErrorListState('There are no albums.')); } else { - emit(AlbumListState.failure( - message: 'There are no albums found for "$query".', - )); + emit(ErrorListState('There are no albums found for "$query".')); } } else { - emit(AlbumListState.success( - results: albumsResponse.results, - isDone: isDone, - )); + emit(ResultsListState.withDone(albumsResponse.results, isDone)); } } on ApiException catch (exception) { - emit(AlbumListState.failure(message: exception.message)); + emit(ErrorListState(exception.message)); } } + @override Future more() async { - final oldState = state; - // Ignore calls to `more()` if there is no data, or already more coming. - if (oldState.isDone || oldState.isLoading || oldState.isLoadingMore) return; + if (state is! ResultsListState || + state is LoadingMoreListState || + state is DoneListState) return; + + final oldState = state as ResultsListState; - emit(oldState.copyWith(isLoadingMore: true)); + emit(LoadingMoreListState.from(oldState)); try { final query = _searchQuery; @@ -92,12 +85,9 @@ class AlbumListCubit extends Cubit { _nextOffset += pageSize; - emit(AlbumListState.success( - results: albums, - isDone: isDone, - )); + emit(ResultsListState.withDone(albums, isDone)); } on ApiException catch (exception) { - emit(AlbumListState.failure(message: exception.message)); + emit(ErrorListState(exception.getMessage())); } } @@ -110,7 +100,7 @@ class AlbumListCubit extends Cubit { _searchDebounceTimer?.cancel(); if (query?.isEmpty ?? false) { /// Don't get results when the query is empty. - emit(const AlbumListState.loading(results: [])); + emit(const LoadingListState()); } else { _searchDebounceTimer = Timer(config.searchDebounceTime, load); } diff --git a/lib/blocs/member_list_cubit.dart b/lib/blocs/member_list_cubit.dart index 456399b22..1ea3bccc0 100644 --- a/lib/blocs/member_list_cubit.dart +++ b/lib/blocs/member_list_cubit.dart @@ -46,9 +46,7 @@ class MemberListCubit extends PaginatedCubit { if (query?.isEmpty ?? true) { emit(const ErrorListState('There are no members.')); } else { - emit(ErrorListState( - 'There are no members found for "$query".', - )); + emit(ErrorListState('There are no members found for "$query".')); } } else { emit(ResultsListState.withDone(membersResponse.results, isDone)); @@ -64,6 +62,7 @@ class MemberListCubit extends PaginatedCubit { if (state is! ResultsListState || state is LoadingMoreListState || state is DoneListState) return; + final oldState = state as ResultsListState; emit(LoadingMoreListState.from(oldState)); diff --git a/lib/ui/screens/albums_screen.dart b/lib/ui/screens/albums_screen.dart index 9405f82d5..4ec79744f 100644 --- a/lib/ui/screens/albums_screen.dart +++ b/lib/ui/screens/albums_screen.dart @@ -3,40 +3,10 @@ import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:google_fonts/google_fonts.dart'; import 'package:reaxit/blocs.dart'; import 'package:reaxit/api/api_repository.dart'; +import 'package:reaxit/models.dart'; import 'package:reaxit/ui/widgets.dart'; -class AlbumsScreen extends StatefulWidget { - @override - State createState() => _AlbumsScreenState(); -} - -class _AlbumsScreenState extends State { - late ScrollController _controller; - late AlbumListCubit _cubit; - - @override - void initState() { - _cubit = BlocProvider.of(context); - _controller = ScrollController()..addListener(_scrollListener); - super.initState(); - } - - void _scrollListener() { - if (_controller.position.pixels >= - _controller.position.maxScrollExtent - 300) { - // Only request loading more if that's not already happening. - if (!_cubit.state.isLoadingMore) { - _cubit.more(); - } - } - } - - @override - void dispose() { - _controller.dispose(); - super.dispose(); - } - +class AlbumsScreen extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( @@ -63,21 +33,9 @@ class _AlbumsScreenState extends State { ), drawer: MenuDrawer(), body: RefreshIndicator( - onRefresh: () async { - await _cubit.load(); - }, - child: BlocBuilder( - builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return AlbumListScrollView( - key: const PageStorageKey('albums'), - controller: _controller, - listState: listState, - ); - } - }, + onRefresh: () => BlocProvider.of(context).load(), + child: PaginatedScrollView( + resultsBuilder: (context, results) => [_AlbumsGrid(results)], ), ), ); @@ -85,22 +43,9 @@ class _AlbumsScreenState extends State { } class AlbumsSearchDelegate extends SearchDelegate { - late final ScrollController _controller; final AlbumListCubit _cubit; - AlbumsSearchDelegate(this._cubit) { - _controller = ScrollController()..addListener(_scrollListener); - } - - void _scrollListener() { - if (_controller.position.pixels >= - _controller.position.maxScrollExtent - 300) { - // Only request loading more if that's not already happening. - if (!_cubit.state.isLoadingMore) { - _cubit.more(); - } - } - } + AlbumsSearchDelegate(this._cubit); @override ThemeData appBarTheme(BuildContext context) { @@ -141,93 +86,47 @@ class AlbumsSearchDelegate extends SearchDelegate { @override Widget buildResults(BuildContext context) { - return BlocBuilder( - bloc: _cubit..search(query), - builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return AlbumListScrollView( - key: const PageStorageKey('albums-search'), - controller: _controller, - listState: listState, - ); - } - }, + return BlocProvider.value( + value: _cubit..search(query), + child: PaginatedScrollView( + resultsBuilder: (_, results) => [_AlbumsGrid(results)], + ), ); } @override Widget buildSuggestions(BuildContext context) { - return BlocBuilder( - bloc: _cubit..search(query), - builder: (context, listState) { - if (listState.hasException) { - return ErrorScrollView(listState.message!); - } else { - return AlbumListScrollView( - key: const PageStorageKey('albums-search'), - controller: _controller, - listState: listState, - ); - } - }, + return BlocProvider.value( + value: _cubit..search(query), + child: PaginatedScrollView( + resultsBuilder: (_, results) => [_AlbumsGrid(results)], + ), ); } } -/// A ScrollView that shows a grid of [AlbumTile]s. -/// -/// This does not take care of communicating with a Bloc. The [controller] -/// should do that. The [listState] also must not have an exception. -class AlbumListScrollView extends StatelessWidget { - final ScrollController controller; - final AlbumListState listState; +class _AlbumsGrid extends StatelessWidget { + const _AlbumsGrid(this.results); - const AlbumListScrollView({ - Key? key, - required this.controller, - required this.listState, - }) : super(key: key); + final List results; @override Widget build(BuildContext context) { - return Scrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - physics: const RangeMaintainingScrollPhysics( - parent: AlwaysScrollableScrollPhysics(), + return SliverPadding( + padding: const EdgeInsets.fromLTRB(8, 8, 8, 0), + sliver: SliverGrid( + gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( + crossAxisCount: 3, + mainAxisSpacing: 8, + crossAxisSpacing: 8, + ), + delegate: SliverChildBuilderDelegate( + (context, index) => AlbumTile( + album: results[index], ), - slivers: [ - SliverPadding( - padding: const EdgeInsets.all(8), - sliver: SliverGrid( - gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( - crossAxisCount: 3, - mainAxisSpacing: 8, - crossAxisSpacing: 8, - ), - delegate: SliverChildBuilderDelegate( - (context, index) => AlbumTile( - album: listState.results[index], - ), - childCount: listState.results.length, - ), - ), - ), - if (listState.isLoadingMore) - const SliverPadding( - padding: EdgeInsets.all(8), - sliver: SliverList( - delegate: SliverChildListDelegate.fixed([ - Center( - child: CircularProgressIndicator(), - ) - ]), - ), - ), - ], - )); + childCount: results.length, + ), + ), + ); } } diff --git a/lib/ui/screens/members_screen.dart b/lib/ui/screens/members_screen.dart index db8d1dbb6..af62b1f19 100644 --- a/lib/ui/screens/members_screen.dart +++ b/lib/ui/screens/members_screen.dart @@ -42,32 +42,6 @@ class MembersScreen extends StatelessWidget { } } -class _MembersGrid extends StatelessWidget { - const _MembersGrid(this.results); - - final List results; - - @override - Widget build(BuildContext context) { - return SliverPadding( - padding: const EdgeInsets.fromLTRB(8, 8, 8, 0), - sliver: SliverGrid( - gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( - crossAxisCount: 3, - mainAxisSpacing: 8, - crossAxisSpacing: 8, - ), - delegate: SliverChildBuilderDelegate( - (context, index) => MemberTile( - member: results[index], - ), - childCount: results.length, - ), - ), - ); - } -} - class MembersSearchDelegate extends SearchDelegate { final MemberListCubit _cubit; @@ -130,3 +104,29 @@ class MembersSearchDelegate extends SearchDelegate { ); } } + +class _MembersGrid extends StatelessWidget { + const _MembersGrid(this.results); + + final List results; + + @override + Widget build(BuildContext context) { + return SliverPadding( + padding: const EdgeInsets.fromLTRB(8, 8, 8, 0), + sliver: SliverGrid( + gridDelegate: const SliverGridDelegateWithFixedCrossAxisCount( + crossAxisCount: 3, + mainAxisSpacing: 8, + crossAxisSpacing: 8, + ), + delegate: SliverChildBuilderDelegate( + (context, index) => MemberTile( + member: results[index], + ), + childCount: results.length, + ), + ), + ); + } +} From fc3e2cce17234c60c7493e575aa6e6ba89bbde41 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Thu, 26 Jan 2023 21:50:10 +0100 Subject: [PATCH 06/12] Extract some EventScreen parts to widgets --- lib/ui/screens/event_screen.dart | 425 +++++++++++++++++-------------- 1 file changed, 229 insertions(+), 196 deletions(-) diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index 535e17470..9d269c43e 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -59,40 +59,6 @@ class _EventScreenState extends State { super.dispose(); } - Widget _makeMap(Event event) { - return Stack( - fit: StackFit.loose, - children: [ - CachedImage( - imageUrl: event.mapsUrl, - placeholder: 'assets/img/map_placeholder.png', - fit: BoxFit.cover, - ), - Positioned.fill( - child: Material( - color: Colors.transparent, - child: InkWell( - onTap: () { - Uri url = Theme.of(context).platform == TargetPlatform.iOS - ? Uri( - scheme: 'maps', - queryParameters: {'daddr': event.location}, - ) - : Uri( - scheme: 'https', - host: 'maps.google.com', - path: 'maps', - queryParameters: {'daddr': event.location}, - ); - launchUrl(url, mode: LaunchMode.externalNonBrowserApplication); - }, - ), - ), - ), - ], - ); - } - /// Create all info of an event until the description, including buttons. Widget _makeEventInfo(Event event) { return Padding( @@ -107,7 +73,7 @@ class _EventScreenState extends State { _makeOptionalRegistrationInfo(event) else _makeNoRegistrationInfo(event), - if (event.hasFoodEvent) _makeFoodButton(event), + if (event.hasFoodEvent) _FoodButton(event), ], ), ); @@ -715,17 +681,6 @@ class _EventScreenState extends State { ); } - Widget _makeFoodButton(Event event) { - return SizedBox( - width: double.infinity, - child: ElevatedButton.icon( - onPressed: () => context.pushNamed('food', extra: event), - icon: const Icon(Icons.local_pizza), - label: const Text('ORDER FOOD'), - ), - ); - } - TextSpan _makeTermsAndConditions(Event event) { final url = config.termsAndConditionsUrl; return TextSpan( @@ -757,40 +712,6 @@ class _EventScreenState extends State { ); } - Widget _makeDescription(Event event) { - return Padding( - padding: const EdgeInsets.symmetric( - horizontal: 16, - vertical: 4, - ), - child: HtmlWidget( - event.description, - onTapUrl: (String url) async { - Uri uri = Uri.parse(url); - if (uri.scheme.isEmpty) uri = uri.replace(scheme: 'https'); - if (isDeepLink(uri)) { - context.go(Uri( - path: uri.path, - query: uri.query, - ).toString()); - return true; - } else { - final messenger = ScaffoldMessenger.of(context); - try { - await launchUrl(uri, mode: LaunchMode.externalApplication); - } catch (_) { - messenger.showSnackBar(SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not open "$url".'), - )); - } - } - return true; - }, - ), - ); - } - SliverPadding _makeRegistrationsHeader(RegistrationsState state) { return SliverPadding( padding: const EdgeInsets.only(left: 16), @@ -846,30 +767,219 @@ class _EventScreenState extends State { } } - Widget _makeShareEventButton(int pk) { - return IconButton( - padding: const EdgeInsets.all(16), - color: Theme.of(context).primaryIconTheme.color, - icon: Icon( - Theme.of(context).platform == TargetPlatform.iOS - ? Icons.ios_share - : Icons.share, + @override + Widget build(BuildContext context) { + return BlocProvider.value( + value: _eventCubit, + child: BlocBuilder( + builder: (context, state) { + if (state is ErrorState) { + return Scaffold( + appBar: ThaliaAppBar( + title: Text(widget.event?.title.toUpperCase() ?? 'EVENT'), + actions: [_ShareEventButton(widget.pk)], + ), + body: RefreshIndicator( + onRefresh: () async { + // Await only the event info. + _registrationsCubit.load(); + await _eventCubit.load(); + }, + child: ErrorScrollView(state.message!), + ), + ); + } else if (state is LoadingState && + state is! ResultState && + widget.event == null) { + return Scaffold( + appBar: ThaliaAppBar( + title: const Text('EVENT'), + actions: [_ShareEventButton(widget.pk)], + ), + body: const Center(child: CircularProgressIndicator()), + ); + } else { + final event = (state.result ?? widget.event)!; + return Scaffold( + appBar: ThaliaAppBar( + title: Text(event.title.toUpperCase()), + actions: [ + _CalendarExportButton(event), + _ShareEventButton(widget.pk), + if (event.userPermissions.manageEvent) + IconButton( + padding: const EdgeInsets.all(16), + icon: const Icon(Icons.settings), + onPressed: () => context.pushNamed( + 'event-admin', + params: {'eventPk': event.pk.toString()}, + ), + ), + ], + ), + body: RefreshIndicator( + onRefresh: () async { + // Await only the event info. + _registrationsCubit.load(); + await _eventCubit.load(); + }, + child: BlocBuilder( + bloc: _registrationsCubit, + builder: (context, listState) { + return Scrollbar( + controller: _controller, + child: CustomScrollView( + controller: _controller, + key: const PageStorageKey('event'), + slivers: [ + SliverToBoxAdapter( + child: Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + _EventMap(event), + const Divider(height: 0), + _makeEventInfo(event), + const Divider(), + _EventDescription(event), + ], + ), + ), + if (event.registrationIsOptional || + event.registrationIsRequired) ...[ + const SliverToBoxAdapter(child: Divider()), + _makeRegistrationsHeader(listState), + _makeRegistrations(listState), + if (listState.isLoadingMore) + const SliverPadding( + padding: EdgeInsets.all(8), + sliver: SliverList( + delegate: SliverChildListDelegate.fixed([ + Center(child: CircularProgressIndicator()), + ]), + ), + ), + ], + ], + ), + ); + }, + ), + ), + ); + } + }, ), - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - try { - await Share.share('https://${config.apiHost}/events/$pk/'); - } catch (_) { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not share the event.'), - )); - } - }, ); } +} + +class _EventMap extends StatelessWidget { + const _EventMap(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return Stack( + fit: StackFit.loose, + children: [ + CachedImage( + imageUrl: event.mapsUrl, + placeholder: 'assets/img/map_placeholder.png', + fit: BoxFit.cover, + ), + Positioned.fill( + child: Material( + color: Colors.transparent, + child: InkWell( + onTap: () { + Uri url = Theme.of(context).platform == TargetPlatform.iOS + ? Uri( + scheme: 'maps', + queryParameters: {'daddr': event.location}, + ) + : Uri( + scheme: 'https', + host: 'maps.google.com', + path: 'maps', + queryParameters: {'daddr': event.location}, + ); + launchUrl(url, mode: LaunchMode.externalNonBrowserApplication); + }, + ), + ), + ), + ], + ); + } +} + +class _FoodButton extends StatelessWidget { + const _FoodButton(this.event); - Widget _makeCalendarExportButton(Event event) { + final Event event; + + @override + Widget build(BuildContext context) { + return SizedBox( + width: double.infinity, + child: ElevatedButton.icon( + onPressed: () => context.pushNamed('food', extra: event), + icon: const Icon(Icons.local_pizza), + label: const Text('ORDER FOOD'), + ), + ); + } +} + +class _EventDescription extends StatelessWidget { + const _EventDescription(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return Padding( + padding: const EdgeInsets.symmetric( + horizontal: 16, + vertical: 4, + ), + child: HtmlWidget( + event.description, + onTapUrl: (String url) async { + Uri uri = Uri.parse(url); + if (uri.scheme.isEmpty) uri = uri.replace(scheme: 'https'); + if (isDeepLink(uri)) { + context.go(Uri( + path: uri.path, + query: uri.query, + ).toString()); + return true; + } else { + final messenger = ScaffoldMessenger.of(context); + try { + await launchUrl(uri, mode: LaunchMode.externalApplication); + } catch (_) { + messenger.showSnackBar(SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not open "$url".'), + )); + } + } + return true; + }, + ), + ); + } +} + +class _CalendarExportButton extends StatelessWidget { + const _CalendarExportButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { return IconButton( padding: const EdgeInsets.all(16), color: Theme.of(context).primaryIconTheme.color, @@ -885,105 +995,28 @@ class _EventScreenState extends State { }, ); } +} + +class _ShareEventButton extends StatelessWidget { + const _ShareEventButton(this.pk); + + final int pk; @override Widget build(BuildContext context) { - return BlocBuilder( - bloc: _eventCubit, - builder: (context, state) { - if (state is ErrorState) { - return Scaffold( - appBar: ThaliaAppBar( - title: Text(widget.event?.title.toUpperCase() ?? 'EVENT'), - actions: [_makeShareEventButton(widget.pk)], - ), - body: RefreshIndicator( - onRefresh: () async { - // Await only the event info. - _registrationsCubit.load(); - await _eventCubit.load(); - }, - child: ErrorScrollView(state.message!), - ), - ); - } else if (state is LoadingState && - state is! ResultState && - widget.event == null) { - return Scaffold( - appBar: ThaliaAppBar( - title: const Text('EVENT'), - actions: [_makeShareEventButton(widget.pk)], - ), - body: const Center(child: CircularProgressIndicator()), - ); - } else { - final event = (state.result ?? widget.event)!; - return Scaffold( - appBar: ThaliaAppBar( - title: Text(event.title.toUpperCase()), - actions: [ - _makeCalendarExportButton(event), - _makeShareEventButton(widget.pk), - if (event.userPermissions.manageEvent) - IconButton( - padding: const EdgeInsets.all(16), - icon: const Icon(Icons.settings), - onPressed: () => context.pushNamed( - 'event-admin', - params: {'eventPk': event.pk.toString()}, - ), - ), - ], - ), - body: RefreshIndicator( - onRefresh: () async { - // Await only the event info. - _registrationsCubit.load(); - await _eventCubit.load(); - }, - child: BlocBuilder( - bloc: _registrationsCubit, - builder: (context, listState) { - return Scrollbar( - controller: _controller, - child: CustomScrollView( - controller: _controller, - key: const PageStorageKey('event'), - slivers: [ - SliverToBoxAdapter( - child: Column( - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - _makeMap(event), - const Divider(height: 0), - _makeEventInfo(event), - const Divider(), - _makeDescription(event), - ], - ), - ), - if (event.registrationIsOptional || - event.registrationIsRequired) ...[ - const SliverToBoxAdapter(child: Divider()), - _makeRegistrationsHeader(listState), - _makeRegistrations(listState), - if (listState.isLoadingMore) - const SliverPadding( - padding: EdgeInsets.all(8), - sliver: SliverList( - delegate: SliverChildListDelegate.fixed([ - Center(child: CircularProgressIndicator()), - ]), - ), - ), - ], - ], - ), - ); - }, - ), - ), - ); + return IconButton( + padding: const EdgeInsets.all(16), + color: Theme.of(context).primaryIconTheme.color, + icon: Icon(Icons.adaptive.share), + onPressed: () async { + final messenger = ScaffoldMessenger.of(context); + try { + await Share.share('https://${config.apiHost}/events/$pk/'); + } catch (_) { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not share the event.'), + )); } }, ); From d4ae310142f7c80552bf88a7a0d78c7546f727a2 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Fri, 27 Jan 2023 08:11:38 +0100 Subject: [PATCH 07/12] Extract more parts --- lib/ui/screens/event_screen.dart | 490 +++++++++++++++++-------------- 1 file changed, 276 insertions(+), 214 deletions(-) diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index 9d269c43e..db07fa7d8 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -66,7 +66,7 @@ class _EventScreenState extends State { child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ - _makeBasicEventInfo(event), + _BasicEventInfo(event), if (event.registrationIsRequired) _makeRequiredRegistrationInfo(event) else if (event.registrationIsOptional) @@ -79,93 +79,6 @@ class _EventScreenState extends State { ); } - /// Create the title, start, end, location and price of an event. - Widget _makeBasicEventInfo(Event event) { - final textTheme = Theme.of(context).textTheme; - return Column( - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - const SizedBox(height: 8), - Text( - event.title.toUpperCase(), - style: textTheme.headline6, - ), - const Divider(height: 24), - Row( - mainAxisSize: MainAxisSize.max, - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Flexible( - fit: FlexFit.tight, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text('FROM', style: textTheme.caption), - const SizedBox(height: 4), - Text( - dateTimeFormatter.format(event.start.toLocal()), - style: textTheme.subtitle2, - ), - ], - ), - ), - const SizedBox(width: 8), - Flexible( - fit: FlexFit.tight, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text('UNTIL', style: textTheme.caption), - const SizedBox(height: 4), - Text( - dateTimeFormatter.format(event.end.toLocal()), - style: textTheme.subtitle2, - ), - ], - ), - ) - ], - ), - const SizedBox(height: 12), - Row( - mainAxisSize: MainAxisSize.max, - children: [ - Flexible( - fit: FlexFit.tight, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text('LOCATION', style: textTheme.caption), - const SizedBox(height: 4), - Text( - event.location, - style: textTheme.subtitle2, - ), - ], - ), - ), - const SizedBox(width: 8), - Flexible( - fit: FlexFit.tight, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text('PRICE', style: textTheme.caption), - const SizedBox(height: 4), - Text( - '€${event.price}', - style: textTheme.subtitle2, - ), - ], - ), - ) - ], - ), - const Divider(height: 24), - ], - ); - } - // Create the info for events with required registration. Widget _makeRequiredRegistrationInfo(Event event) { assert(event.registrationIsRequired); @@ -192,19 +105,23 @@ class _EventScreenState extends State { textSpans.add(TextSpan( text: event.cancelTooLateMessage, )); - final text = 'The deadline has passed, are you sure you want ' - 'to cancel your registration and pay the estimated full costs of ' - '€${event.fine}? You will not be able to undo this!'; - registrationButton = _makeCancelRegistrationButton(event, text); + registrationButton = _CancelRegistrationButton( + event: event, + warningText: 'The deadline has passed, are you sure you want ' + 'to cancel your registration and pay the estimated full costs of ' + '€${event.fine}? You will not be able to undo this!', + ); } else { // Cancel button. - const text = 'Are you sure you want to cancel your registration?'; - registrationButton = _makeCancelRegistrationButton(event, text); + registrationButton = _CancelRegistrationButton( + event: event, + warningText: 'Are you sure you want to cancel your registration?', + ); } } if (event.canUpdateRegistration) { - updateButton = _makeUpdateButton(event); + updateButton = _UpdateRegistrationButton(event); } if (event.canCreateRegistration || !event.isRegistered) { @@ -402,7 +319,7 @@ class _EventScreenState extends State { final textSpans = []; Widget registrationButton = const SizedBox.shrink(); if (event.canCancelRegistration) { - registrationButton = _makeIWontBeThereButton(event); + registrationButton = _IWontBeThereButton(event); } if (event.isInvited) { @@ -413,7 +330,7 @@ class _EventScreenState extends State { 'can still register to give an indication of who will be there, as ' 'well as mark the event as "registered" in your calendar. ', )); - registrationButton = _makeIllBeThereButton(event); + registrationButton = _IllBeThereButton(event); } if (event.noRegistrationMessage?.isNotEmpty ?? false) { @@ -425,7 +342,7 @@ class _EventScreenState extends State { Widget updateButton = const SizedBox.shrink(); if (event.canUpdateRegistration) { - updateButton = _makeUpdateButton(event); + updateButton = _UpdateRegistrationButton(event); } return Column( @@ -468,50 +385,6 @@ class _EventScreenState extends State { ); } - Widget _makeIllBeThereButton(Event event) { - return ElevatedButton.icon( - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - try { - final calendarCubit = BlocProvider.of(context); - await _eventCubit.register(); - await _registrationsCubit.load(); - calendarCubit.load(); - } on ApiException { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not register for the event.'), - )); - } - }, - icon: const Icon(Icons.check), - label: const Text("I'LL BE THERE"), - ); - } - - Widget _makeIWontBeThereButton(Event event) { - return ElevatedButton.icon( - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - try { - final calendarCubit = BlocProvider.of(context); - await _eventCubit.cancelRegistration( - registrationPk: event.registration!.pk, - ); - await _registrationsCubit.load(); - calendarCubit.load(); - } on ApiException { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not cancel your registration.'), - )); - } - }, - icon: const Icon(Icons.clear), - label: const Text("I WON'T BE THERE"), - ); - } - Widget _makeCreateRegistrationButton(Event event) { return ElevatedButton.icon( onPressed: () async { @@ -609,78 +482,6 @@ class _EventScreenState extends State { ); } - Widget _makeCancelRegistrationButton(Event event, String warningText) { - return ElevatedButton.icon( - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - final calendarCubit = BlocProvider.of(context); - final welcomeCubit = BlocProvider.of(context); - final confirmed = await showDialog( - context: context, - builder: (context) { - return AlertDialog( - title: const Text('Cancel registration'), - content: Text( - warningText, - style: Theme.of(context).textTheme.bodyText2, - ), - actions: [ - TextButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(false), - icon: const Icon(Icons.clear), - label: const Text('NO'), - ), - ElevatedButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(true), - icon: const Icon(Icons.check), - label: const Text('YES'), - ), - ], - ); - }, - ); - - if (confirmed ?? false) { - try { - await _eventCubit.cancelRegistration( - registrationPk: event.registration!.pk, - ); - } on ApiException { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not cancel your registration.'), - )); - } - } - await _registrationsCubit.load(); - calendarCubit.load(); - await welcomeCubit.load(); - }, - icon: const Icon(Icons.delete_forever_outlined), - label: const Text('CANCEL REGISTRATION'), - ); - } - - Widget _makeUpdateButton(Event event) { - return ElevatedButton.icon( - onPressed: () => context.pushNamed( - 'event-registration', - params: { - 'eventPk': event.pk.toString(), - 'registrationPk': event.registration!.pk.toString(), - }, - ), - icon: const Icon(Icons.build), - label: const Text('UPDATE REGISTRATION'), - ); - } - TextSpan _makeTermsAndConditions(Event event) { final url = config.termsAndConditionsUrl; return TextSpan( @@ -873,6 +674,263 @@ class _EventScreenState extends State { } } +class _IWontBeThereButton extends StatelessWidget { + const _IWontBeThereButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () async { + final messenger = ScaffoldMessenger.of(context); + final calendarCubit = BlocProvider.of(context); + final eventCubit = BlocProvider.of(context); + final registrationsCubit = BlocProvider.of(context); + + try { + await eventCubit.cancelRegistration( + registrationPk: event.registration!.pk, + ); + registrationsCubit.load(); + calendarCubit.load(); + } on ApiException { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not cancel your registration.'), + )); + } + }, + icon: const Icon(Icons.clear), + label: const Text("I WON'T BE THERE"), + ); + } +} + +class _IllBeThereButton extends StatelessWidget { + const _IllBeThereButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () async { + final messenger = ScaffoldMessenger.of(context); + final calendarCubit = BlocProvider.of(context); + final eventCubit = BlocProvider.of(context); + final registrationsCubit = BlocProvider.of(context); + + try { + await eventCubit.register(); + registrationsCubit.load(); + calendarCubit.load(); + } on ApiException { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not register for the event.'), + )); + } + }, + icon: const Icon(Icons.check), + label: const Text("I'LL BE THERE"), + ); + } +} + +/// A button that allows the user to cancel their registration for an event. +class _CancelRegistrationButton extends StatelessWidget { + const _CancelRegistrationButton({ + Key? key, + required this.event, + required this.warningText, + }) : super(key: key); + + final Event event; + final String warningText; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () async { + final messenger = ScaffoldMessenger.of(context); + final calendarCubit = BlocProvider.of(context); + final welcomeCubit = BlocProvider.of(context); + final eventCubit = BlocProvider.of(context); + final registrationsCubit = BlocProvider.of(context); + + final confirmed = await showDialog( + context: context, + builder: (context) { + return AlertDialog( + title: const Text('Cancel registration'), + content: Text( + warningText, + style: Theme.of(context).textTheme.bodyText2, + ), + actions: [ + TextButton.icon( + onPressed: () => Navigator.of( + context, + rootNavigator: true, + ).pop(false), + icon: const Icon(Icons.clear), + label: const Text('NO'), + ), + ElevatedButton.icon( + onPressed: () => Navigator.of( + context, + rootNavigator: true, + ).pop(true), + icon: const Icon(Icons.check), + label: const Text('YES'), + ), + ], + ); + }, + ); + + if (confirmed ?? false) { + try { + await eventCubit.cancelRegistration( + registrationPk: event.registration!.pk, + ); + } on ApiException { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not cancel your registration.'), + )); + } + } + registrationsCubit.load(); + calendarCubit.load(); + welcomeCubit.load(); + }, + icon: const Icon(Icons.delete_forever_outlined), + label: const Text('CANCEL REGISTRATION'), + ); + } +} + +/// A button that opens the registration fields page. +class _UpdateRegistrationButton extends StatelessWidget { + const _UpdateRegistrationButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () => context.pushNamed( + 'event-registration', + params: { + 'eventPk': event.pk.toString(), + 'registrationPk': event.registration!.pk.toString(), + }, + ), + icon: const Icon(Icons.build), + label: const Text('UPDATE REGISTRATION'), + ); + } +} + +/// The basic event info: title, time, location, price. +class _BasicEventInfo extends StatelessWidget { + static final dateTimeFormatter = DateFormat('E d MMM y, HH:mm'); + + const _BasicEventInfo(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + final textTheme = Theme.of(context).textTheme; + return Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + const SizedBox(height: 8), + Text( + event.title.toUpperCase(), + style: textTheme.headline6, + ), + const Divider(height: 24), + Row( + mainAxisSize: MainAxisSize.max, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Flexible( + fit: FlexFit.tight, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text('FROM', style: textTheme.caption), + const SizedBox(height: 4), + Text( + dateTimeFormatter.format(event.start.toLocal()), + style: textTheme.subtitle2, + ), + ], + ), + ), + const SizedBox(width: 8), + Flexible( + fit: FlexFit.tight, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text('UNTIL', style: textTheme.caption), + const SizedBox(height: 4), + Text( + dateTimeFormatter.format(event.end.toLocal()), + style: textTheme.subtitle2, + ), + ], + ), + ) + ], + ), + const SizedBox(height: 12), + Row( + mainAxisSize: MainAxisSize.max, + children: [ + Flexible( + fit: FlexFit.tight, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text('LOCATION', style: textTheme.caption), + const SizedBox(height: 4), + Text( + event.location, + style: textTheme.subtitle2, + ), + ], + ), + ), + const SizedBox(width: 8), + Flexible( + fit: FlexFit.tight, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text('PRICE', style: textTheme.caption), + const SizedBox(height: 4), + Text( + '€${event.price}', + style: textTheme.subtitle2, + ), + ], + ), + ) + ], + ), + const Divider(height: 24), + ], + ); + } +} + +/// A map of the event location, that opens a maps app when tapped. class _EventMap extends StatelessWidget { const _EventMap(this.event); @@ -914,6 +972,7 @@ class _EventMap extends StatelessWidget { } } +/// A button that opens the food ordering page. class _FoodButton extends StatelessWidget { const _FoodButton(this.event); @@ -932,6 +991,7 @@ class _FoodButton extends StatelessWidget { } } +/// A widget that displays the event's description HTML. class _EventDescription extends StatelessWidget { const _EventDescription(this.event); @@ -973,6 +1033,7 @@ class _EventDescription extends StatelessWidget { } } +/// A button that creates a calendar entry for the event. class _CalendarExportButton extends StatelessWidget { const _CalendarExportButton(this.event); @@ -997,6 +1058,7 @@ class _CalendarExportButton extends StatelessWidget { } } +/// A button that shares the event's URL. class _ShareEventButton extends StatelessWidget { const _ShareEventButton(this.pk); From b1b4787053534a4f2932bff276012f5a231eb0d3 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Tue, 31 Jan 2023 10:50:09 +0100 Subject: [PATCH 08/12] Refactor more --- lib/ui/screens/event_screen.dart | 263 ++++++++++++++++--------------- 1 file changed, 140 insertions(+), 123 deletions(-) diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index db07fa7d8..04ec27383 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -95,9 +95,9 @@ class _EventScreenState extends State { if (event.canCreateRegistration) { if (event.reachedMaxParticipants) { - registrationButton = _makeJoinQueueButton(event); + registrationButton = _JoinQueueButton(event); } else { - registrationButton = _makeCreateRegistrationButton(event); + registrationButton = _CreateRegistrationButton(event); } } else if (event.canCancelRegistration) { if (event.cancelDeadlinePassed()) { @@ -385,103 +385,6 @@ class _EventScreenState extends State { ); } - Widget _makeCreateRegistrationButton(Event event) { - return ElevatedButton.icon( - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - final calendarCubit = BlocProvider.of(context); - final router = GoRouter.of(context); - var confirmed = !event.cancelDeadlinePassed(); - if (!confirmed) { - confirmed = await showDialog( - context: context, - builder: (context) { - return AlertDialog( - title: const Text('Register'), - content: Text( - 'Are you sure you want to register? The ' - 'cancellation deadline has already passed.', - style: Theme.of(context).textTheme.bodyText2, - ), - actions: [ - TextButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(false), - icon: const Icon(Icons.clear), - label: const Text('NO'), - ), - ElevatedButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(true), - icon: const Icon(Icons.check), - label: const Text('YES'), - ), - ], - ); - }, - ) ?? - false; - } - - if (confirmed) { - try { - final registration = await _eventCubit.register(); - if (event.hasFields) { - router.pushNamed('event-registration', params: { - 'eventPk': event.pk.toString(), - 'registrationPk': registration.pk.toString(), - }); - } - calendarCubit.load(); - } on ApiException { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not register for the event.'), - )); - } - await _registrationsCubit.load(); - } - }, - icon: const Icon(Icons.create_outlined), - label: const Text('REGISTER'), - ); - } - - Widget _makeJoinQueueButton(Event event) { - return ElevatedButton.icon( - onPressed: () async { - final messenger = ScaffoldMessenger.of(context); - final calendarCubit = BlocProvider.of(context); - final router = GoRouter.of(context); - try { - final registration = await _eventCubit.register(); - if (event.hasFields) { - router.pushNamed( - 'event-registration', - params: { - 'eventPk': event.pk.toString(), - 'registrationPk': registration.pk.toString(), - }, - ); - } - calendarCubit.load(); - } on ApiException { - messenger.showSnackBar(const SnackBar( - behavior: SnackBarBehavior.floating, - content: Text('Could not join the waiting list for the event.'), - )); - } - await _registrationsCubit.load(); - }, - icon: const Icon(Icons.create_outlined), - label: const Text('JOIN QUEUE'), - ); - } - TextSpan _makeTermsAndConditions(Event event) { final url = config.termsAndConditionsUrl; return TextSpan( @@ -674,6 +577,100 @@ class _EventScreenState extends State { } } +class _JoinQueueButton extends StatelessWidget { + const _JoinQueueButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () async { + final router = GoRouter.of(context); + final messenger = ScaffoldMessenger.of(context); + final calendarCubit = BlocProvider.of(context); + final eventCubit = BlocProvider.of(context); + final registrationsCubit = BlocProvider.of(context); + + try { + final registration = await eventCubit.register(); + if (event.hasFields) { + router.pushNamed( + 'event-registration', + params: { + 'eventPk': event.pk.toString(), + 'registrationPk': registration.pk.toString(), + }, + ); + } + calendarCubit.load(); + } on ApiException { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not join the waiting list for the event.'), + )); + } + registrationsCubit.load(); + }, + icon: const Icon(Icons.create_outlined), + label: const Text('JOIN QUEUE'), + ); + } +} + +class _CreateRegistrationButton extends StatelessWidget { + const _CreateRegistrationButton(this.event); + + final Event event; + + @override + Widget build(BuildContext context) { + return ElevatedButton.icon( + onPressed: () async { + final router = GoRouter.of(context); + final messenger = ScaffoldMessenger.of(context); + final calendarCubit = BlocProvider.of(context); + final eventCubit = BlocProvider.of(context); + final registrationsCubit = BlocProvider.of(context); + + var confirmed = !event.cancelDeadlinePassed(); + if (!confirmed) { + confirmed = await showDialog( + context: context, + builder: (context) => const _ConfirmationDialog( + titleText: 'Register', + warningText: 'Are you sure you want to register? ' + 'The cancellation deadline has already passed.', + ), + ) ?? + false; + } + + if (confirmed) { + try { + final registration = await eventCubit.register(); + if (event.hasFields) { + router.pushNamed('event-registration', params: { + 'eventPk': event.pk.toString(), + 'registrationPk': registration.pk.toString(), + }); + } + calendarCubit.load(); + } on ApiException { + messenger.showSnackBar(const SnackBar( + behavior: SnackBarBehavior.floating, + content: Text('Could not register for the event.'), + )); + } + registrationsCubit.load(); + } + }, + icon: const Icon(Icons.create_outlined), + label: const Text('REGISTER'), + ); + } +} + class _IWontBeThereButton extends StatelessWidget { const _IWontBeThereButton(this.event); @@ -762,30 +759,9 @@ class _CancelRegistrationButton extends StatelessWidget { final confirmed = await showDialog( context: context, builder: (context) { - return AlertDialog( - title: const Text('Cancel registration'), - content: Text( - warningText, - style: Theme.of(context).textTheme.bodyText2, - ), - actions: [ - TextButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(false), - icon: const Icon(Icons.clear), - label: const Text('NO'), - ), - ElevatedButton.icon( - onPressed: () => Navigator.of( - context, - rootNavigator: true, - ).pop(true), - icon: const Icon(Icons.check), - label: const Text('YES'), - ), - ], + return _ConfirmationDialog( + titleText: 'Cancel registration', + warningText: warningText, ); }, ); @@ -812,6 +788,47 @@ class _CancelRegistrationButton extends StatelessWidget { } } +/// A dialog that shows a message with buttons 'YES' and 'NO', popping with a bool. +class _ConfirmationDialog extends StatelessWidget { + const _ConfirmationDialog({ + Key? key, + required this.titleText, + required this.warningText, + }) : super(key: key); + + final String titleText; + final String warningText; + + @override + Widget build(BuildContext context) { + return AlertDialog( + title: Text(titleText), + content: Text( + warningText, + style: Theme.of(context).textTheme.bodyText2, + ), + actions: [ + TextButton.icon( + onPressed: () => Navigator.of( + context, + rootNavigator: true, + ).pop(false), + icon: const Icon(Icons.clear), + label: const Text('NO'), + ), + ElevatedButton.icon( + onPressed: () => Navigator.of( + context, + rootNavigator: true, + ).pop(true), + icon: const Icon(Icons.check), + label: const Text('YES'), + ), + ], + ); + } +} + /// A button that opens the registration fields page. class _UpdateRegistrationButton extends StatelessWidget { const _UpdateRegistrationButton(this.event); From 54d62b3ad60dea7998ff1fb009cb8f96910ccda5 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Tue, 31 Jan 2023 11:13:32 +0100 Subject: [PATCH 09/12] Use new ListState for eventregistrations --- lib/blocs/registrations_cubit.dart | 41 +++++++++++------------- lib/ui/screens/event_screen.dart | 51 ++++++++++++++---------------- 2 files changed, 42 insertions(+), 50 deletions(-) diff --git a/lib/blocs/registrations_cubit.dart b/lib/blocs/registrations_cubit.dart index 314198d80..928c52220 100644 --- a/lib/blocs/registrations_cubit.dart +++ b/lib/blocs/registrations_cubit.dart @@ -1,26 +1,21 @@ -import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:reaxit/api/api_repository.dart'; import 'package:reaxit/api/exceptions.dart'; import 'package:reaxit/blocs.dart'; import 'package:reaxit/models.dart'; +import 'package:reaxit/ui/widgets.dart'; -typedef RegistrationsState = ListState; - -class RegistrationsCubit extends Cubit { +class RegistrationsCubit extends PaginatedCubit { final ApiRepository api; final int eventPk; - static const int firstPageSize = 60; - static const int pageSize = 30; - /// The offset to be used for the next paginated request. int _nextOffset = 0; RegistrationsCubit(this.api, {required this.eventPk}) - : super(const RegistrationsState.loading(results: [])); + : super(firstPageSize: 60, pageSize: 30); + @override Future load() async { - emit(state.copyWith(isLoading: true)); try { final listResponse = await api.getEventRegistrations( pk: eventPk, limit: firstPageSize, offset: 0); @@ -30,27 +25,27 @@ class RegistrationsCubit extends Cubit { _nextOffset = firstPageSize; if (listResponse.results.isNotEmpty) { - emit(RegistrationsState.success( - results: listResponse.results, isDone: isDone)); + emit(ResultsListState.withDone(listResponse.results, isDone)); } else { - emit(const RegistrationsState.failure( - message: 'There are no registrations yet.', - )); + emit(const ErrorListState('There are no registrations yet.')); } } on ApiException catch (exception) { - emit(RegistrationsState.failure( - message: exception.getMessage(notFound: 'The event does not exist.'), + emit(ErrorListState( + exception.getMessage(notFound: 'The event does not exist.'), )); } } + @override Future more() async { - final oldState = state; - - if (oldState.isDone || oldState.isLoading || oldState.isLoadingMore) return; + // Ignore calls to `more()` if there is no data, or already more coming. + if (state is! ResultsListState || + state is LoadingMoreListState || + state is DoneListState) return; - emit(oldState.copyWith(isLoadingMore: true)); + final oldState = state as ResultsListState; + emit(LoadingMoreListState.from(oldState)); try { var listResponse = await api.getEventRegistrations( pk: eventPk, @@ -63,10 +58,10 @@ class RegistrationsCubit extends Cubit { _nextOffset += pageSize; - emit(RegistrationsState.success(results: registrations, isDone: isDone)); + emit(ResultsListState.withDone(registrations, isDone)); } on ApiException catch (exception) { - emit(RegistrationsState.failure( - message: exception.getMessage(notFound: 'The event does not exist.'), + emit(ErrorListState( + exception.getMessage(notFound: 'The event does not exist.'), )); } } diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index 04ec27383..4ea7ad8e8 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -46,7 +46,8 @@ class _EventScreenState extends State { if (_controller.position.pixels >= _controller.position.maxScrollExtent - 300) { // Only request loading more if that's not already happening. - if (!_registrationsCubit.state.isLoadingMore) { + if (_registrationsCubit.state is ResultsListState && + _registrationsCubit.state is! LoadingMoreListState) { _registrationsCubit.more(); } } @@ -416,7 +417,7 @@ class _EventScreenState extends State { ); } - SliverPadding _makeRegistrationsHeader(RegistrationsState state) { + SliverPadding _makeRegistrationsHeader() { return SliverPadding( padding: const EdgeInsets.only(left: 16), sliver: SliverToBoxAdapter( @@ -428,22 +429,21 @@ class _EventScreenState extends State { ); } - SliverPadding _makeRegistrations(RegistrationsState state) { - if (state.isLoading && state.results.isEmpty) { + SliverPadding _makeRegistrations(XListState state) { + if (state is ErrorListState) { + return SliverPadding( + padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 16), + sliver: SliverToBoxAdapter(child: Text(state.message!)), + ); + } else if (state is LoadingListState) { return const SliverPadding( padding: EdgeInsets.all(16), sliver: SliverToBoxAdapter( - child: Center( - child: CircularProgressIndicator(), - ), + child: Center(child: CircularProgressIndicator()), ), ); - } else if (state.hasException) { - return SliverPadding( - padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 16), - sliver: SliverToBoxAdapter(child: Text(state.message!)), - ); } else { + final results = state.results; return SliverPadding( padding: const EdgeInsets.only(left: 16, right: 16, top: 8, bottom: 16), sliver: SliverGrid( @@ -453,18 +453,10 @@ class _EventScreenState extends State { crossAxisSpacing: 8, ), delegate: SliverChildBuilderDelegate( - (context, index) { - if (state.results[index].member != null) { - return MemberTile( - member: state.results[index].member!, - ); - } else { - return DefaultMemberTile( - name: state.results[index].name!, - ); - } - }, - childCount: state.results.length, + childCount: results.length, + (context, index) => results[index].member != null + ? MemberTile(member: results[index].member!) + : DefaultMemberTile(name: results[index].name!), ), ), ); @@ -527,7 +519,8 @@ class _EventScreenState extends State { _registrationsCubit.load(); await _eventCubit.load(); }, - child: BlocBuilder( + child: BlocBuilder>( bloc: _registrationsCubit, builder: (context, listState) { return Scrollbar( @@ -551,9 +544,9 @@ class _EventScreenState extends State { if (event.registrationIsOptional || event.registrationIsRequired) ...[ const SliverToBoxAdapter(child: Divider()), - _makeRegistrationsHeader(listState), + _makeRegistrationsHeader(), _makeRegistrations(listState), - if (listState.isLoadingMore) + if (listState is LoadingMoreListState) const SliverPadding( padding: EdgeInsets.all(8), sliver: SliverList( @@ -563,6 +556,10 @@ class _EventScreenState extends State { ), ), ], + const SliverSafeArea( + minimum: EdgeInsets.only(bottom: 8), + sliver: SliverPadding(padding: EdgeInsets.zero), + ), ], ), ); From 786b50c190ead0c8c5f1445589d3d73f3104c7ce Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Tue, 31 Jan 2023 11:14:28 +0100 Subject: [PATCH 10/12] Remove old liststate It's now used only for the calendar, as that will be refactored a lot soon. --- lib/blocs/calendar_cubit.dart | 83 ++++++++++++++++++++- lib/blocs/list_state.dart | 91 ++--------------------- lib/ui/screens/event_screen.dart | 4 +- lib/ui/widgets/paginated_scroll_view.dart | 4 +- 4 files changed, 90 insertions(+), 92 deletions(-) diff --git a/lib/blocs/calendar_cubit.dart b/lib/blocs/calendar_cubit.dart index c0939e367..e66d24132 100644 --- a/lib/blocs/calendar_cubit.dart +++ b/lib/blocs/calendar_cubit.dart @@ -1,10 +1,10 @@ import 'dart:async'; +import 'package:equatable/equatable.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:intl/intl.dart'; import 'package:reaxit/api/api_repository.dart'; import 'package:reaxit/api/exceptions.dart'; -import 'package:reaxit/blocs.dart'; import 'package:reaxit/config.dart' as config; import 'package:reaxit/models.dart'; @@ -103,7 +103,86 @@ class CalendarEvent { ); } -typedef CalendarState = ListState; +/// Generic class to be used as state for paginated lists. +class CalendarState extends Equatable { + /// The results to be shown. These are outdated if `isLoading` is true. + final List results; + + /// A message describing why there are no results. + final String? message; + + /// Different results are being loaded. The results are outdated. + final bool isLoading; + + /// More of the same results are being loaded. The results are not outdated. + final bool isLoadingMore; + + /// The last results have been loaded. There are no more pages left. + final bool isDone; + + bool get hasException => message != null; + + const CalendarState({ + required this.results, + required this.message, + required this.isLoading, + required this.isLoadingMore, + required this.isDone, + }); + + CalendarState copyWith({ + List? results, + String? message, + bool? isLoading, + bool? isLoadingMore, + bool? isDone, + }) => + CalendarState( + results: results ?? this.results, + message: message ?? this.message, + isLoading: isLoading ?? this.isLoading, + isLoadingMore: isLoadingMore ?? this.isLoadingMore, + isDone: isDone ?? this.isDone, + ); + + @override + List get props => [ + results, + message, + isLoading, + isLoadingMore, + isDone, + ]; + + @override + String toString() { + return 'ListState<$CalendarEvent>(isLoading: $isLoading, isLoadingMore: $isLoadingMore,' + ' isDone: $isDone, message: $message, ${results.length} ${CalendarEvent}s)'; + } + + const CalendarState.loading({required this.results}) + : message = null, + isLoading = true, + isLoadingMore = false, + isDone = true; + + const CalendarState.loadingMore({required this.results}) + : message = null, + isLoading = false, + isLoadingMore = true, + isDone = true; + + const CalendarState.success({required this.results, required this.isDone}) + : message = null, + isLoading = false, + isLoadingMore = false; + + const CalendarState.failure({required String this.message}) + : results = const [], + isLoading = false, + isLoadingMore = false, + isDone = true; +} class CalendarCubit extends Cubit { static const int firstPageSize = 20; diff --git a/lib/blocs/list_state.dart b/lib/blocs/list_state.dart index e0bf5972f..05d551c5d 100644 --- a/lib/blocs/list_state.dart +++ b/lib/blocs/list_state.dart @@ -1,86 +1,5 @@ import 'package:equatable/equatable.dart'; -/// Generic class to be used as state for paginated lists. -class ListState extends Equatable { - /// The results to be shown. These are outdated if `isLoading` is true. - final List results; - - /// A message describing why there are no results. - final String? message; - - /// Different results are being loaded. The results are outdated. - final bool isLoading; - - /// More of the same results are being loaded. The results are not outdated. - final bool isLoadingMore; - - /// The last results have been loaded. There are no more pages left. - final bool isDone; - - bool get hasException => message != null; - - const ListState({ - required this.results, - required this.message, - required this.isLoading, - required this.isLoadingMore, - required this.isDone, - }); - - ListState copyWith({ - List? results, - String? message, - bool? isLoading, - bool? isLoadingMore, - bool? isDone, - }) => - ListState( - results: results ?? this.results, - message: message ?? this.message, - isLoading: isLoading ?? this.isLoading, - isLoadingMore: isLoadingMore ?? this.isLoadingMore, - isDone: isDone ?? this.isDone, - ); - - @override - List get props => [ - results, - message, - isLoading, - isLoadingMore, - isDone, - ]; - - @override - String toString() { - return 'ListState<$T>(isLoading: $isLoading, isLoadingMore: $isLoadingMore,' - ' isDone: $isDone, message: $message, ${results.length} ${T}s)'; - } - - const ListState.loading({required this.results}) - : message = null, - isLoading = true, - isLoadingMore = false, - isDone = true; - - const ListState.loadingMore({required this.results}) - : message = null, - isLoading = false, - isLoadingMore = true, - isDone = true; - - const ListState.success({required this.results, required this.isDone}) - : message = null, - isLoading = false, - isLoadingMore = false; - - const ListState.failure({required String this.message}) - : results = const [], - isLoading = false, - isLoadingMore = false, - isDone = true; -} - /// Generic type for states with a paginated list of results. /// /// There are a number of subtypes: @@ -89,8 +8,8 @@ class ListState extends Equatable { /// * [ResultsListState] - indicates that there are results. /// * [DoneListState] - indicates that there are no more results. /// * [LoadingMoreListState] - indicates that we are loading more results. -abstract class XListState extends Equatable { - const XListState(); +abstract class ListState extends Equatable { + const ListState(); /// A convenience method to get the results if they are available. /// @@ -108,11 +27,11 @@ abstract class XListState extends Equatable { List get props => []; } -class LoadingListState extends XListState { +class LoadingListState extends ListState { const LoadingListState(); } -class ErrorListState extends XListState { +class ErrorListState extends ListState { @override final String message; @@ -122,7 +41,7 @@ class ErrorListState extends XListState { List get props => [message]; } -class ResultsListState extends XListState { +class ResultsListState extends ListState { @override final List results; diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index 4ea7ad8e8..6edea2c70 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -429,7 +429,7 @@ class _EventScreenState extends State { ); } - SliverPadding _makeRegistrations(XListState state) { + SliverPadding _makeRegistrations(ListState state) { if (state is ErrorListState) { return SliverPadding( padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 16), @@ -520,7 +520,7 @@ class _EventScreenState extends State { await _eventCubit.load(); }, child: BlocBuilder>( + ListState>( bloc: _registrationsCubit, builder: (context, listState) { return Scrollbar( diff --git a/lib/ui/widgets/paginated_scroll_view.dart b/lib/ui/widgets/paginated_scroll_view.dart index be6a4d08b..1354f82ff 100644 --- a/lib/ui/widgets/paginated_scroll_view.dart +++ b/lib/ui/widgets/paginated_scroll_view.dart @@ -2,7 +2,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:reaxit/blocs/list_state.dart'; -abstract class PaginatedCubit extends Cubit> { +abstract class PaginatedCubit extends Cubit> { final int firstPageSize; final int pageSize; @@ -76,7 +76,7 @@ class _PaginatedScrollViewState> @override Widget build(BuildContext context) { - return BlocBuilder>( + return BlocBuilder>( builder: (context, state) { late final List slivers; if (state is ErrorListState) { From b6c050fa7f07743dd620ad1dc1d1f8c54d156156 Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Tue, 31 Jan 2023 11:23:22 +0100 Subject: [PATCH 11/12] Change loading indicator to match PaginatedScrollView --- lib/ui/screens/event_screen.dart | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/lib/ui/screens/event_screen.dart b/lib/ui/screens/event_screen.dart index 6edea2c70..2e7bca893 100644 --- a/lib/ui/screens/event_screen.dart +++ b/lib/ui/screens/event_screen.dart @@ -548,11 +548,10 @@ class _EventScreenState extends State { _makeRegistrations(listState), if (listState is LoadingMoreListState) const SliverPadding( - padding: EdgeInsets.all(8), - sliver: SliverList( - delegate: SliverChildListDelegate.fixed([ - Center(child: CircularProgressIndicator()), - ]), + padding: EdgeInsets.only(top: 16), + sliver: SliverToBoxAdapter( + child: Center( + child: CircularProgressIndicator()), ), ), ], From 259946e1fecd36d080a1e52dd33a545a3937593a Mon Sep 17 00:00:00 2001 From: Dirk Doesburg Date: Wed, 1 Feb 2023 19:17:02 +0100 Subject: [PATCH 12/12] Fix multiple `state` evaluations --- lib/blocs/album_list_cubit.dart | 13 +++++++------ lib/blocs/member_list_cubit.dart | 13 +++++++------ lib/blocs/registrations_cubit.dart | 13 +++++++------ 3 files changed, 21 insertions(+), 18 deletions(-) diff --git a/lib/blocs/album_list_cubit.dart b/lib/blocs/album_list_cubit.dart index 4d379a5dc..bcffcccbc 100644 --- a/lib/blocs/album_list_cubit.dart +++ b/lib/blocs/album_list_cubit.dart @@ -59,13 +59,14 @@ class AlbumListCubit extends PaginatedCubit { @override Future more() async { // Ignore calls to `more()` if there is no data, or already more coming. - if (state is! ResultsListState || - state is LoadingMoreListState || - state is DoneListState) return; + final oldState = state; + if (oldState is! ResultsListState || + oldState is LoadingMoreListState || + oldState is DoneListState) return; - final oldState = state as ResultsListState; + final resultsState = oldState as ResultsListState; - emit(LoadingMoreListState.from(oldState)); + emit(LoadingMoreListState.from(resultsState)); try { final query = _searchQuery; @@ -80,7 +81,7 @@ class AlbumListCubit extends PaginatedCubit { // changed since the request was made. if (query != _searchQuery) return; - final albums = state.results + albumsResponse.results; + final albums = resultsState.results + albumsResponse.results; final isDone = albums.length == albumsResponse.count; _nextOffset += pageSize; diff --git a/lib/blocs/member_list_cubit.dart b/lib/blocs/member_list_cubit.dart index 1ea3bccc0..596ecc333 100644 --- a/lib/blocs/member_list_cubit.dart +++ b/lib/blocs/member_list_cubit.dart @@ -59,13 +59,14 @@ class MemberListCubit extends PaginatedCubit { @override Future more() async { // Ignore calls to `more()` if there is no data, or already more coming. - if (state is! ResultsListState || - state is LoadingMoreListState || - state is DoneListState) return; + final oldState = state; + if (oldState is! ResultsListState || + oldState is LoadingMoreListState || + oldState is DoneListState) return; - final oldState = state as ResultsListState; + final resultsState = oldState as ResultsListState; - emit(LoadingMoreListState.from(oldState)); + emit(LoadingMoreListState.from(resultsState)); try { final query = _searchQuery; @@ -80,7 +81,7 @@ class MemberListCubit extends PaginatedCubit { // changed since the request was made. if (query != _searchQuery) return; - final members = oldState.results + membersResponse.results; + final members = resultsState.results + membersResponse.results; final isDone = members.length == membersResponse.count; _nextOffset += pageSize; diff --git a/lib/blocs/registrations_cubit.dart b/lib/blocs/registrations_cubit.dart index 928c52220..59e58ce91 100644 --- a/lib/blocs/registrations_cubit.dart +++ b/lib/blocs/registrations_cubit.dart @@ -39,13 +39,14 @@ class RegistrationsCubit extends PaginatedCubit { @override Future more() async { // Ignore calls to `more()` if there is no data, or already more coming. - if (state is! ResultsListState || - state is LoadingMoreListState || - state is DoneListState) return; + final oldState = state; + if (oldState is! ResultsListState || + oldState is LoadingMoreListState || + oldState is DoneListState) return; - final oldState = state as ResultsListState; + final resultsState = oldState as ResultsListState; - emit(LoadingMoreListState.from(oldState)); + emit(LoadingMoreListState.from(resultsState)); try { var listResponse = await api.getEventRegistrations( pk: eventPk, @@ -53,7 +54,7 @@ class RegistrationsCubit extends PaginatedCubit { offset: _nextOffset, ); - final registrations = state.results + listResponse.results; + final registrations = resultsState.results + listResponse.results; final isDone = registrations.length == listResponse.count; _nextOffset += pageSize;