Article thumbnail

This commit is contained in:
LittleSheep 2024-12-07 17:43:44 +08:00
parent 599dd4827b
commit b583780cfc
10 changed files with 370 additions and 178 deletions

View File

@ -257,8 +257,13 @@
"addAttachmentFromAlbum": "Add from album", "addAttachmentFromAlbum": "Add from album",
"addAttachmentFromClipboard": "Paste file", "addAttachmentFromClipboard": "Paste file",
"attachmentPastedImage": "Pasted Image", "attachmentPastedImage": "Pasted Image",
"notificationUnread": "未读", "attachmentInsertLink": "Insert Link",
"notificationRead": "已读", "attachmentSetAsPostThumbnail": "Set as post thumbnail",
"attachmentUnsetAsPostThumbnail": "Unset as post thumbnail",
"attachmentSetThumbnail": "Set thumbnail",
"attachmentUpload": "Upload",
"notificationUnread": "Unread",
"notificationRead": "Read",
"notificationMarkAllRead": "Mark all notifications as read", "notificationMarkAllRead": "Mark all notifications as read",
"notificationMarkAllReadDescription": "Are you sure you want to mark all notifications as read? This operation is irreversible.", "notificationMarkAllReadDescription": "Are you sure you want to mark all notifications as read? This operation is irreversible.",
"notificationMarkAllReadPrompt": { "notificationMarkAllReadPrompt": {
@ -377,5 +382,6 @@
"accountStatus": "Status", "accountStatus": "Status",
"accountStatusOnline": "Online", "accountStatusOnline": "Online",
"accountStatusOffline": "Offline", "accountStatusOffline": "Offline",
"accountStatusLastSeen": "Last seen at {}" "accountStatusLastSeen": "Last seen at {}",
"postArticle": "Article on the Solar Network"
} }

View File

@ -257,6 +257,11 @@
"addAttachmentFromAlbum": "从相册中添加附件", "addAttachmentFromAlbum": "从相册中添加附件",
"addAttachmentFromClipboard": "粘贴附件", "addAttachmentFromClipboard": "粘贴附件",
"attachmentPastedImage": "粘贴的图片", "attachmentPastedImage": "粘贴的图片",
"attachmentInsertLink": "插入连接",
"attachmentSetAsPostThumbnail": "设置为帖子缩略图",
"attachmentUnsetAsPostThumbnail": "取消设置为帖子缩略图",
"attachmentSetThumbnail": "设置缩略图",
"attachmentUpload": "上传",
"notificationUnread": "未读", "notificationUnread": "未读",
"notificationRead": "已读", "notificationRead": "已读",
"notificationMarkAllRead": "已读所有通知", "notificationMarkAllRead": "已读所有通知",
@ -377,5 +382,6 @@
"accountStatus": "状态", "accountStatus": "状态",
"accountStatusOnline": "在线", "accountStatusOnline": "在线",
"accountStatusOffline": "离线", "accountStatusOffline": "离线",
"accountStatusLastSeen": "最后一次在 {} 上线" "accountStatusLastSeen": "最后一次在 {} 上线",
"postArticle": "Solar Network 上的文章"
} }

View File

@ -28,6 +28,8 @@ class PostWriteMedia {
final XFile? file; final XFile? file;
final Uint8List? raw; final Uint8List? raw;
PostWriteMedia? thumbnail;
PostWriteMedia(this.attachment, {this.file, this.raw}) { PostWriteMedia(this.attachment, {this.file, this.raw}) {
name = attachment!.name; name = attachment!.name;
@ -67,8 +69,7 @@ class PostWriteMedia {
} }
} }
PostWriteMedia.fromBytes(this.raw, this.name, this.type, PostWriteMedia.fromBytes(this.raw, this.name, this.type, {this.attachment, this.file});
{this.attachment, this.file});
bool get isEmpty => attachment == null && file == null && raw == null; bool get isEmpty => attachment == null && file == null && raw == null;
@ -102,8 +103,7 @@ class PostWriteMedia {
}) { }) {
if (attachment != null) { if (attachment != null) {
final sn = context.read<SnNetworkProvider>(); final sn = context.read<SnNetworkProvider>();
final ImageProvider provider = final ImageProvider provider = UniversalImage.provider(sn.getAttachmentUrl(attachment!.rid));
UniversalImage.provider(sn.getAttachmentUrl(attachment!.rid));
if (width != null && height != null) { if (width != null && height != null) {
return ResizeImage( return ResizeImage(
provider, provider,
@ -114,8 +114,7 @@ class PostWriteMedia {
} }
return provider; return provider;
} else if (file != null) { } else if (file != null) {
final ImageProvider provider = final ImageProvider provider = kIsWeb ? NetworkImage(file!.path) : FileImage(File(file!.path));
kIsWeb ? NetworkImage(file!.path) : FileImage(File(file!.path));
if (width != null && height != null) { if (width != null && height != null) {
return ResizeImage( return ResizeImage(
provider, provider,
@ -162,9 +161,10 @@ class PostWriteController extends ChangeNotifier {
String mode = kTitleMap.keys.first; String mode = kTitleMap.keys.first;
String get title => titleController.text; String get title => titleController.text;
String get description => descriptionController.text; String get description => descriptionController.text;
bool get isRelatedNull =>
![editingPost, repostingPost, replyingPost].any((ele) => ele != null); bool get isRelatedNull => ![editingPost, repostingPost, replyingPost].any((ele) => ele != null);
bool isLoading = false, isBusy = false; bool isLoading = false, isBusy = false;
double? progress; double? progress;
@ -176,6 +176,7 @@ class PostWriteController extends ChangeNotifier {
List<int> visibleUsers = List.empty(); List<int> visibleUsers = List.empty();
List<int> invisibleUsers = List.empty(); List<int> invisibleUsers = List.empty();
List<String> tags = List.empty(); List<String> tags = List.empty();
PostWriteMedia? thumbnail;
List<PostWriteMedia> attachments = List.empty(growable: true); List<PostWriteMedia> attachments = List.empty(growable: true);
DateTime? publishedAt, publishedUntil; DateTime? publishedAt, publishedUntil;
@ -203,9 +204,11 @@ class PostWriteController extends ChangeNotifier {
invisibleUsers = List.from(post.invisibleUsersList ?? []); invisibleUsers = List.from(post.invisibleUsersList ?? []);
visibility = post.visibility; visibility = post.visibility;
tags = List.from(post.tags.map((ele) => ele.alias)); tags = List.from(post.tags.map((ele) => ele.alias));
attachments.addAll( attachments.addAll(post.preload?.attachments?.map((ele) => PostWriteMedia(ele)) ?? []);
post.preload?.attachments?.map((ele) => PostWriteMedia(ele)) ?? [],
); if (post.preload?.thumbnail != null) {
thumbnail = PostWriteMedia(post.preload!.thumbnail);
}
editingPost = post; editingPost = post;
} }
@ -228,6 +231,43 @@ class PostWriteController extends ChangeNotifier {
} }
} }
Future<SnAttachment> _uploadAttachment(BuildContext context, PostWriteMedia media) async {
final attach = context.read<SnAttachmentProvider>();
final place = await attach.chunkedUploadInitialize(
(await media.length())!,
media.name,
'interactive',
null,
mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null,
);
final item = await attach.chunkedUploadParts(
media.toFile()!,
place.$1,
place.$2,
onProgress: (progress) {
progress = progress;
notifyListeners();
},
);
return item;
}
Future<void> uploadSingleAttachment(BuildContext context, int idx) async {
if (isBusy) return;
final media = idx == -1 ? thumbnail! : attachments[idx];
isBusy = true;
notifyListeners();
final item = await _uploadAttachment(context, media);
attachments[idx] = PostWriteMedia(item);
notifyListeners();
}
Future<void> post(BuildContext context) async { Future<void> post(BuildContext context) async {
if (isBusy || publisher == null) return; if (isBusy || publisher == null) return;
@ -240,6 +280,11 @@ class PostWriteController extends ChangeNotifier {
// Uploading attachments // Uploading attachments
try { try {
if (thumbnail != null && thumbnail!.attachment == null) {
final thumb = await _uploadAttachment(context, thumbnail!);
thumbnail = PostWriteMedia(thumb);
}
for (int i = 0; i < attachments.length; i++) { for (int i = 0; i < attachments.length; i++) {
final media = attachments[i]; final media = attachments[i];
if (media.attachment != null) continue; // Already uploaded, skip if (media.attachment != null) continue; // Already uploaded, skip
@ -250,9 +295,7 @@ class PostWriteController extends ChangeNotifier {
media.name, media.name,
'interactive', 'interactive',
null, null,
mimetype: media.raw != null && media.type == PostWriteMediaType.image mimetype: media.raw != null && media.type == PostWriteMediaType.image ? 'image/png' : null,
? 'image/png'
: null,
); );
final item = await attach.chunkedUploadParts( final item = await attach.chunkedUploadParts(
@ -261,8 +304,7 @@ class PostWriteController extends ChangeNotifier {
place.$2, place.$2,
onProgress: (progress) { onProgress: (progress) {
// Calculate overall progress for attachments // Calculate overall progress for attachments
progress = ((i + progress) / attachments.length) * progress = ((i + progress) / attachments.length) * kAttachmentProgressWeight;
kAttachmentProgressWeight;
notifyListeners(); notifyListeners();
}, },
); );
@ -292,32 +334,24 @@ class PostWriteController extends ChangeNotifier {
'publisher': publisher!.id, 'publisher': publisher!.id,
'content': contentController.text, 'content': contentController.text,
if (titleController.text.isNotEmpty) 'title': titleController.text, if (titleController.text.isNotEmpty) 'title': titleController.text,
if (descriptionController.text.isNotEmpty) if (descriptionController.text.isNotEmpty) 'description': descriptionController.text,
'description': descriptionController.text, if (thumbnail != null && thumbnail!.attachment != null) 'thumbnail': thumbnail!.attachment!.rid,
'attachments': attachments 'attachments': attachments.where((e) => e.attachment != null).map((e) => e.attachment!.rid).toList(),
.where((e) => e.attachment != null)
.map((e) => e.attachment!.rid)
.toList(),
'tags': tags.map((ele) => {'alias': ele}).toList(), 'tags': tags.map((ele) => {'alias': ele}).toList(),
'visibility': visibility, 'visibility': visibility,
'visible_users_list': visibleUsers, 'visible_users_list': visibleUsers,
'invisible_users_list': invisibleUsers, 'invisible_users_list': invisibleUsers,
if (publishedAt != null) if (publishedAt != null) 'published_at': publishedAt!.toUtc().toIso8601String(),
'published_at': publishedAt!.toUtc().toIso8601String(), if (publishedUntil != null) 'published_until': publishedAt!.toUtc().toIso8601String(),
if (publishedUntil != null)
'published_until': publishedAt!.toUtc().toIso8601String(),
if (replyingPost != null) 'reply_to': replyingPost!.id, if (replyingPost != null) 'reply_to': replyingPost!.id,
if (repostingPost != null) 'repost_to': repostingPost!.id, if (repostingPost != null) 'repost_to': repostingPost!.id,
}, },
onSendProgress: (count, total) { onSendProgress: (count, total) {
progress = progress = baseProgressVal + (count / total) * (kPostingProgressWeight / 2);
baseProgressVal + (count / total) * (kPostingProgressWeight / 2);
notifyListeners(); notifyListeners();
}, },
onReceiveProgress: (count, total) { onReceiveProgress: (count, total) {
progress = baseProgressVal + progress = baseProgressVal + (kPostingProgressWeight / 2) + (count / total) * (kPostingProgressWeight / 2);
(kPostingProgressWeight / 2) +
(count / total) * (kPostingProgressWeight / 2);
notifyListeners(); notifyListeners();
}, },
options: Options( options: Options(
@ -339,12 +373,31 @@ class PostWriteController extends ChangeNotifier {
} }
void setAttachmentAt(int idx, PostWriteMedia item) { void setAttachmentAt(int idx, PostWriteMedia item) {
attachments[idx] = item; if (idx == -1) {
thumbnail = item;
} else {
attachments[idx] = item;
}
notifyListeners(); notifyListeners();
} }
void removeAttachmentAt(int idx) { void removeAttachmentAt(int idx) {
attachments.removeAt(idx); if (idx == -1) {
thumbnail = null;
} else {
attachments.removeAt(idx);
}
notifyListeners();
}
void setThumbnail(int? idx) {
if (idx == null) {
attachments.add(thumbnail!);
thumbnail = null;
} else {
thumbnail = attachments[idx];
attachments.removeAt(idx);
}
notifyListeners(); notifyListeners();
} }
@ -383,11 +436,21 @@ class PostWriteController extends ChangeNotifier {
notifyListeners(); notifyListeners();
} }
void setProgress(double? value) {
progress = value;
notifyListeners();
}
void setIsBusy(bool value) { void setIsBusy(bool value) {
isBusy = value; isBusy = value;
notifyListeners(); notifyListeners();
} }
void setMode(String value) {
mode = value;
notifyListeners();
}
void reset() { void reset() {
publishedAt = null; publishedAt = null;
publishedUntil = null; publishedUntil = null;

View File

@ -45,10 +45,6 @@ void main() async {
options: DefaultFirebaseOptions.currentPlatform, options: DefaultFirebaseOptions.currentPlatform,
); );
if (!kReleaseMode) {
// debugInvertOversizedImages = true;
}
GoRouter.optionURLReflectsImperativeAPIs = true; GoRouter.optionURLReflectsImperativeAPIs = true;
usePathUrlStrategy(); usePathUrlStrategy();

View File

@ -110,6 +110,7 @@ class _PostDetailScreenState extends State<PostDetailScreen> {
data: _data!, data: _data!,
maxWidth: 640, maxWidth: 640,
showComments: false, showComments: false,
showFullPost: true,
onChanged: (data) { onChanged: (data) {
setState(() => _data = data); setState(() => _data = data);
}, },

View File

@ -25,6 +25,7 @@ class PostEditorScreen extends StatefulWidget {
final int? postEditId; final int? postEditId;
final int? postReplyId; final int? postReplyId;
final int? postRepostId; final int? postRepostId;
const PostEditorScreen({ const PostEditorScreen({
super.key, super.key,
required this.mode, required this.mode,
@ -41,6 +42,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
final PostWriteController _writeController = PostWriteController(); final PostWriteController _writeController = PostWriteController();
bool _isFetching = false; bool _isFetching = false;
bool get _isLoading => _isFetching || _writeController.isLoading; bool get _isLoading => _isFetching || _writeController.isLoading;
List<SnPublisher>? _publishers; List<SnPublisher>? _publishers;
@ -105,6 +107,8 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
if (!PostWriteController.kTitleMap.keys.contains(widget.mode)) { if (!PostWriteController.kTitleMap.keys.contains(widget.mode)) {
context.showErrorDialog('Unknown post type'); context.showErrorDialog('Unknown post type');
Navigator.pop(context); Navigator.pop(context);
} else {
_writeController.setMode(widget.mode);
} }
_fetchPublishers(); _fetchPublishers();
_writeController.fetchRelatedPost( _writeController.fetchRelatedPost(
@ -131,21 +135,13 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
textAlign: TextAlign.center, textAlign: TextAlign.center,
text: TextSpan(children: [ text: TextSpan(children: [
TextSpan( TextSpan(
text: _writeController.title.isNotEmpty text: _writeController.title.isNotEmpty ? _writeController.title : 'untitled'.tr(),
? _writeController.title style: Theme.of(context).textTheme.titleLarge!.copyWith(color: Colors.white),
: 'untitled'.tr(),
style: Theme.of(context)
.textTheme
.titleLarge!
.copyWith(color: Colors.white),
), ),
const TextSpan(text: '\n'), const TextSpan(text: '\n'),
TextSpan( TextSpan(
text: PostWriteController.kTitleMap[widget.mode]!.tr(), text: PostWriteController.kTitleMap[widget.mode]!.tr(),
style: Theme.of(context) style: Theme.of(context).textTheme.bodySmall!.copyWith(color: Colors.white),
.textTheme
.bodySmall!
.copyWith(color: Colors.white),
), ),
]), ]),
), ),
@ -181,17 +177,11 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
Expanded( Expanded(
child: Column( child: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: crossAxisAlignment: CrossAxisAlignment.start,
CrossAxisAlignment.start,
children: [ children: [
Text(item.nick).textStyle( Text(item.nick).textStyle(Theme.of(context).textTheme.bodyMedium!),
Theme.of(context)
.textTheme
.bodyMedium!),
Text('@${item.name}') Text('@${item.name}')
.textStyle(Theme.of(context) .textStyle(Theme.of(context).textTheme.bodySmall!)
.textTheme
.bodySmall!)
.fontSize(12), .fontSize(12),
], ],
), ),
@ -208,8 +198,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
CircleAvatar( CircleAvatar(
radius: 16, radius: 16,
backgroundColor: Colors.transparent, backgroundColor: Colors.transparent,
foregroundColor: foregroundColor: Theme.of(context).colorScheme.onSurface,
Theme.of(context).colorScheme.onSurface,
child: const Icon(Symbols.add), child: const Icon(Symbols.add),
), ),
const Gap(8), const Gap(8),
@ -218,8 +207,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Text('publishersNew').tr().textStyle( Text('publishersNew').tr().textStyle(Theme.of(context).textTheme.bodyMedium!),
Theme.of(context).textTheme.bodyMedium!),
], ],
), ),
), ),
@ -230,9 +218,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
value: _writeController.publisher, value: _writeController.publisher,
onChanged: (SnPublisher? value) { onChanged: (SnPublisher? value) {
if (value == null) { if (value == null) {
GoRouter.of(context) GoRouter.of(context).pushNamed('accountPublisherNew').then((value) {
.pushNamed('accountPublisherNew')
.then((value) {
if (value == true) { if (value == true) {
_publishers = null; _publishers = null;
_fetchPublishers(); _fetchPublishers();
@ -267,16 +253,11 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
child: ExpansionTile( child: ExpansionTile(
minTileHeight: 48, minTileHeight: 48,
leading: leading: const Icon(Symbols.reply).padding(left: 4),
const Icon(Symbols.reply).padding(left: 4),
title: Text('postReplyingNotice') title: Text('postReplyingNotice')
.fontSize(15) .fontSize(15)
.tr(args: [ .tr(args: ['@${_writeController.replyingPost!.publisher.name}']),
'@${_writeController.replyingPost!.publisher.name}' children: <Widget>[PostItem(data: _writeController.replyingPost!)],
]),
children: <Widget>[
PostItem(data: _writeController.replyingPost!)
],
), ),
), ),
const Divider(height: 1), const Divider(height: 1),
@ -292,13 +273,10 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
child: ExpansionTile( child: ExpansionTile(
minTileHeight: 48, minTileHeight: 48,
leading: const Icon(Symbols.forward) leading: const Icon(Symbols.forward).padding(left: 4),
.padding(left: 4),
title: Text('postRepostingNotice') title: Text('postRepostingNotice')
.fontSize(15) .fontSize(15)
.tr(args: [ .tr(args: ['@${_writeController.repostingPost!.publisher.name}']),
'@${_writeController.repostingPost!.publisher.name}'
]),
children: <Widget>[ children: <Widget>[
PostItem( PostItem(
data: _writeController.repostingPost!, data: _writeController.repostingPost!,
@ -319,16 +297,11 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
child: ExpansionTile( child: ExpansionTile(
minTileHeight: 48, minTileHeight: 48,
leading: const Icon(Symbols.edit_note) leading: const Icon(Symbols.edit_note).padding(left: 4),
.padding(left: 4),
title: Text('postEditingNotice') title: Text('postEditingNotice')
.fontSize(15) .fontSize(15)
.tr(args: [ .tr(args: ['@${_writeController.editingPost!.publisher.name}']),
'@${_writeController.editingPost!.publisher.name}' children: <Widget>[PostItem(data: _writeController.editingPost!)],
]),
children: <Widget>[
PostItem(data: _writeController.editingPost!)
],
), ),
), ),
const Divider(height: 1), const Divider(height: 1),
@ -347,14 +320,12 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
border: InputBorder.none, border: InputBorder.none,
), ),
onTapOutside: (_) => onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(),
FocusManager.instance.primaryFocus?.unfocus(),
), ),
] ]
.expandIndexed( .expandIndexed(
(idx, ele) => [ (idx, ele) => [
if (idx != 0 || _writeController.isRelatedNull) if (idx != 0 || _writeController.isRelatedNull) const Gap(8),
const Gap(8),
ele, ele,
], ],
) )
@ -362,10 +333,21 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
), ),
), ),
if (_writeController.attachments.isNotEmpty) if (_writeController.attachments.isNotEmpty || _writeController.thumbnail != null)
PostMediaPendingList( PostMediaPendingList(
thumbnail: _writeController.thumbnail,
attachments: _writeController.attachments, attachments: _writeController.attachments,
isBusy: _writeController.isBusy, isBusy: _writeController.isBusy,
onUpload: (int idx) async {
await _writeController.uploadSingleAttachment(context, idx);
},
onPostSetThumbnail: (int? idx) {
_writeController.setThumbnail(idx);
},
onInsertLink: (int idx) async {
_writeController.contentController.text +=
'\n![](solink://attachments/${_writeController.attachments[idx].attachment!.rid})';
},
onUpdate: (int idx, PostWriteMedia updatedMedia) async { onUpdate: (int idx, PostWriteMedia updatedMedia) async {
_writeController.setIsBusy(true); _writeController.setIsBusy(true);
try { try {
@ -390,13 +372,11 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
LoadingIndicator(isActive: _isLoading), LoadingIndicator(isActive: _isLoading),
if (_writeController.isBusy && if (_writeController.isBusy && _writeController.progress != null)
_writeController.progress != null)
TweenAnimationBuilder<double>( TweenAnimationBuilder<double>(
tween: Tween(begin: 0, end: _writeController.progress), tween: Tween(begin: 0, end: _writeController.progress),
duration: Duration(milliseconds: 300), duration: Duration(milliseconds: 300),
builder: (context, value, _) => builder: (context, value, _) => LinearProgressIndicator(value: value, minHeight: 2),
LinearProgressIndicator(value: value, minHeight: 2),
) )
else if (_writeController.isBusy) else if (_writeController.isBusy)
const LinearProgressIndicator(value: null, minHeight: 2), const LinearProgressIndicator(value: null, minHeight: 2),
@ -413,8 +393,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
PopupMenuButton( PopupMenuButton(
icon: Icon( icon: Icon(
Symbols.add_photo_alternate, Symbols.add_photo_alternate,
color: color: Theme.of(context).colorScheme.primary,
Theme.of(context).colorScheme.primary,
), ),
itemBuilder: (context) => [ itemBuilder: (context) => [
PopupMenuItem( PopupMenuItem(
@ -434,8 +413,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
children: [ children: [
const Icon(Symbols.content_paste), const Icon(Symbols.content_paste),
const Gap(16), const Gap(16),
Text('addAttachmentFromClipboard') Text('addAttachmentFromClipboard').tr(),
.tr(),
], ],
), ),
onTap: () { onTap: () {
@ -450,8 +428,7 @@ class _PostEditorScreenState extends State<PostEditorScreen> {
), ),
), ),
TextButton.icon( TextButton.icon(
onPressed: (_writeController.isBusy || onPressed: (_writeController.isBusy || _writeController.publisher == null)
_writeController.publisher == null)
? null ? null
: () { : () {
_writeController.post(context).then((_) { _writeController.post(context).then((_) {

View File

@ -56,7 +56,7 @@ class _AppNavigationDrawerState extends State<AppNavigationDrawer> {
], ],
).padding( ).padding(
horizontal: 32, horizontal: 32,
top: MediaQuery.of(context).padding.top > 16 ? 8 : 24, top: MediaQuery.of(context).padding.top > 32 ? 8 : 32,
bottom: 8, bottom: 8,
), ),
...destinations.where((ele) => ele.isPinned).map((ele) { ...destinations.where((ele) => ele.isPinned).map((ele) {

View File

@ -21,21 +21,25 @@ import 'package:surface/widgets/post/post_comment_list.dart';
import 'package:surface/widgets/post/post_meta_editor.dart'; import 'package:surface/widgets/post/post_meta_editor.dart';
import 'package:surface/widgets/post/post_reaction.dart'; import 'package:surface/widgets/post/post_reaction.dart';
import 'package:surface/widgets/post/publisher_popover.dart'; import 'package:surface/widgets/post/publisher_popover.dart';
import 'package:surface/widgets/universal_image.dart';
class PostItem extends StatelessWidget { class PostItem extends StatelessWidget {
final SnPost data; final SnPost data;
final bool showReactions; final bool showReactions;
final bool showComments; final bool showComments;
final bool showMenu; final bool showMenu;
final bool showFullPost;
final double? maxWidth; final double? maxWidth;
final Function(SnPost data)? onChanged; final Function(SnPost data)? onChanged;
final Function()? onDeleted; final Function()? onDeleted;
const PostItem({ const PostItem({
super.key, super.key,
required this.data, required this.data,
this.showReactions = true, this.showReactions = true,
this.showComments = true, this.showComments = true,
this.showMenu = true, this.showMenu = true,
this.showFullPost = false,
this.maxWidth, this.maxWidth,
this.onChanged, this.onChanged,
this.onDeleted, this.onDeleted,
@ -47,6 +51,75 @@ class PostItem extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final sn = context.read<SnNetworkProvider>();
// Article headline preview
if (!showFullPost && data.type == 'article') {
return Container(
constraints: BoxConstraints(maxWidth: maxWidth ?? double.infinity),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_PostContentHeader(
data: data,
onDeleted: () {
if (onDeleted != null) {}
},
).padding(horizontal: 12, top: 8, bottom: 4),
Container(
width: double.infinity,
margin: const EdgeInsets.only(bottom: 4, left: 12, right: 12),
decoration: BoxDecoration(
borderRadius: const BorderRadius.all(Radius.circular(8)),
border: Border.all(
color: Theme.of(context).dividerColor,
width: 1,
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (data.preload?.thumbnail != null)
AspectRatio(
aspectRatio: 16 / 9,
child: ClipRRect(
borderRadius: const BorderRadius.only(
topLeft: Radius.circular(8),
topRight: Radius.circular(8),
),
child: AutoResizeUniversalImage(
sn.getAttachmentUrl(data.preload!.thumbnail!.rid),
fit: BoxFit.cover,
),
),
),
const Gap(8),
_PostHeadline(data: data).padding(horizontal: 14),
const Gap(4),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (data.visibility > 0) _PostVisibilityHint(data: data),
_PostTruncatedHint(data: data),
],
).padding(horizontal: 12),
const Gap(8),
],
),
),
Text('postArticle').tr().fontSize(13).opacity(0.75).padding(horizontal: 24, bottom: 8),
if (data.tags.isNotEmpty) _PostTagsList(data: data).padding(horizontal: 16, bottom: 6),
_PostBottomAction(
data: data,
showComments: showComments,
showReactions: showReactions,
onChanged: _onChanged,
).padding(left: 8, right: 14),
],
),
);
}
return Column( return Column(
crossAxisAlignment: CrossAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.center,
children: [ children: [
@ -62,11 +135,9 @@ class PostItem extends StatelessWidget {
if (onDeleted != null) onDeleted!(); if (onDeleted != null) onDeleted!();
}, },
).padding(horizontal: 12, vertical: 8), ).padding(horizontal: 12, vertical: 8),
if (data.body['title'] != null || if (data.body['title'] != null || data.body['description'] != null)
data.body['description'] != null)
_PostHeadline(data: data).padding(horizontal: 16, bottom: 8), _PostHeadline(data: data).padding(horizontal: 16, bottom: 8),
_PostContentBody(data: data.body) _PostContentBody(data: data.body).padding(horizontal: 16, bottom: 6),
.padding(horizontal: 16, bottom: 6),
if (data.repostTo != null) if (data.repostTo != null)
_PostQuoteContent(child: data.repostTo!).padding( _PostQuoteContent(child: data.repostTo!).padding(
horizontal: 12, horizontal: 12,
@ -81,8 +152,7 @@ class PostItem extends StatelessWidget {
horizontal: 16, horizontal: 16,
vertical: 4, vertical: 4,
), ),
if (data.tags.isNotEmpty) if (data.tags.isNotEmpty) _PostTagsList(data: data).padding(horizontal: 16, bottom: 6),
_PostTagsList(data: data).padding(horizontal: 16, bottom: 6),
], ],
), ),
), ),
@ -116,6 +186,7 @@ class _PostBottomAction extends StatelessWidget {
final bool showComments; final bool showComments;
final bool showReactions; final bool showReactions;
final Function(SnPost data) onChanged; final Function(SnPost data) onChanged;
const _PostBottomAction({ const _PostBottomAction({
required this.data, required this.data,
required this.showComments, required this.showComments,
@ -130,9 +201,7 @@ class _PostBottomAction extends StatelessWidget {
); );
final String? mostTypicalReaction = data.metric.reactionList.isNotEmpty final String? mostTypicalReaction = data.metric.reactionList.isNotEmpty
? data.metric.reactionList.entries ? data.metric.reactionList.entries.reduce((a, b) => a.value > b.value ? a : b).key
.reduce((a, b) => a.value > b.value ? a : b)
.key
: null; : null;
return Row( return Row(
@ -145,8 +214,7 @@ class _PostBottomAction extends StatelessWidget {
InkWell( InkWell(
child: Row( child: Row(
children: [ children: [
if (mostTypicalReaction == null || if (mostTypicalReaction == null || kTemplateReactions[mostTypicalReaction] == null)
kTemplateReactions[mostTypicalReaction] == null)
Icon(Symbols.add_reaction, size: 20, color: iconColor) Icon(Symbols.add_reaction, size: 20, color: iconColor)
else else
Text( Text(
@ -158,8 +226,7 @@ class _PostBottomAction extends StatelessWidget {
), ),
), ),
const Gap(8), const Gap(8),
if (data.totalUpvote > 0 && if (data.totalUpvote > 0 && data.totalUpvote >= data.totalDownvote)
data.totalUpvote >= data.totalDownvote)
Text('postReactionUpvote').plural( Text('postReactionUpvote').plural(
data.totalUpvote, data.totalUpvote,
) )
@ -178,12 +245,8 @@ class _PostBottomAction extends StatelessWidget {
data: data, data: data,
onChanged: (value, attr, delta) { onChanged: (value, attr, delta) {
onChanged(data.copyWith( onChanged(data.copyWith(
totalUpvote: attr == 1 totalUpvote: attr == 1 ? data.totalUpvote + delta : data.totalUpvote,
? data.totalUpvote + delta totalDownvote: attr == 2 ? data.totalDownvote + delta : data.totalDownvote,
: data.totalUpvote,
totalDownvote: attr == 2
? data.totalDownvote + delta
: data.totalDownvote,
metric: data.metric.copyWith(reactionList: value), metric: data.metric.copyWith(reactionList: value),
)); ));
}, },
@ -229,6 +292,7 @@ class _PostBottomAction extends StatelessWidget {
class _PostHeadline extends StatelessWidget { class _PostHeadline extends StatelessWidget {
final SnPost data; final SnPost data;
const _PostHeadline({super.key, required this.data}); const _PostHeadline({super.key, required this.data});
@override @override
@ -256,6 +320,7 @@ class _PostContentHeader extends StatelessWidget {
final bool isCompact; final bool isCompact;
final bool showMenu; final bool showMenu;
final Function onDeleted; final Function onDeleted;
const _PostContentHeader({ const _PostContentHeader({
required this.data, required this.data,
this.isCompact = false, this.isCompact = false,
@ -438,6 +503,7 @@ class _PostContentHeader extends StatelessWidget {
class _PostContentBody extends StatelessWidget { class _PostContentBody extends StatelessWidget {
final dynamic data; final dynamic data;
const _PostContentBody({this.data}); const _PostContentBody({this.data});
@override @override
@ -449,6 +515,7 @@ class _PostContentBody extends StatelessWidget {
class _PostQuoteContent extends StatelessWidget { class _PostQuoteContent extends StatelessWidget {
final SnPost child; final SnPost child;
const _PostQuoteContent({super.key, required this.child}); const _PostQuoteContent({super.key, required this.child});
@override @override
@ -479,6 +546,7 @@ class _PostQuoteContent extends StatelessWidget {
class _PostTagsList extends StatelessWidget { class _PostTagsList extends StatelessWidget {
final SnPost data; final SnPost data;
const _PostTagsList({super.key, required this.data}); const _PostTagsList({super.key, required this.data});
@override @override
@ -505,6 +573,7 @@ class _PostTagsList extends StatelessWidget {
class _PostVisibilityHint extends StatelessWidget { class _PostVisibilityHint extends StatelessWidget {
final SnPost data; final SnPost data;
const _PostVisibilityHint({super.key, required this.data}); const _PostVisibilityHint({super.key, required this.data});
static const List<IconData> kVisibilityIcons = [ static const List<IconData> kVisibilityIcons = [
@ -529,6 +598,7 @@ class _PostVisibilityHint extends StatelessWidget {
class _PostTruncatedHint extends StatelessWidget { class _PostTruncatedHint extends StatelessWidget {
final SnPost data; final SnPost data;
const _PostTruncatedHint({super.key, required this.data}); const _PostTruncatedHint({super.key, required this.data});
static const int kHumanReadSpeed = 238; static const int kHumanReadSpeed = 238;
@ -544,13 +614,11 @@ class _PostTruncatedHint extends StatelessWidget {
const Gap(4), const Gap(4),
Text('postReadEstimate').tr(args: [ Text('postReadEstimate').tr(args: [
'${Duration( '${Duration(
seconds: (data.body['content_length'] as num).toDouble() * seconds: (data.body['content_length'] as num).toDouble() * 60 ~/ kHumanReadSpeed,
60 ~/
kHumanReadSpeed,
).inSeconds}s', ).inSeconds}s',
]), ]),
], ],
).padding(right: 12), ).padding(right: 8),
if (data.body['content_length'] != null) if (data.body['content_length'] != null)
Row( Row(
children: [ children: [

View File

@ -17,18 +17,26 @@ import 'package:surface/widgets/attachment/attachment_detail.dart';
import 'package:surface/widgets/dialog.dart'; import 'package:surface/widgets/dialog.dart';
class PostMediaPendingList extends StatelessWidget { class PostMediaPendingList extends StatelessWidget {
final PostWriteMedia? thumbnail;
final List<PostWriteMedia> attachments; final List<PostWriteMedia> attachments;
final bool isBusy; final bool isBusy;
final Future<void> Function(int idx, PostWriteMedia updatedMedia)? onUpdate; final Future<void> Function(int idx, PostWriteMedia updatedMedia)? onUpdate;
final Future<void> Function(int idx)? onRemove; final Future<void> Function(int idx)? onRemove;
final Future<void> Function(int idx)? onUpload;
final void Function(int? idx)? onPostSetThumbnail;
final void Function(int idx)? onInsertLink;
final void Function(bool state)? onUpdateBusy; final void Function(bool state)? onUpdateBusy;
const PostMediaPendingList({ const PostMediaPendingList({
super.key, super.key,
this.thumbnail,
required this.attachments, required this.attachments,
required this.isBusy, required this.isBusy,
this.onUpdate, this.onUpdate,
this.onRemove, this.onRemove,
this.onUpload,
this.onPostSetThumbnail,
this.onInsertLink,
this.onUpdateBusy, this.onUpdateBusy,
}); });
@ -50,10 +58,7 @@ class PostMediaPendingList extends StatelessWidget {
if (result == null) return; if (result == null) return;
final rawBytes = final rawBytes = (await result.uiImage.toByteData(format: ImageByteFormat.png))!.buffer.asUint8List();
(await result.uiImage.toByteData(format: ImageByteFormat.png))!
.buffer
.asUint8List();
if (onUpdate != null) { if (onUpdate != null) {
final updatedMedia = PostWriteMedia.fromBytes( final updatedMedia = PostWriteMedia.fromBytes(
@ -66,7 +71,7 @@ class PostMediaPendingList extends StatelessWidget {
} }
Future<void> _deleteAttachment(BuildContext context, int idx) async { Future<void> _deleteAttachment(BuildContext context, int idx) async {
final media = attachments[idx]; final media = idx == -1 ? thumbnail! : attachments[idx];
if (media.attachment == null) return; if (media.attachment == null) return;
try { try {
@ -82,10 +87,40 @@ class PostMediaPendingList extends StatelessWidget {
} }
} }
ContextMenu _buildContextMenu( ContextMenu _buildContextMenu(BuildContext context, int idx, PostWriteMedia media) {
BuildContext context, int idx, PostWriteMedia media) {
return ContextMenu( return ContextMenu(
entries: [ entries: [
if (media.attachment == null && onUpload != null)
MenuItem(
label: 'attachmentUpload'.tr(),
icon: Symbols.upload,
onSelected: () {
onUpload!(idx);
}),
if (media.attachment != null && onPostSetThumbnail != null && idx != -1)
MenuItem(
label: 'attachmentSetAsPostThumbnail'.tr(),
icon: Symbols.gallery_thumbnail,
onSelected: () {
onPostSetThumbnail!(idx);
},
)
else if (media.attachment != null && onPostSetThumbnail != null)
MenuItem(
label: 'attachmentUnsetAsPostThumbnail'.tr(),
icon: Symbols.cancel,
onSelected: () {
onPostSetThumbnail!(null);
},
),
if (media.attachment != null && onInsertLink != null)
MenuItem(
label: 'attachmentInsertLink'.tr(),
icon: Symbols.add_link,
onSelected: () {
onInsertLink!(idx);
},
),
if (media.type == PostWriteMediaType.image && media.attachment != null) if (media.type == PostWriteMediaType.image && media.attachment != null)
MenuItem( MenuItem(
label: 'preview'.tr(), label: 'preview'.tr(),
@ -135,51 +170,91 @@ class PostMediaPendingList extends StatelessWidget {
return Container( return Container(
constraints: const BoxConstraints(maxHeight: 120), constraints: const BoxConstraints(maxHeight: 120),
child: ListView.separated( child: Row(
scrollDirection: Axis.horizontal, children: [
padding: const EdgeInsets.symmetric(horizontal: 8), const Gap(8),
separatorBuilder: (context, index) => const Gap(8), if (thumbnail != null)
itemCount: attachments.length, ContextMenuRegion(
itemBuilder: (context, idx) { contextMenu: _buildContextMenu(context, -1, thumbnail!),
final media = attachments[idx]; child: Container(
return ContextMenuRegion( decoration: BoxDecoration(
contextMenu: _buildContextMenu(context, idx, media), border: Border.all(
child: Container( color: Theme.of(context).dividerColor,
decoration: BoxDecoration( width: 1,
border: Border.all( ),
color: Theme.of(context).dividerColor, borderRadius: BorderRadius.circular(8),
width: 1,
), ),
borderRadius: BorderRadius.circular(8), child: ClipRRect(
), borderRadius: const BorderRadius.all(Radius.circular(8)),
child: ClipRRect( child: AspectRatio(
borderRadius: const BorderRadius.all(Radius.circular(8)), aspectRatio: 1,
child: AspectRatio( child: switch (thumbnail!.type) {
aspectRatio: 1, PostWriteMediaType.image => LayoutBuilder(builder: (context, constraints) {
child: switch (media.type) { return Image(
PostWriteMediaType.image => image: thumbnail!.getImageProvider(
LayoutBuilder(builder: (context, constraints) { context,
return Image( width: (constraints.maxWidth * devicePixelRatio).round(),
image: media.getImageProvider( height: (constraints.maxHeight * devicePixelRatio).round(),
context, )!,
width: (constraints.maxWidth * devicePixelRatio) fit: BoxFit.cover,
.round(), );
height: (constraints.maxHeight * devicePixelRatio) }),
.round(), _ => Container(
)!, color: Theme.of(context).colorScheme.surface,
fit: BoxFit.cover, child: const Icon(Symbols.docs).center(),
); ),
}), },
_ => Container( ),
color: Theme.of(context).colorScheme.surface,
child: const Icon(Symbols.docs).center(),
),
},
), ),
), ),
), ),
); if (thumbnail != null) const VerticalDivider(width: 1).padding(horizontal: 8),
}, Expanded(
child: ListView.separated(
scrollDirection: Axis.horizontal,
padding: const EdgeInsets.only(right: 8),
separatorBuilder: (context, index) => const Gap(8),
itemCount: attachments.length,
itemBuilder: (context, idx) {
final media = attachments[idx];
return ContextMenuRegion(
contextMenu: _buildContextMenu(context, idx, media),
child: Container(
decoration: BoxDecoration(
border: Border.all(
color: Theme.of(context).dividerColor,
width: 1,
),
borderRadius: BorderRadius.circular(8),
),
child: ClipRRect(
borderRadius: const BorderRadius.all(Radius.circular(8)),
child: AspectRatio(
aspectRatio: 1,
child: switch (media.type) {
PostWriteMediaType.image => LayoutBuilder(builder: (context, constraints) {
return Image(
image: media.getImageProvider(
context,
width: (constraints.maxWidth * devicePixelRatio).round(),
height: (constraints.maxHeight * devicePixelRatio).round(),
)!,
fit: BoxFit.cover,
);
}),
_ => Container(
color: Theme.of(context).colorScheme.surface,
child: const Icon(Symbols.docs).center(),
),
},
),
),
),
);
},
),
),
],
), ),
); );
} }

View File

@ -94,8 +94,8 @@ class PostMetaEditor extends StatelessWidget {
onTapOutside: (_) => onTapOutside: (_) =>
FocusManager.instance.primaryFocus?.unfocus(), FocusManager.instance.primaryFocus?.unfocus(),
).padding(horizontal: 24), ).padding(horizontal: 24),
if (controller.mode == 'article') const Gap(4), if (controller.mode == 'articles') const Gap(4),
if (controller.mode == 'article') if (controller.mode == 'articles')
TextField( TextField(
controller: controller.descriptionController, controller: controller.descriptionController,
maxLines: null, maxLines: null,