mirror of
https://github.com/twonlyapp/twonly-app.git
synced 2026-01-15 17:08:40 +00:00
Some checks are pending
Flutter analyze & test / flutter_analyze_and_test (push) Waiting to run
258 lines
8.6 KiB
Dart
258 lines
8.6 KiB
Dart
import 'dart:async';
|
|
import 'dart:collection';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
import 'package:twonly/globals.dart';
|
|
import 'package:twonly/src/database/daos/contacts.dao.dart';
|
|
import 'package:twonly/src/database/twonly.db.dart';
|
|
import 'package:twonly/src/utils/misc.dart';
|
|
import 'package:twonly/src/views/components/avatar_icon.component.dart';
|
|
import 'package:twonly/src/views/components/flame.dart';
|
|
import 'package:twonly/src/views/components/user_context_menu.component.dart';
|
|
|
|
class SelectAdditionalUsers extends StatefulWidget {
|
|
const SelectAdditionalUsers({
|
|
required this.alreadySelected,
|
|
required this.limit,
|
|
super.key,
|
|
});
|
|
final List<int> alreadySelected;
|
|
final int limit;
|
|
@override
|
|
State<SelectAdditionalUsers> createState() => _SelectAdditionalUsers();
|
|
}
|
|
|
|
class _SelectAdditionalUsers extends State<SelectAdditionalUsers> {
|
|
List<Contact> contacts = [];
|
|
List<Contact> allContacts = [];
|
|
final TextEditingController searchUserName = TextEditingController();
|
|
late StreamSubscription<List<Contact>> contactSub;
|
|
|
|
final HashSet<int> selectedUsers = HashSet();
|
|
late HashSet<int> _alreadySelected;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
|
|
_alreadySelected = HashSet.from(widget.alreadySelected);
|
|
|
|
final stream = twonlyDB.contactsDao.watchAllAcceptedContacts();
|
|
|
|
contactSub = stream.listen((update) async {
|
|
update.sort(
|
|
(a, b) => getContactDisplayName(a).compareTo(getContactDisplayName(b)),
|
|
);
|
|
setState(() {
|
|
allContacts = update;
|
|
});
|
|
await filterUsers();
|
|
});
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
unawaited(contactSub.cancel());
|
|
super.dispose();
|
|
}
|
|
|
|
Future<void> filterUsers() async {
|
|
if (searchUserName.value.text.isEmpty) {
|
|
setState(() {
|
|
contacts = allContacts;
|
|
});
|
|
return;
|
|
}
|
|
final usersFiltered = allContacts
|
|
.where(
|
|
(user) => getContactDisplayName(user)
|
|
.toLowerCase()
|
|
.contains(searchUserName.value.text.toLowerCase()),
|
|
)
|
|
.toList();
|
|
setState(() {
|
|
contacts = usersFiltered;
|
|
});
|
|
}
|
|
|
|
void toggleSelectedUser(int userId) {
|
|
if (_alreadySelected.contains(userId)) return;
|
|
if (!selectedUsers.contains(userId)) {
|
|
if (selectedUsers.length < widget.limit) {
|
|
selectedUsers.add(userId);
|
|
}
|
|
} else {
|
|
selectedUsers.remove(userId);
|
|
}
|
|
setState(() {});
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return GestureDetector(
|
|
onTap: () => FocusScope.of(context).unfocus(),
|
|
child: Scaffold(
|
|
appBar: AppBar(
|
|
title: Text(context.lang.additionalUserSelectTitle),
|
|
),
|
|
floatingActionButton: FilledButton.icon(
|
|
onPressed: selectedUsers.isEmpty
|
|
? null
|
|
: () => Navigator.pop(context, selectedUsers.toList()),
|
|
label: Text(
|
|
context.lang.additionalUserSelectButton(
|
|
widget.limit,
|
|
selectedUsers.length + widget.alreadySelected.length,
|
|
),
|
|
),
|
|
icon: const FaIcon(FontAwesomeIcons.userPlus),
|
|
),
|
|
body: SafeArea(
|
|
child: Padding(
|
|
padding:
|
|
const EdgeInsets.only(bottom: 40, left: 10, top: 20, right: 10),
|
|
child: Column(
|
|
children: [
|
|
Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 10),
|
|
child: TextField(
|
|
onChanged: (_) async {
|
|
await filterUsers();
|
|
},
|
|
controller: searchUserName,
|
|
decoration: getInputDecoration(
|
|
context,
|
|
context.lang.shareImageSearchAllContacts,
|
|
),
|
|
),
|
|
),
|
|
const SizedBox(height: 10),
|
|
Expanded(
|
|
child: ListView.builder(
|
|
restorationId: 'new_message_users_list',
|
|
itemCount:
|
|
contacts.length + (selectedUsers.isEmpty ? 0 : 2),
|
|
itemBuilder: (BuildContext context, int i) {
|
|
if (selectedUsers.isNotEmpty) {
|
|
final selected = selectedUsers.toList();
|
|
if (i == 0) {
|
|
return Container(
|
|
padding: const EdgeInsets.symmetric(horizontal: 18),
|
|
constraints: const BoxConstraints(
|
|
maxHeight: 150,
|
|
),
|
|
child: SingleChildScrollView(
|
|
child: LayoutBuilder(
|
|
builder: (context, constraints) {
|
|
return Wrap(
|
|
spacing: 8,
|
|
children: selected.map((w) {
|
|
return _Chip(
|
|
contact: allContacts
|
|
.firstWhere((t) => t.userId == w),
|
|
onTap: toggleSelectedUser,
|
|
);
|
|
}).toList(),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
);
|
|
}
|
|
if (i == 1) {
|
|
return const Divider();
|
|
}
|
|
i -= 2;
|
|
}
|
|
final user = contacts[i];
|
|
return UserContextMenu(
|
|
key: ValueKey(user.userId),
|
|
contact: user,
|
|
child: ListTile(
|
|
title: Row(
|
|
children: [
|
|
Text(getContactDisplayName(user)),
|
|
FlameCounterWidget(
|
|
contactId: user.userId,
|
|
prefix: true,
|
|
),
|
|
],
|
|
),
|
|
subtitle: (_alreadySelected.contains(user.userId))
|
|
? Text(context.lang.alreadyInGroup)
|
|
: null,
|
|
leading: AvatarIcon(
|
|
contactId: user.userId,
|
|
fontSize: 13,
|
|
),
|
|
trailing: Checkbox(
|
|
value: selectedUsers.contains(user.userId) |
|
|
_alreadySelected.contains(user.userId),
|
|
side: WidgetStateBorderSide.resolveWith(
|
|
(states) {
|
|
if (states.contains(WidgetState.selected)) {
|
|
return const BorderSide(width: 0);
|
|
}
|
|
return BorderSide(
|
|
color: Theme.of(context).colorScheme.outline,
|
|
);
|
|
},
|
|
),
|
|
onChanged: (bool? value) {
|
|
toggleSelectedUser(user.userId);
|
|
},
|
|
),
|
|
onTap: () {
|
|
toggleSelectedUser(user.userId);
|
|
},
|
|
),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
class _Chip extends StatelessWidget {
|
|
const _Chip({
|
|
required this.contact,
|
|
required this.onTap,
|
|
});
|
|
final Contact contact;
|
|
final void Function(int) onTap;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return GestureDetector(
|
|
onTap: () => onTap(contact.userId),
|
|
child: Chip(
|
|
avatar: AvatarIcon(
|
|
contactId: contact.userId,
|
|
fontSize: 10,
|
|
),
|
|
label: Row(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
Text(
|
|
getContactDisplayName(contact),
|
|
style: const TextStyle(fontSize: 14),
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
const SizedBox(width: 15),
|
|
const FaIcon(
|
|
FontAwesomeIcons.xmark,
|
|
color: Colors.grey,
|
|
size: 12,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|