feat: im settings

Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
This commit is contained in:
bggRGjQaUbCoE
2025-05-09 21:54:23 +08:00
parent a282baf5a2
commit 0f41d5b2f8
14 changed files with 865 additions and 9 deletions

View File

@@ -0,0 +1,58 @@
import 'package:PiliPlus/grpc/bilibili/app/im/v1.pb.dart';
import 'package:PiliPlus/grpc/im.dart';
import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/pages/common/common_list_controller.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
class WhisperBlockController extends CommonListController<
KeywordBlockingListReply, KeywordBlockingItem> {
@override
void onInit() {
super.onInit();
queryData();
}
RxInt count = 0.obs;
int? listLimit;
int? charLimit;
@override
List<KeywordBlockingItem>? getDataList(KeywordBlockingListReply response) {
count.value = response.items.length;
listLimit = response.listLimit;
charLimit = response.charLimit;
return response.items;
}
@override
Future<LoadingState<KeywordBlockingListReply>> customGetData() =>
ImGrpc.keywordBlockingList();
Future<void> onAdd(String keyword) async {
var res = await ImGrpc.keywordBlockingAdd(keyword);
if (res['status']) {
Get.back();
loadingState
..value.data!.add(KeywordBlockingItem(keyword: keyword))
..refresh();
count.value += 1;
SmartDialog.showToast('添加成功');
} else {
SmartDialog.showToast(res['msg']);
}
}
Future<void> onRemove(KeywordBlockingItem item) async {
var res = await ImGrpc.keywordBlockingDelete(item.keyword);
if (res['status']) {
loadingState
..value.data!.remove(item)
..refresh();
count.value -= 1;
SmartDialog.showToast('删除成功');
} else {
SmartDialog.showToast(res['msg']);
}
}
}

View File

@@ -0,0 +1,219 @@
import 'package:PiliPlus/common/widgets/dialog/dialog.dart';
import 'package:PiliPlus/common/widgets/loading_widget/loading_widget.dart';
import 'package:PiliPlus/grpc/bilibili/app/im/v1.pb.dart'
show KeywordBlockingItem;
import 'package:PiliPlus/http/loading_state.dart';
import 'package:PiliPlus/pages/search/widgets/search_text.dart';
import 'package:PiliPlus/pages/whisper_block/controller.dart';
import 'package:flutter/material.dart';
import 'package:flutter_svg/svg.dart';
import 'package:get/get.dart';
class WhisperBlockPage extends StatefulWidget {
const WhisperBlockPage({
super.key,
});
@override
State<WhisperBlockPage> createState() => _WhisperBlockPageState();
}
class _WhisperBlockPageState extends State<WhisperBlockPage> {
final _controller = Get.put(WhisperBlockController());
@override
Widget build(BuildContext context) {
final ThemeData theme = Theme.of(context);
return Scaffold(
resizeToAvoidBottomInset: false,
appBar: AppBar(title: const Text('消息屏蔽词')),
body: Obx(() => _buildBody(theme, _controller.loadingState.value)),
);
}
Widget _buildBody(
ThemeData theme, LoadingState<List<KeywordBlockingItem>?> loadingState) {
return switch (loadingState) {
Loading() => loadingWidget,
Success() => loadingState.response?.isNotEmpty == true
? Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.symmetric(horizontal: 12),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'点击屏蔽词即可删除',
style: TextStyle(
fontSize: 13,
color: theme.colorScheme.outline,
),
),
if (_controller.listLimit != null)
Obx(
() => Text(
'${_controller.count.value}/${_controller.listLimit}',
style: TextStyle(
fontSize: 13,
color: theme.colorScheme.outline,
),
),
),
],
),
),
Expanded(
child: Padding(
padding: const EdgeInsets.all(12),
child: SingleChildScrollView(
child: Wrap(
spacing: 12,
runSpacing: 12,
children: loadingState.response!
.map((e) => SearchText(
text: e.keyword,
onTap: (keyword) {
showConfirmDialog(
context: context,
title: '删除屏蔽词?',
content: '该屏蔽词将不再生效',
onConfirm: () {
_controller.onRemove(e);
},
);
},
))
.toList(),
),
),
),
),
Padding(
padding: EdgeInsets.only(
left: 25,
right: 25,
bottom: MediaQuery.paddingOf(context).bottom + 10,
),
child: FilledButton.tonal(
onPressed: _onAdd,
child: const Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [Icon(Icons.add, size: 22), Text('添加消息屏蔽词')],
),
),
),
],
)
: SizedBox.expand(
child: Column(
children: [
const Spacer(),
SvgPicture.asset("assets/images/error.svg", height: 156),
const SizedBox(height: 6),
const Text(
'还未添加屏蔽词',
style: TextStyle(fontSize: 15, fontWeight: FontWeight.bold),
),
const SizedBox(height: 6),
const Text('添加后,将不再接受包含屏蔽词的消息'),
const SizedBox(height: 6),
FilledButton.tonal(
onPressed: _onAdd,
style: FilledButton.styleFrom(
visualDensity: VisualDensity.compact),
child: const Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(Icons.add, size: 22),
Text('添加'),
],
),
),
const Spacer(flex: 2),
],
),
),
Error() => scrollErrorWidget(
errMsg: loadingState.errMsg,
onReload: _controller.onReload,
),
};
}
void _onAdd() {
String keyword = '';
showModalBottomSheet(
context: context,
enableDrag: false,
useSafeArea: true,
isScrollControlled: true,
builder: (context) {
final theme = Theme.of(context);
return Padding(
padding: const EdgeInsets.symmetric(horizontal: 14, vertical: 12) +
EdgeInsets.only(
bottom: MediaQuery.paddingOf(context).bottom +
MediaQuery.viewInsetsOf(context).bottom),
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
const Text(
'添加消息屏蔽词',
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.bold,
),
),
GestureDetector(
onTap: Get.back,
child: Icon(
Icons.clear,
color: theme.colorScheme.onSurfaceVariant,
),
),
],
),
const SizedBox(height: 12),
TextFormField(
autofocus: true,
maxLength: _controller.charLimit,
decoration: InputDecoration(
isDense: true,
hintText: '请输入',
hintStyle: const TextStyle(fontSize: 14),
contentPadding:
const EdgeInsets.symmetric(horizontal: 14, vertical: 6),
border: const OutlineInputBorder(
borderSide: BorderSide.none,
borderRadius: BorderRadius.all(Radius.circular(25)),
),
filled: true,
fillColor: theme.colorScheme.onInverseSurface,
),
onChanged: (value) => keyword = value,
),
const SizedBox(height: 12),
FilledButton.tonal(
onPressed: () {
if (keyword.isNotEmpty) {
_controller.onAdd(keyword);
}
},
child: const Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [Icon(Icons.add, size: 22), Text('添加消息屏蔽词')],
),
),
],
),
);
},
);
}
}