mirror of
https://github.com/twonlyapp/twonly-app.git
synced 2026-01-15 21:38:39 +00:00
228 lines
6.5 KiB
Dart
228 lines
6.5 KiB
Dart
import 'dart:collection';
|
|
import 'dart:convert';
|
|
|
|
import 'package:flutter/material.dart';
|
|
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
import 'package:twonly/src/database/tables/messages_table.dart';
|
|
import 'package:twonly/src/database/twonly_database.dart';
|
|
import 'package:twonly/src/model/json/message.dart';
|
|
import 'package:twonly/src/utils/misc.dart';
|
|
import 'package:twonly/src/views/components/animate_icon.dart';
|
|
|
|
enum MessageSendState {
|
|
received,
|
|
receivedOpened,
|
|
receiving,
|
|
send,
|
|
sendOpened,
|
|
sending,
|
|
}
|
|
|
|
MessageSendState messageSendStateFromMessage(Message msg) {
|
|
MessageSendState state;
|
|
|
|
if (!msg.acknowledgeByServer) {
|
|
if (msg.messageOtherId == null) {
|
|
state = MessageSendState.sending;
|
|
} else {
|
|
state = MessageSendState.receiving;
|
|
}
|
|
} else {
|
|
if (msg.messageOtherId == null) {
|
|
// message send
|
|
if (msg.openedAt == null) {
|
|
state = MessageSendState.send;
|
|
} else {
|
|
state = MessageSendState.sendOpened;
|
|
}
|
|
} else {
|
|
// message received
|
|
if (msg.openedAt == null) {
|
|
state = MessageSendState.received;
|
|
} else {
|
|
state = MessageSendState.receivedOpened;
|
|
}
|
|
}
|
|
}
|
|
return state;
|
|
}
|
|
|
|
class MessageSendStateIcon extends StatefulWidget {
|
|
const MessageSendStateIcon(
|
|
this.messages, {
|
|
super.key,
|
|
this.canBeReopened = false,
|
|
this.mainAxisAlignment = MainAxisAlignment.end,
|
|
});
|
|
final List<Message> messages;
|
|
final MainAxisAlignment mainAxisAlignment;
|
|
final bool canBeReopened;
|
|
|
|
@override
|
|
State<MessageSendStateIcon> createState() => _MessageSendStateIconState();
|
|
}
|
|
|
|
class _MessageSendStateIconState extends State<MessageSendStateIcon> {
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
}
|
|
|
|
Widget getLoaderIcon(Color color) {
|
|
return Row(
|
|
children: [
|
|
SizedBox(
|
|
width: 10,
|
|
height: 10,
|
|
child: CircularProgressIndicator(strokeWidth: 1, color: color),
|
|
),
|
|
const SizedBox(width: 2),
|
|
],
|
|
);
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final icons = <Widget>[];
|
|
var text = '';
|
|
|
|
final kindsAlreadyShown = HashSet<MessageKind>();
|
|
Widget? textWidget;
|
|
|
|
for (final message in widget.messages) {
|
|
if (icons.length == 2) break;
|
|
if (kindsAlreadyShown.contains(message.kind)) continue;
|
|
kindsAlreadyShown.add(message.kind);
|
|
|
|
final state = messageSendStateFromMessage(message);
|
|
late Color color;
|
|
MessageContent? content;
|
|
|
|
if (message.contentJson == null) {
|
|
color = getMessageColorFromType(TextMessageContent(text: ''), context);
|
|
} else {
|
|
content = MessageContent.fromJson(
|
|
message.kind,
|
|
jsonDecode(message.contentJson!) as Map,
|
|
);
|
|
if (content == null) continue;
|
|
color = getMessageColorFromType(content, context);
|
|
}
|
|
|
|
Widget icon = const Placeholder();
|
|
textWidget = null;
|
|
|
|
switch (state) {
|
|
case MessageSendState.receivedOpened:
|
|
icon = Icon(Icons.crop_square, size: 14, color: color);
|
|
if (content is TextMessageContent) {
|
|
if (isEmoji(content.text)) {
|
|
icon = Text(content.text, style: const TextStyle(fontSize: 12));
|
|
}
|
|
}
|
|
text = context.lang.messageSendState_Received;
|
|
if (widget.canBeReopened) {
|
|
textWidget = Text(
|
|
context.lang.doubleClickToReopen,
|
|
style: const TextStyle(fontSize: 9),
|
|
);
|
|
}
|
|
case MessageSendState.sendOpened:
|
|
icon = FaIcon(FontAwesomeIcons.paperPlane, size: 12, color: color);
|
|
text = context.lang.messageSendState_Opened;
|
|
case MessageSendState.received:
|
|
icon = Icon(Icons.square_rounded, size: 14, color: color);
|
|
text = context.lang.messageSendState_Received;
|
|
if (message.kind == MessageKind.media) {
|
|
if (message.downloadState == DownloadState.pending) {
|
|
text = context.lang.messageSendState_TapToLoad;
|
|
}
|
|
if (message.downloadState == DownloadState.downloading) {
|
|
text = context.lang.messageSendState_Loading;
|
|
icon = getLoaderIcon(color);
|
|
}
|
|
}
|
|
case MessageSendState.send:
|
|
icon =
|
|
FaIcon(FontAwesomeIcons.solidPaperPlane, size: 12, color: color);
|
|
text = context.lang.messageSendState_Send;
|
|
case MessageSendState.sending:
|
|
icon = getLoaderIcon(color);
|
|
text = context.lang.messageSendState_Sending;
|
|
case MessageSendState.receiving:
|
|
icon = getLoaderIcon(color);
|
|
text = context.lang.messageSendState_Received;
|
|
}
|
|
|
|
if (message.kind == MessageKind.storedMediaFile) {
|
|
icon = FaIcon(FontAwesomeIcons.floppyDisk, size: 12, color: color);
|
|
text = context.lang.messageStoredInGallery;
|
|
}
|
|
|
|
if (message.kind == MessageKind.reopenedMedia) {
|
|
icon = FaIcon(FontAwesomeIcons.repeat, size: 12, color: color);
|
|
text = context.lang.messageReopened;
|
|
}
|
|
|
|
if (message.errorWhileSending) {
|
|
icon =
|
|
FaIcon(FontAwesomeIcons.circleExclamation, size: 12, color: color);
|
|
text = 'Error';
|
|
}
|
|
|
|
if (message.mediaRetransmissionState == MediaRetransmitting.requested) {
|
|
icon = FaIcon(FontAwesomeIcons.clockRotateLeft, size: 12, color: color);
|
|
textWidget = Text(
|
|
context.lang.retransmissionRequested,
|
|
style: const TextStyle(fontSize: 9),
|
|
);
|
|
}
|
|
|
|
if (message.kind == MessageKind.media) {
|
|
icons.insert(0, icon);
|
|
} else {
|
|
icons.add(icon);
|
|
}
|
|
}
|
|
|
|
if (icons.isEmpty) return Container();
|
|
|
|
var icon = icons[0];
|
|
|
|
if (icons.length == 2) {
|
|
icon = Stack(
|
|
alignment: Alignment.center,
|
|
children: <Widget>[
|
|
// First icon (bottom icon)
|
|
icons[0],
|
|
|
|
Transform(
|
|
transform: Matrix4.identity()
|
|
..scale(0.7) // Scale to half
|
|
..translate(3.0, 5),
|
|
// Move down by 10 pixels (adjust as needed)
|
|
alignment: Alignment.center,
|
|
child: icons[1],
|
|
),
|
|
// Second icon (top icon, slightly offset)
|
|
],
|
|
);
|
|
}
|
|
|
|
return Row(
|
|
mainAxisAlignment: widget.mainAxisAlignment,
|
|
children: [
|
|
icon,
|
|
const SizedBox(width: 3),
|
|
if (textWidget != null)
|
|
textWidget
|
|
else
|
|
Text(
|
|
text,
|
|
style: const TextStyle(fontSize: 12),
|
|
),
|
|
const SizedBox(width: 5),
|
|
],
|
|
);
|
|
}
|
|
}
|