git-touch-android-ios-app/lib/screens/notifications.dart

288 lines
7.4 KiB
Dart
Raw Normal View History

2019-02-06 06:06:11 +01:00
import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart';
2019-09-02 14:40:20 +02:00
import 'package:provider/provider.dart';
import '../scaffolds/refresh_stateless.dart';
2019-09-02 14:40:20 +02:00
import 'package:git_touch/models/notification.dart';
2019-02-07 07:35:19 +01:00
import '../providers/settings.dart';
import '../widgets/notification_item.dart';
import '../widgets/list_group.dart';
2019-02-06 06:06:11 +01:00
import '../widgets/link.dart';
2019-02-10 05:16:52 +01:00
import '../widgets/empty.dart';
import '../utils/utils.dart';
2019-02-06 12:14:11 +01:00
String getRepoKey(NotificationGroup group) {
// Add heading _ to fix number case
// - => __
// . => ___
return ('_' + group.owner + '_' + group.name)
.replaceAll('-', '__')
.replaceAll('.', '___');
2019-02-06 12:14:11 +01:00
}
String getItemKey(NotificationPayload item) {
return '_' + item.number.toString();
}
class NotificationGroup {
2019-02-06 12:14:11 +01:00
String owner;
String name;
get repo => owner + '/' + name;
2019-02-06 06:06:11 +01:00
List<NotificationPayload> items = [];
2019-02-06 12:14:11 +01:00
NotificationGroup(this.owner, this.name);
}
class NotificationScreen extends StatefulWidget {
@override
NotificationScreenState createState() => NotificationScreenState();
}
class NotificationScreenState extends State<NotificationScreen> {
String error = '';
int active = 0;
2019-02-10 12:15:50 +01:00
bool loading = false;
2019-02-06 06:06:11 +01:00
Map<String, NotificationGroup> groupMap = {};
2019-02-06 14:35:52 +01:00
@override
void initState() {
super.initState();
nextTick(_onSwitchTab);
2019-02-06 14:35:52 +01:00
}
2019-02-07 07:35:19 +01:00
Future<Map<String, NotificationGroup>> fetchNotifications(int index) async {
List items = await SettingsProvider.of(context).getWithCredentials(
'/notifications?all=${index == 2}&participating=${index == 1}');
var ns = items.map((item) => NotificationPayload.fromJson(item)).toList();
if (index == 0) {
2019-09-02 14:40:20 +02:00
Provider.of<NotificationModel>(context).setCount(ns.length);
2019-02-07 07:35:19 +01:00
}
Map<String, NotificationGroup> _groupMap = {};
ns.forEach((item) {
String repo = item.owner + '/' + item.name;
if (_groupMap[repo] == null) {
_groupMap[repo] = NotificationGroup(item.owner, item.name);
}
_groupMap[repo].items.add(item);
});
if (_groupMap.isNotEmpty) {
// query state of issues and pull requests
var schema = '{';
_groupMap.forEach((repo, group) {
var repoKey = getRepoKey(group);
2019-03-19 13:11:35 +01:00
// Check if issue and pull request exist
if (group.items.where((item) {
return item.type == 'Issue' || item.type == 'PullRequest';
}).isEmpty) {
return;
}
schema +=
'$repoKey: repository(owner: "${group.owner}", name: "${group.name}") {';
group.items.forEach((item) {
var key = getItemKey(item);
switch (item.type) {
case 'Issue':
schema += '''
2019-02-07 07:35:19 +01:00
$key: issue(number: ${item.number}) {
state
}
''';
break;
case 'PullRequest':
schema += '''
2019-02-07 07:35:19 +01:00
$key: pullRequest(number: ${item.number}) {
state
}
''';
break;
}
});
2019-02-07 07:35:19 +01:00
schema += '}';
});
2019-02-07 07:35:19 +01:00
schema += '}';
// print(schema);
var data = await SettingsProvider.of(context).query(schema);
_groupMap.forEach((repo, group) {
group.items.forEach((item) {
2019-03-19 13:11:35 +01:00
var groupData = data[getRepoKey(group)];
if (groupData == null) return;
var itemData = data[getRepoKey(group)][getItemKey(item)];
if (itemData != null) {
item.state = itemData['state'];
}
});
2019-02-07 07:35:19 +01:00
});
// print(data);
}
2019-02-07 07:35:19 +01:00
return _groupMap;
}
Widget _buildGroupItem(
2019-02-07 13:28:48 +01:00
BuildContext context,
MapEntry<String, NotificationGroup> entry,
) {
2019-02-06 14:35:52 +01:00
var group = entry.value;
2019-02-06 06:06:11 +01:00
var repo = group.repo;
return ListGroup(
2019-03-10 14:26:05 +01:00
title: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: <Widget>[
Text(
repo,
style: TextStyle(fontSize: 16, fontWeight: FontWeight.w600),
),
Link(
material: false,
onTap: () async {
await SettingsProvider.of(context)
.putWithCredentials('/repos/$repo/notifications');
await _onSwitchTab();
2019-02-06 06:06:11 +01:00
},
2019-03-10 14:26:05 +01:00
child: Icon(
Octicons.check,
color: Colors.black45,
size: 24,
),
),
],
),
items: group.items,
itemBuilder: (item, index) {
return NotificationItem(
payload: item,
markAsRead: () {
if (mounted) {
setState(() {
groupMap[entry.key].items[index].unread = false;
});
}
},
);
},
);
}
2019-02-07 13:28:48 +01:00
Future<void> _onSwitchTab([int index]) async {
if (loading) return;
setState(() {
error = '';
if (index != null) {
active = index;
}
2019-02-06 14:35:52 +01:00
loading = true;
});
try {
groupMap = await fetchNotifications(active);
} catch (err) {
error = err.toString();
throw err;
} finally {
if (mounted) {
setState(() {
loading = false;
});
}
2019-02-06 14:35:52 +01:00
}
}
2019-02-06 14:35:52 +01:00
var textMap = {
0: 'Unread',
1: 'Paticipating',
2: 'All',
};
2019-02-07 13:28:48 +01:00
Widget _buildTitle() {
switch (SettingsProvider.of(context).theme) {
case ThemeMap.cupertino:
// var textStyle = DefaultTextStyle.of(context).style;
return DefaultTextStyle(
style: TextStyle(fontSize: 16),
child: SizedBox.expand(
child: CupertinoSegmentedControl(
groupValue: active,
onValueChanged: _onSwitchTab,
children: textMap.map((key, text) => MapEntry(key, Text(text))),
),
),
);
default:
return Text('Notifications');
}
}
void _confirm() async {
var value = await showConfirm(context, 'Mark all as read?');
if (value) {
await SettingsProvider.of(context).putWithCredentials('/notifications');
await _onSwitchTab();
2019-02-07 13:28:48 +01:00
}
}
2019-02-06 14:35:52 +01:00
@override
Widget build(context) {
return RefreshStatelessScaffold(
2019-02-07 13:28:48 +01:00
title: _buildTitle(),
bottom: TabBar(
onTap: _onSwitchTab,
2019-03-10 09:09:26 +01:00
tabs: textMap.entries
.map((entry) => Tab(text: entry.value.toUpperCase()))
.toList(),
2019-02-06 14:35:52 +01:00
),
2019-02-07 13:28:48 +01:00
// trailing: GestureDetector(
// child: Icon(Icons.more_vert, size: 20),
// onTap: () async {
// int value = await showCupertinoDialog(
// context: context,
// builder: (context) {
// return CupertinoAlertDialog(
// title: Text('Select filter'),
// actions: textMap.entries.map((entry) {
// return CupertinoDialogAction(
// child: Text(entry.value),
// onPressed: () {
// Navigator.pop(context, entry.key);
// },
// );
// }).toList(),
// );
// },
// );
// _onSwitchTab(value);
// },
// ),
2019-02-06 14:35:52 +01:00
actions: <Widget>[
2019-02-07 13:28:48 +01:00
IconButton(
2019-02-10 12:15:50 +01:00
icon: Icon(Icons.done_all),
2019-02-07 13:28:48 +01:00
onPressed: _confirm,
2019-02-06 14:35:52 +01:00
)
],
onRefresh: _onSwitchTab,
2019-02-06 14:35:52 +01:00
loading: loading,
error: error,
2019-02-06 06:06:11 +01:00
bodyBuilder: () {
2019-02-10 05:16:52 +01:00
return groupMap.isEmpty
? EmptyWidget()
2019-05-12 08:01:12 +02:00
: Column(children: [
Padding(padding: EdgeInsets.only(top: 10)),
...groupMap.entries
.map((entry) => _buildGroupItem(context, entry))
.toList()
]);
2019-02-06 06:06:11 +01:00
},
);
}
}