fluffychat/lib/pages/chat/events/message_content.dart

331 lines
11 KiB
Dart
Raw Normal View History

2020-01-01 18:10:13 +00:00
import 'package:flutter/material.dart';
2021-10-26 16:50:34 +00:00
import 'package:flutter_gen/gen_l10n/l10n.dart';
import 'package:flutter_linkify/flutter_linkify.dart';
2021-10-26 16:50:34 +00:00
import 'package:matrix/matrix.dart';
2021-12-27 08:35:07 +00:00
import 'package:fluffychat/pages/chat/events/video_player.dart';
2023-01-07 09:29:34 +00:00
import 'package:fluffychat/utils/adaptive_bottom_sheet.dart';
import 'package:fluffychat/utils/date_time_extension.dart';
2022-12-30 16:54:01 +00:00
import 'package:fluffychat/utils/matrix_sdk_extensions/matrix_locals.dart';
import 'package:fluffychat/widgets/avatar.dart';
2021-11-09 20:32:16 +00:00
import 'package:fluffychat/widgets/matrix.dart';
import '../../../config/app_config.dart';
import '../../../utils/platform_infos.dart';
import '../../../utils/url_launcher.dart';
import '../../bootstrap/bootstrap_dialog.dart';
2021-11-09 20:32:16 +00:00
import 'audio_player.dart';
import 'cute_events.dart';
2021-10-26 16:50:34 +00:00
import 'html_message.dart';
2021-11-09 20:32:16 +00:00
import 'image_bubble.dart';
2021-08-01 07:53:43 +00:00
import 'map_bubble.dart';
2021-10-26 16:50:34 +00:00
import 'message_download_content.dart';
import 'sticker.dart';
2020-01-01 18:10:13 +00:00
class MessageContent extends StatelessWidget {
final Event event;
final Color textColor;
2022-01-29 11:35:03 +00:00
final void Function(Event)? onInfoTab;
2020-01-01 18:10:13 +00:00
const MessageContent(
this.event, {
this.onInfoTab,
Key? key,
required this.textColor,
}) : super(key: key);
2020-01-01 18:10:13 +00:00
2020-11-22 10:46:31 +00:00
void _verifyOrRequestKey(BuildContext context) async {
final l10n = L10n.of(context)!;
2020-11-22 10:46:31 +00:00
if (event.content['can_request_session'] != true) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
2021-04-03 11:09:20 +00:00
content: Text(
event.type == EventTypes.Encrypted
? l10n.needPantalaimonWarning
: event.calcLocalizedBodyFallback(
MatrixLocals(l10n),
),
),
),
);
2020-11-22 10:46:31 +00:00
return;
}
final client = Matrix.of(context).client;
2022-01-29 11:35:03 +00:00
if (client.isUnknownSession && client.encryption!.crossSigning.enabled) {
final success = await BootstrapDialog(
client: Matrix.of(context).client,
).show(context);
if (success != true) return;
2020-11-22 10:46:31 +00:00
}
event.requestKey();
final sender = event.senderFromMemoryOrFallback;
2023-01-07 09:29:34 +00:00
await showAdaptiveBottomSheet(
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),
),
2023-08-18 05:24:31 +00:00
),
],
),
),
),
);
2020-11-22 10:46:31 +00:00
}
2020-01-01 18:10:13 +00:00
@override
Widget build(BuildContext context) {
2021-11-13 12:06:36 +00:00
final fontSize = AppConfig.messageFontSize * AppConfig.fontSizeFactor;
2023-07-22 11:09:03 +00:00
final buttonTextColor = textColor;
2020-01-01 18:10:13 +00:00
switch (event.type) {
case EventTypes.Message:
2020-02-21 08:45:37 +00:00
case EventTypes.Encrypted:
case EventTypes.Sticker:
2020-03-29 18:13:25 +00:00
switch (event.messageType) {
case MessageTypes.Image:
2021-11-19 09:01:35 +00:00
return ImageBubble(
event,
width: 400,
height: 300,
fit: BoxFit.cover,
);
case MessageTypes.Sticker:
if (event.redacted) continue textmessage;
2021-11-19 09:01:35 +00:00
return Sticker(event);
case CuteEventContent.eventType:
return CuteContent(event);
case MessageTypes.Audio:
if (PlatformInfos.isMobile ||
2023-02-14 16:21:06 +00:00
PlatformInfos.isMacOS ||
PlatformInfos.isWeb
// Disabled until https://github.com/bleonard252/just_audio_mpv/issues/3
// is fixed
// || PlatformInfos.isLinux
) {
return AudioPlayerWidget(
event,
color: textColor,
);
}
return MessageDownloadContent(event, textColor);
2020-03-13 20:58:48 +00:00
case MessageTypes.Video:
if (PlatformInfos.isMobile || PlatformInfos.isWeb) {
2021-12-27 08:35:07 +00:00
return EventVideoPlayer(event);
2021-08-08 15:55:00 +00:00
}
return MessageDownloadContent(event, textColor);
2020-03-13 20:58:48 +00:00
case MessageTypes.File:
return MessageDownloadContent(event, textColor);
case MessageTypes.Text:
2020-05-09 11:36:41 +00:00
case MessageTypes.Notice:
case MessageTypes.Emote:
if (AppConfig.renderHtml &&
2020-05-13 13:58:59 +00:00
!event.redacted &&
2020-09-21 07:44:13 +00:00
event.isRichMessage) {
var html = event.formattedText;
2020-05-09 11:36:41 +00:00
if (event.messageType == MessageTypes.Emote) {
2020-05-13 13:58:59 +00:00
html = '* $html';
2020-05-09 11:36:41 +00:00
}
return HtmlMessage(
html: html,
textColor: textColor,
2020-05-14 05:43:21 +00:00
room: event.room,
2020-05-09 11:36:41 +00:00
);
}
// else we fall through to the normal message rendering
continue textmessage;
case MessageTypes.BadEncrypted:
2020-11-22 10:46:31 +00:00
case EventTypes.Encrypted:
2021-11-13 12:06:36 +00:00
return _ButtonContent(
2021-11-15 06:59:51 +00:00
textColor: buttonTextColor,
2020-11-22 10:46:31 +00:00
onPressed: () => _verifyOrRequestKey(context),
icon: '🔒',
2022-01-29 11:35:03 +00:00
label: L10n.of(context)!.encrypted,
2023-08-11 12:07:51 +00:00
fontSize: fontSize,
2020-11-22 10:46:31 +00:00
);
case MessageTypes.Location:
2021-08-01 07:53:43 +00:00
final geoUri =
2022-01-29 11:35:03 +00:00
Uri.tryParse(event.content.tryGet<String>('geo_uri')!);
if (geoUri != null && geoUri.scheme == 'geo') {
2021-08-01 07:53:43 +00: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 11:35:03 +00:00
latitude: latlong.first!,
longitude: latlong.last!,
2021-08-01 07:53:43 +00:00
),
2021-10-14 16:09:30 +00:00
const SizedBox(height: 6),
2021-08-01 07:53:43 +00:00
OutlinedButton.icon(
icon: Icon(Icons.location_on_outlined, color: textColor),
onPressed:
UrlLauncher(context, geoUri.toString()).launchUrl,
label: Text(
2022-01-29 11:35:03 +00:00
L10n.of(context)!.openInMaps,
2021-08-01 07:53:43 +00:00
style: TextStyle(color: textColor),
),
),
],
);
}
}
continue textmessage;
case MessageTypes.None:
2020-05-09 11:36:41 +00:00
textmessage:
2020-02-21 08:45:37 +00:00
default:
2020-11-22 20:45:23 +00:00
if (event.redacted) {
return FutureBuilder<User?>(
future: event.redactedBecause?.fetchSenderUser(),
builder: (context, snapshot) {
final reason =
event.redactedBecause?.content.tryGet<String>('reason');
final redactedBy = snapshot.data?.calcDisplayname() ??
event.redactedBecause?.senderId.localpart ??
L10n.of(context)!.user;
return _ButtonContent(
label: reason == null
? L10n.of(context)!.redactedBy(redactedBy)
: L10n.of(context)!.redactedByBecause(
redactedBy,
reason,
),
icon: '🗑️',
textColor: buttonTextColor,
onPressed: () => onInfoTab!(event),
2023-08-11 12:07:51 +00:00
fontSize: fontSize,
);
},
);
2020-11-22 20:45:23 +00:00
}
final bigEmotes = event.onlyEmotes &&
event.numberEmotes > 0 &&
event.numberEmotes <= 10;
return FutureBuilder<String>(
future: event.calcLocalizedBody(
MatrixLocals(L10n.of(context)!),
hideReply: true,
),
builder: (context, snapshot) {
return Linkify(
text: snapshot.data ??
event.calcLocalizedBodyFallback(
MatrixLocals(L10n.of(context)!),
hideReply: true,
),
style: TextStyle(
color: textColor,
fontSize: bigEmotes ? fontSize * 3 : fontSize,
decoration:
event.redacted ? TextDecoration.lineThrough : null,
),
2023-05-23 06:22:34 +00:00
options: const LinkifyOptions(humanize: false),
linkStyle: TextStyle(
color: textColor.withAlpha(150),
fontSize: bigEmotes ? fontSize * 3 : fontSize,
decoration: TextDecoration.underline,
decorationColor: textColor.withAlpha(150),
),
onOpen: (url) => UrlLauncher(context, url.url).launchUrl(),
);
},
);
2020-01-04 08:37:09 +00:00
}
case EventTypes.CallInvite:
return FutureBuilder<User?>(
future: event.fetchSenderUser(),
builder: (context, snapshot) {
return _ButtonContent(
label: L10n.of(context)!.startedACall(
snapshot.data?.calcDisplayname() ??
event.senderFromMemoryOrFallback.calcDisplayname(),
),
icon: '📞',
textColor: buttonTextColor,
onPressed: () => onInfoTab!(event),
2023-08-11 12:07:51 +00:00
fontSize: fontSize,
);
},
);
2020-01-19 14:07:42 +00:00
default:
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: '',
textColor: buttonTextColor,
onPressed: () => onInfoTab!(event),
2023-08-11 12:07:51 +00:00
fontSize: fontSize,
);
},
);
2020-01-01 18:10:13 +00:00
}
}
}
2021-11-13 12:06:36 +00:00
class _ButtonContent extends StatelessWidget {
final void Function() onPressed;
final String label;
final String icon;
2022-01-29 11:35:03 +00:00
final Color? textColor;
2023-08-11 12:07:51 +00:00
final double fontSize;
2021-11-13 12:06:36 +00:00
const _ButtonContent({
2022-01-29 11:35:03 +00:00
required this.label,
required this.icon,
required this.textColor,
required this.onPressed,
2023-08-11 12:07:51 +00:00
required this.fontSize,
2022-01-29 11:35:03 +00:00
Key? key,
2021-11-13 12:06:36 +00:00
}) : super(key: key);
@override
Widget build(BuildContext context) {
2023-08-11 12:07:51 +00:00
return InkWell(
onTap: onPressed,
child: Text(
'$icon $label',
style: TextStyle(
color: textColor,
fontSize: fontSize,
),
),
2021-11-13 12:06:36 +00:00
);
}
}