Better(?) comment nesting

This commit is contained in:
LittleSheep 2025-03-16 21:41:38 +08:00
parent fc5a79b29b
commit 7b96013406
7 changed files with 132 additions and 8 deletions

View File

@ -207,6 +207,7 @@
"one": "{} comment",
"other": "{} comments"
},
"postCommentExpand": "Show comments",
"settingsAppearance": "Appearance",
"settingsCustomFonts": "Custom Fonts",
"settingsCustomFontsDescription": "Set custom fonts for the application.",

View File

@ -205,6 +205,7 @@
"one": "{} 条评论",
"other": "{} 条评论"
},
"postCommentExpand": "展开评论",
"settingsAppearance": "外观",
"settingsCustomFonts": "自定义字体",
"settingsCustomFontsDescription": "设置应用程序使用的字体。",

View File

@ -205,6 +205,7 @@
"one": "{} 條評論",
"other": "{} 條評論"
},
"postCommentExpand": "展開評論",
"settingsAppearance": "外觀",
"settingsCustomFonts": "自定義字體",
"settingsCustomFontsDescription": "設置應用程序使用的字體。",

View File

@ -205,6 +205,7 @@
"one": "{} 條評論",
"other": "{} 條評論"
},
"postCommentExpand": "展開評論",
"settingsAppearance": "外觀",
"settingsCustomFonts": "自定義字體",
"settingsCustomFontsDescription": "設置應用程序使用的字體。",

View File

@ -14,6 +14,7 @@ class AccountImage extends StatelessWidget {
final Widget? fallbackWidget;
final Widget? badge;
final Offset? badgeOffset;
final FilterQuality? filterQuality;
const AccountImage({
super.key,
@ -25,6 +26,7 @@ class AccountImage extends StatelessWidget {
this.fallbackWidget,
this.badge,
this.badgeOffset,
this.filterQuality,
});
@override
@ -54,6 +56,7 @@ class AccountImage extends StatelessWidget {
)
: AutoResizeUniversalImage(
sn.getAttachmentUrl(url),
filterQuality: filterQuality,
key: Key('attachment-${content.hashCode}'),
fit: BoxFit.cover,
),

View File

@ -138,6 +138,7 @@ class PostCommentSliverListState extends State<PostCommentSliverList> {
child: PostItem(
data: _posts[idx],
maxWidth: widget.maxWidth,
showExpandableComments: true,
onSelectAnswer: widget.parentPost.type == 'question'
? () => _selectAnswer(_posts[idx])
: null,
@ -209,6 +210,7 @@ class _PostCommentListPopupState extends State<PostCommentListPopup> {
if (ua.isAuthorized)
SliverToBoxAdapter(
child: Container(
margin: const EdgeInsets.only(bottom: 8),
height: 240,
decoration: BoxDecoration(
border: Border.symmetric(

View File

@ -50,6 +50,7 @@ class OpenablePostItem extends StatelessWidget {
final bool showComments;
final bool showMenu;
final bool showFullPost;
final bool showExpandableComments;
final double? maxWidth;
final Function(SnPost data)? onChanged;
final Function()? onDeleted;
@ -62,6 +63,7 @@ class OpenablePostItem extends StatelessWidget {
this.showComments = true,
this.showMenu = true,
this.showFullPost = false,
this.showExpandableComments = false,
this.maxWidth,
this.onChanged,
this.onDeleted,
@ -83,6 +85,7 @@ class OpenablePostItem extends StatelessWidget {
maxWidth: maxWidth,
showComments: showComments,
showFullPost: showFullPost,
showExpandableComments: showExpandableComments,
onChanged: onChanged,
onDeleted: onDeleted,
onSelectAnswer: onSelectAnswer,
@ -115,6 +118,7 @@ class PostItem extends StatelessWidget {
final bool showComments;
final bool showMenu;
final bool showFullPost;
final bool showExpandableComments;
final double? maxWidth;
final Function(SnPost data)? onChanged;
final Function()? onDeleted;
@ -127,6 +131,7 @@ class PostItem extends StatelessWidget {
this.showComments = true,
this.showMenu = true,
this.showFullPost = false,
this.showExpandableComments = false,
this.maxWidth,
this.onChanged,
this.onDeleted,
@ -354,14 +359,17 @@ class PostItem extends StatelessWidget {
),
if (data.preload?.poll != null)
PostPoll(poll: data.preload!.poll!)
.padding(horizontal: 12, vertical: 4),
.padding(left: 60, right: 12, top: 12, bottom: 4),
if (data.body['content'] != null &&
(cfg.prefs.getBool(kAppExpandPostLink) ?? true))
LinkPreviewWidget(
text: data.body['content'],
).padding(left: 60, right: 4),
_PostFeaturedComment(data: data, maxWidth: maxWidth)
.padding(left: 60, right: 12),
if (showExpandableComments)
_PostCommentIntent(data: data).padding(left: 60, right: 12)
else
_PostFeaturedComment(data: data, maxWidth: maxWidth)
.padding(left: 60, right: 12),
Padding(
padding: const EdgeInsets.only(top: 4),
child: _PostReactionList(
@ -404,13 +412,24 @@ class PostShareImageWidget extends StatelessWidget {
child: AutoResizeUniversalImage(
sn.getAttachmentUrl(data.preload!.thumbnail!.rid),
fit: BoxFit.cover,
filterQuality: FilterQuality.high,
),
),
).padding(bottom: 8),
_PostContentHeader(
data: data,
isRelativeDate: false,
).padding(horizontal: 16, bottom: 8),
Row(
children: [
_PostAvatar(
data: data,
isCompact: false,
filterQuality: FilterQuality.high,
),
const Gap(12),
_PostContentHeader(
data: data,
isRelativeDate: false,
).padding(horizontal: 16, bottom: 8),
],
),
if (data.type == 'question')
_PostQuestionHint(data: data).padding(horizontal: 16, bottom: 8),
_PostHeadline(
@ -803,7 +822,12 @@ class _PostHeadline extends StatelessWidget {
class _PostAvatar extends StatelessWidget {
final SnPost data;
final bool isCompact;
const _PostAvatar({required this.data, required this.isCompact});
final FilterQuality? filterQuality;
const _PostAvatar({
required this.data,
required this.isCompact,
this.filterQuality,
});
@override
Widget build(BuildContext context) {
@ -815,6 +839,7 @@ class _PostAvatar extends StatelessWidget {
return GestureDetector(
child: data.preload?.realm == null
? AccountImage(
filterQuality: filterQuality,
content: data.publisher.avatar,
radius: isCompact ? 12 : 20,
borderRadius: data.publisher.type == 1 ? (isCompact ? 4 : 8) : 20,
@ -827,6 +852,7 @@ class _PostAvatar extends StatelessWidget {
: null,
)
: AccountImage(
filterQuality: filterQuality,
content: data.preload!.realm!.avatar,
radius: isCompact ? 12 : 20,
borderRadius: isCompact ? 4 : 8,
@ -1395,6 +1421,95 @@ class _PostTruncatedHint extends StatelessWidget {
}
}
class _PostCommentIntent extends StatefulWidget {
final SnPost data;
const _PostCommentIntent({required this.data});
@override
State<_PostCommentIntent> createState() => _PostCommentIntentState();
}
class _PostCommentIntentState extends State<_PostCommentIntent> {
bool _isBusy = false;
int? _totalCount;
final List<SnPost> _comments = List.empty(growable: true);
bool get _isAllLoaded =>
_totalCount != null && _comments.length >= _totalCount!;
Future<void> _fetchComments() async {
setState(() => _isBusy = true);
try {
final sn = context.read<SnNetworkProvider>();
final resp = await sn.client.get(
'/cgi/co/posts/${widget.data.id}/replies',
queryParameters: {
'take': 10,
'offset': _comments.length,
},
);
_totalCount = resp.data['count'];
_comments.addAll(
resp.data['data'].map((ele) => SnPost.fromJson(ele)).cast<SnPost>(),
);
} catch (err) {
if (!mounted) return;
context.showErrorDialog(err);
} finally {
setState(() => _isBusy = false);
}
}
@override
Widget build(BuildContext context) {
return Column(
children: [
if (_comments.isNotEmpty)
Card(
margin: EdgeInsets.zero,
child: Column(
spacing: 8,
children: [
for (final ele in _comments)
PostItem(
data: ele,
showExpandableComments: true,
maxWidth: double.infinity,
).padding(vertical: 8),
],
),
).padding(bottom: 8),
Row(
children: [
Transform.flip(
flipX: true,
child: const Icon(Symbols.comment, size: 20),
),
const Gap(4),
Text('postCommentsDetailed'.plural(widget.data.metric.replyCount)),
if (widget.data.metric.replyCount > 0 && !_isAllLoaded)
SizedBox(
width: 20,
height: 20,
child: IconButton(
visualDensity: VisualDensity(horizontal: -4, vertical: -4),
constraints: const BoxConstraints(),
padding: EdgeInsets.zero,
icon: const Icon(Symbols.expand_more, size: 18),
onPressed: _isBusy
? null
: () {
_fetchComments();
},
),
).padding(left: 8),
],
).opacity(0.75).padding(horizontal: 4),
],
);
}
}
class _PostFeaturedComment extends StatefulWidget {
final SnPost data;
final double? maxWidth;