Able to edit visibility

This commit is contained in:
LittleSheep 2024-07-30 20:49:01 +08:00
parent bb77b74356
commit 19751617cb
10 changed files with 406 additions and 51 deletions

View File

@ -3,11 +3,11 @@ import 'dart:convert';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:get/get_rx/get_rx.dart';
import 'package:solian/models/post.dart'; import 'package:solian/models/post.dart';
import 'package:solian/models/realm.dart'; import 'package:solian/models/realm.dart';
import 'package:solian/widgets/attachments/attachment_publish.dart'; import 'package:solian/widgets/attachments/attachment_editor.dart';
import 'package:solian/widgets/posts/editor/post_editor_overview.dart'; import 'package:solian/widgets/posts/editor/post_editor_overview.dart';
import 'package:solian/widgets/posts/editor/post_editor_visibility.dart';
import 'package:textfield_tags/textfield_tags.dart'; import 'package:textfield_tags/textfield_tags.dart';
import 'package:shared_preferences/shared_preferences.dart'; import 'package:shared_preferences/shared_preferences.dart';
@ -28,6 +28,10 @@ class PostEditorController extends GetxController {
Rx<Realm?> realmZone = Rx(null); Rx<Realm?> realmZone = Rx(null);
RxList<int> attachments = RxList<int>.empty(growable: true); RxList<int> attachments = RxList<int>.empty(growable: true);
RxList<int> visibleUsers = RxList.empty(growable: true);
RxList<int> invisibleUsers = RxList.empty(growable: true);
RxInt visibility = 0.obs;
RxBool isDraft = false.obs; RxBool isDraft = false.obs;
RxBool isRestoreFromLocal = false.obs; RxBool isRestoreFromLocal = false.obs;
@ -66,11 +70,20 @@ class PostEditorController extends GetxController {
); );
} }
Future<void> editVisibility(BuildContext context) {
return showDialog(
context: context,
builder: (context) => PostEditorVisibilityDialog(
controller: this,
),
);
}
Future<void> editAttachment(BuildContext context) { Future<void> editAttachment(BuildContext context) {
return showModalBottomSheet( return showModalBottomSheet(
context: context, context: context,
isScrollControlled: true, isScrollControlled: true,
builder: (context) => AttachmentPublishPopup( builder: (context) => AttachmentEditorPopup(
usage: 'i.attachment', usage: 'i.attachment',
current: attachments, current: attachments,
onUpdate: (value) { onUpdate: (value) {
@ -116,6 +129,9 @@ class PostEditorController extends GetxController {
contentController.clear(); contentController.clear();
tagController.clearTags(); tagController.clearTags();
attachments.clear(); attachments.clear();
visibleUsers.clear();
invisibleUsers.clear();
visibility.value = 0;
isDraft.value = false; isDraft.value = false;
isRestoreFromLocal.value = false; isRestoreFromLocal.value = false;
lastSaveTime.value = null; lastSaveTime.value = null;
@ -145,7 +161,7 @@ class PostEditorController extends GetxController {
} }
String get typeEndpoint { String get typeEndpoint {
switch(mode.value) { switch (mode.value) {
case 0: case 0:
return 'stories'; return 'stories';
case 1: case 1:
@ -156,7 +172,7 @@ class PostEditorController extends GetxController {
} }
String get type { String get type {
switch(mode.value) { switch (mode.value) {
case 0: case 0:
return 'story'; return 'story';
case 1: case 1:
@ -193,6 +209,9 @@ class PostEditorController extends GetxController {
'tags': tagController.getTags?.map((x) => {'alias': x}).toList() ?? 'tags': tagController.getTags?.map((x) => {'alias': x}).toList() ??
List.empty(), List.empty(),
'attachments': attachments, 'attachments': attachments,
'visible_users': visibleUsers,
'invisible_users': invisibleUsers,
'visibility': visibility.value,
'is_draft': isDraft.value, 'is_draft': isDraft.value,
if (replyTo.value != null) 'reply_to': replyTo.value!.id, if (replyTo.value != null) 'reply_to': replyTo.value!.id,
if (repostTo.value != null) 'repost_to': repostTo.value!.id, if (repostTo.value != null) 'repost_to': repostTo.value!.id,
@ -207,7 +226,14 @@ class PostEditorController extends GetxController {
contentController.text = value['content'] ?? ''; contentController.text = value['content'] ?? '';
attachments.value = value['attachments'].cast<int>() ?? List.empty(); attachments.value = value['attachments'].cast<int>() ?? List.empty();
attachments.refresh(); attachments.refresh();
visibility.value = value['visibility'];
isDraft.value = value['is_draft']; isDraft.value = value['is_draft'];
if (value['visible_users'] != null) {
visibleUsers.value = value['visible_users'].cast<int>();
}
if (value['invisible_users'] != null) {
invisibleUsers.value = value['invisible_users'].cast<int>();
}
if (value['reply_to'] != null) { if (value['reply_to'] != null) {
replyTo.value = Post.fromJson(value['reply_to']); replyTo.value = Post.fromJson(value['reply_to']);
} }

View File

@ -46,12 +46,4 @@ class Relationship {
'related': related.toJson(), 'related': related.toJson(),
'status': status, 'status': status,
}; };
Account getOtherside(int selfId) {
if (accountId != selfId) {
return account;
} else {
return related;
}
}
} }

View File

@ -148,7 +148,7 @@ class _PostPublishScreenState extends State<PostPublishScreen> {
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),
contentPadding: const EdgeInsets.only( contentPadding: const EdgeInsets.only(
left: 16, left: 17,
right: 8, right: 8,
top: 0, top: 0,
bottom: 0, bottom: 0,
@ -225,15 +225,18 @@ class _PostPublishScreenState extends State<PostPublishScreen> {
actions: notifyBannerActions, actions: notifyBannerActions,
), ),
Container( Container(
padding: padding: const EdgeInsets.symmetric(
const EdgeInsets.symmetric(horizontal: 16, vertical: 8), horizontal: 16,
vertical: 8,
),
child: TextField( child: TextField(
maxLines: null, maxLines: null,
autofocus: true, autofocus: true,
autocorrect: true, autocorrect: true,
keyboardType: TextInputType.multiline, keyboardType: TextInputType.multiline,
controller: _editorController.contentController, controller: _editorController.contentController,
decoration: InputDecoration.collapsed( decoration: InputDecoration(
border: InputBorder.none,
hintText: 'postContentPlaceholder'.tr, hintText: 'postContentPlaceholder'.tr,
), ),
onTapOutside: (_) => onTapOutside: (_) =>
@ -299,7 +302,10 @@ class _PostPublishScreenState extends State<PostPublishScreen> {
], ],
), ),
) )
.animate(target: doShow ? 1 : 0) .animate(
key: const Key('post-editor-hint-animation'),
target: doShow ? 1 : 0,
)
.fade(curve: Curves.easeInOut, duration: 300.ms); .fade(curve: Curves.easeInOut, duration: 300.ms);
}), }),
if (_editorController.mode.value == 0) if (_editorController.mode.value == 0)
@ -325,22 +331,39 @@ class _PostPublishScreenState extends State<PostPublishScreen> {
child: ListView( child: ListView(
scrollDirection: Axis.horizontal, scrollDirection: Axis.horizontal,
children: [ children: [
Obx( Obx(() {
() => IconButton( final isDraft = _editorController.isDraft.value;
icon: _editorController.isDraft.value return IconButton(
? const Icon(Icons.drive_file_rename_outline) icon: const Icon(
: const Icon(Icons.public), Icons.drive_file_rename_outline,
color: _editorController.isDraft.value color: Colors.grey,
? Colors.grey.shade600 )
: Colors.green.shade700, .animate(
target: isDraft ? 0 : 1,
)
.fadeOut(duration: 150.ms)
.swap(
duration: 150.ms,
builder: (_, __) => const Icon(
Icons.public,
color: Colors.green,
).animate().fadeIn(duration: 150.ms),
),
onPressed: () { onPressed: () {
_editorController.toggleDraftMode(); _editorController.toggleDraftMode();
}, },
), );
}),
IconButton(
icon: const Icon(Icons.disabled_visible),
color: Theme.of(context).colorScheme.primary,
onPressed: () {
_editorController.editVisibility(context);
},
), ),
IconButton( IconButton(
icon: Obx( icon: Obx(() {
() => badges.Badge( return badges.Badge(
badgeContent: Text( badgeContent: Text(
_editorController.attachments.length.toString(), _editorController.attachments.length.toString(),
style: const TextStyle(color: Colors.white), style: const TextStyle(color: Colors.white),
@ -351,12 +374,13 @@ class _PostPublishScreenState extends State<PostPublishScreen> {
top: -12, top: -12,
end: -8, end: -8,
), ),
child: const Icon(Icons.camera_alt), child: const Icon(Icons.file_present_rounded),
), );
), }),
color: Theme.of(context).colorScheme.primary, color: Theme.of(context).colorScheme.primary,
onPressed: () => onPressed: () {
_editorController.editAttachment(context), _editorController.editAttachment(context);
},
), ),
], ],
).paddingSymmetric(horizontal: 6, vertical: 8), ).paddingSymmetric(horizontal: 6, vertical: 8),

View File

@ -98,6 +98,14 @@ const i18nEnglish = {
'unpinPost': 'Unpin this post', 'unpinPost': 'Unpin this post',
'postRestoreFromLocal': 'Restore from local', 'postRestoreFromLocal': 'Restore from local',
'postAutoSaveAt': 'Auto saved at @date', 'postAutoSaveAt': 'Auto saved at @date',
'postVisibility': 'Visibility',
'postVisibilityAll': 'Everyone',
'postVisibilityFriends': 'Friends',
'postVisibilitySelected': 'Selected visible',
'postVisibilityFiltered': 'Selected invisible',
'postVisibilityNone': 'Only me',
'postVisibleUsers': 'Visible users',
'postInvisibleUsers': 'Invisible users',
'postOverview': 'Overview', 'postOverview': 'Overview',
'postPinned': 'Pinned', 'postPinned': 'Pinned',
'postListNews': 'News', 'postListNews': 'News',

View File

@ -92,6 +92,14 @@ const i18nSimplifiedChinese = {
'unpinPost': '取消置顶本帖', 'unpinPost': '取消置顶本帖',
'postRestoreFromLocal': '内容从本地暂存回复', 'postRestoreFromLocal': '内容从本地暂存回复',
'postAutoSaveAt': '已自动保存于 @date', 'postAutoSaveAt': '已自动保存于 @date',
'postVisibility': '帖子可见性',
'postVisibilityAll': '所有人可见',
'postVisibilityFriends': '仅好友可见',
'postVisibilitySelected': '选中者可见',
'postVisibilityFiltered': '选中者不可见',
'postVisibilityNone': '仅自己可见',
'postVisibleUsers': '可见帖子者',
'postInvisibleUsers': '隐藏帖子者',
'postOverview': '帖子概览', 'postOverview': '帖子概览',
'postPinned': '已置顶', 'postPinned': '已置顶',
'postEditorModeStory': '发个帖子', 'postEditorModeStory': '发个帖子',

View File

@ -0,0 +1,176 @@
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:solian/models/account.dart';
import 'package:solian/models/relations.dart';
import 'package:solian/providers/auth.dart';
import 'package:solian/providers/relation.dart';
import 'package:solian/services.dart';
import 'package:solian/widgets/account/account_avatar.dart';
class AccountSelector extends StatefulWidget {
final String title;
final Widget? Function(Account item)? trailingBuilder;
final List<int>? initialSelection;
final Function(List<Account>)? onMultipleSelect;
const AccountSelector({
super.key,
required this.title,
this.trailingBuilder,
this.initialSelection,
this.onMultipleSelect,
});
@override
State<AccountSelector> createState() => _AccountSelectorState();
}
class _AccountSelectorState extends State<AccountSelector> {
final TextEditingController _probeController = TextEditingController();
final List<Account> _relativeUsers = List.empty(growable: true);
final List<Account> _pendingUsers = List.empty(growable: true);
final List<Account> _selectedUsers = List.empty(growable: true);
int _accountId = 0;
_revertSelectedUsers() async {
if (widget.initialSelection?.isEmpty ?? true) return;
final client = ServiceFinder.configureClient('auth');
final idQuery = widget.initialSelection!.join(',');
final resp = await client.get('/users?id=$idQuery');
setState(() {
_selectedUsers.addAll(
resp.body
.map((e) => Account.fromJson(e))
.toList()
.cast<Account>(),
);
});
}
_getFriends() async {
final AuthProvider auth = Get.find();
_accountId = auth.userProfile.value!['id'];
final RelationshipProvider provider = Get.find();
final resp = await provider.listRelationWithStatus(1);
setState(() {
_relativeUsers.addAll(
resp.body
.map((e) => Relationship.fromJson(e).related)
.toList()
.cast<Account>(),
);
});
}
_searchAccount() async {
final AuthProvider auth = Get.find();
_accountId = auth.userProfile.value!['id'];
if (_probeController.text.isEmpty) return;
final client = auth.configureClient('auth');
final resp = await client.get(
'/users/search?probe=${_probeController.text}',
);
setState(() {
_pendingUsers.clear();
_pendingUsers.addAll(
resp.body.map((e) => Account.fromJson(e)).toList().cast<Account>(),
);
});
}
bool _checkSelected(Account item) {
return _selectedUsers.any((x) => x.id == item.id);
}
@override
void initState() {
super.initState();
_getFriends();
_revertSelectedUsers();
}
@override
void dispose() {
super.dispose();
_probeController.dispose();
}
@override
Widget build(BuildContext context) {
return SizedBox(
height: MediaQuery.of(context).size.height * 0.85,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
widget.title,
style: Theme.of(context).textTheme.headlineSmall,
).paddingOnly(left: 24, right: 24, top: 32, bottom: 16),
Container(
color: Theme.of(context).colorScheme.secondaryContainer,
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 16),
child: TextField(
controller: _probeController,
decoration: InputDecoration(
isCollapsed: true,
border: InputBorder.none,
hintText: 'search'.tr,
),
onSubmitted: (_) {
_searchAccount();
},
),
),
Expanded(
child: ListView.builder(
itemCount: _pendingUsers.isEmpty
? _relativeUsers.length
: _pendingUsers.length,
itemBuilder: (context, index) {
var element = _pendingUsers.isEmpty
? _relativeUsers[index]
: _pendingUsers[index];
return ListTile(
title: Text(element.nick),
subtitle: Text(element.name),
leading: AccountAvatar(content: element.avatar),
trailing: widget.trailingBuilder != null
? widget.trailingBuilder!(element)
: _checkSelected(element)
? const Icon(Icons.check)
: null,
onTap: element.id == _accountId
? null
: () {
if (widget.onMultipleSelect == null) {
Navigator.pop(context, element);
return;
}
setState(() {
final idx = _selectedUsers.indexWhere((x) => x.id == element.id);
if (idx != -1) {
_selectedUsers.removeAt(idx);
} else {
_selectedUsers.add(element);
}
});
widget.onMultipleSelect!(_selectedUsers);
},
);
},
),
),
],
),
);
}
}

View File

@ -2,7 +2,6 @@ import 'package:flutter/material.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:solian/models/account.dart'; import 'package:solian/models/account.dart';
import 'package:solian/models/relations.dart'; import 'package:solian/models/relations.dart';
import 'package:solian/providers/auth.dart';
import 'package:solian/providers/relation.dart'; import 'package:solian/providers/relation.dart';
import 'package:solian/widgets/account/account_avatar.dart'; import 'package:solian/widgets/account/account_avatar.dart';
@ -10,21 +9,17 @@ class RelativeSelector extends StatefulWidget {
final String title; final String title;
final Widget? Function(Account item)? trailingBuilder; final Widget? Function(Account item)? trailingBuilder;
const RelativeSelector({super.key, required this.title, this.trailingBuilder}); const RelativeSelector(
{super.key, required this.title, this.trailingBuilder});
@override @override
State<RelativeSelector> createState() => _RelativeSelectorState(); State<RelativeSelector> createState() => _RelativeSelectorState();
} }
class _RelativeSelectorState extends State<RelativeSelector> { class _RelativeSelectorState extends State<RelativeSelector> {
int _accountId = 0;
final List<Relationship> _friends = List.empty(growable: true); final List<Relationship> _friends = List.empty(growable: true);
getFriends() async { _getFriends() async {
final AuthProvider auth = Get.find();
_accountId = auth.userProfile.value!['id'];
final RelationshipProvider provider = Get.find(); final RelationshipProvider provider = Get.find();
final resp = await provider.listRelationWithStatus(1); final resp = await provider.listRelationWithStatus(1);
@ -39,8 +34,7 @@ class _RelativeSelectorState extends State<RelativeSelector> {
@override @override
void initState() { void initState() {
super.initState(); super.initState();
_getFriends();
getFriends();
} }
@override @override
@ -58,7 +52,7 @@ class _RelativeSelectorState extends State<RelativeSelector> {
child: ListView.builder( child: ListView.builder(
itemCount: _friends.length, itemCount: _friends.length,
itemBuilder: (context, index) { itemBuilder: (context, index) {
var element = _friends[index].getOtherside(_accountId); var element = _friends[index].related;
return ListTile( return ListTile(
title: Text(element.nick), title: Text(element.nick),
subtitle: Text(element.name), subtitle: Text(element.name),

View File

@ -18,12 +18,12 @@ import 'package:solian/providers/auth.dart';
import 'package:solian/providers/content/attachment.dart'; import 'package:solian/providers/content/attachment.dart';
import 'package:solian/widgets/attachments/attachment_item.dart'; import 'package:solian/widgets/attachments/attachment_item.dart';
class AttachmentPublishPopup extends StatefulWidget { class AttachmentEditorPopup extends StatefulWidget {
final String usage; final String usage;
final List<int> current; final List<int> current;
final void Function(List<int> data) onUpdate; final void Function(List<int> data) onUpdate;
const AttachmentPublishPopup({ const AttachmentEditorPopup({
super.key, super.key,
required this.usage, required this.usage,
required this.current, required this.current,
@ -31,10 +31,10 @@ class AttachmentPublishPopup extends StatefulWidget {
}); });
@override @override
State<AttachmentPublishPopup> createState() => _AttachmentPublishPopupState(); State<AttachmentEditorPopup> createState() => _AttachmentEditorPopupState();
} }
class _AttachmentPublishPopupState extends State<AttachmentPublishPopup> { class _AttachmentEditorPopupState extends State<AttachmentEditorPopup> {
final _imagePicker = ImagePicker(); final _imagePicker = ImagePicker();
bool _isBusy = false; bool _isBusy = false;

View File

@ -6,7 +6,7 @@ import 'package:solian/models/account.dart';
import 'package:solian/models/channel.dart'; import 'package:solian/models/channel.dart';
import 'package:solian/models/event.dart'; import 'package:solian/models/event.dart';
import 'package:solian/providers/auth.dart'; import 'package:solian/providers/auth.dart';
import 'package:solian/widgets/attachments/attachment_publish.dart'; import 'package:solian/widgets/attachments/attachment_editor.dart';
import 'package:solian/widgets/chat/chat_event.dart'; import 'package:solian/widgets/chat/chat_event.dart';
import 'package:uuid/uuid.dart'; import 'package:uuid/uuid.dart';
@ -47,7 +47,7 @@ class _ChatMessageInputState extends State<ChatMessageInput> {
showModalBottomSheet( showModalBottomSheet(
context: context, context: context,
isScrollControlled: true, isScrollControlled: true,
builder: (context) => AttachmentPublishPopup( builder: (context) => AttachmentEditorPopup(
usage: 'm.attachment', usage: 'm.attachment',
current: _attachments, current: _attachments,
onUpdate: (value) => _attachments = value, onUpdate: (value) => _attachments = value,

View File

@ -0,0 +1,127 @@
import 'package:dropdown_button2/dropdown_button2.dart';
import 'package:flutter/material.dart';
import 'package:get/get.dart';
import 'package:solian/controllers/post_editor_controller.dart';
import 'package:solian/widgets/account/account_select.dart';
class PostEditorVisibilityDialog extends StatelessWidget {
final PostEditorController controller;
const PostEditorVisibilityDialog({super.key, required this.controller});
static List<(int, String)> visibilityLevels = [
(0, 'postVisibilityAll'),
(1, 'postVisibilityFriends'),
(2, 'postVisibilitySelected'),
(3, 'postVisibilityFiltered'),
(4, 'postVisibilityNone'),
];
@override
Widget build(BuildContext context) {
return AlertDialog(
title: Text('postVisibility'.tr),
content: Column(
mainAxisSize: MainAxisSize.min,
children: [
Obx(() {
return DropdownButtonFormField2<int>(
isExpanded: true,
decoration: const InputDecoration(
border: OutlineInputBorder(
borderRadius: BorderRadius.all(Radius.circular(8)),
),
),
items: visibilityLevels
.map(
(entry) => DropdownMenuItem<int>(
value: entry.$1,
child: Text(
entry.$2.tr,
style: const TextStyle(fontSize: 14),
),
),
)
.toList(),
value: controller.visibility.value,
onChanged: (int? value) {
if (value != null) {
controller.visibility.value = value;
}
},
buttonStyleData: const ButtonStyleData(height: 20),
menuItemStyleData: const MenuItemStyleData(height: 40),
);
}),
Obx(() {
if (controller.visibility.value != 2 &&
controller.visibility.value != 3) {
return const SizedBox(height: 8);
}
return const SizedBox();
}),
Obx(() {
if (controller.visibility.value == 2) {
return ListTile(
shape: const RoundedRectangleBorder(
borderRadius: BorderRadius.all(Radius.circular(8)),
),
contentPadding: const EdgeInsets.only(left: 16, right: 13),
trailing: const Icon(Icons.chevron_right),
title: Text('postVisibleUsers'.tr),
onTap: () {
showModalBottomSheet(
context: context,
builder: (context) => AccountSelector(
title: 'postVisibleUsers'.tr,
initialSelection: controller.visibleUsers,
onMultipleSelect: (value) {
controller.visibleUsers.value =
value.map((e) => e.id).toList();
controller.visibleUsers.refresh();
},
),
);
},
);
}
return const SizedBox();
}),
Obx(() {
if (controller.visibility.value == 3) {
return ListTile(
shape: const RoundedRectangleBorder(
borderRadius: BorderRadius.all(Radius.circular(8)),
),
contentPadding: const EdgeInsets.only(left: 16, right: 13),
trailing: const Icon(Icons.chevron_right),
title: Text('postInvisibleUsers'.tr),
onTap: () {
showModalBottomSheet(
context: context,
builder: (context) => AccountSelector(
title: 'postInvisibleUsers'.tr,
initialSelection: controller.invisibleUsers,
onMultipleSelect: (value) {
controller.invisibleUsers.value =
value.map((e) => e.id).toList();
controller.invisibleUsers.refresh();
},
),
);
},
);
}
return const SizedBox();
}),
],
),
actions: [
TextButton(
onPressed: () => Navigator.pop(context),
child: Text('confirm'.tr),
),
],
);
}
}