159 lines
4.3 KiB
Dart
159 lines
4.3 KiB
Dart
|
import 'package:flutter/material.dart';
|
||
|
import 'package:get/get.dart';
|
||
|
import 'package:solian/exts.dart';
|
||
|
import 'package:solian/models/account.dart';
|
||
|
import 'package:solian/models/channel.dart';
|
||
|
import 'package:solian/models/message.dart';
|
||
|
import 'package:solian/providers/auth.dart';
|
||
|
import 'package:solian/services.dart';
|
||
|
import 'package:uuid/uuid.dart';
|
||
|
|
||
|
class ChatMessageInput extends StatefulWidget {
|
||
|
final Message? edit;
|
||
|
final Message? reply;
|
||
|
final Channel channel;
|
||
|
final String realm;
|
||
|
final Function(Message) onSent;
|
||
|
|
||
|
const ChatMessageInput({
|
||
|
super.key,
|
||
|
this.edit,
|
||
|
this.reply,
|
||
|
required this.channel,
|
||
|
required this.realm,
|
||
|
required this.onSent,
|
||
|
});
|
||
|
|
||
|
@override
|
||
|
State<ChatMessageInput> createState() => _ChatMessageInputState();
|
||
|
}
|
||
|
|
||
|
class _ChatMessageInputState extends State<ChatMessageInput> {
|
||
|
final TextEditingController _textController = TextEditingController();
|
||
|
final FocusNode _focusNode = FocusNode();
|
||
|
|
||
|
List<int> _attachments = List.empty(growable: true);
|
||
|
|
||
|
Map<String, dynamic> encodeMessage(String content) {
|
||
|
// TODO Impl E2EE
|
||
|
|
||
|
return {
|
||
|
'value': content,
|
||
|
'keypair_id': null,
|
||
|
'algorithm': 'plain',
|
||
|
};
|
||
|
}
|
||
|
|
||
|
Future<void> sendMessage() async {
|
||
|
_focusNode.requestFocus();
|
||
|
|
||
|
final AuthProvider auth = Get.find();
|
||
|
final prof = await auth.getProfile();
|
||
|
if (!await auth.isAuthorized) return;
|
||
|
|
||
|
final client = GetConnect(maxAuthRetries: 3);
|
||
|
client.httpClient.baseUrl = ServiceFinder.services['messaging'];
|
||
|
client.httpClient.addAuthenticator(auth.requestAuthenticator);
|
||
|
|
||
|
final payload = {
|
||
|
'uuid': const Uuid().v4(),
|
||
|
'type': 'm.text',
|
||
|
'content': encodeMessage(_textController.value.text),
|
||
|
'attachments': _attachments,
|
||
|
'reply_to': widget.reply?.id,
|
||
|
};
|
||
|
|
||
|
// The mock data
|
||
|
final sender = Sender(
|
||
|
id: 0,
|
||
|
createdAt: DateTime.now(),
|
||
|
updatedAt: DateTime.now(),
|
||
|
account: Account.fromJson(prof.body),
|
||
|
channelId: widget.channel.id,
|
||
|
accountId: prof.body['id'],
|
||
|
notify: 0,
|
||
|
);
|
||
|
final message = Message(
|
||
|
id: 0,
|
||
|
uuid: payload['uuid'] as String,
|
||
|
createdAt: DateTime.now(),
|
||
|
updatedAt: DateTime.now(),
|
||
|
content: payload['content'] as Map<String, dynamic>,
|
||
|
type: payload['type'] as String,
|
||
|
sender: sender,
|
||
|
replyId: widget.reply?.id,
|
||
|
replyTo: widget.reply,
|
||
|
channelId: widget.channel.id,
|
||
|
senderId: sender.id,
|
||
|
);
|
||
|
|
||
|
widget.onSent(message);
|
||
|
resetInput();
|
||
|
|
||
|
Response resp;
|
||
|
if (widget.edit != null) {
|
||
|
resp = await client.put(
|
||
|
'/api/channels/${widget.realm}/${widget.channel.alias}/messages/${widget.edit!.id}',
|
||
|
payload,
|
||
|
);
|
||
|
} else {
|
||
|
resp = await client.post(
|
||
|
'/api/channels/${widget.realm}/${widget.channel.alias}/messages',
|
||
|
payload,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
if (resp.statusCode != 200) {
|
||
|
context.showErrorDialog(resp.bodyString);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void resetInput() {
|
||
|
_textController.clear();
|
||
|
}
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
const double height = 56;
|
||
|
const borderRadius = BorderRadius.all(Radius.circular(height / 2));
|
||
|
|
||
|
return Material(
|
||
|
borderRadius: borderRadius,
|
||
|
elevation: 2,
|
||
|
child: ClipRRect(
|
||
|
borderRadius: borderRadius,
|
||
|
child: SizedBox(
|
||
|
height: height,
|
||
|
child: Row(
|
||
|
mainAxisAlignment: MainAxisAlignment.start,
|
||
|
children: [
|
||
|
Expanded(
|
||
|
child: TextField(
|
||
|
controller: _textController,
|
||
|
focusNode: _focusNode,
|
||
|
maxLines: null,
|
||
|
autocorrect: true,
|
||
|
keyboardType: TextInputType.text,
|
||
|
decoration: InputDecoration.collapsed(
|
||
|
hintText: 'messageInputPlaceholder'.trParams(
|
||
|
{'channel': '#${widget.channel.alias}'},
|
||
|
),
|
||
|
),
|
||
|
onSubmitted: (_) => sendMessage(),
|
||
|
onTapOutside: (_) =>
|
||
|
FocusManager.instance.primaryFocus?.unfocus(),
|
||
|
),
|
||
|
),
|
||
|
IconButton(
|
||
|
icon: const Icon(Icons.send),
|
||
|
color: Theme.of(context).colorScheme.primary,
|
||
|
onPressed: () => sendMessage(),
|
||
|
)
|
||
|
],
|
||
|
).paddingOnly(left: 16, right: 4),
|
||
|
),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|