2024-05-26 13:39:21 +08:00
|
|
|
import 'package:flutter/material.dart';
|
2024-06-02 23:38:34 +08:00
|
|
|
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
2024-05-26 13:39:21 +08:00
|
|
|
import 'package:get/get.dart';
|
|
|
|
import 'package:solian/exts.dart';
|
|
|
|
import 'package:solian/models/account.dart';
|
|
|
|
import 'package:solian/models/channel.dart';
|
2024-06-28 00:05:43 +08:00
|
|
|
import 'package:solian/models/event.dart';
|
2024-08-01 22:36:00 +08:00
|
|
|
import 'package:solian/providers/attachment_uploader.dart';
|
2024-05-26 13:39:21 +08:00
|
|
|
import 'package:solian/providers/auth.dart';
|
2024-07-30 20:49:01 +08:00
|
|
|
import 'package:solian/widgets/attachments/attachment_editor.dart';
|
2024-06-28 00:59:11 +08:00
|
|
|
import 'package:solian/widgets/chat/chat_event.dart';
|
2024-08-01 22:36:00 +08:00
|
|
|
import 'package:badges/badges.dart' as badges;
|
2024-05-26 13:39:21 +08:00
|
|
|
import 'package:uuid/uuid.dart';
|
|
|
|
|
|
|
|
class ChatMessageInput extends StatefulWidget {
|
2024-06-28 00:05:43 +08:00
|
|
|
final Event? edit;
|
|
|
|
final Event? reply;
|
2024-05-30 22:02:54 +08:00
|
|
|
final String? placeholder;
|
2024-05-26 13:39:21 +08:00
|
|
|
final Channel channel;
|
|
|
|
final String realm;
|
2024-06-28 00:05:43 +08:00
|
|
|
final Function(Event) onSent;
|
2024-05-30 23:14:29 +08:00
|
|
|
final Function()? onReset;
|
2024-05-26 13:39:21 +08:00
|
|
|
|
|
|
|
const ChatMessageInput({
|
|
|
|
super.key,
|
|
|
|
this.edit,
|
|
|
|
this.reply,
|
2024-05-30 22:02:54 +08:00
|
|
|
this.placeholder,
|
2024-05-26 13:39:21 +08:00
|
|
|
required this.channel,
|
|
|
|
required this.realm,
|
|
|
|
required this.onSent,
|
2024-05-30 23:14:29 +08:00
|
|
|
this.onReset,
|
2024-05-26 13:39:21 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<ChatMessageInput> createState() => _ChatMessageInputState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ChatMessageInputState extends State<ChatMessageInput> {
|
|
|
|
final TextEditingController _textController = TextEditingController();
|
|
|
|
final FocusNode _focusNode = FocusNode();
|
|
|
|
|
2024-08-01 22:13:08 +08:00
|
|
|
final List<int> _attachments = List.empty(growable: true);
|
2024-05-26 13:39:21 +08:00
|
|
|
|
2024-06-28 00:05:43 +08:00
|
|
|
Event? _editTo;
|
|
|
|
Event? _replyTo;
|
2024-06-02 23:38:34 +08:00
|
|
|
|
2024-08-01 14:01:12 +08:00
|
|
|
void _editAttachments() {
|
2024-05-26 21:03:25 +08:00
|
|
|
showModalBottomSheet(
|
|
|
|
context: context,
|
2024-07-30 20:49:01 +08:00
|
|
|
builder: (context) => AttachmentEditorPopup(
|
2024-05-26 21:03:25 +08:00
|
|
|
usage: 'm.attachment',
|
2024-08-01 22:13:08 +08:00
|
|
|
initialAttachments: _attachments,
|
|
|
|
onAdd: (value) {
|
|
|
|
setState(() {
|
|
|
|
_attachments.add(value);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
onRemove: (value) {
|
|
|
|
setState(() {
|
|
|
|
_attachments.remove(value);
|
|
|
|
});
|
|
|
|
},
|
2024-05-26 21:03:25 +08:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-08-01 14:01:12 +08:00
|
|
|
List<String> _findMentionedUsers(String text) {
|
|
|
|
RegExp regExp = RegExp(r'@[a-zA-Z0-9_]+');
|
|
|
|
Iterable<RegExpMatch> matches = regExp.allMatches(text);
|
|
|
|
|
|
|
|
List<String> mentionedUsers =
|
|
|
|
matches.map((match) => match.group(0)!.substring(1)).toList();
|
|
|
|
|
|
|
|
return mentionedUsers;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _sendMessage() async {
|
2024-05-26 13:39:21 +08:00
|
|
|
_focusNode.requestFocus();
|
|
|
|
|
|
|
|
final AuthProvider auth = Get.find();
|
2024-07-25 01:18:47 +08:00
|
|
|
final prof = auth.userProfile.value!;
|
|
|
|
if (auth.isAuthorized.isFalse) return;
|
2024-05-26 13:39:21 +08:00
|
|
|
|
2024-08-01 22:36:00 +08:00
|
|
|
final AttachmentUploaderController uploader = Get.find();
|
|
|
|
if (uploader.queueOfUpload.any(
|
|
|
|
((x) => x.usage == 'm.attachment' && x.isUploading),
|
|
|
|
)) {
|
|
|
|
context.showErrorDialog('attachmentUploadInProgress'.tr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-08-01 14:01:12 +08:00
|
|
|
Response resp;
|
|
|
|
|
|
|
|
final mentionedUserNames = _findMentionedUsers(_textController.text);
|
|
|
|
final mentionedUserIds = List<int>.empty(growable: true);
|
2024-05-26 13:39:21 +08:00
|
|
|
|
2024-08-01 14:01:12 +08:00
|
|
|
var client = auth.configureClient('auth');
|
|
|
|
if (mentionedUserNames.isNotEmpty) {
|
|
|
|
resp = await client.get('/users?name=${mentionedUserNames.join(',')}');
|
|
|
|
if (resp.statusCode != 200) {
|
|
|
|
context.showErrorDialog(resp.bodyString);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
mentionedUserIds.addAll(
|
|
|
|
resp.body.map((x) => Account.fromJson(x).id).toList().cast<int>(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
client = auth.configureClient('messaging');
|
2024-06-28 00:59:11 +08:00
|
|
|
|
2024-06-28 04:34:35 +08:00
|
|
|
const uuid = Uuid();
|
2024-05-26 13:39:21 +08:00
|
|
|
final payload = {
|
2024-06-28 04:34:35 +08:00
|
|
|
'uuid': uuid.v4(),
|
2024-06-28 00:59:11 +08:00
|
|
|
'type': _editTo == null ? 'messages.new' : 'messages.edit',
|
|
|
|
'body': {
|
2024-08-01 14:01:12 +08:00
|
|
|
'text': _textController.text,
|
2024-06-28 00:59:11 +08:00
|
|
|
'algorithm': 'plain',
|
|
|
|
'attachments': List.from(_attachments),
|
|
|
|
'related_users': [
|
|
|
|
if (_replyTo != null) _replyTo!.sender.accountId,
|
2024-08-01 14:01:12 +08:00
|
|
|
...mentionedUserIds,
|
2024-06-28 00:59:11 +08:00
|
|
|
],
|
|
|
|
if (_replyTo != null) 'quote_event': _replyTo!.id,
|
|
|
|
if (_editTo != null) 'related_event': _editTo!.id,
|
2024-08-01 14:01:12 +08:00
|
|
|
if (_editTo != null && _editTo!.body['quote_event'] != null)
|
|
|
|
'quote_event': _editTo!.body['quote_event'],
|
2024-06-28 00:59:11 +08:00
|
|
|
}
|
2024-05-26 13:39:21 +08:00
|
|
|
};
|
|
|
|
|
2024-06-28 00:59:11 +08:00
|
|
|
// The local mock data
|
2024-05-26 13:39:21 +08:00
|
|
|
final sender = Sender(
|
|
|
|
id: 0,
|
|
|
|
createdAt: DateTime.now(),
|
|
|
|
updatedAt: DateTime.now(),
|
2024-07-25 01:18:47 +08:00
|
|
|
account: Account.fromJson(prof),
|
2024-05-26 13:39:21 +08:00
|
|
|
channelId: widget.channel.id,
|
2024-07-25 01:18:47 +08:00
|
|
|
accountId: prof['id'],
|
2024-05-26 13:39:21 +08:00
|
|
|
notify: 0,
|
|
|
|
);
|
2024-06-28 00:05:43 +08:00
|
|
|
final message = Event(
|
2024-05-26 13:39:21 +08:00
|
|
|
id: 0,
|
|
|
|
uuid: payload['uuid'] as String,
|
|
|
|
createdAt: DateTime.now(),
|
|
|
|
updatedAt: DateTime.now(),
|
2024-06-28 00:59:11 +08:00
|
|
|
body: payload['body'] as Map<String, dynamic>,
|
2024-05-26 13:39:21 +08:00
|
|
|
type: payload['type'] as String,
|
|
|
|
sender: sender,
|
|
|
|
channelId: widget.channel.id,
|
|
|
|
senderId: sender.id,
|
|
|
|
);
|
2024-06-04 23:29:05 +08:00
|
|
|
|
2024-06-08 21:35:50 +08:00
|
|
|
if (_editTo == null) {
|
2024-06-28 00:59:11 +08:00
|
|
|
message.isPending = true;
|
2024-06-08 21:35:50 +08:00
|
|
|
widget.onSent(message);
|
|
|
|
}
|
2024-05-26 13:39:21 +08:00
|
|
|
|
|
|
|
resetInput();
|
|
|
|
|
2024-06-02 23:38:34 +08:00
|
|
|
if (_editTo != null) {
|
2024-05-26 13:39:21 +08:00
|
|
|
resp = await client.put(
|
2024-07-16 19:46:53 +08:00
|
|
|
'/channels/${widget.realm}/${widget.channel.alias}/messages/${_editTo!.id}',
|
2024-05-26 13:39:21 +08:00
|
|
|
payload,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
resp = await client.post(
|
2024-07-16 19:46:53 +08:00
|
|
|
'/channels/${widget.realm}/${widget.channel.alias}/messages',
|
2024-05-26 13:39:21 +08:00
|
|
|
payload,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resp.statusCode != 200) {
|
|
|
|
context.showErrorDialog(resp.bodyString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetInput() {
|
2024-05-30 23:14:29 +08:00
|
|
|
if (widget.onReset != null) widget.onReset!();
|
2024-06-02 23:38:34 +08:00
|
|
|
_editTo = null;
|
|
|
|
_replyTo = null;
|
2024-05-26 13:39:21 +08:00
|
|
|
_textController.clear();
|
2024-06-08 21:35:50 +08:00
|
|
|
_attachments.clear();
|
2024-06-02 23:38:34 +08:00
|
|
|
setState(() {});
|
2024-05-26 13:39:21 +08:00
|
|
|
}
|
|
|
|
|
2024-05-30 23:14:29 +08:00
|
|
|
void syncWidget() {
|
2024-06-28 00:59:11 +08:00
|
|
|
if (widget.edit != null && widget.edit!.type.startsWith('messages')) {
|
|
|
|
final body = EventMessageBody.fromJson(widget.edit!.body);
|
2024-06-02 23:38:34 +08:00
|
|
|
_editTo = widget.edit!;
|
2024-06-28 00:59:11 +08:00
|
|
|
_textController.text = body.text;
|
2024-05-30 23:14:29 +08:00
|
|
|
}
|
2024-06-02 23:38:34 +08:00
|
|
|
if (widget.reply != null) {
|
|
|
|
_replyTo = widget.reply!;
|
|
|
|
}
|
|
|
|
|
|
|
|
setState(() {});
|
2024-05-30 23:14:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void didUpdateWidget(covariant ChatMessageInput oldWidget) {
|
|
|
|
syncWidget();
|
|
|
|
super.didUpdateWidget(oldWidget);
|
|
|
|
}
|
|
|
|
|
2024-05-26 13:39:21 +08:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-05-30 23:14:29 +08:00
|
|
|
final notifyBannerActions = [
|
|
|
|
TextButton(
|
|
|
|
onPressed: resetInput,
|
|
|
|
child: Text('cancel'.tr),
|
|
|
|
)
|
|
|
|
];
|
2024-05-26 13:39:21 +08:00
|
|
|
|
2024-06-08 21:35:50 +08:00
|
|
|
return Column(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
children: [
|
|
|
|
if (_replyTo != null)
|
|
|
|
MaterialBanner(
|
|
|
|
leading: const FaIcon(FontAwesomeIcons.reply, size: 18),
|
|
|
|
dividerColor: Colors.transparent,
|
2024-06-09 00:09:01 +08:00
|
|
|
padding: const EdgeInsets.only(left: 20),
|
|
|
|
backgroundColor: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.surfaceContainerHighest
|
|
|
|
.withOpacity(0.5),
|
2024-06-28 00:59:11 +08:00
|
|
|
content: ChatEvent(
|
2024-06-08 21:35:50 +08:00
|
|
|
item: _replyTo!,
|
|
|
|
isContentPreviewing: true,
|
2024-06-05 23:55:21 +08:00
|
|
|
),
|
2024-06-08 21:35:50 +08:00
|
|
|
actions: notifyBannerActions,
|
|
|
|
),
|
|
|
|
if (_editTo != null)
|
|
|
|
MaterialBanner(
|
|
|
|
leading: const Icon(Icons.edit),
|
|
|
|
dividerColor: Colors.transparent,
|
2024-06-09 00:09:01 +08:00
|
|
|
padding: const EdgeInsets.only(left: 20),
|
|
|
|
backgroundColor: Theme.of(context)
|
|
|
|
.colorScheme
|
|
|
|
.surfaceContainerHighest
|
|
|
|
.withOpacity(0.5),
|
2024-06-28 00:59:11 +08:00
|
|
|
content: ChatEvent(
|
2024-06-08 21:35:50 +08:00
|
|
|
item: _editTo!,
|
|
|
|
isContentPreviewing: true,
|
2024-06-05 23:55:21 +08:00
|
|
|
),
|
2024-06-08 21:35:50 +08:00
|
|
|
actions: notifyBannerActions,
|
|
|
|
),
|
|
|
|
SizedBox(
|
|
|
|
height: 56,
|
|
|
|
child: Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
Expanded(
|
|
|
|
child: TextField(
|
|
|
|
controller: _textController,
|
|
|
|
focusNode: _focusNode,
|
|
|
|
maxLines: null,
|
|
|
|
autocorrect: true,
|
|
|
|
keyboardType: TextInputType.text,
|
|
|
|
decoration: InputDecoration.collapsed(
|
|
|
|
hintText: widget.placeholder ??
|
|
|
|
'messageInputPlaceholder'.trParams(
|
|
|
|
{'channel': '#${widget.channel.alias}'},
|
|
|
|
),
|
2024-05-30 23:14:29 +08:00
|
|
|
),
|
2024-08-01 14:01:12 +08:00
|
|
|
onSubmitted: (_) => _sendMessage(),
|
2024-06-08 21:35:50 +08:00
|
|
|
onTapOutside: (_) =>
|
|
|
|
FocusManager.instance.primaryFocus?.unfocus(),
|
2024-06-05 23:55:21 +08:00
|
|
|
),
|
2024-06-08 21:35:50 +08:00
|
|
|
),
|
|
|
|
IconButton(
|
2024-08-01 22:36:00 +08:00
|
|
|
icon: badges.Badge(
|
|
|
|
badgeContent: Text(
|
|
|
|
_attachments.length.toString(),
|
|
|
|
style: const TextStyle(color: Colors.white),
|
|
|
|
),
|
|
|
|
showBadge: _attachments.isNotEmpty,
|
|
|
|
position: badges.BadgePosition.topEnd(
|
|
|
|
top: -12,
|
|
|
|
end: -8,
|
|
|
|
),
|
|
|
|
child: const Icon(Icons.file_present_rounded),
|
|
|
|
),
|
2024-06-08 21:35:50 +08:00
|
|
|
color: Colors.teal,
|
2024-08-01 14:01:12 +08:00
|
|
|
onPressed: () => _editAttachments(),
|
2024-06-08 21:35:50 +08:00
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
icon: const Icon(Icons.send),
|
|
|
|
color: Theme.of(context).colorScheme.primary,
|
2024-08-01 14:01:12 +08:00
|
|
|
onPressed: () => _sendMessage(),
|
2024-06-08 21:35:50 +08:00
|
|
|
)
|
|
|
|
],
|
|
|
|
).paddingOnly(left: 20, right: 16),
|
|
|
|
),
|
|
|
|
],
|
2024-05-26 13:39:21 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|