✨ Article thumbnail
This commit is contained in:
parent
599dd4827b
commit
b583780cfc
@ -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"
|
||||||
}
|
}
|
||||||
|
@ -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 上的文章"
|
||||||
}
|
}
|
||||||
|
@ -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) {
|
||||||
|
if (idx == -1) {
|
||||||
|
thumbnail = item;
|
||||||
|
} else {
|
||||||
attachments[idx] = item;
|
attachments[idx] = item;
|
||||||
|
}
|
||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeAttachmentAt(int idx) {
|
void removeAttachmentAt(int idx) {
|
||||||
|
if (idx == -1) {
|
||||||
|
thumbnail = null;
|
||||||
|
} else {
|
||||||
attachments.removeAt(idx);
|
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;
|
||||||
|
@ -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();
|
||||||
|
|
||||||
|
@ -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);
|
||||||
},
|
},
|
||||||
|
@ -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';
|
||||||
|
},
|
||||||
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((_) {
|
||||||
|
@ -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) {
|
||||||
|
@ -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: [
|
||||||
|
@ -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,9 +170,49 @@ class PostMediaPendingList extends StatelessWidget {
|
|||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
constraints: const BoxConstraints(maxHeight: 120),
|
constraints: const BoxConstraints(maxHeight: 120),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
const Gap(8),
|
||||||
|
if (thumbnail != null)
|
||||||
|
ContextMenuRegion(
|
||||||
|
contextMenu: _buildContextMenu(context, -1, thumbnail!),
|
||||||
|
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 (thumbnail!.type) {
|
||||||
|
PostWriteMediaType.image => LayoutBuilder(builder: (context, constraints) {
|
||||||
|
return Image(
|
||||||
|
image: thumbnail!.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(),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (thumbnail != null) const VerticalDivider(width: 1).padding(horizontal: 8),
|
||||||
|
Expanded(
|
||||||
child: ListView.separated(
|
child: ListView.separated(
|
||||||
scrollDirection: Axis.horizontal,
|
scrollDirection: Axis.horizontal,
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 8),
|
padding: const EdgeInsets.only(right: 8),
|
||||||
separatorBuilder: (context, index) => const Gap(8),
|
separatorBuilder: (context, index) => const Gap(8),
|
||||||
itemCount: attachments.length,
|
itemCount: attachments.length,
|
||||||
itemBuilder: (context, idx) {
|
itemBuilder: (context, idx) {
|
||||||
@ -157,15 +232,12 @@ class PostMediaPendingList extends StatelessWidget {
|
|||||||
child: AspectRatio(
|
child: AspectRatio(
|
||||||
aspectRatio: 1,
|
aspectRatio: 1,
|
||||||
child: switch (media.type) {
|
child: switch (media.type) {
|
||||||
PostWriteMediaType.image =>
|
PostWriteMediaType.image => LayoutBuilder(builder: (context, constraints) {
|
||||||
LayoutBuilder(builder: (context, constraints) {
|
|
||||||
return Image(
|
return Image(
|
||||||
image: media.getImageProvider(
|
image: media.getImageProvider(
|
||||||
context,
|
context,
|
||||||
width: (constraints.maxWidth * devicePixelRatio)
|
width: (constraints.maxWidth * devicePixelRatio).round(),
|
||||||
.round(),
|
height: (constraints.maxHeight * devicePixelRatio).round(),
|
||||||
height: (constraints.maxHeight * devicePixelRatio)
|
|
||||||
.round(),
|
|
||||||
)!,
|
)!,
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
);
|
);
|
||||||
@ -181,6 +253,9 @@ class PostMediaPendingList extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user