feat: 视频搜索条件筛选v1

This commit is contained in:
guozhigq
2023-08-22 09:49:23 +08:00
parent 5812b5cff1
commit 8703d9f576
6 changed files with 186 additions and 24 deletions

View File

@ -46,14 +46,19 @@ class SearchHttp {
required SearchType searchType, required SearchType searchType,
required String keyword, required String keyword,
required page, required page,
String? order,
}) async { }) async {
var res = await Request().get(Api.searchByType, data: { Map<String, dynamic> reqData = {
'search_type': searchType.type, 'search_type': searchType.type,
'keyword': keyword, 'keyword': keyword,
// 'order_sort': 0, // 'order_sort': 0,
// 'user_type': 0, // 'user_type': 0,
'page': page 'page': page,
}); };
if (order != null && order != '') {
reqData['order'] = order;
}
var res = await Request().get(Api.searchByType, data: reqData);
if (res.data['code'] == 0 && res.data['data']['numPages'] > 0) { if (res.data['code'] == 0 && res.data['data']['numPages'] > 0) {
Object data; Object data;
switch (searchType) { switch (searchType) {

View File

@ -27,3 +27,20 @@ extension SearchTypeExtension on SearchType {
['video', 'media_bangumi', 'live_room', 'bili_user'][index]; ['video', 'media_bangumi', 'live_room', 'bili_user'][index];
String get label => ['视频', '番剧', '直播间', '用户'][index]; String get label => ['视频', '番剧', '直播间', '用户'][index];
} }
// 搜索类型为视频、专栏及相簿时
enum ArchiveFilterType {
totalrank,
click,
pubdate,
dm,
stow,
scores,
// 专栏
// attention,
}
extension ArchiveFilterTypeExtension on ArchiveFilterType {
String get description =>
['默认排序', '播放多', '新发布', '弹幕多', '收藏多', '评论多', '最多喜欢'][index];
}

View File

@ -45,8 +45,9 @@ class _SearchPageState extends State<SearchPage> with RouteAware {
return OpenContainer( return OpenContainer(
closedElevation: 0, closedElevation: 0,
openElevation: 0, openElevation: 0,
onClosed: (_) { onClosed: (_) async {
// 在 openBuilder 关闭时触发的回调函数 // 在 openBuilder 关闭时触发的回调函数
await Future.delayed(const Duration(milliseconds: 500));
_searchController.onClear(); _searchController.onClear();
}, },
openColor: Theme.of(context).colorScheme.background, openColor: Theme.of(context).colorScheme.background,

View File

@ -12,17 +12,22 @@ class SearchPanelController extends GetxController {
SearchType? searchType; SearchType? searchType;
RxInt page = 1.obs; RxInt page = 1.obs;
RxList resultList = [].obs; RxList resultList = [].obs;
RxString order = ''.obs;
Future onSearch({type = 'init'}) async { Future onSearch({type = 'init'}) async {
var result = await SearchHttp.searchByType( var result = await SearchHttp.searchByType(
searchType: searchType!, keyword: keyword!, page: page.value); searchType: searchType!,
keyword: keyword!,
page: page.value,
order: searchType!.type != 'video' ? '' : order.value,
);
if (result['status']) { if (result['status']) {
if (type == 'init' || type == 'onLoad') { if (type == 'onRefresh') {
page.value++;
resultList.addAll(result['data'].list);
} else if (type == 'onRefresh') {
resultList.value = result['data'].list; resultList.value = result['data'].list;
} else {
resultList.addAll(result['data'].list);
} }
page.value++;
onPushDetail(keyword, resultList); onPushDetail(keyword, resultList);
} }
return result; return result;
@ -30,7 +35,7 @@ class SearchPanelController extends GetxController {
Future onRefresh() async { Future onRefresh() async {
page.value = 1; page.value = 1;
onSearch(type: 'onRefresh'); await onSearch(type: 'onRefresh');
} }
// 返回顶部并刷新 // 返回顶部并刷新

View File

@ -28,7 +28,6 @@ class _SearchPanelState extends State<SearchPanel>
with AutomaticKeepAliveClientMixin { with AutomaticKeepAliveClientMixin {
late SearchPanelController _searchPanelController; late SearchPanelController _searchPanelController;
bool _isLoadingMore = false;
late Future _futureBuilderFuture; late Future _futureBuilderFuture;
late ScrollController scrollController; late ScrollController scrollController;
@ -76,12 +75,15 @@ class _SearchPanelState extends State<SearchPanel>
if (snapshot.connectionState == ConnectionState.done) { if (snapshot.connectionState == ConnectionState.done) {
Map data = snapshot.data; Map data = snapshot.data;
var ctr = _searchPanelController; var ctr = _searchPanelController;
List list = ctr.resultList; RxList list = ctr.resultList;
if (data['status']) { if (data['status']) {
return Obx(() { return Obx(() {
switch (widget.searchType) { switch (widget.searchType) {
case SearchType.video: case SearchType.video:
return searchVideoPanel(context, ctr, list); return SearchVideoPanel(
ctr: _searchPanelController,
list: list.value,
);
case SearchType.media_bangumi: case SearchType.media_bangumi:
return searchMbangumiPanel(context, ctr, list); return searchMbangumiPanel(context, ctr, list);
case SearchType.bili_user: case SearchType.bili_user:

View File

@ -1,15 +1,147 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
import 'package:get/get.dart';
import 'package:pilipala/common/widgets/video_card_h.dart'; import 'package:pilipala/common/widgets/video_card_h.dart';
import 'package:pilipala/models/common/search_type.dart';
import 'package:pilipala/pages/searchPanel/index.dart';
Widget searchVideoPanel(BuildContext context, ctr, list) { class SearchVideoPanel extends StatelessWidget {
return ListView.builder( SearchVideoPanel({
this.ctr,
this.list,
Key? key,
}) : super(key: key);
final SearchPanelController? ctr;
final List? list;
final VideoPanelController controller = Get.put(VideoPanelController());
@override
Widget build(BuildContext context) {
return Stack(
alignment: Alignment.topCenter,
children: [
Padding(
padding: const EdgeInsets.only(top: 34),
child: ListView.builder(
controller: ctr!.scrollController, controller: ctr!.scrollController,
addAutomaticKeepAlives: false, addAutomaticKeepAlives: false,
addRepaintBoundaries: false, addRepaintBoundaries: false,
itemCount: list!.length, itemCount: list!.length,
itemBuilder: (context, index) { itemBuilder: (context, index) {
var i = list![index]; var i = list![index];
return VideoCardH(videoItem: i); return Padding(
padding: index == 0
? const EdgeInsets.only(top: 2)
: EdgeInsets.zero,
child: VideoCardH(videoItem: i),
);
}, },
),
),
// 分类筛选
Container(
width: double.infinity,
height: 34,
padding: const EdgeInsets.only(left: 8, top: 0, right: 12),
// decoration: BoxDecoration(
// border: Border(
// bottom: BorderSide(
// color: Theme.of(context).colorScheme.primary.withOpacity(0.1),
// ),
// ),
// ),
child: SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Obx(
() => Wrap(
// spacing: ,
children: [
for (var i in controller.filterList) ...[
CustomFilterChip(
label: i['label'],
type: i['type'],
selectedType: controller.selectedType.value,
callFn: (bool selected) async {
controller.selectedType.value = i['type'];
ctr!.order.value = i['type'].toString().split('.').last;
SmartDialog.showLoading(msg: 'loooad');
await ctr!.onRefresh();
SmartDialog.dismiss();
},
),
]
],
),
),
),
), // 放置在ListView.builder()上方的组件
],
); );
} }
}
class CustomFilterChip extends StatelessWidget {
const CustomFilterChip({
this.label,
this.type,
this.selectedType,
this.callFn,
Key? key,
}) : super(key: key);
final String? label;
final ArchiveFilterType? type;
final ArchiveFilterType? selectedType;
final Function? callFn;
@override
Widget build(BuildContext context) {
return SizedBox(
height: 32,
child: FilterChip(
padding: const EdgeInsets.only(left: 11, right: 11),
labelPadding: EdgeInsets.zero,
label: Text(
label!,
style: const TextStyle(fontSize: 13),
),
labelStyle: TextStyle(
color: type == selectedType
? Theme.of(context).colorScheme.primary
: Theme.of(context).colorScheme.outline),
selected: type == selectedType,
showCheckmark: false,
shape: ContinuousRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
selectedColor: Colors.transparent,
// backgroundColor:
// Theme.of(context).colorScheme.surfaceVariant.withOpacity(0.5),
backgroundColor: Colors.transparent,
side: BorderSide.none,
onSelected: (bool selected) => callFn!(selected),
),
);
}
}
class VideoPanelController extends GetxController {
RxList<Map> filterList = [{}].obs;
Rx<ArchiveFilterType> selectedType = ArchiveFilterType.values.first.obs;
@override
void onInit() {
List<Map<String, dynamic>> list = ArchiveFilterType.values
.map((type) => {
'label': type.description,
'type': type,
})
.toList();
filterList.value = list;
super.onInit();
}
onSelect() {}
}