FurryChat/lib/views/invitation_selection.dart

204 lines
7.2 KiB
Dart
Raw Normal View History

2020-02-16 11:07:48 +00:00
import 'dart:async';
2020-01-01 18:10:13 +00:00
import 'package:famedlysdk/famedlysdk.dart';
import 'package:fluffychat/components/adaptive_page_layout.dart';
import 'package:fluffychat/components/avatar.dart';
2020-04-27 11:36:39 +00:00
import 'package:fluffychat/components/dialogs/simple_dialogs.dart';
2020-01-01 18:10:13 +00:00
import 'package:fluffychat/components/matrix.dart';
2020-05-07 05:52:40 +00:00
import 'package:fluffychat/l10n/l10n.dart';
2020-01-01 18:10:13 +00:00
import 'package:flutter/material.dart';
2020-05-13 08:45:50 +00:00
import 'package:bot_toast/bot_toast.dart';
2020-01-01 18:10:13 +00:00
import 'chat_list.dart';
2020-02-16 11:07:48 +00:00
class InvitationSelection extends StatefulWidget {
2020-01-01 18:10:13 +00:00
final Room room;
const InvitationSelection(this.room, {Key key}) : super(key: key);
2020-02-16 11:07:48 +00:00
@override
_InvitationSelectionState createState() => _InvitationSelectionState();
}
class _InvitationSelectionState extends State<InvitationSelection> {
TextEditingController controller = TextEditingController();
String currentSearchTerm;
bool loading = false;
List<Map<String, dynamic>> foundProfiles = [];
Timer coolDown;
2020-01-01 18:10:13 +00:00
Future<List<User>> getContacts(BuildContext context) async {
2020-05-13 13:58:59 +00:00
var client2 = Matrix.of(context).client;
final client = client2;
var participants = await widget.room.requestParticipants();
2020-04-02 11:14:39 +00:00
participants.removeWhere(
(u) => ![Membership.join, Membership.invite].contains(u.membership),
);
2020-05-13 13:58:59 +00:00
var contacts = <User>[];
var userMap = <String, bool>{};
for (var i = 0; i < client.rooms.length; i++) {
var roomUsers = client.rooms[i].getParticipants();
2020-04-02 11:14:39 +00:00
2020-05-13 13:58:59 +00:00
for (var j = 0; j < roomUsers.length; j++) {
2020-01-01 18:10:13 +00:00
if (userMap[roomUsers[j].id] != true &&
2020-01-02 21:31:39 +00:00
participants.indexWhere((u) => u.id == roomUsers[j].id) == -1) {
2020-01-01 18:10:13 +00:00
contacts.add(roomUsers[j]);
2020-01-02 21:31:39 +00:00
}
2020-01-01 18:10:13 +00:00
userMap[roomUsers[j].id] = true;
}
}
2020-04-02 11:14:39 +00:00
contacts.sort(
(a, b) => a.calcDisplayname().toLowerCase().compareTo(
b.calcDisplayname().toLowerCase(),
),
);
2020-01-01 18:10:13 +00:00
return contacts;
}
void inviteAction(BuildContext context, String id) async {
2020-04-27 11:36:39 +00:00
final success = await SimpleDialogs(context).tryRequestWithLoadingDialog(
2020-02-16 11:07:48 +00:00
widget.room.invite(id),
2020-01-01 18:10:13 +00:00
);
2020-01-02 21:31:39 +00:00
if (success != false) {
2020-05-13 08:45:50 +00:00
BotToast.showText(text: L10n.of(context).contactHasBeenInvitedToTheGroup);
2020-01-02 21:31:39 +00:00
}
2020-01-01 18:10:13 +00:00
}
2020-02-16 11:07:48 +00:00
void searchUserWithCoolDown(BuildContext context, String text) async {
coolDown?.cancel();
coolDown = Timer(
Duration(seconds: 1),
() => searchUser(context, text),
);
}
void searchUser(BuildContext context, String text) async {
coolDown?.cancel();
if (text.isEmpty) {
setState(() {
foundProfiles = [];
});
}
currentSearchTerm = text;
if (currentSearchTerm.isEmpty) return;
if (loading) return;
setState(() => loading = true);
2020-05-13 13:58:59 +00:00
final matrix = Matrix.of(context);
2020-04-27 11:36:39 +00:00
final response = await SimpleDialogs(context).tryRequestWithErrorToast(
2020-02-16 11:07:48 +00:00
matrix.client.jsonRequest(
type: HTTPType.POST,
2020-05-13 13:58:59 +00:00
action: '/client/r0/user_directory/search',
2020-02-16 11:07:48 +00:00
data: {
2020-05-13 13:58:59 +00:00
'search_term': text,
'limit': 10,
2020-02-16 11:07:48 +00:00
}),
);
setState(() => loading = false);
if (response == false ||
!(response is Map) ||
2020-05-13 13:58:59 +00:00
(response['results'] == null)) return;
2020-02-16 11:07:48 +00:00
setState(() {
2020-05-13 13:58:59 +00:00
foundProfiles = List<Map<String, dynamic>>.from(response['results']);
if ('@$text'.isValidMatrixId &&
2020-02-16 11:07:48 +00:00
foundProfiles
2020-05-13 13:58:59 +00:00
.indexWhere((profile) => '@$text' == profile['user_id']) ==
2020-02-16 11:07:48 +00:00
-1) {
setState(() => foundProfiles = [
2020-05-13 13:58:59 +00:00
{'user_id': '@$text'}
2020-02-16 11:07:48 +00:00
]);
}
foundProfiles.removeWhere((profile) =>
widget.room
.getParticipants()
2020-05-13 13:58:59 +00:00
.indexWhere((u) => u.id == profile['user_id']) !=
2020-02-16 11:07:48 +00:00
-1);
});
}
2020-01-01 18:10:13 +00:00
@override
Widget build(BuildContext context) {
2020-05-13 13:58:59 +00:00
final groupName = widget.room.name?.isEmpty ?? false
2020-05-07 05:52:40 +00:00
? L10n.of(context).group
2020-02-16 11:07:48 +00:00
: widget.room.name;
2020-01-01 18:10:13 +00:00
return AdaptivePageLayout(
primaryPage: FocusPage.SECOND,
2020-02-16 11:07:48 +00:00
firstScaffold: ChatList(activeChat: widget.room.id),
2020-01-01 18:10:13 +00:00
secondScaffold: Scaffold(
appBar: AppBar(
2020-05-07 05:52:40 +00:00
title: Text(L10n.of(context).inviteContact),
2020-02-16 11:07:48 +00:00
bottom: PreferredSize(
2020-02-23 07:49:58 +00:00
preferredSize: Size.fromHeight(92),
2020-02-16 11:07:48 +00:00
child: Padding(
padding: const EdgeInsets.all(16.0),
child: TextField(
controller: controller,
autofocus: true,
autocorrect: false,
textInputAction: TextInputAction.search,
onChanged: (String text) =>
searchUserWithCoolDown(context, text),
onSubmitted: (String text) => searchUser(context, text),
decoration: InputDecoration(
border: OutlineInputBorder(),
2020-05-13 13:58:59 +00:00
prefixText: '@',
2020-05-07 05:52:40 +00:00
hintText: L10n.of(context).username,
labelText: L10n.of(context).inviteContactToGroup(groupName),
2020-02-16 11:07:48 +00:00
suffixIcon: loading
? Container(
padding: const EdgeInsets.all(8.0),
width: 12,
height: 12,
child: CircularProgressIndicator(),
)
: Icon(Icons.search),
2020-01-18 12:22:22 +00:00
),
2020-01-01 18:10:13 +00:00
),
2020-02-16 11:07:48 +00:00
),
),
),
body: foundProfiles.isNotEmpty
? ListView.builder(
itemCount: foundProfiles.length,
itemBuilder: (BuildContext context, int i) => ListTile(
leading: Avatar(
2020-05-13 13:58:59 +00:00
foundProfiles[i]['avatar_url'] == null
2020-04-28 12:11:56 +00:00
? null
2020-05-13 13:58:59 +00:00
: Uri.parse(foundProfiles[i]['avatar_url']),
foundProfiles[i]['display_name'] ??
foundProfiles[i]['user_id'],
2020-02-16 11:07:48 +00:00
),
title: Text(
2020-05-13 13:58:59 +00:00
foundProfiles[i]['display_name'] ??
(foundProfiles[i]['user_id'] as String).localpart,
2020-02-16 11:07:48 +00:00
),
2020-05-13 13:58:59 +00:00
subtitle: Text(foundProfiles[i]['user_id']),
2020-02-16 11:07:48 +00:00
onTap: () =>
2020-05-13 13:58:59 +00:00
inviteAction(context, foundProfiles[i]['user_id']),
2020-02-16 11:07:48 +00:00
),
)
: FutureBuilder<List<User>>(
future: getContacts(context),
builder: (BuildContext context, snapshot) {
if (!snapshot.hasData) {
return Center(
child: CircularProgressIndicator(),
);
}
2020-05-13 13:58:59 +00:00
var contacts = snapshot.data;
2020-02-16 11:07:48 +00:00
return ListView.builder(
itemCount: contacts.length,
itemBuilder: (BuildContext context, int i) => ListTile(
leading: Avatar(
contacts[i].avatarUrl,
contacts[i].calcDisplayname(),
),
title: Text(contacts[i].calcDisplayname()),
subtitle: Text(contacts[i].id),
onTap: () => inviteAction(context, contacts[i].id),
),
);
},
)),
2020-01-01 18:10:13 +00:00
);
}
}