From 126048b4fad4193b7b956c8565f4ba2fe10ca30f Mon Sep 17 00:00:00 2001 From: LittleSheep Date: Thu, 23 Oct 2025 01:27:40 +0800 Subject: [PATCH] :lipstick: Optimize post metadata --- assets/i18n/en-US.json | 1 + lib/screens/posts/post_category_detail.dart | 2 +- lib/screens/posts/post_detail.dart | 2 +- lib/widgets/post/post_shared.dart | 195 ++++++++++++-------- 4 files changed, 116 insertions(+), 84 deletions(-) diff --git a/assets/i18n/en-US.json b/assets/i18n/en-US.json index 514818ae..5991b2ba 100644 --- a/assets/i18n/en-US.json +++ b/assets/i18n/en-US.json @@ -871,6 +871,7 @@ "pollShortTextAnswerPreview": "Short text answer (preview)", "award": "Award", "awardPost": "Award Post", + "awardPoints": "Awarded {} points", "awardMessage": "Message", "awardMessageHint": "Enter your award message...", "awardAttitude": "Attitude", diff --git a/lib/screens/posts/post_category_detail.dart b/lib/screens/posts/post_category_detail.dart index 4614c642..e2befaa3 100644 --- a/lib/screens/posts/post_category_detail.dart +++ b/lib/screens/posts/post_category_detail.dart @@ -237,7 +237,7 @@ class PostCategoryDetailScreen extends HookConsumerWidget { loading: () => ResponseLoadingWidget(), ), ), - ), + ).padding(horizontal: 8), ), ), const SliverGap(4), diff --git a/lib/screens/posts/post_detail.dart b/lib/screens/posts/post_detail.dart index cd198b19..33bb8bea 100644 --- a/lib/screens/posts/post_detail.dart +++ b/lib/screens/posts/post_detail.dart @@ -289,7 +289,7 @@ class PostActionButtons extends HookConsumerWidget { builder: (context) => PostAwardHistorySheet(postId: post.id), ); }, - icon: const Icon(Symbols.star), + icon: const Icon(Symbols.emoji_events), label: post.awardedScore > 0 ? Text('${formatScore(post.awardedScore)} pts') diff --git a/lib/widgets/post/post_shared.dart b/lib/widgets/post/post_shared.dart index 731cde30..3f79cf2c 100644 --- a/lib/widgets/post/post_shared.dart +++ b/lib/widgets/post/post_shared.dart @@ -647,35 +647,13 @@ class PostHeader extends StatelessWidget { ]), ], ), - Row( - spacing: 6, - crossAxisAlignment: CrossAxisAlignment.end, - children: [ - Text( - !isFullPost && isRelativeTime - ? (item.publishedAt ?? item.createdAt)! - .formatRelative(context) - : (item.publishedAt ?? item.createdAt)! - .formatSystem(), - ).fontSize(10), - if (item.editedAt != null) - Text( - 'editedAt'.tr( - args: [ - !isFullPost && isRelativeTime - ? item.editedAt!.formatRelative(context) - : item.editedAt!.formatSystem(), - ], - ), - ).fontSize(10), - if (item.visibility != 0) - Text( - PostVisibilityHelpers.getVisibilityText( - item.visibility, - ).tr(), - ).fontSize(10), - ], - ), + Text( + !isFullPost && isRelativeTime + ? (item.publishedAt ?? item.createdAt)!.formatRelative( + context, + ) + : (item.publishedAt ?? item.createdAt)!.formatSystem(), + ).fontSize(10), ], ), ), @@ -694,6 +672,7 @@ class PostBody extends ConsumerWidget { final Widget? translationSection; final bool isInteractive; final EdgeInsets renderingPadding; + final bool isRelativeTime; const PostBody({ super.key, @@ -703,10 +682,113 @@ class PostBody extends ConsumerWidget { this.translationSection, this.isInteractive = true, this.renderingPadding = EdgeInsets.zero, + this.isRelativeTime = true, }); @override Widget build(BuildContext context, WidgetRef ref) { + final metadataChildren = []; + + if (item.tags.isNotEmpty) { + metadataChildren.add( + Wrap( + runAlignment: WrapAlignment.center, + spacing: 8, + children: [ + const Icon(Symbols.label, size: 16).padding(top: 2), + for (final tag in isFullPost ? item.tags : item.tags.take(3)) + InkWell( + onTap: + isInteractive + ? () { + GoRouter.of(context).pushNamed( + 'postTagDetail', + pathParameters: {'slug': tag.slug}, + ); + } + : null, + child: Text('#${tag.name ?? tag.slug}'), + ), + if (!isFullPost && item.tags.length > 3) + Text('+${item.tags.length - 3}').opacity(0.6), + ], + ), + ); + } + if (item.categories.isNotEmpty) { + metadataChildren.add( + Wrap( + runAlignment: WrapAlignment.center, + spacing: 8, + children: [ + const Icon(Symbols.category, size: 16).padding(top: 2), + for (final category + in isFullPost ? item.categories : item.categories.take(2)) + InkWell( + onTap: + isInteractive + ? () { + GoRouter.of(context).pushNamed( + 'postCategoryDetail', + pathParameters: {'slug': category.slug}, + ); + } + : null, + child: Text(category.categoryDisplayTitle), + ), + if (!isFullPost && item.categories.length > 2) + Text('+${item.categories.length - 2}').opacity(0.6), + ], + ), + ); + } + if (item.editedAt != null) { + metadataChildren.add( + Row( + spacing: 8, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + const Icon(Symbols.edit, size: 16), + Text( + 'editedAt'.tr( + args: [ + !isFullPost && isRelativeTime + ? item.editedAt!.formatRelative(context) + : item.editedAt!.formatSystem(), + ], + ), + ).fontSize(13), + ], + ), + ); + } + if (item.visibility != 0) { + metadataChildren.add( + Row( + spacing: 8, + children: [ + const Icon(Symbols.visibility_lock, size: 16).padding(top: 2), + Text( + PostVisibilityHelpers.getVisibilityText(item.visibility).tr(), + ).fontSize(13), + ], + ), + ); + } + if (item.awardedScore != 0) { + metadataChildren.add( + Row( + spacing: 8, + children: [ + const Icon(Symbols.emoji_events, size: 16).padding(top: 2), + Text( + 'awardPoints'.tr(args: [item.awardedScore.toString()]), + ).fontSize(13), + ], + ), + ); + } + return Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ @@ -819,63 +901,12 @@ class PostBody extends ConsumerWidget { vertical: 4, ), ), - if (item.tags.isNotEmpty || item.categories.isNotEmpty) + if (metadataChildren.isNotEmpty) Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, spacing: 2, - children: [ - if (item.tags.isNotEmpty) - Wrap( - runAlignment: WrapAlignment.center, - spacing: 8, - children: [ - const Icon(Symbols.label, size: 16).padding(top: 2), - for (final tag - in isFullPost ? item.tags : item.tags.take(3)) - InkWell( - onTap: - isInteractive - ? () { - GoRouter.of(context).pushNamed( - 'postTagDetail', - pathParameters: {'slug': tag.slug}, - ); - } - : null, - child: Text('#${tag.name ?? tag.slug}'), - ), - if (!isFullPost && item.tags.length > 3) - Text('+${item.tags.length - 3}').opacity(0.6), - ], - ), - if (item.categories.isNotEmpty) - Wrap( - runAlignment: WrapAlignment.center, - spacing: 8, - children: [ - const Icon(Symbols.category, size: 16).padding(top: 2), - for (final category - in isFullPost - ? item.categories - : item.categories.take(2)) - InkWell( - onTap: - isInteractive - ? () { - GoRouter.of(context).pushNamed( - 'postCategoryDetail', - pathParameters: {'slug': category.slug}, - ); - } - : null, - child: Text(category.categoryDisplayTitle), - ), - if (!isFullPost && item.categories.length > 2) - Text('+${item.categories.length - 2}').opacity(0.6), - ], - ), - ], + children: metadataChildren, ).padding(horizontal: renderingPadding.horizontal + 4, top: 4), if (item.meta?['embeds'] != null) EmbedListWidget(