git-touch-android-ios-app/lib/scaffolds/tab.dart

173 lines
4.2 KiB
Dart
Raw Normal View History

2019-09-24 13:58:34 +02:00
import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart';
import 'package:git_touch/models/theme.dart';
import 'package:provider/provider.dart';
import '../widgets/loading.dart';
import '../widgets/error_reload.dart';
class TabScaffold<T> extends StatefulWidget {
final Widget title;
2019-09-24 14:45:55 +02:00
final Widget Function(T payload, int activeTab) bodyBuilder;
2019-09-24 13:58:34 +02:00
final Future<T> Function(int activeTab) onRefresh;
final List<String> tabs;
2019-09-24 14:35:37 +02:00
final Widget Function(T payload) trailingBuilder;
2019-09-24 13:58:34 +02:00
TabScaffold({
@required this.title,
@required this.bodyBuilder,
@required this.onRefresh,
2019-09-24 14:35:37 +02:00
@required this.tabs,
2019-09-24 13:58:34 +02:00
this.trailingBuilder,
});
@override
_TabScaffoldState createState() => _TabScaffoldState();
}
class _TabScaffoldState<T> extends State<TabScaffold<T>> {
bool _loading;
2019-09-24 14:35:37 +02:00
T _payload0;
T _payload1;
T _payload2;
2019-09-24 13:58:34 +02:00
String _error = '';
int _activeTab = 0;
2019-09-24 14:35:37 +02:00
T _getPayload(int selected) {
switch (selected) {
case 0:
return _payload0;
case 1:
return _payload1;
case 2:
return _payload2;
default:
throw '';
}
}
T get _payload => _getPayload(_activeTab);
set _payload(T v) {
switch (_activeTab) {
case 0:
_payload0 = v;
break;
case 1:
_payload1 = v;
break;
case 2:
_payload2 = v;
break;
}
}
2019-09-24 13:58:34 +02:00
@override
void initState() {
super.initState();
_refresh();
}
Widget _buildBody() {
if (_error.isNotEmpty) {
return ErrorReload(text: _error, onTap: _refresh);
} else if (_payload == null) {
return Loading(more: false);
} else {
2019-09-24 14:45:55 +02:00
return widget.bodyBuilder(_payload, _activeTab);
2019-09-24 13:58:34 +02:00
}
}
2019-09-24 14:35:37 +02:00
Future<void> _refresh() async {
2019-09-24 13:58:34 +02:00
try {
setState(() {
_error = '';
_loading = true;
});
2019-09-24 14:35:37 +02:00
_payload = await widget.onRefresh(_activeTab);
2019-09-24 13:58:34 +02:00
} catch (err) {
_error = err.toString();
throw err;
} finally {
if (mounted) {
setState(() {
_loading = false;
});
}
}
}
2019-09-24 14:45:55 +02:00
Future<void> _switch(int selected) async {
2019-09-24 14:35:37 +02:00
setState(() {
_activeTab = selected;
});
if (_getPayload(selected) == null) {
2019-09-24 14:45:55 +02:00
await _refresh();
2019-09-24 14:35:37 +02:00
}
}
2019-09-24 13:58:34 +02:00
Widget _buildTrailing() {
if (_payload == null || widget.trailingBuilder == null) return null;
return widget.trailingBuilder(_payload);
}
List<Widget> _buildActions() {
if (_payload == null || widget.trailingBuilder == null) return null;
var w = widget.trailingBuilder(_payload);
return [if (w != null) w];
}
@override
Widget build(BuildContext context) {
switch (Provider.of<ThemeModel>(context).theme) {
case AppThemeType.cupertino:
return CupertinoPageScaffold(
navigationBar: CupertinoNavigationBar(
middle: DefaultTextStyle(
2019-09-24 14:35:37 +02:00
style: TextStyle(fontSize: 16),
2019-09-24 13:58:34 +02:00
child: CupertinoSegmentedControl(
groupValue: _activeTab,
2019-09-24 14:35:37 +02:00
onValueChanged: _switch,
2019-09-24 13:58:34 +02:00
children: widget.tabs.asMap().map((key, text) => MapEntry(
key,
Padding(
2019-09-24 14:35:37 +02:00
padding: const EdgeInsets.symmetric(horizontal: 8),
2019-09-24 13:58:34 +02:00
child: Text(text),
))),
),
),
trailing: null, // TODO:
),
child: SafeArea(
child: CustomScrollView(
slivers: <Widget>[
CupertinoSliverRefreshControl(onRefresh: _refresh),
SliverToBoxAdapter(child: _buildBody())
],
),
),
);
default:
return DefaultTabController(
length: widget.tabs.length,
child: Scaffold(
appBar: AppBar(
title: widget.title,
actions: _buildActions(),
bottom: TabBar(
2019-09-24 14:35:37 +02:00
onTap: _switch,
2019-09-24 13:58:34 +02:00
tabs: widget.tabs
.map((text) => Tab(text: text.toUpperCase()))
.toList(),
),
),
body: RefreshIndicator(
onRefresh: _refresh,
child: SingleChildScrollView(child: _buildBody()),
),
),
);
}
}
}