mirror of
https://gitlab.com/famedly/fluffychat.git
synced 2024-12-21 04:42:37 +01:00
289 lines
10 KiB
Dart
289 lines
10 KiB
Dart
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
import 'package:future_loading_dialog/future_loading_dialog.dart';
|
|
import 'package:matrix/encryption/utils/key_verification.dart';
|
|
import 'package:matrix/matrix.dart';
|
|
import 'package:matrix_link_text/link_text.dart';
|
|
import 'package:url_launcher/url_launcher.dart';
|
|
|
|
import 'package:fluffychat/pages/key_verification/key_verification_dialog.dart';
|
|
import 'package:fluffychat/utils/matrix_sdk_extensions.dart/event_extension.dart';
|
|
import 'package:fluffychat/utils/matrix_sdk_extensions.dart/matrix_locals.dart';
|
|
import 'package:fluffychat/widgets/matrix.dart';
|
|
import '../../../config/app_config.dart';
|
|
import '../../../pages/video_viewer/video_viewer.dart';
|
|
import '../../../utils/platform_infos.dart';
|
|
import '../../../utils/url_launcher.dart';
|
|
import 'audio_player.dart';
|
|
import 'html_message.dart';
|
|
import 'image_bubble.dart';
|
|
import 'map_bubble.dart';
|
|
import 'message_download_content.dart';
|
|
import 'sticker.dart';
|
|
|
|
class MessageContent extends StatelessWidget {
|
|
final Event event;
|
|
final Color textColor;
|
|
final void Function(Event) onInfoTab;
|
|
|
|
const MessageContent(this.event, {this.onInfoTab, Key key, this.textColor})
|
|
: super(key: key);
|
|
|
|
void _verifyOrRequestKey(BuildContext context) async {
|
|
if (event.content['can_request_session'] != true) {
|
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
|
content: Text(
|
|
event.type == EventTypes.Encrypted
|
|
? L10n.of(context).needPantalaimonWarning
|
|
: event.getLocalizedBody(
|
|
MatrixLocals(L10n.of(context)),
|
|
),
|
|
)));
|
|
return;
|
|
}
|
|
final client = Matrix.of(context).client;
|
|
if (client.isUnknownSession && client.encryption.crossSigning.enabled) {
|
|
final req =
|
|
await client.userDeviceKeys[client.userID].startVerification();
|
|
req.onUpdate = () async {
|
|
if (req.state == KeyVerificationState.done) {
|
|
for (var i = 0; i < 12; i++) {
|
|
if (await client.encryption.keyManager.isCached()) {
|
|
break;
|
|
}
|
|
await Future.delayed(const Duration(seconds: 1));
|
|
}
|
|
final timeline = await event.room.getTimeline();
|
|
timeline.requestKeys();
|
|
timeline.cancelSubscriptions();
|
|
}
|
|
};
|
|
await KeyVerificationDialog(request: req).show(context);
|
|
} else {
|
|
final success = await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () => event.requestKey(),
|
|
);
|
|
if (success.error == null) {
|
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
|
content: Text(L10n.of(context).requestToReadOlderMessages)));
|
|
}
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final fontSize = AppConfig.messageFontSize * AppConfig.fontSizeFactor;
|
|
final buttonTextColor =
|
|
event.senderId == Matrix.of(context).client.userID ? textColor : null;
|
|
switch (event.type) {
|
|
case EventTypes.Message:
|
|
case EventTypes.Encrypted:
|
|
case EventTypes.Sticker:
|
|
switch (event.messageType) {
|
|
case MessageTypes.Image:
|
|
if (event.showThumbnail) {
|
|
return ImageBubble(
|
|
event,
|
|
width: 400,
|
|
height: 300,
|
|
fit: BoxFit.cover,
|
|
);
|
|
}
|
|
return MessageDownloadContent(event, textColor);
|
|
case MessageTypes.Sticker:
|
|
if (event.showThumbnail) {
|
|
return Sticker(event);
|
|
}
|
|
return MessageDownloadContent(event, textColor);
|
|
case MessageTypes.Audio:
|
|
if (PlatformInfos.isMobile) {
|
|
return AudioPlayerWidget(
|
|
event,
|
|
color: textColor,
|
|
);
|
|
}
|
|
return MessageDownloadContent(event, textColor);
|
|
case MessageTypes.Video:
|
|
if (event.showThumbnail &&
|
|
(PlatformInfos.isMobile || PlatformInfos.isWeb)) {
|
|
return Stack(
|
|
alignment: Alignment.center,
|
|
children: <Widget>[
|
|
ImageBubble(
|
|
event,
|
|
width: 400,
|
|
height: 300,
|
|
fit: BoxFit.cover,
|
|
tapToView: false,
|
|
),
|
|
FloatingActionButton.extended(
|
|
onPressed: () => showDialog(
|
|
context: Matrix.of(context).navigatorContext,
|
|
useRootNavigator: false,
|
|
builder: (_) => VideoViewer(event),
|
|
),
|
|
label: Text(L10n.of(context).play('Video')),
|
|
icon: const Icon(Icons.video_camera_front_outlined),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
return MessageDownloadContent(event, textColor);
|
|
case MessageTypes.File:
|
|
return MessageDownloadContent(event, textColor);
|
|
|
|
case MessageTypes.Text:
|
|
case MessageTypes.Notice:
|
|
case MessageTypes.Emote:
|
|
if (AppConfig.renderHtml &&
|
|
!event.redacted &&
|
|
event.isRichMessage) {
|
|
var html = event.formattedText;
|
|
if (event.messageType == MessageTypes.Emote) {
|
|
html = '* $html';
|
|
}
|
|
final bigEmotes = event.onlyEmotes &&
|
|
event.numberEmotes > 0 &&
|
|
event.numberEmotes <= 10;
|
|
return HtmlMessage(
|
|
html: html,
|
|
defaultTextStyle: TextStyle(
|
|
color: textColor,
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
|
),
|
|
linkStyle: TextStyle(
|
|
color: textColor.withAlpha(150),
|
|
fontSize: bigEmotes ? fontSize * 3 : fontSize,
|
|
decoration: TextDecoration.underline,
|
|
),
|
|
room: event.room,
|
|
emoteSize: bigEmotes ? fontSize * 3 : fontSize * 1.5,
|
|
);
|
|
}
|
|
// else we fall through to the normal message rendering
|
|
continue textmessage;
|
|
case MessageTypes.BadEncrypted:
|
|
case EventTypes.Encrypted:
|
|
return _ButtonContent(
|
|
textColor: buttonTextColor,
|
|
onPressed: () => _verifyOrRequestKey(context),
|
|
icon: const Icon(Icons.lock_outline),
|
|
label: L10n.of(context).encrypted,
|
|
);
|
|
case MessageTypes.Location:
|
|
final geoUri =
|
|
Uri.tryParse(event.content.tryGet<String>('geo_uri'));
|
|
if (geoUri != null &&
|
|
geoUri.scheme == 'geo' &&
|
|
geoUri.path != null) {
|
|
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(
|
|
latitude: latlong.first,
|
|
longitude: latlong.last,
|
|
),
|
|
const SizedBox(height: 6),
|
|
OutlinedButton.icon(
|
|
icon: Icon(Icons.location_on_outlined, color: textColor),
|
|
onPressed:
|
|
UrlLauncher(context, geoUri.toString()).launchUrl,
|
|
label: Text(
|
|
L10n.of(context).openInMaps,
|
|
style: TextStyle(color: textColor),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
}
|
|
continue textmessage;
|
|
case MessageTypes.None:
|
|
textmessage:
|
|
default:
|
|
if (event.content['msgtype'] == Matrix.callNamespace) {
|
|
return _ButtonContent(
|
|
onPressed: () => launch(event.body),
|
|
icon: const Icon(Icons.phone_outlined, color: Colors.green),
|
|
label: L10n.of(context).videoCall,
|
|
textColor: buttonTextColor,
|
|
);
|
|
}
|
|
if (event.redacted) {
|
|
return _ButtonContent(
|
|
label: L10n.of(context)
|
|
.redactedAnEvent(event.sender.calcDisplayname()),
|
|
icon: const Icon(Icons.delete_outlined),
|
|
textColor: buttonTextColor,
|
|
onPressed: () => onInfoTab(event),
|
|
);
|
|
}
|
|
final bigEmotes = event.onlyEmotes &&
|
|
event.numberEmotes > 0 &&
|
|
event.numberEmotes <= 10;
|
|
return LinkText(
|
|
text: event.getLocalizedBody(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,
|
|
),
|
|
onLinkTap: (url) => UrlLauncher(context, url).launchUrl(),
|
|
);
|
|
}
|
|
break;
|
|
default:
|
|
return _ButtonContent(
|
|
label: L10n.of(context)
|
|
.userSentUnknownEvent(event.sender.calcDisplayname(), event.type),
|
|
icon: const Icon(Icons.info_outlined),
|
|
textColor: buttonTextColor,
|
|
onPressed: () => onInfoTab(event),
|
|
);
|
|
}
|
|
return Container(); // else flutter analyze complains
|
|
}
|
|
}
|
|
|
|
class _ButtonContent extends StatelessWidget {
|
|
final void Function() onPressed;
|
|
final String label;
|
|
final Icon icon;
|
|
final Color textColor;
|
|
|
|
const _ButtonContent({
|
|
@required this.label,
|
|
@required this.icon,
|
|
@required this.textColor,
|
|
@required this.onPressed,
|
|
Key key,
|
|
}) : super(key: key);
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return TextButton.icon(
|
|
onPressed: onPressed,
|
|
icon: icon,
|
|
label: Text(label, overflow: TextOverflow.ellipsis),
|
|
style: TextButton.styleFrom(primary: textColor),
|
|
);
|
|
}
|
|
}
|