From 059164002263a6f240c120660a8f8dafa5436d14 Mon Sep 17 00:00:00 2001 From: ChenDoXiu <2839673621@qq.com> Date: Mon, 5 Feb 2024 01:20:59 +0800 Subject: [PATCH] ### Feature - The preview of the original note is now displayed in the note create dialog - The username mention is now automatically added to the reply dialog ### Fixed - Some mention avatars on the notes page don't display in some cases - The link for MFM is not opening --- CHANGELOG.md | 15 +- lib/models/note.dart | 14 ++ lib/models/user_simple.dart | 4 + lib/networks/apis.dart | 2 - lib/networks/user.dart | 10 +- lib/pages/notes/note_page.dart | 4 +- lib/pages/users/user_overview.dart | 81 ++++------ lib/pages/users/user_page.dart | 30 ++-- lib/widgets/mfm_text/mfm_text.dart | 151 ++++++++++-------- .../note_create_dialog.dart | 100 +++++++++++- .../note_create_dialog_state.dart | 6 +- lib/widgets/notes/note_card.dart | 25 ++- pubspec.lock | 56 +++---- pubspec.yaml | 6 +- 14 files changed, 321 insertions(+), 183 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f9dd39b..fd05aeb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,17 @@ # Changelog -# 0.1.1 +## 0.2.0 -# 0.1.0 +### Feature +- The preview of the original note is now displayed in the note create dialog +- The username mention is now automatically added to the reply dialog + +### Fixed + +- Some mention avatars on the notes page don't display in some cases +- The link for MFM is not opening + +### Other + +- Upgrade flutter dependencies \ No newline at end of file diff --git a/lib/models/note.dart b/lib/models/note.dart index 46f71da..7e52975 100644 --- a/lib/models/note.dart +++ b/lib/models/note.dart @@ -59,6 +59,20 @@ class NoteModel { return 'NoteModel{id: $id, clippedCount: $clippedCount, createdAt: $createdAt, cw: $cw, emojis: $emojis, files: $files, localOnly: $localOnly, reactionAcceptance: $reactionAcceptance, reactionEmojis: $reactionEmojis, reactions: $reactions, myReaction: $myReaction, renote: $renote, renoteCount: $renoteCount, renoteId: $renoteId, repliesCount: $repliesCount, reply: $reply, replyId: $replyId, text: $text, uri: $uri, user: $user, userId: $userId, visibility: $visibility, poll: $poll}'; } + String createReplyAtText(String? currentUserId) { + var initText = ""; + if (user.id != currentUserId) { + initText += user.getAtUserName(); + } + if (reply?.user != null && + reply!.user.id != user.id && + reply!.user.id != currentUserId) { + initText += " "; + initText += reply!.user.getAtUserName(); + } + return initText; + } + factory NoteModel.fromMap(dynamic map) { List files = []; for (var item in map['files']) { diff --git a/lib/models/user_simple.dart b/lib/models/user_simple.dart index aa8e162..db27e31 100644 --- a/lib/models/user_simple.dart +++ b/lib/models/user_simple.dart @@ -30,6 +30,10 @@ class UserSimpleModel { required this.username, required this.badgeRoles}); + String getAtUserName() { + return "@$username${host != null ? "@$host" : ""}"; + } + UserSimpleModel copyWith( {bool? isCat, Map? emojis, diff --git a/lib/networks/apis.dart b/lib/networks/apis.dart index 73d47fb..94020e0 100644 --- a/lib/networks/apis.dart +++ b/lib/networks/apis.dart @@ -5,7 +5,6 @@ import 'package:flutter/services.dart'; import 'package:moekey/networks/dio.dart'; import 'package:riverpod_annotation/riverpod_annotation.dart'; -import '../main.dart'; import '../state/themes.dart'; part 'apis.g.dart'; @@ -25,7 +24,6 @@ class ServerInfoState extends _$ServerInfoState { FutureOr build() async { var colors = ref.read(themeColorsProvider.notifier); var meta = await ref.watch(apiMetaProvider.future); - logger.d("123"); colors.updateThemes(meta); return meta; } diff --git a/lib/networks/user.dart b/lib/networks/user.dart index 32dbb9a..934e65e 100644 --- a/lib/networks/user.dart +++ b/lib/networks/user.dart @@ -30,12 +30,16 @@ class UserFollowing extends _$UserFollowing { @riverpod Future userInfo(UserInfoRef ref, - {required String userId}) async { + {String? username, String? host, String? userId}) async { var http = await ref.read(httpProvider.future); var user = await ref.read(currentLoginUserProvider.future); - var res = await http - .post("/users/show", data: {"userId": userId, "i": user?.token}); + var res = await http.post("/users/show", data: { + if (username != null) "username": username, + if (host != null) "host": host, + "i": user?.token, + if (userId != null) "userId": userId, + }); return UserFullModel.fromMap(res.data); } diff --git a/lib/pages/notes/note_page.dart b/lib/pages/notes/note_page.dart index 017ccd7..7cb244a 100644 --- a/lib/pages/notes/note_page.dart +++ b/lib/pages/notes/note_page.dart @@ -356,7 +356,7 @@ class NotesPageNoteCard extends HookConsumerWidget { MFMText( text: data.text ?? "", emojis: data.emojis, - currentServerHost: data.user.host ?? "", + currentServerHost: data.user.host, isSelection: true, ), const SizedBox(height: 4), @@ -379,7 +379,7 @@ class NotesPageNoteCard extends HookConsumerWidget { borderRadius: const BorderRadius.all(Radius.circular(8))), padding: const EdgeInsets.all(12), child: TimeLineNoteCardComponent( - data: data.renote, + data: data.renote!, isShowAction: false, disableReactions: true), ), diff --git a/lib/pages/users/user_overview.dart b/lib/pages/users/user_overview.dart index 5cc2c24..40a3acb 100644 --- a/lib/pages/users/user_overview.dart +++ b/lib/pages/users/user_overview.dart @@ -36,14 +36,8 @@ class UserOverview extends HookConsumerWidget { } else { padding = 0; } - var dataProvider = userNotesListProvider( - userId: userId, - withRenotes: true, - withChannelNotes: false, - withFiles: false, - withReplies: false); + var dataProvider = userNotesListProvider(userId: userId); var data = ref.watch(dataProvider); - print(data.error); var themes = ref.watch(themeColorsProvider); var user = ref.watch(userInfoProvider(userId: userId)); return RefreshIndicator.adaptive( @@ -96,17 +90,11 @@ class UserOverview extends HookConsumerWidget { pined: true, ); } - print(index); return NoteCard( key: ValueKey( data.valueOrNull!.list[index - pinCount].id), borderRadius: borderRadius, data: data.valueOrNull!.list[index - pinCount]); - // return KeepAliveWrapper( - // child: TimelineCardComponent( - // data: data.valueOrNull![index], - // borderRadius: borderRadius, - // )); }, separatorBuilder: (BuildContext context, int index) { return SizedBox( @@ -235,7 +223,7 @@ class UserHomeCard extends HookConsumerWidget { ? Alignment.bottomCenter : const Alignment(-1, 1), child: Transform.translate( - offset: Offset(isSmall ? 0 : 32, isSmall ? 50 : 60), + offset: Offset(isSmall ? 0 : 16, isSmall ? 50 : 60), child: Container( decoration: BoxDecoration( border: Border.all( @@ -244,14 +232,14 @@ class UserHomeCard extends HookConsumerWidget { Radius.circular(200), ), color: themes.panelColor, - // boxShadow: [ - // if (!isSmall) - // BoxShadow( - // color: Colors.black.withOpacity(0.1), - // offset: const Offset(0, 1), - // blurRadius: 2, - // ), - // ], + boxShadow: [ + if (!isSmall) + BoxShadow( + color: Colors.black.withOpacity(0.1), + offset: const Offset(0, 1), + blurRadius: 2, + ), + ], ), height: isSmall ? 120 : 160, width: isSmall ? 120 : 160, @@ -280,7 +268,7 @@ class UserHomeCard extends HookConsumerWidget { bottom: 0, left: 180, child: Padding( - padding: const EdgeInsets.fromLTRB(32, 0, 0, 16), + padding: const EdgeInsets.all(12), child: Column( mainAxisAlignment: MainAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start, @@ -384,10 +372,10 @@ class UserHomeCard extends HookConsumerWidget { ) ], ), - ), - if (!isSmall) + ) + else if (userData.description != null) Padding( - padding: const EdgeInsets.fromLTRB(212, 10, 24, 10), + padding: const EdgeInsets.fromLTRB(190, 10, 24, 10), child: LayoutBuilder( builder: (context, constraints) { return ConstrainedBox( @@ -400,7 +388,7 @@ class UserHomeCard extends HookConsumerWidget { .style .copyWith(fontSize: 13), child: MFMText( - text: userData.description ?? "这个用户什么都没有写", + text: userData.description!, bigEmojiCode: false, emojis: userData.emojis, currentServerHost: userData.host, @@ -417,28 +405,29 @@ class UserHomeCard extends HookConsumerWidget { color: themes.dividerColor, ), if (isSmall) ...[ - Padding( - padding: const EdgeInsets.all(12), - child: Column( - children: [ - Align( - alignment: Alignment.center, - child: DefaultTextStyle( - style: DefaultTextStyle.of(context) - .style - .copyWith(fontSize: 13), - child: MFMText( - text: userData.description ?? "这个用户什么都没有写", - bigEmojiCode: false, - emojis: userData.emojis, - textAlign: TextAlign.center, - currentServerHost: userData.host, + if (userData.description != null) + Padding( + padding: const EdgeInsets.all(12), + child: Column( + children: [ + Align( + alignment: Alignment.center, + child: DefaultTextStyle( + style: DefaultTextStyle.of(context) + .style + .copyWith(fontSize: 13), + child: MFMText( + text: userData.description!, + bigEmojiCode: false, + emojis: userData.emojis, + textAlign: TextAlign.center, + currentServerHost: userData.host, + ), ), ), - ), - ], + ], + ), ), - ), Container( height: 1, color: themes.dividerColor, diff --git a/lib/pages/users/user_page.dart b/lib/pages/users/user_page.dart index 718b682..34e46cf 100644 --- a/lib/pages/users/user_page.dart +++ b/lib/pages/users/user_page.dart @@ -19,14 +19,18 @@ import '../../state/themes.dart'; class UserPage extends HookConsumerWidget { const UserPage({ super.key, - required this.userId, + this.username, + this.host, + this.userId, }); - final String userId; + final String? username; + final String? host; + final String? userId; @override Widget build(BuildContext context, WidgetRef ref) { - var padding = MediaQuery.paddingOf(context); var themes = ref.watch(themeColorsProvider); - var userProvider = userInfoProvider(userId: userId); + var userProvider = + userInfoProvider(username: username, host: host, userId: this.userId); var user = ref.watch(userProvider); if (user.isLoading) { return MkScaffold( @@ -59,7 +63,7 @@ class UserPage extends HookConsumerWidget { )); } var userData = user.valueOrNull; - + var userId = userData!.id; logger.d(user); var tabs = [ const Tab( @@ -84,7 +88,7 @@ class UserPage extends HookConsumerWidget { ], ), ), - if (userData?.publicReactions ?? false) + if (userData.publicReactions) const Tab( child: Row( children: [ @@ -112,7 +116,7 @@ class UserPage extends HookConsumerWidget { children: [ KeepAliveWrapper(child: UserOverview(userId: userId)), KeepAliveWrapper(child: UserNotesPage(userId: userId)), - if (userData?.publicReactions ?? false) + if (userData.publicReactions) KeepAliveWrapper(child: UserReactionsPage(userId: userId)), // Text("data"), ], @@ -129,7 +133,7 @@ class UserPage extends HookConsumerWidget { width: 32, height: 32, child: MkImage( - userData?.avatarUrl ?? "", + userData.avatarUrl ?? "", width: 32, height: 32, shape: BoxShape.circle, @@ -149,8 +153,8 @@ class UserPage extends HookConsumerWidget { .style .copyWith(fontWeight: FontWeight.bold), child: MFMText( - text: userData?.name ?? userData?.username ?? "", - emojis: userData?.emojis, + text: userData.name ?? userData.username, + emojis: userData.emojis, maxLines: 1, overflow: TextOverflow.ellipsis, bigEmojiCode: false, @@ -167,12 +171,12 @@ class UserPage extends HookConsumerWidget { TextSpan( children: [ TextSpan( - text: "@${userData?.username ?? ""}", + text: "@${userData.username}", style: textStyle.copyWith(fontSize: 12)), TextSpan( - text: userData?.host != null - ? "@${userData?.host}" + text: userData.host != null + ? "@${userData.host}" : "", style: textStyle.copyWith( color: diff --git a/lib/widgets/mfm_text/mfm_text.dart b/lib/widgets/mfm_text/mfm_text.dart index f4285ba..56d5e05 100644 --- a/lib/widgets/mfm_text/mfm_text.dart +++ b/lib/widgets/mfm_text/mfm_text.dart @@ -10,6 +10,8 @@ import 'package:twemoji_v2/twemoji_v2.dart'; import 'package:url_launcher/url_launcher_string.dart'; import '../../main.dart'; +import '../../pages/users/user_page.dart'; +import '../../router/main_router_delegate.dart'; import '../../state/themes.dart'; import '../hover_builder.dart'; import '../mk_image.dart'; @@ -150,7 +152,7 @@ _getParse({ color: themes.accentColor, size: textStyle.fontSize! + 2, ), - alignment: PlaceholderAlignment.top) + alignment: PlaceholderAlignment.middle) ], ); } else { @@ -158,43 +160,48 @@ _getParse({ } }, "link": (MfmLink item, TextStyle textStyle) { - var parse = _getParse( - themes: themes, - loginServerUrl: loginServerUrl, - systemEmojis: systemEmojis); var a = item.children; if (a != null) { textStyle = textStyle.copyWith( color: themes.accentColor, ); return WidgetSpan( - child: Tooltip( - message: item.url, - child: Text.rich(TextSpan( - // text: item.url, - style: textStyle.copyWith( - color: themes.accentColor, - ), - children: [ - for (var item in a) - if (parse[item.type] != null) - parse[item.type](item, textStyle) - else - TextSpan(text: item.toString(), style: textStyle), - WidgetSpan( - child: Icon( - TablerIcons.external_link, - color: themes.accentColor, - size: textStyle.fontSize! + 2, + child: Tooltip( + message: item.url, + child: Text.rich( + TextSpan( + // text: item.url, + style: textStyle.copyWith( + color: themes.accentColor, + ), + children: [ + for (var item1 in a) + TextSpan( + text: (item1 as MfmText).text, + style: textStyle, + recognizer: TapGestureRecognizer() + ..onTap = () { + launchUrlString(item.url); + }), + WidgetSpan( + child: GestureDetector( + onTap: () { + launchUrlString(item.url); + }, + child: Icon( + TablerIcons.external_link, + color: themes.accentColor, + size: textStyle.fontSize! + 2, + ), ), - alignment: PlaceholderAlignment.top) - ], - mouseCursor: SystemMouseCursors.click, - recognizer: TapGestureRecognizer() - ..onTap = () { - launchUrlString(item.url); - })), - )); + alignment: PlaceholderAlignment.middle, + ) + ], + mouseCursor: SystemMouseCursors.click, + ), + ), + ), + ); } }, "emojiCode": (MfmEmojiCode item, TextStyle textStyle) { @@ -247,43 +254,54 @@ _getParse({ builder: (context) { String user = item.props?["username"]!; var host = item.props?["host"] ?? currentServerHost; - return DecoratedBox( - decoration: BoxDecoration( - color: themes.mentionColor.withOpacity(0.1), - borderRadius: const BorderRadius.all(Radius.circular(100)), - ), - child: Padding( - padding: const EdgeInsets.fromLTRB(4, 4, 8, 4), - child: IntrinsicWidth( - child: Row( - mainAxisSize: MainAxisSize.min, - children: [ - MkImage( - "$loginServerUrl/avatar/@$user${host != null ? "@$host" : ""}", - height: textStyle.fontSize! * 1.5, - shape: BoxShape.circle, - ), - Expanded( - child: Text.rich( - maxLines: 1, - overflow: TextOverflow.ellipsis, - TextSpan( - children: [ - TextSpan( - text: "@$user", - style: textStyle.copyWith( - color: themes.mentionColor)), - if (host != null && host != currentHost) + return GestureDetector( + onTap: () { + MainRouterDelegate.of(context).setNewRoutePath(RouterItem( + path: "user/@$user${host != null ? "@$host" : ""}", + page: () { + return UserPage(username: user, host: host); + }, + )); + }, + child: DecoratedBox( + decoration: BoxDecoration( + color: themes.mentionColor.withOpacity(0.1), + borderRadius: + const BorderRadius.all(Radius.circular(100)), + ), + child: Padding( + padding: const EdgeInsets.fromLTRB(4, 4, 8, 4), + child: IntrinsicWidth( + child: Row( + mainAxisSize: MainAxisSize.min, + children: [ + MkImage( + "$loginServerUrl/avatar/@$user${host != null ? "@$host" : ""}", + height: textStyle.fontSize! * 1.5, + shape: BoxShape.circle, + ), + Expanded( + child: Text.rich( + maxLines: 1, + overflow: TextOverflow.ellipsis, + TextSpan( + children: [ TextSpan( - text: "@$host", + text: "@$user", style: textStyle.copyWith( - color: themes.mentionColor - .withOpacity(0.5), - )), - ], - ), - )) - ], + color: themes.mentionColor)), + if (host != null && host != currentHost) + TextSpan( + text: "@$host", + style: textStyle.copyWith( + color: themes.mentionColor + .withOpacity(0.5), + )), + ], + ), + )) + ], + ), ), ), ), @@ -344,7 +362,6 @@ _getParse({ alignment: PlaceholderAlignment.middle); }, "quote": (MfmQuote item, TextStyle textStyle) { - var code = item.children; var parse = _getParse( themes: themes, loginServerUrl: loginServerUrl, diff --git a/lib/widgets/note_create_dialog/note_create_dialog.dart b/lib/widgets/note_create_dialog/note_create_dialog.dart index a3c9f7c..23703f5 100644 --- a/lib/widgets/note_create_dialog/note_create_dialog.dart +++ b/lib/widgets/note_create_dialog/note_create_dialog.dart @@ -19,16 +19,25 @@ import 'package:moekey/widgets/user_select_dialog/user_select_dialog.dart'; import '../../main.dart'; import '../../models/note.dart'; +import '../../utils/time_ago_since_date.dart'; import '../driver/driver_select_dialog/driver_select_dialog.dart'; import '../hashtag/hashtag_select_dialog.dart'; import '../input_decoration.dart'; import '../mk_switch.dart'; +import '../notes/note_card.dart'; class NoteCreateDialog extends HookConsumerWidget { - NoteCreateDialog({super.key, this.noteId, this.noteType = NoteType.note}); + NoteCreateDialog( + {super.key, + this.noteId, + this.noteType = NoteType.note, + this.note, + this.initText}); final GlobalKey myKey = GlobalKey(); final String? noteId; final NoteType noteType; + final NoteModel? note; + final String? initText; @override Widget build(BuildContext context, WidgetRef ref) { var themes = ref.watch(themeColorsProvider); @@ -100,7 +109,12 @@ class NoteCreateDialog extends HookConsumerWidget { .watch(noteCreateDialogStateProvider(noteId, noteType).notifier) .getDriverSelectDialogStateProvider()); var form = ref.watch(noteCreateDialogStateProvider(noteId, noteType)); - var contentController = useTextEditingController(text: form.text); + var contentController = + useTextEditingController(text: form.text ?? initText); + if (note != null && note?.cw != null) { + form.cw = note!.cw!; + form.isCw = true; + } contentController.addListener(() { ref .read(noteCreateDialogStateProvider(noteId, noteType).notifier) @@ -197,6 +211,7 @@ class NoteCreateDialog extends HookConsumerWidget { mainAxisAlignment: MainAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start, children: [ + if (note != null) buildNotePreview(themes), if (!form.preview) buildTextField(data, fullscreen, contentController) else @@ -236,6 +251,69 @@ class NoteCreateDialog extends HookConsumerWidget { ); } + Padding buildNotePreview(ThemeColorModel themes) { + return Padding( + padding: const EdgeInsets.all(8.0), + child: Row( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisAlignment: MainAxisAlignment.start, + children: [ + MkImage( + note!.user.avatarUrl ?? "", + shape: BoxShape.circle, + width: 48, + height: 48, + ), + const SizedBox( + width: 10, + ), + Expanded( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisAlignment: MainAxisAlignment.start, + children: [ + Row( + mainAxisSize: MainAxisSize.max, + children: [ + Expanded(child: UserNameRichText(data: note!.user)), + Text(timeAgoSinceDate(note!.createdAt), + style: const TextStyle(fontSize: 12)), + const SizedBox( + width: 6, + ), + if (NoteVisibility.getIcon(note!.visibility) != null) + Icon( + NoteVisibility.getIcon(note!.visibility)!, + size: 14, + color: themes.fgColor, + ) + ], + ), + if (note!.cw != null) + MFMText( + key: ValueKey(note!.cw ?? ""), + text: note!.cw ?? "", + emojis: note!.emojis, + currentServerHost: note!.user.host, + maxLines: 4, + overflow: TextOverflow.ellipsis, + ), + MFMText( + key: ValueKey(note!.text ?? ""), + text: note!.text ?? "", + emojis: note!.emojis, + currentServerHost: note!.user.host, + maxLines: 5, + overflow: TextOverflow.ellipsis, + ), + ], + ), + ) + ], + ), + ); + } + Widget buildRecipient() { return HookConsumer( builder: (context, ref, child) { @@ -973,8 +1051,8 @@ class NoteCreateDialog extends HookConsumerWidget { decoration: InputDecoration( border: InputBorder.none, hintText: [ - if (noteType == NoteType.reply) "回复给...", - if (noteType == NoteType.reNote) "引用...", + if (noteType == NoteType.reply) "回复这个帖子...", + if (noteType == NoteType.reNote) "引用这个帖子...", "发生了什么...", ][0], isDense: true, @@ -1366,13 +1444,21 @@ class NoteCreateDialog extends HookConsumerWidget { } static PageRouteBuilder getRouter( - {String? noteId, NoteType type = NoteType.note}) { + {String? noteId, + NoteType type = NoteType.note, + NoteModel? note, + String? initText}) { return PageRouteBuilder( opaque: false, transitionDuration: const Duration(milliseconds: 200), reverseTransitionDuration: const Duration(milliseconds: 110), pageBuilder: (context, animation, secondaryAnimation) { - return NoteCreateDialog(noteId: noteId, noteType: type); + return NoteCreateDialog( + noteId: noteId, + noteType: type, + note: note, + initText: initText, + ); }, transitionsBuilder: (context, animation, secondaryAnimation, child) { var tween = animation.drive(Tween(begin: 0.95, end: 1.0) @@ -1418,7 +1504,7 @@ class NoteCreateDialog extends HookConsumerWidget { MFMText(text: state.cw), const Divider(), ], - MFMText(text: state.text), + MFMText(text: state.text ?? ""), ], ), ); diff --git a/lib/widgets/note_create_dialog/note_create_dialog_state.dart b/lib/widgets/note_create_dialog/note_create_dialog_state.dart index 05862b6..a7110d0 100644 --- a/lib/widgets/note_create_dialog/note_create_dialog_state.dart +++ b/lib/widgets/note_create_dialog/note_create_dialog_state.dart @@ -16,7 +16,7 @@ part 'note_create_dialog_state.g.dart'; class NoteCreateDialogStateModel { NoteVisibility visibility = NoteVisibility.public; // 可见性 LinkedHashMap visibleUserIds = LinkedHashMap(); // 当 可见性为specified 时的可见用户列表 - String text = ''; // 文本 + String? text = null; // 文本 String cw = ''; //敏感内容 bool isCw = false; bool localOnly = false; // 禁用联合 @@ -35,7 +35,7 @@ class NoteCreateDialogStateModel { 'visibility': visibility.value, if (visibility == NoteVisibility.specified) 'visibleUserIds': visibleUserIds.keys.toList(), - 'text': text, + 'text': text ?? "", if (isCw) 'cw': cw, 'localOnly': localOnly, if (reactionAcceptance != null && visibility != NoteVisibility.specified) @@ -250,7 +250,7 @@ class NoteCreateDialogState extends _$NoteCreateDialogState { state.fileIds = imageSelect.keys.toList(); // 参数验证 - if (state.text.isEmpty) { + if (state.text != null && state.text!.isEmpty) { throw Exception("内容不能为空"); } if (state.isCw && state.cw.isEmpty) { diff --git a/lib/widgets/notes/note_card.dart b/lib/widgets/notes/note_card.dart index eba1d0e..288749b 100644 --- a/lib/widgets/notes/note_card.dart +++ b/lib/widgets/notes/note_card.dart @@ -12,6 +12,7 @@ import 'package:moekey/models/note.dart'; import 'package:moekey/models/user_simple.dart'; import 'package:moekey/networks/notes.dart'; import 'package:moekey/pages/users/user_page.dart'; +import 'package:moekey/state/server.dart'; import 'package:moekey/state/themes.dart'; import 'package:moekey/utils/format_duration.dart'; import 'package:moekey/widgets/context_menu.dart'; @@ -90,7 +91,7 @@ class NoteCard extends ConsumerWidget { Opacity( opacity: 0.8, child: TimeLineNoteCardComponent( - data: thisData.reply, + data: thisData.reply!, reply: true, // isShowReactions: false, disableReactions: true, @@ -113,7 +114,7 @@ class NoteCard extends ConsumerWidget { const BorderRadius.all(Radius.circular(8))), padding: const EdgeInsets.all(12), child: TimeLineNoteCardComponent( - data: thisData.renote, + data: thisData.renote!, isShowAction: false, isShowReactions: false, limit: 500, @@ -143,7 +144,7 @@ class TimeLineNoteCardComponent extends HookConsumerWidget { this.limit = 1000, this.height = 400, }); - final dynamic data; + final NoteModel data; final bool reply; final bool isShowAction; final bool isShowReactions; @@ -726,12 +727,13 @@ class TimeLineActions extends StatelessWidget { required this.fontsize, required this.data, }); - final dynamic data; + final NoteModel data; final double fontsize; @override Widget build(BuildContext context) { return HookConsumer(builder: (context, ref, child) { + var currentUser = ref.watch(currentLoginUserProvider); return Row( children: [ TimelineActionButton( @@ -739,8 +741,14 @@ class TimeLineActions extends StatelessWidget { icon: TablerIcons.arrow_back_up, count: data.repliesCount, onTap: () { - globalNav.currentState?.push(NoteCreateDialog.getRouter( - noteId: data.id, type: NoteType.reply)); + globalNav.currentState?.push( + NoteCreateDialog.getRouter( + noteId: data.id, + type: NoteType.reply, + note: data, + initText: "${data.createReplyAtText(currentUser.value!.id)} ", + ), + ); }, ), const SizedBox( @@ -766,7 +774,10 @@ class TimeLineActions extends StatelessWidget { label: "引用", onTap: () { globalNav.currentState?.push(NoteCreateDialog.getRouter( - type: NoteType.reNote, noteId: data.id)); + type: NoteType.reNote, + noteId: data.id, + note: data, + )); return false; }, ) diff --git a/pubspec.lock b/pubspec.lock index 3942186..23d2446 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -237,26 +237,26 @@ packages: dependency: "direct dev" description: name: custom_lint - sha256: "22bd87a362f433ba6aae127a7bac2838645270737f3721b180916d7c5946cb5d" + sha256: f89ff83efdba7c8996e86bb3bad0b759d58f9b19ae4d0e277a386ddd8b481217 url: "https://pub.flutter-io.cn" source: hosted - version: "0.5.11" + version: "0.6.0" custom_lint_builder: dependency: transitive description: name: custom_lint_builder - sha256: "7d0b094266b5c357769fffb920826b1a08373290f3c5c44b86253aae6873d5fc" + sha256: "9cdd9987feaa6925ec5f98d64de4fbbb5d94248ff77bbf2489366efad6c4baef" url: "https://pub.flutter-io.cn" source: hosted - version: "0.5.11" + version: "0.6.0" custom_lint_core: dependency: transitive description: name: custom_lint_core - sha256: "77dd37e9afe5ed86fc75de22ed3dfae5afb75303111358766550af23e7de53cd" + sha256: "9003a91409c9f1db6e2e50b4870d1d5e802e5923b25f7261bf3cb3e11ea9d4fb" url: "https://pub.flutter-io.cn" source: hosted - version: "0.5.11" + version: "0.6.0" dart_style: dependency: transitive description: @@ -394,10 +394,10 @@ packages: dependency: "direct dev" description: name: flutter_lints - sha256: a25a15ebbdfc33ab1cd26c63a6ee519df92338a9c10f122adda92938253bef04 + sha256: e2a421b7e59244faef694ba7b30562e489c2b489866e505074eb005cd7060db7 url: "https://pub.flutter-io.cn" source: hosted - version: "2.0.3" + version: "3.0.1" flutter_localizations: dependency: "direct main" description: flutter @@ -415,10 +415,10 @@ packages: dependency: transitive description: name: flutter_riverpod - sha256: da9591d1f8d5881628ccd5c25c40e74fc3eef50ba45e40c3905a06e1712412d5 + sha256: "4bce556b7ecbfea26109638d5237684538d4abc509d253e6c5c4c5733b360098" url: "https://pub.flutter-io.cn" source: hosted - version: "2.4.9" + version: "2.4.10" flutter_staggered_grid_view: dependency: "direct main" description: @@ -439,10 +439,10 @@ packages: dependency: "direct main" description: name: flutter_tabler_icons - sha256: b2abc371e50352f1a52b984eeed80e05657ee3cd7d1abbcd6793e0a8e50118b8 + sha256: e72647c386049e78b0880467f7f1e56809dde7d7421d14c5a58dee1c64fa35e0 url: "https://pub.flutter-io.cn" source: hosted - version: "1.20.0" + version: "1.21.0" flutter_test: dependency: "direct dev" description: flutter @@ -505,10 +505,10 @@ packages: dependency: "direct main" description: name: hooks_riverpod - sha256: c12a456e03ef9be65b0be66963596650ad7a3220e96c7e7b0a048562ea32d6ae + sha256: "758b07eba336e3cbacbd81dba481f2228a14102083fdde07045e8514e8054c49" url: "https://pub.flutter-io.cn" source: hosted - version: "2.4.9" + version: "2.4.10" hotreloader: dependency: transitive description: @@ -561,10 +561,10 @@ packages: dependency: transitive description: name: image - sha256: "004a2e90ce080f8627b5a04aecb4cdfac87d2c3f3b520aa291260be5a32c033d" + sha256: "49a0d4b0c12402853d3f227fe7c315601b238d126aa4caa5dbb2dcf99421aa4a" url: "https://pub.flutter-io.cn" source: hosted - version: "4.1.4" + version: "4.1.6" image_compression: dependency: "direct main" description: @@ -609,10 +609,10 @@ packages: dependency: transitive description: name: lints - sha256: "0a217c6c989d21039f1498c3ed9f3ed71b354e69873f13a8dfc3c9fe76f1b452" + sha256: cbf8d4b858bb0134ef3ef87841abdf8d63bfc255c266b7bf6b39daa1085c4290 url: "https://pub.flutter-io.cn" source: hosted - version: "2.1.1" + version: "3.0.0" logger: dependency: "direct main" description: @@ -913,42 +913,42 @@ packages: dependency: transitive description: name: riverpod - sha256: "942999ee48b899f8a46a860f1e13cee36f2f77609eb54c5b7a669bb20d550b11" + sha256: "548e2192eb7aeb826eb89387f814edb76594f3363e2c0bb99dd733d795ba3589" url: "https://pub.flutter-io.cn" source: hosted - version: "2.4.9" + version: "2.5.0" riverpod_analyzer_utils: dependency: transitive description: name: riverpod_analyzer_utils - sha256: d4dabc35358413bf4611fcb6abb46308a67c4ef4cd5e69fd3367b11925c59f57 + sha256: "8b71f03fc47ae27d13769496a1746332df4cec43918aeba9aff1e232783a780f" url: "https://pub.flutter-io.cn" source: hosted - version: "0.5.0" + version: "0.5.1" riverpod_annotation: dependency: "direct main" description: name: riverpod_annotation - sha256: b70e95fbd5ca7ce42f5148092022971bb2e9843b6ab71e97d479e8ab52e98979 + sha256: "77e5d51afa4fa3e67903fb8746f33d368728d7051a0b6c292bcee60aeba46d95" url: "https://pub.flutter-io.cn" source: hosted - version: "2.3.3" + version: "2.3.4" riverpod_generator: dependency: "direct dev" description: name: riverpod_generator - sha256: ff8f064f1d7ef3cc6af481bba8e9a3fcdb4d34df34fac1b39bbc003167065be0 + sha256: "359068f04879347ae4edbe66c81cc95f83fa1743806d1a0c86e55dd3c33ebb32" url: "https://pub.flutter-io.cn" source: hosted - version: "2.3.9" + version: "2.3.11" riverpod_lint: dependency: "direct dev" description: name: riverpod_lint - sha256: "944929ef82c9bfeaa455ccab97920abcf847a0ffed5c9f6babc520a95db25176" + sha256: e9bbd02e9e89e18eecb183bbca556d7b523a0669024da9b8167c08903f442937 url: "https://pub.flutter-io.cn" source: hosted - version: "2.3.7" + version: "2.3.9" rxdart: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index b91b7b0..7d87f00 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -16,7 +16,7 @@ publish_to: 'none' # Remove this line if you wish to publish to pub.dev # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html # In Windows, build-name is used as the major, minor, and patch parts # of the product and file versions while build-number is used as the build suffix. -version: 0.1.1+2 +version: 0.2.0+3 environment: sdk: '>=3.1.0 <4.0.0' @@ -83,10 +83,10 @@ dev_dependencies: # activated in the `analysis_options.yaml` file located at the root of your # package. See that file for information about deactivating specific lint # rules and activating additional ones. - flutter_lints: ^2.0.0 + flutter_lints: ^3.0.1 riverpod_generator: ^2.3.8 build_runner: ^2.4.6 - custom_lint: ^0.5.7 + custom_lint: ^0.6.0 riverpod_lint: ^2.3.6 # For information on the generic Dart part of this file, see the