Add new user search
This commit is contained in:
parent
5ad49c03a3
commit
7c81657a9b
|
@ -1,12 +1,27 @@
|
||||||
|
import 'dart:async';
|
||||||
|
|
||||||
import 'package:famedlysdk/famedlysdk.dart';
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
|
import 'package:fluffychat/components/avatar.dart';
|
||||||
import 'package:fluffychat/views/chat.dart';
|
import 'package:fluffychat/views/chat.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:share/share.dart';
|
||||||
|
|
||||||
import '../matrix.dart';
|
import '../matrix.dart';
|
||||||
|
|
||||||
class NewPrivateChatDialog extends StatelessWidget {
|
class NewPrivateChatDialog extends StatefulWidget {
|
||||||
final TextEditingController controller = TextEditingController();
|
@override
|
||||||
|
_NewPrivateChatDialogState createState() => _NewPrivateChatDialogState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _NewPrivateChatDialogState extends State<NewPrivateChatDialog> {
|
||||||
|
TextEditingController controller = TextEditingController();
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
|
bool loading = false;
|
||||||
|
String currentSearchTerm;
|
||||||
|
Map<String, dynamic> foundProfile;
|
||||||
|
Timer coolDown;
|
||||||
|
bool get correctMxId =>
|
||||||
|
foundProfile != null && foundProfile["user_id"] == "@$currentSearchTerm";
|
||||||
|
|
||||||
void submitAction(BuildContext context) async {
|
void submitAction(BuildContext context) async {
|
||||||
if (controller.text.isEmpty) return;
|
if (controller.text.isEmpty) return;
|
||||||
|
@ -31,12 +46,53 @@ class NewPrivateChatDialog extends StatelessWidget {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void searchUserWithCoolDown(BuildContext context, String text) async {
|
||||||
|
coolDown?.cancel();
|
||||||
|
coolDown = Timer(
|
||||||
|
Duration(seconds: 1),
|
||||||
|
() => searchUser(context, text),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void searchUser(BuildContext context, String text) async {
|
||||||
|
if (text.isEmpty) {
|
||||||
|
setState(() {
|
||||||
|
foundProfile = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
currentSearchTerm = text;
|
||||||
|
if (currentSearchTerm.isEmpty) return;
|
||||||
|
if (loading) return;
|
||||||
|
setState(() => loading = true);
|
||||||
|
final MatrixState matrix = Matrix.of(context);
|
||||||
|
final response = await matrix.tryRequestWithErrorToast(
|
||||||
|
matrix.client.jsonRequest(
|
||||||
|
type: HTTPType.POST,
|
||||||
|
action: "/client/r0/user_directory/search",
|
||||||
|
data: {
|
||||||
|
"search_term": text,
|
||||||
|
"limit": 1,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
print(response);
|
||||||
|
setState(() => loading = false);
|
||||||
|
if (response == false ||
|
||||||
|
!(response is Map) ||
|
||||||
|
(response["results"]?.isEmpty ?? true)) return;
|
||||||
|
print("Set...");
|
||||||
|
setState(() {
|
||||||
|
foundProfile = response["results"].first;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
final String defaultDomain = Matrix.of(context).client.userID.split(":")[1];
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
title: Text("New private chat"),
|
title: Text("New private chat"),
|
||||||
content: Column(
|
content: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Form(
|
Form(
|
||||||
key: _formKey,
|
key: _formKey,
|
||||||
|
@ -44,6 +100,7 @@ class NewPrivateChatDialog extends StatelessWidget {
|
||||||
controller: controller,
|
controller: controller,
|
||||||
autofocus: true,
|
autofocus: true,
|
||||||
autocorrect: false,
|
autocorrect: false,
|
||||||
|
onChanged: (String text) => searchUserWithCoolDown(context, text),
|
||||||
textInputAction: TextInputAction.go,
|
textInputAction: TextInputAction.go,
|
||||||
onFieldSubmitted: (s) => submitAction(context),
|
onFieldSubmitted: (s) => submitAction(context),
|
||||||
validator: (value) {
|
validator: (value) {
|
||||||
|
@ -55,30 +112,84 @@ class NewPrivateChatDialog extends StatelessWidget {
|
||||||
if (mxid == matrix.client.userID) {
|
if (mxid == matrix.client.userID) {
|
||||||
return "You cannot invite yourself";
|
return "You cannot invite yourself";
|
||||||
}
|
}
|
||||||
if(!mxid.contains("@")) {
|
if (!mxid.contains("@")) {
|
||||||
return "Make sure the identifier is valid";
|
return "Make sure the identifier is valid";
|
||||||
}
|
}
|
||||||
if(!mxid.contains(":")) {
|
if (!mxid.contains(":")) {
|
||||||
return "Make sure the identifier is valid";
|
return "Make sure the identifier is valid";
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: "Enter a username",
|
labelText: "Enter a username",
|
||||||
icon: Icon(Icons.account_circle),
|
icon: loading
|
||||||
|
? Container(
|
||||||
|
width: 24,
|
||||||
|
height: 24,
|
||||||
|
child: CircularProgressIndicator(strokeWidth: 1),
|
||||||
|
)
|
||||||
|
: correctMxId
|
||||||
|
? Avatar(
|
||||||
|
MxContent(foundProfile["avatar_url"] ?? ""),
|
||||||
|
foundProfile["display_name"] ??
|
||||||
|
foundProfile["user_id"],
|
||||||
|
size: 24,
|
||||||
|
)
|
||||||
|
: Icon(Icons.account_circle),
|
||||||
prefixText: "@",
|
prefixText: "@",
|
||||||
hintText: "username:homeserver",
|
hintText: "username:$defaultDomain",
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(height: 16),
|
if (foundProfile != null && !correctMxId)
|
||||||
Text(
|
ListTile(
|
||||||
"Your username is ${Matrix.of(context).client.userID}",
|
onTap: () {
|
||||||
style: TextStyle(
|
setState(() {
|
||||||
color: Colors.blueGrey,
|
controller.text =
|
||||||
fontSize: 12,
|
currentSearchTerm = foundProfile["user_id"].substring(1);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
leading: Avatar(
|
||||||
|
MxContent(foundProfile["avatar_url"] ?? ""),
|
||||||
|
foundProfile["display_name"] ?? foundProfile["user_id"],
|
||||||
|
//size: 24,
|
||||||
|
),
|
||||||
|
contentPadding: EdgeInsets.all(0),
|
||||||
|
title: Text(
|
||||||
|
foundProfile["display_name"] ??
|
||||||
|
foundProfile["user_id"].split(":").first.substring(1),
|
||||||
|
style: TextStyle(),
|
||||||
|
maxLines: 1,
|
||||||
|
),
|
||||||
|
subtitle: Text(
|
||||||
|
foundProfile["user_id"],
|
||||||
|
maxLines: 1,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 12,
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
if (foundProfile == null || correctMxId)
|
||||||
|
ListTile(
|
||||||
|
trailing: Icon(
|
||||||
|
Icons.share,
|
||||||
|
size: 16,
|
||||||
|
),
|
||||||
|
contentPadding: EdgeInsets.all(0),
|
||||||
|
onTap: () => Share.share(
|
||||||
|
"https://matrix.to/#/${Matrix.of(context).client.userID}"),
|
||||||
|
title: Text(
|
||||||
|
"Your own username:",
|
||||||
|
style: TextStyle(
|
||||||
|
color: Colors.blueGrey,
|
||||||
|
fontSize: 12,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
subtitle: Text(
|
||||||
|
Matrix.of(context).client.userID,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Divider(height: 1),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
actions: <Widget>[
|
actions: <Widget>[
|
||||||
|
|
|
@ -214,11 +214,8 @@ extension LocalizedBody on Event {
|
||||||
|
|
||||||
// Hide quotes
|
// Hide quotes
|
||||||
if (hideQuotes) {
|
if (hideQuotes) {
|
||||||
print("+++ Hide quites +++");
|
|
||||||
List<String> lines = localizedBody.split("\n");
|
List<String> lines = localizedBody.split("\n");
|
||||||
print("Lines with quotes: ${lines.length}");
|
|
||||||
lines.removeWhere((s) => s.startsWith("> "));
|
lines.removeWhere((s) => s.startsWith("> "));
|
||||||
print("Lines without quotes: ${lines.length}");
|
|
||||||
localizedBody = lines.join("\n");
|
localizedBody = lines.join("\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue