2023-05-04 12:16:19 -04:00
|
|
|
import 'dart:async';
|
|
|
|
|
2024-07-15 00:23:04 -04:00
|
|
|
import 'package:OpenContacts/apis/user_api.dart';
|
|
|
|
import 'package:OpenContacts/client_holder.dart';
|
|
|
|
import 'package:OpenContacts/clients/messaging_client.dart';
|
|
|
|
import 'package:OpenContacts/models/users/user.dart';
|
|
|
|
import 'package:OpenContacts/widgets/default_error_widget.dart';
|
|
|
|
import 'package:OpenContacts/widgets/friends/user_list_tile.dart';
|
2023-05-04 12:16:19 -04:00
|
|
|
import 'package:flutter/material.dart';
|
2023-05-06 13:01:15 -04:00
|
|
|
import 'package:provider/provider.dart';
|
2023-05-04 12:16:19 -04:00
|
|
|
|
|
|
|
class SearchError {
|
|
|
|
final String message;
|
|
|
|
final IconData icon;
|
|
|
|
|
|
|
|
const SearchError({required this.message, required this.icon});
|
|
|
|
}
|
|
|
|
|
|
|
|
class UserSearch extends StatefulWidget {
|
2023-05-06 13:01:15 -04:00
|
|
|
const UserSearch({super.key});
|
2023-05-04 12:16:19 -04:00
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _UserSearchState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _UserSearchState extends State<UserSearch> {
|
|
|
|
final TextEditingController _searchInputController = TextEditingController();
|
|
|
|
Timer? _searchDebouncer;
|
2023-05-04 13:38:35 -04:00
|
|
|
late Future<List<User>?>? _usersFuture = _emptySearch;
|
2023-05-04 12:16:19 -04:00
|
|
|
|
2023-05-04 13:38:35 -04:00
|
|
|
Future<List<User>> get _emptySearch =>
|
|
|
|
Future(() =>
|
|
|
|
throw const SearchError(
|
|
|
|
message: "Start typing to search for users", icon: Icons.search)
|
|
|
|
);
|
2023-05-04 12:16:19 -04:00
|
|
|
|
|
|
|
void _querySearch(BuildContext context, String needle) {
|
|
|
|
if (needle.isEmpty) {
|
|
|
|
_usersFuture = _emptySearch;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_usersFuture = UserApi.searchUsers(ClientHolder
|
2023-05-04 13:38:35 -04:00
|
|
|
.of(context)
|
|
|
|
.apiClient, needle: needle).then((value) {
|
2023-05-04 12:16:19 -04:00
|
|
|
final res = value.toList();
|
|
|
|
if (res.isEmpty) throw SearchError(message: "No user found with username '$needle'", icon: Icons.search_off);
|
|
|
|
res.sort(
|
|
|
|
(a, b) => a.username.length.compareTo(b.username.length)
|
|
|
|
);
|
|
|
|
return res;
|
2023-05-04 13:38:35 -04:00
|
|
|
});
|
2023-05-04 12:16:19 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-05-06 13:01:15 -04:00
|
|
|
final mClient = Provider.of<MessagingClient>(context, listen: false);
|
2023-05-04 12:16:19 -04:00
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
2024-07-15 02:08:17 -04:00
|
|
|
title: const Text("Add Users"),
|
2023-05-04 12:16:19 -04:00
|
|
|
),
|
2023-10-10 04:57:14 -04:00
|
|
|
body: SafeArea(
|
|
|
|
top: false,
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
Expanded(
|
|
|
|
child: FutureBuilder(
|
|
|
|
future: _usersFuture,
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
if (snapshot.hasData) {
|
|
|
|
final users = snapshot.data as List<User>;
|
|
|
|
return ListView.builder(
|
|
|
|
itemCount: users.length,
|
|
|
|
itemBuilder: (context, index) {
|
|
|
|
final user = users[index];
|
|
|
|
return UserListTile(user: user, onChanged: () {
|
|
|
|
mClient.refreshFriendsList();
|
|
|
|
}, isFriend: mClient.getAsFriend(user.id) != null,);
|
|
|
|
},
|
2023-05-04 12:16:19 -04:00
|
|
|
);
|
2023-10-10 04:57:14 -04:00
|
|
|
} else if (snapshot.hasError) {
|
|
|
|
final err = snapshot.error;
|
|
|
|
if (err is SearchError) {
|
|
|
|
return DefaultErrorWidget(
|
|
|
|
title: err.message,
|
|
|
|
iconOverride: err.icon,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
FlutterError.reportError(
|
|
|
|
FlutterErrorDetails(exception: snapshot.error!, stack: snapshot.stackTrace));
|
|
|
|
return DefaultErrorWidget(title: "${snapshot.error}",);
|
|
|
|
}
|
2023-05-04 12:16:19 -04:00
|
|
|
} else {
|
2023-10-10 04:57:14 -04:00
|
|
|
return const Column(
|
|
|
|
children: [
|
|
|
|
LinearProgressIndicator(),
|
|
|
|
],
|
|
|
|
);
|
2023-05-04 12:16:19 -04:00
|
|
|
}
|
2023-10-10 04:57:14 -04:00
|
|
|
},
|
2023-05-04 12:16:19 -04:00
|
|
|
),
|
2023-10-10 04:57:14 -04:00
|
|
|
),
|
|
|
|
Padding(
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 4),
|
|
|
|
child: TextField(
|
|
|
|
decoration: InputDecoration(
|
|
|
|
isDense: true,
|
|
|
|
hintText: "Search for users...",
|
|
|
|
contentPadding: const EdgeInsets.all(16),
|
|
|
|
border: OutlineInputBorder(
|
|
|
|
borderRadius: BorderRadius.circular(24)
|
|
|
|
)
|
|
|
|
),
|
|
|
|
autocorrect: false,
|
|
|
|
controller: _searchInputController,
|
|
|
|
onChanged: (String value) {
|
|
|
|
_searchDebouncer?.cancel();
|
|
|
|
if (value.isEmpty) {
|
|
|
|
setState(() {
|
|
|
|
_querySearch(context, value);
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
2023-05-04 12:16:19 -04:00
|
|
|
setState(() {
|
2023-10-10 04:57:14 -04:00
|
|
|
_usersFuture = Future(() => null);
|
2023-05-04 12:16:19 -04:00
|
|
|
});
|
2023-10-10 04:57:14 -04:00
|
|
|
_searchDebouncer = Timer(const Duration(milliseconds: 300), () {
|
|
|
|
setState(() {
|
|
|
|
_querySearch(context, value);
|
|
|
|
});
|
2023-05-04 12:16:19 -04:00
|
|
|
});
|
2023-10-10 04:57:14 -04:00
|
|
|
},
|
|
|
|
),
|
2023-05-04 12:16:19 -04:00
|
|
|
),
|
2023-10-10 04:57:14 -04:00
|
|
|
],
|
|
|
|
),
|
2023-05-04 12:16:19 -04:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|