mirror of
https://gitlab.com/famedly/fluffychat.git
synced 2025-05-08 03:07:39 +02:00
chore: Clickable links in stories
This commit is contained in:
parent
86ebf3b80a
commit
bf0a1985e6
@ -88,7 +88,8 @@ class AddStoryView extends StatelessWidget {
|
|||||||
child: TextField(
|
child: TextField(
|
||||||
controller: controller.controller,
|
controller: controller.controller,
|
||||||
minLines: 1,
|
minLines: 1,
|
||||||
maxLines: 20,
|
maxLines: 15,
|
||||||
|
maxLength: 500,
|
||||||
autofocus: true,
|
autofocus: true,
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
|
@ -3,10 +3,12 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:adaptive_dialog/adaptive_dialog.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
import 'package:emoji_picker_flutter/emoji_picker_flutter.dart';
|
import 'package:emoji_picker_flutter/emoji_picker_flutter.dart';
|
||||||
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
||||||
|
import 'package:future_loading_dialog/future_loading_dialog.dart';
|
||||||
import 'package:matrix/matrix.dart';
|
import 'package:matrix/matrix.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:video_player/video_player.dart';
|
import 'package:video_player/video_player.dart';
|
||||||
@ -227,6 +229,55 @@ class StoryPageController extends State<StoryPage> {
|
|||||||
|
|
||||||
final Map<String, Future<MatrixFile>> _fileCache = {};
|
final Map<String, Future<MatrixFile>> _fileCache = {};
|
||||||
|
|
||||||
|
void report(_) async {
|
||||||
|
_modalOpened = true;
|
||||||
|
final event = currentEvent;
|
||||||
|
if (event == null) return;
|
||||||
|
final score = await showConfirmationDialog<int>(
|
||||||
|
context: context,
|
||||||
|
title: L10n.of(context)!.reportMessage,
|
||||||
|
message: L10n.of(context)!.howOffensiveIsThisContent,
|
||||||
|
cancelLabel: L10n.of(context)!.cancel,
|
||||||
|
okLabel: L10n.of(context)!.ok,
|
||||||
|
actions: [
|
||||||
|
AlertDialogAction(
|
||||||
|
key: -100,
|
||||||
|
label: L10n.of(context)!.extremeOffensive,
|
||||||
|
),
|
||||||
|
AlertDialogAction(
|
||||||
|
key: -50,
|
||||||
|
label: L10n.of(context)!.offensive,
|
||||||
|
),
|
||||||
|
AlertDialogAction(
|
||||||
|
key: 0,
|
||||||
|
label: L10n.of(context)!.inoffensive,
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
if (score == null) return;
|
||||||
|
final reason = await showTextInputDialog(
|
||||||
|
useRootNavigator: false,
|
||||||
|
context: context,
|
||||||
|
title: L10n.of(context)!.whyDoYouWantToReportThis,
|
||||||
|
okLabel: L10n.of(context)!.ok,
|
||||||
|
cancelLabel: L10n.of(context)!.cancel,
|
||||||
|
textFields: [DialogTextField(hintText: L10n.of(context)!.reason)]);
|
||||||
|
if (reason == null || reason.single.isEmpty) return;
|
||||||
|
final result = await showFutureLoadingDialog(
|
||||||
|
context: context,
|
||||||
|
future: () => Matrix.of(context).client.reportContent(
|
||||||
|
roomId,
|
||||||
|
event.eventId,
|
||||||
|
reason: reason.single,
|
||||||
|
score: score,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
_modalOpened = false;
|
||||||
|
if (result.error != null) return;
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(content: Text(L10n.of(context)!.contentHasBeenReported)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
Future<MatrixFile> downloadAndDecryptAttachment(
|
Future<MatrixFile> downloadAndDecryptAttachment(
|
||||||
Event event, bool getThumbnail) async {
|
Event event, bool getThumbnail) async {
|
||||||
return _fileCache[event.eventId] ??=
|
return _fileCache[event.eventId] ??=
|
||||||
@ -362,3 +413,8 @@ extension on List<Event> {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum PopupStoryAction {
|
||||||
|
report,
|
||||||
|
message,
|
||||||
|
}
|
||||||
|
@ -6,6 +6,7 @@ import 'package:flutter/services.dart';
|
|||||||
import 'package:flutter_blurhash/flutter_blurhash.dart';
|
import 'package:flutter_blurhash/flutter_blurhash.dart';
|
||||||
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
||||||
import 'package:matrix/matrix.dart';
|
import 'package:matrix/matrix.dart';
|
||||||
|
import 'package:matrix_link_text/link_text.dart';
|
||||||
import 'package:video_player/video_player.dart';
|
import 'package:video_player/video_player.dart';
|
||||||
|
|
||||||
import 'package:fluffychat/pages/story/story_page.dart';
|
import 'package:fluffychat/pages/story/story_page.dart';
|
||||||
@ -13,6 +14,7 @@ import 'package:fluffychat/utils/date_time_extension.dart';
|
|||||||
import 'package:fluffychat/utils/localized_exception_extension.dart';
|
import 'package:fluffychat/utils/localized_exception_extension.dart';
|
||||||
import 'package:fluffychat/utils/platform_infos.dart';
|
import 'package:fluffychat/utils/platform_infos.dart';
|
||||||
import 'package:fluffychat/utils/string_color.dart';
|
import 'package:fluffychat/utils/string_color.dart';
|
||||||
|
import 'package:fluffychat/utils/url_launcher.dart';
|
||||||
import 'package:fluffychat/widgets/avatar.dart';
|
import 'package:fluffychat/widgets/avatar.dart';
|
||||||
|
|
||||||
class StoryView extends StatelessWidget {
|
class StoryView extends StatelessWidget {
|
||||||
@ -69,6 +71,20 @@ class StoryView extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
actions: [
|
||||||
|
AnimatedOpacity(
|
||||||
|
duration: const Duration(seconds: 1),
|
||||||
|
opacity: controller.isHold ? 0 : 1,
|
||||||
|
child: PopupMenuButton<bool>(
|
||||||
|
onSelected: controller.report,
|
||||||
|
itemBuilder: (context) => [
|
||||||
|
PopupMenuItem(
|
||||||
|
value: true,
|
||||||
|
child: Text(L10n.of(context)!.reportMessage),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)),
|
||||||
|
],
|
||||||
systemOverlayStyle: SystemUiOverlayStyle.light,
|
systemOverlayStyle: SystemUiOverlayStyle.light,
|
||||||
iconTheme: const IconThemeData(color: Colors.white),
|
iconTheme: const IconThemeData(color: Colors.white),
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
@ -189,18 +205,35 @@ class StoryView extends StatelessWidget {
|
|||||||
: null,
|
: null,
|
||||||
),
|
),
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
child: Text(
|
child: ListView(
|
||||||
controller.loadingMode
|
shrinkWrap: true,
|
||||||
? L10n.of(context)!.loadingPleaseWait
|
children: [
|
||||||
: event.content.tryGet<String>('body') ?? '',
|
LinkText(
|
||||||
textAlign: TextAlign.center,
|
text: controller.loadingMode
|
||||||
style: TextStyle(
|
? L10n.of(context)!.loadingPleaseWait
|
||||||
fontSize: 24,
|
: event.content.tryGet<String>('body') ?? '',
|
||||||
color: Colors.white,
|
textAlign: TextAlign.center,
|
||||||
backgroundColor: event.messageType == MessageTypes.Text
|
onLinkTap: (url) =>
|
||||||
? null
|
UrlLauncher(context, url).launchUrl(),
|
||||||
: Colors.black,
|
linkStyle: TextStyle(
|
||||||
),
|
fontSize: 24,
|
||||||
|
color: Colors.blue.shade50,
|
||||||
|
decoration: TextDecoration.underline,
|
||||||
|
backgroundColor:
|
||||||
|
event.messageType == MessageTypes.Text
|
||||||
|
? null
|
||||||
|
: Colors.black,
|
||||||
|
),
|
||||||
|
textStyle: TextStyle(
|
||||||
|
fontSize: 24,
|
||||||
|
color: Colors.white,
|
||||||
|
backgroundColor:
|
||||||
|
event.messageType == MessageTypes.Text
|
||||||
|
? null
|
||||||
|
: Colors.black,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Positioned(
|
Positioned(
|
||||||
|
Loading…
x
Reference in New Issue
Block a user