Files
PiliPlus/lib/pages/whisper_block/view.dart
bggRGjQaUbCoE 0f41d5b2f8 feat: im settings
Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
2025-05-09 22:17:31 +08:00

220 lines
8.0 KiB
Dart

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('添加消息屏蔽词')],
),
),
],
),
);
},
);
}
}