import 'package:flutter/material.dart'; import 'package:flutter/cupertino.dart'; import 'package:flutter/widgets.dart'; import 'package:git_touch/models/theme.dart'; import 'package:git_touch/scaffolds/common.dart'; import 'package:git_touch/utils/utils.dart'; import 'package:provider/provider.dart'; import 'package:git_touch/widgets/error_reload.dart'; import 'package:git_touch/widgets/loading.dart'; import 'package:git_touch/widgets/empty.dart'; export 'package:git_touch/utils/utils.dart'; // This is a scaffold for infinite scroll screens class ListStatefulScaffold extends StatefulWidget { final Widget title; final Widget Function()? actionBuilder; final Widget Function(T payload) itemBuilder; final Future> Function(K? cursor) fetch; ListStatefulScaffold({ required this.title, required this.fetch, required this.itemBuilder, this.actionBuilder, }); @override _ListStatefulScaffoldState createState() => _ListStatefulScaffoldState(); } class _ListStatefulScaffoldState extends State> { bool loading = false; bool loadingMore = false; String error = ''; List items = []; K? cursor; bool? hasMore; ScrollController _controller = ScrollController(); @override void initState() { super.initState(); _refresh(); _controller.addListener(() { if (_controller.position.maxScrollExtent - _controller.offset < 100 && !_controller.position.outOfRange && !loading && !loadingMore && hasMore != false) { _loadMore(); } }); } @override void dispose() { _controller.dispose(); super.dispose(); } Future _refresh() async { // Fimber.d('list scaffold refresh'); setState(() { error = ''; loading = true; }); try { final ListPayload p = await widget.fetch(null); items = p.items.toList(); cursor = p.cursor; hasMore = p.hasMore; } catch (err) { error = err.toString(); throw err; } finally { if (mounted) { setState(() { loading = false; }); } } } Future _loadMore() async { // Fimber.d('list scaffold load more'); setState(() { loadingMore = true; }); try { ListPayload p = await widget.fetch(cursor); items.addAll(p.items); cursor = p.cursor; hasMore = p.hasMore; } catch (err) { error = err.toString(); throw err; } finally { if (mounted) { setState(() { loadingMore = false; }); } } } Widget _buildItem(BuildContext context, int index) { if (index == 2 * items.length) { if (hasMore != false) { return Loading(more: true); } else { return Container(); } } else if (index % 2 == 1) { return CommonStyle.border; } else { return widget.itemBuilder(items[index ~/ 2]); } } Widget _buildCupertinoSliver() { if (error.isNotEmpty) { return SliverToBoxAdapter( child: ErrorReload(text: error, onTap: _refresh), ); } else if (loading && items.isEmpty) { return SliverToBoxAdapter(child: Loading(more: false)); } else if (items.isEmpty) { return SliverToBoxAdapter(child: EmptyWidget()); } else { return SliverList( delegate: SliverChildBuilderDelegate( _buildItem, childCount: 2 * items.length + 1, ), ); } } Widget _buildMaterial() { if (error.isNotEmpty) { return ErrorReload(text: error, onTap: _refresh); } else if (loading && items.isEmpty) { return Loading(more: false); } else if (items.isEmpty) { return EmptyWidget(); } else { return Scrollbar( child: ListView.builder( controller: _controller, itemCount: 2 * items.length + 1, itemBuilder: _buildItem, ), ); } } Widget _buildBody() { switch (Provider.of(context).theme) { case AppThemeType.cupertino: return CupertinoScrollbar( child: CustomScrollView( controller: _controller, slivers: [ CupertinoSliverRefreshControl(onRefresh: _refresh), _buildCupertinoSliver(), ], ), ); default: return RefreshIndicator( onRefresh: _refresh, child: _buildMaterial(), ); } } @override Widget build(BuildContext context) { return CommonScaffold( title: widget.title, body: _buildBody(), action: widget.actionBuilder?.call(), ); } }