2020-01-01 19:10:13 +01:00
|
|
|
import 'package:flutter/material.dart';
|
2021-10-26 18:50:34 +02:00
|
|
|
|
2020-10-03 13:11:07 +02:00
|
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
2021-10-26 18:50:34 +02:00
|
|
|
import 'package:matrix/matrix.dart';
|
2020-09-05 13:45:03 +02:00
|
|
|
import 'package:matrix_link_text/link_text.dart';
|
2020-10-03 13:11:07 +02:00
|
|
|
|
2021-12-27 09:35:07 +01:00
|
|
|
import 'package:fluffychat/pages/chat/events/video_player.dart';
|
2023-01-07 10:29:34 +01:00
|
|
|
import 'package:fluffychat/utils/adaptive_bottom_sheet.dart';
|
2022-09-08 09:12:08 +02:00
|
|
|
import 'package:fluffychat/utils/date_time_extension.dart';
|
2022-12-30 17:54:01 +01:00
|
|
|
import 'package:fluffychat/utils/matrix_sdk_extensions/matrix_locals.dart';
|
2022-09-08 09:12:08 +02:00
|
|
|
import 'package:fluffychat/widgets/avatar.dart';
|
2021-11-09 21:32:16 +01:00
|
|
|
import 'package:fluffychat/widgets/matrix.dart';
|
|
|
|
import '../../../config/app_config.dart';
|
|
|
|
import '../../../utils/platform_infos.dart';
|
|
|
|
import '../../../utils/url_launcher.dart';
|
2021-11-19 09:22:43 +01:00
|
|
|
import '../../bootstrap/bootstrap_dialog.dart';
|
2021-11-09 21:32:16 +01:00
|
|
|
import 'audio_player.dart';
|
2022-11-15 11:39:50 +01:00
|
|
|
import 'cute_events.dart';
|
2021-10-26 18:50:34 +02:00
|
|
|
import 'html_message.dart';
|
2021-11-09 21:32:16 +01:00
|
|
|
import 'image_bubble.dart';
|
2021-08-01 09:53:43 +02:00
|
|
|
import 'map_bubble.dart';
|
2021-10-26 18:50:34 +02:00
|
|
|
import 'message_download_content.dart';
|
2021-08-07 19:36:42 +02:00
|
|
|
import 'sticker.dart';
|
2020-01-01 19:10:13 +01:00
|
|
|
|
|
|
|
class MessageContent extends StatelessWidget {
|
|
|
|
final Event event;
|
2022-02-14 18:44:37 +01:00
|
|
|
final Color textColor;
|
2022-01-29 12:35:03 +01:00
|
|
|
final void Function(Event)? onInfoTab;
|
2020-01-01 19:10:13 +01:00
|
|
|
|
2022-02-14 18:44:37 +01:00
|
|
|
const MessageContent(this.event,
|
|
|
|
{this.onInfoTab, Key? key, required this.textColor})
|
2021-11-13 13:06:36 +01:00
|
|
|
: super(key: key);
|
2020-01-01 19:10:13 +01:00
|
|
|
|
2020-11-22 11:46:31 +01:00
|
|
|
void _verifyOrRequestKey(BuildContext context) async {
|
2022-09-08 09:12:08 +02:00
|
|
|
final l10n = L10n.of(context)!;
|
2020-11-22 11:46:31 +01:00
|
|
|
if (event.content['can_request_session'] != true) {
|
2021-05-23 13:11:55 +02:00
|
|
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
2021-04-03 13:09:20 +02:00
|
|
|
content: Text(
|
|
|
|
event.type == EventTypes.Encrypted
|
2022-09-08 09:12:08 +02:00
|
|
|
? l10n.needPantalaimonWarning
|
2022-05-30 13:44:05 +02:00
|
|
|
: event.calcLocalizedBodyFallback(
|
2022-09-08 09:12:08 +02:00
|
|
|
MatrixLocals(l10n),
|
2020-11-22 11:46:31 +01:00
|
|
|
),
|
2021-04-03 13:09:20 +02:00
|
|
|
)));
|
2020-11-22 11:46:31 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
final client = Matrix.of(context).client;
|
2022-01-29 12:35:03 +01:00
|
|
|
if (client.isUnknownSession && client.encryption!.crossSigning.enabled) {
|
2022-09-08 09:12:08 +02:00
|
|
|
final success = await BootstrapDialog(
|
2021-11-19 09:22:43 +01:00
|
|
|
client: Matrix.of(context).client,
|
|
|
|
).show(context);
|
2022-09-08 09:12:08 +02:00
|
|
|
if (success != true) return;
|
2020-11-22 11:46:31 +01:00
|
|
|
}
|
2022-09-08 09:12:08 +02:00
|
|
|
event.requestKey();
|
|
|
|
final sender = event.senderFromMemoryOrFallback;
|
2023-01-07 10:29:34 +01:00
|
|
|
await showAdaptiveBottomSheet(
|
2022-09-08 09:12:08 +02:00
|
|
|
context: context,
|
|
|
|
builder: (context) => Scaffold(
|
|
|
|
appBar: AppBar(
|
|
|
|
leading: CloseButton(onPressed: Navigator.of(context).pop),
|
|
|
|
title: Text(
|
|
|
|
l10n.whyIsThisMessageEncrypted,
|
|
|
|
style: const TextStyle(fontSize: 16),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
body: SafeArea(
|
|
|
|
child: ListView(
|
|
|
|
padding: const EdgeInsets.all(16),
|
|
|
|
children: [
|
|
|
|
ListTile(
|
|
|
|
contentPadding: EdgeInsets.zero,
|
|
|
|
leading: Avatar(
|
|
|
|
mxContent: sender.avatarUrl,
|
|
|
|
name: sender.calcDisplayname(),
|
|
|
|
),
|
|
|
|
title: Text(sender.calcDisplayname()),
|
|
|
|
subtitle: Text(event.originServerTs.localizedTime(context)),
|
|
|
|
trailing: const Icon(Icons.lock_outlined),
|
|
|
|
),
|
|
|
|
const Divider(),
|
|
|
|
Text(
|
|
|
|
event.calcLocalizedBodyFallback(
|
|
|
|
MatrixLocals(l10n),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
2020-11-22 11:46:31 +01:00
|
|
|
}
|
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2021-11-13 13:06:36 +01:00
|
|
|
final fontSize = AppConfig.messageFontSize * AppConfig.fontSizeFactor;
|
2021-11-15 07:59:51 +01:00
|
|
|
final buttonTextColor =
|
|
|
|
event.senderId == Matrix.of(context).client.userID ? textColor : null;
|
2020-01-01 19:10:13 +01:00
|
|
|
switch (event.type) {
|
2020-01-04 13:53:49 +01:00
|
|
|
case EventTypes.Message:
|
2020-02-21 09:45:37 +01:00
|
|
|
case EventTypes.Encrypted:
|
2020-01-04 13:53:49 +01:00
|
|
|
case EventTypes.Sticker:
|
2020-03-29 20:13:25 +02:00
|
|
|
switch (event.messageType) {
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.Image:
|
2021-11-19 10:01:35 +01:00
|
|
|
return ImageBubble(
|
|
|
|
event,
|
|
|
|
width: 400,
|
|
|
|
height: 300,
|
|
|
|
fit: BoxFit.cover,
|
|
|
|
);
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.Sticker:
|
2022-07-29 18:24:59 +02:00
|
|
|
if (event.redacted) continue textmessage;
|
2021-11-19 10:01:35 +01:00
|
|
|
return Sticker(event);
|
2022-11-15 11:39:50 +01:00
|
|
|
case CuteEventContent.eventType:
|
|
|
|
return CuteContent(event);
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.Audio:
|
2023-02-14 09:15:47 +01:00
|
|
|
if (PlatformInfos.isMobile ||
|
2023-02-14 17:21:06 +01:00
|
|
|
PlatformInfos.isMacOS ||
|
|
|
|
PlatformInfos.isWeb
|
|
|
|
// Disabled until https://github.com/bleonard252/just_audio_mpv/issues/3
|
|
|
|
// is fixed
|
|
|
|
// || PlatformInfos.isLinux
|
|
|
|
) {
|
2021-08-21 13:40:36 +02:00
|
|
|
return AudioPlayerWidget(
|
|
|
|
event,
|
2022-02-14 18:44:37 +01:00
|
|
|
color: textColor,
|
2021-08-21 13:40:36 +02:00
|
|
|
);
|
|
|
|
}
|
2022-02-14 18:44:37 +01:00
|
|
|
return MessageDownloadContent(event, textColor);
|
2020-03-13 21:58:48 +01:00
|
|
|
case MessageTypes.Video:
|
2021-12-22 09:05:09 +01:00
|
|
|
if (PlatformInfos.isMobile || PlatformInfos.isWeb) {
|
2021-12-27 09:35:07 +01:00
|
|
|
return EventVideoPlayer(event);
|
2021-08-08 17:55:00 +02:00
|
|
|
}
|
2022-02-14 18:44:37 +01:00
|
|
|
return MessageDownloadContent(event, textColor);
|
2020-03-13 21:58:48 +01:00
|
|
|
case MessageTypes.File:
|
2022-02-14 18:44:37 +01:00
|
|
|
return MessageDownloadContent(event, textColor);
|
2021-08-21 13:40:36 +02:00
|
|
|
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.Text:
|
2020-05-09 13:36:41 +02:00
|
|
|
case MessageTypes.Notice:
|
|
|
|
case MessageTypes.Emote:
|
2020-11-07 12:00:41 +01:00
|
|
|
if (AppConfig.renderHtml &&
|
2020-05-13 15:58:59 +02:00
|
|
|
!event.redacted &&
|
2020-09-21 09:44:13 +02:00
|
|
|
event.isRichMessage) {
|
2020-12-27 16:47:03 +01:00
|
|
|
var html = event.formattedText;
|
2020-05-09 13:36:41 +02:00
|
|
|
if (event.messageType == MessageTypes.Emote) {
|
2020-05-13 15:58:59 +02:00
|
|
|
html = '* $html';
|
2020-05-09 13:36:41 +02:00
|
|
|
}
|
2020-09-20 19:16:23 +02:00
|
|
|
final bigEmotes = event.onlyEmotes &&
|
|
|
|
event.numberEmotes > 0 &&
|
|
|
|
event.numberEmotes <= 10;
|
2020-05-09 13:36:41 +02:00
|
|
|
return HtmlMessage(
|
|
|
|
html: html,
|
2020-05-15 07:47:32 +02:00
|
|
|
defaultTextStyle: TextStyle(
|
|
|
|
color: textColor,
|
2020-09-20 11:35:28 +02:00
|
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
2020-05-15 07:47:32 +02:00
|
|
|
),
|
2021-01-27 20:08:09 +01:00
|
|
|
linkStyle: TextStyle(
|
2022-02-14 18:44:37 +01:00
|
|
|
color: textColor.withAlpha(150),
|
2021-01-27 20:08:09 +01:00
|
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
|
|
|
decoration: TextDecoration.underline,
|
2023-02-14 14:05:18 +01:00
|
|
|
decorationColor: textColor.withAlpha(150),
|
2021-01-27 20:08:09 +01:00
|
|
|
),
|
2020-05-14 07:43:21 +02:00
|
|
|
room: event.room,
|
2020-09-20 11:35:28 +02:00
|
|
|
emoteSize: bigEmotes ? fontSize * 3 : fontSize * 1.5,
|
2020-05-09 13:36:41 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
// else we fall through to the normal message rendering
|
|
|
|
continue textmessage;
|
|
|
|
case MessageTypes.BadEncrypted:
|
2020-11-22 11:46:31 +01:00
|
|
|
case EventTypes.Encrypted:
|
2021-11-13 13:06:36 +01:00
|
|
|
return _ButtonContent(
|
2021-11-15 07:59:51 +01:00
|
|
|
textColor: buttonTextColor,
|
2020-11-22 11:46:31 +01:00
|
|
|
onPressed: () => _verifyOrRequestKey(context),
|
2021-10-14 18:09:30 +02:00
|
|
|
icon: const Icon(Icons.lock_outline),
|
2022-01-29 12:35:03 +01:00
|
|
|
label: L10n.of(context)!.encrypted,
|
2020-11-22 11:46:31 +01:00
|
|
|
);
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.Location:
|
2021-08-01 09:53:43 +02:00
|
|
|
final geoUri =
|
2022-01-29 12:35:03 +01:00
|
|
|
Uri.tryParse(event.content.tryGet<String>('geo_uri')!);
|
|
|
|
if (geoUri != null && geoUri.scheme == 'geo') {
|
2021-08-01 09:53:43 +02:00
|
|
|
final latlong = geoUri.path
|
|
|
|
.split(';')
|
|
|
|
.first
|
|
|
|
.split(',')
|
|
|
|
.map((s) => double.tryParse(s))
|
|
|
|
.toList();
|
|
|
|
if (latlong.length == 2 &&
|
|
|
|
latlong.first != null &&
|
|
|
|
latlong.last != null) {
|
|
|
|
return Column(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
|
|
|
MapBubble(
|
2022-01-29 12:35:03 +01:00
|
|
|
latitude: latlong.first!,
|
|
|
|
longitude: latlong.last!,
|
2021-08-01 09:53:43 +02:00
|
|
|
),
|
2021-10-14 18:09:30 +02:00
|
|
|
const SizedBox(height: 6),
|
2021-08-01 09:53:43 +02:00
|
|
|
OutlinedButton.icon(
|
|
|
|
icon: Icon(Icons.location_on_outlined, color: textColor),
|
|
|
|
onPressed:
|
|
|
|
UrlLauncher(context, geoUri.toString()).launchUrl,
|
|
|
|
label: Text(
|
2022-01-29 12:35:03 +01:00
|
|
|
L10n.of(context)!.openInMaps,
|
2021-08-01 09:53:43 +02:00
|
|
|
style: TextStyle(color: textColor),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue textmessage;
|
2020-01-04 13:53:49 +01:00
|
|
|
case MessageTypes.None:
|
2020-05-09 13:36:41 +02:00
|
|
|
textmessage:
|
2020-02-21 09:45:37 +01:00
|
|
|
default:
|
2020-11-22 21:45:23 +01:00
|
|
|
if (event.redacted) {
|
2022-05-30 13:44:05 +02:00
|
|
|
return FutureBuilder<User?>(
|
2022-12-30 09:01:43 +01:00
|
|
|
future: event.redactedBecause?.fetchSenderUser(),
|
2022-05-30 13:44:05 +02:00
|
|
|
builder: (context, snapshot) {
|
|
|
|
return _ButtonContent(
|
|
|
|
label: L10n.of(context)!.redactedAnEvent(snapshot.data
|
|
|
|
?.calcDisplayname() ??
|
|
|
|
event.senderFromMemoryOrFallback.calcDisplayname()),
|
|
|
|
icon: const Icon(Icons.delete_outlined),
|
|
|
|
textColor: buttonTextColor,
|
|
|
|
onPressed: () => onInfoTab!(event),
|
|
|
|
);
|
|
|
|
});
|
2020-11-22 21:45:23 +01:00
|
|
|
}
|
2020-09-20 19:16:23 +02:00
|
|
|
final bigEmotes = event.onlyEmotes &&
|
|
|
|
event.numberEmotes > 0 &&
|
|
|
|
event.numberEmotes <= 10;
|
2022-05-30 13:44:05 +02:00
|
|
|
return FutureBuilder<String>(
|
|
|
|
future: event.calcLocalizedBody(MatrixLocals(L10n.of(context)!),
|
|
|
|
hideReply: true),
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
return LinkText(
|
|
|
|
text: snapshot.data ??
|
|
|
|
event.calcLocalizedBodyFallback(
|
|
|
|
MatrixLocals(L10n.of(context)!),
|
|
|
|
hideReply: true),
|
|
|
|
textStyle: TextStyle(
|
|
|
|
color: textColor,
|
|
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
|
|
|
decoration:
|
|
|
|
event.redacted ? TextDecoration.lineThrough : null,
|
|
|
|
),
|
|
|
|
linkStyle: TextStyle(
|
|
|
|
color: textColor.withAlpha(150),
|
|
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
|
|
|
decoration: TextDecoration.underline,
|
2023-02-14 14:05:18 +01:00
|
|
|
decorationColor: textColor.withAlpha(150),
|
2022-05-30 13:44:05 +02:00
|
|
|
),
|
|
|
|
onLinkTap: (url) => UrlLauncher(context, url).launchUrl(),
|
|
|
|
);
|
|
|
|
});
|
2020-01-04 09:37:09 +01:00
|
|
|
}
|
2022-02-19 11:05:21 +01:00
|
|
|
case EventTypes.CallInvite:
|
2022-05-30 13:44:05 +02:00
|
|
|
return FutureBuilder<User?>(
|
|
|
|
future: event.fetchSenderUser(),
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
return _ButtonContent(
|
|
|
|
label: L10n.of(context)!.startedACall(
|
|
|
|
snapshot.data?.calcDisplayname() ??
|
|
|
|
event.senderFromMemoryOrFallback.calcDisplayname()),
|
|
|
|
icon: const Icon(Icons.phone_outlined),
|
|
|
|
textColor: buttonTextColor,
|
|
|
|
onPressed: () => onInfoTab!(event),
|
|
|
|
);
|
|
|
|
});
|
2020-01-19 15:07:42 +01:00
|
|
|
default:
|
2022-05-30 13:44:05 +02:00
|
|
|
return FutureBuilder<User?>(
|
|
|
|
future: event.fetchSenderUser(),
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
return _ButtonContent(
|
|
|
|
label: L10n.of(context)!.userSentUnknownEvent(
|
|
|
|
snapshot.data?.calcDisplayname() ??
|
|
|
|
event.senderFromMemoryOrFallback.calcDisplayname(),
|
|
|
|
event.type),
|
|
|
|
icon: const Icon(Icons.info_outlined),
|
|
|
|
textColor: buttonTextColor,
|
|
|
|
onPressed: () => onInfoTab!(event),
|
|
|
|
);
|
|
|
|
});
|
2020-01-01 19:10:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-11-13 13:06:36 +01:00
|
|
|
|
|
|
|
class _ButtonContent extends StatelessWidget {
|
|
|
|
final void Function() onPressed;
|
|
|
|
final String label;
|
|
|
|
final Icon icon;
|
2022-01-29 12:35:03 +01:00
|
|
|
final Color? textColor;
|
2021-11-13 13:06:36 +01:00
|
|
|
|
|
|
|
const _ButtonContent({
|
2022-01-29 12:35:03 +01:00
|
|
|
required this.label,
|
|
|
|
required this.icon,
|
|
|
|
required this.textColor,
|
|
|
|
required this.onPressed,
|
|
|
|
Key? key,
|
2021-11-13 13:06:36 +01:00
|
|
|
}) : super(key: key);
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2022-02-19 11:05:21 +01:00
|
|
|
return OutlinedButton.icon(
|
2021-11-13 13:06:36 +01:00
|
|
|
onPressed: onPressed,
|
|
|
|
icon: icon,
|
2021-11-14 12:00:49 +01:00
|
|
|
label: Text(label, overflow: TextOverflow.ellipsis),
|
2022-02-19 11:05:21 +01:00
|
|
|
style: OutlinedButton.styleFrom(
|
2022-08-31 19:54:22 +02:00
|
|
|
foregroundColor: textColor,
|
2022-02-19 11:05:21 +01:00
|
|
|
backgroundColor: Colors.white.withAlpha(64),
|
|
|
|
),
|
2021-11-13 13:06:36 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|