git-touch-android-ios-app/lib/widgets/list_scaffold.dart

152 lines
3.5 KiB
Dart
Raw Normal View History

2019-02-03 07:42:50 +01:00
import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart';
import 'package:flutter/widgets.dart';
import '../providers/settings.dart';
import 'loading.dart';
typedef RefreshCallback = Future<void> Function();
class ListScaffold extends StatefulWidget {
2019-02-05 13:57:05 +01:00
final Widget title;
2019-02-03 07:42:50 +01:00
final Widget header;
final int itemCount;
final IndexedWidgetBuilder itemBuilder;
final RefreshCallback onRefresh;
final RefreshCallback onLoadMore;
ListScaffold({
@required this.title,
this.header,
@required this.itemCount,
@required this.itemBuilder,
@required this.onRefresh,
this.onLoadMore,
});
@override
_ListScaffoldState createState() => _ListScaffoldState();
}
class _ListScaffoldState extends State<ListScaffold> {
bool loading = false;
bool loadingMore = false;
ScrollController _controller = ScrollController();
@override
void initState() {
super.initState();
_refresh();
_controller.addListener(() {
if (_controller.offset + 100 > _controller.position.maxScrollExtent &&
!_controller.position.outOfRange &&
!loading &&
!loadingMore) {
_loadMore();
}
});
}
Future<void> _refresh() async {
print('refresh');
setState(() {
loading = true;
});
try {
await widget.onRefresh();
} catch (err) {
print(err);
} finally {
setState(() {
loading = false;
});
}
}
Future<void> _loadMore() async {
print('more');
setState(() {
loadingMore = true;
});
try {
await widget.onLoadMore();
} catch (err) {
print(err);
} finally {
loadingMore = false;
}
}
Widget _buildItem(BuildContext context, int index) {
2019-02-05 13:57:05 +01:00
if (index == 2 * widget.itemCount) {
2019-02-03 07:42:50 +01:00
return Loading(more: true);
}
2019-02-05 13:57:05 +01:00
if (index % 2 == 1) {
return Container(
decoration: BoxDecoration(
border: Border(bottom: BorderSide(color: Colors.black12)),
),
);
}
return widget.itemBuilder(context, index ~/ 2);
2019-02-03 07:42:50 +01:00
}
Widget _buildSliver(BuildContext context) {
if (loading) {
return SliverToBoxAdapter(child: Loading(more: false));
} else {
return SliverList(
delegate: SliverChildBuilderDelegate(
_buildItem,
2019-02-05 13:57:05 +01:00
childCount: 2 * widget.itemCount + 1,
2019-02-03 07:42:50 +01:00
),
);
}
}
Widget _buildBody(BuildContext context) {
if (loading) {
return Loading(more: false);
} else {
return ListView.builder(
controller: _controller,
2019-02-05 13:57:05 +01:00
itemCount: 2 * widget.itemCount + 1,
2019-02-03 07:42:50 +01:00
itemBuilder: _buildItem,
);
}
}
@override
Widget build(BuildContext context) {
switch (SettingsProvider.of(context).layout) {
case LayoutMap.cupertino:
List<Widget> slivers = [
CupertinoSliverRefreshControl(onRefresh: widget.onRefresh)
];
if (widget.header != null) {
slivers.add(SliverToBoxAdapter(child: widget.header));
}
slivers.add(_buildSliver(context));
return CupertinoPageScaffold(
2019-02-05 13:57:05 +01:00
navigationBar: CupertinoNavigationBar(middle: widget.title),
2019-02-03 07:42:50 +01:00
child: SafeArea(
child: CustomScrollView(
controller: _controller,
slivers: slivers,
),
),
);
default:
return Scaffold(
2019-02-05 13:57:05 +01:00
appBar: AppBar(title: widget.title),
2019-02-03 07:42:50 +01:00
body: RefreshIndicator(
onRefresh: widget.onRefresh,
child: _buildBody(context),
),
);
}
}
}