import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; import 'package:get/get.dart'; import 'package:hive/hive.dart'; import 'package:pilipala/http/constants.dart'; import 'package:pilipala/http/user.dart'; import 'package:pilipala/http/video.dart'; import 'package:pilipala/models/user/fav_folder.dart'; import 'package:pilipala/models/video/ai.dart'; import 'package:pilipala/models/video_detail_res.dart'; import 'package:pilipala/pages/video/detail/controller.dart'; import 'package:pilipala/pages/video/detail/reply/index.dart'; import 'package:pilipala/plugin/pl_player/models/play_repeat.dart'; import 'package:pilipala/utils/feed_back.dart'; import 'package:pilipala/utils/id_utils.dart'; import 'package:pilipala/utils/storage.dart'; import 'package:share_plus/share_plus.dart'; import 'widgets/group_panel.dart'; class VideoIntroController extends GetxController { // 视频bvid String bvid = Get.parameters['bvid']!; // 是否预渲染 骨架屏 bool preRender = false; // 视频详情 上个页面传入 Map? videoItem = {}; // 请求状态 RxBool isLoading = false.obs; // 视频详情 请求返回 Rx videoDetail = VideoDetailData().obs; // up主粉丝数 Map userStat = {'follower': '-'}; // 是否点赞 RxBool hasLike = false.obs; // 是否投币 RxBool hasCoin = false.obs; // 是否收藏 RxBool hasFav = false.obs; Box userInfoCache = GStrorage.userInfo; bool userLogin = false; Rx favFolderData = FavFolderData().obs; List addMediaIdsNew = []; List delMediaIdsNew = []; // 关注状态 默认未关注 RxMap followStatus = {}.obs; int _tempThemeValue = -1; RxInt lastPlayCid = 0.obs; var userInfo; // 同时观看 bool isShowOnlineTotal = false; RxString total = '1'.obs; Timer? timer; bool isPaused = false; String heroTag = ''; late ModelResult modelResult; @override void onInit() { super.onInit(); userInfo = userInfoCache.get('userInfoCache'); try { heroTag = Get.arguments['heroTag']; } catch (_) {} if (Get.arguments.isNotEmpty) { if (Get.arguments.containsKey('videoItem')) { preRender = true; var args = Get.arguments['videoItem']; var keys = Get.arguments.keys.toList(); videoItem!['pic'] = args.pic; if (args.title is String) { videoItem!['title'] = args.title; } else { String str = ''; for (Map map in args.title) { str += map['text']; } videoItem!['title'] = str; } videoItem!['stat'] = keys.contains('stat') && args.stat; videoItem!['pubdate'] = keys.contains('pubdate') && args.pubdate; videoItem!['owner'] = keys.contains('owner') && args.owner; } } userLogin = userInfo != null; lastPlayCid.value = int.parse(Get.parameters['cid']!); isShowOnlineTotal = setting.get(SettingBoxKey.enableOnlineTotal, defaultValue: false); if (isShowOnlineTotal) { queryOnlineTotal(); startTimer(); // 在页面加载时启动定时器 } } // 获取视频简介&分p Future queryVideoIntro() async { var result = await VideoHttp.videoIntro(bvid: bvid); if (result['status']) { videoDetail.value = result['data']!; if (videoDetail.value.pages!.isNotEmpty && lastPlayCid.value == 0) { lastPlayCid.value = videoDetail.value.pages!.first.cid!; } // Get.find(tag: heroTag).tabs.value = [ // '简介', // '评论 ${result['data']!.stat!.reply}' // ]; // 获取到粉丝数再返回 await queryUserStat(); } if (userLogin) { // 获取点赞状态 queryHasLikeVideo(); // 获取投币状态 queryHasCoinVideo(); // 获取收藏状态 queryHasFavVideo(); // queryFollowStatus(); } return result; } // 获取up主粉丝数 Future queryUserStat() async { var result = await UserHttp.userStat(mid: videoDetail.value.owner!.mid!); if (result['status']) { userStat = result['data']; } } // 获取点赞状态 Future queryHasLikeVideo() async { var result = await VideoHttp.hasLikeVideo(bvid: bvid); // data num 被点赞标志 0:未点赞 1:已点赞 hasLike.value = result["data"] == 1 ? true : false; } // 获取投币状态 Future queryHasCoinVideo() async { var result = await VideoHttp.hasCoinVideo(bvid: bvid); if (result['status']) { hasCoin.value = result["data"]['multiply'] == 0 ? false : true; } } // 获取收藏状态 Future queryHasFavVideo() async { /// fix 延迟查询 await Future.delayed(const Duration(milliseconds: 200)); var result = await VideoHttp.hasFavVideo(aid: IdUtils.bv2av(bvid)); if (result['status']) { hasFav.value = result["data"]['favoured']; } else { hasFav.value = false; } } // 一键三连 Future actionOneThree() async { if (userInfo == null) { SmartDialog.showToast('账号未登录'); return; } if (hasLike.value && hasCoin.value && hasFav.value) { // 已点赞、投币、收藏 SmartDialog.showToast('🙏 UP已经收到了~'); return false; } SmartDialog.show( useSystem: true, animationType: SmartAnimationType.centerFade_otherSlide, builder: (BuildContext context) { return AlertDialog( title: const Text('提示'), content: const Text('一键三连 给UP送温暖'), actions: [ TextButton( onPressed: () => SmartDialog.dismiss(), child: const Text('点错了')), TextButton( onPressed: () async { var result = await VideoHttp.oneThree(bvid: bvid); if (result['status']) { hasLike.value = result["data"]["like"]; hasCoin.value = result["data"]["coin"]; hasFav.value = result["data"]["fav"]; SmartDialog.showToast('三连成功 🎉'); } else { SmartDialog.showToast(result['msg']); } SmartDialog.dismiss(); }, child: const Text('确认'), ) ], ); }, ); } // (取消)点赞 Future actionLikeVideo() async { var result = await VideoHttp.likeVideo(bvid: bvid, type: !hasLike.value); if (result['status']) { // hasLike.value = result["data"] == 1 ? true : false; if (!hasLike.value) { SmartDialog.showToast('点赞成功 👍'); hasLike.value = true; videoDetail.value.stat!.like = videoDetail.value.stat!.like! + 1; } else if (hasLike.value) { SmartDialog.showToast('取消赞'); hasLike.value = false; videoDetail.value.stat!.like = videoDetail.value.stat!.like! - 1; } hasLike.refresh(); } else { SmartDialog.showToast(result['msg']); } } // 投币 Future actionCoinVideo() async { if (userInfo == null) { SmartDialog.showToast('账号未登录'); return; } showDialog( context: Get.context!, builder: (context) { return AlertDialog( title: const Text('选择投币个数'), contentPadding: const EdgeInsets.fromLTRB(0, 12, 0, 12), content: StatefulBuilder(builder: (context, StateSetter setState) { return Column( mainAxisSize: MainAxisSize.min, children: [ RadioListTile( value: 1, title: const Text('1枚'), groupValue: _tempThemeValue, onChanged: (value) { _tempThemeValue = value!; Get.appUpdate(); }, ), RadioListTile( value: 2, title: const Text('2枚'), groupValue: _tempThemeValue, onChanged: (value) { _tempThemeValue = value!; Get.appUpdate(); }, ), ], ); }), actions: [ TextButton(onPressed: () => Get.back(), child: const Text('取消')), TextButton( onPressed: () async { var res = await VideoHttp.coinVideo( bvid: bvid, multiply: _tempThemeValue); if (res['status']) { SmartDialog.showToast('投币成功 👏'); hasCoin.value = true; videoDetail.value.stat!.coin = videoDetail.value.stat!.coin! + _tempThemeValue; } else { SmartDialog.showToast(res['msg']); } Get.back(); }, child: const Text('确定')) ], ); }); } // (取消)收藏 Future actionFavVideo({type = 'choose'}) async { // 收藏至默认文件夹 if (type == 'default') { await queryVideoInFolder(); int defaultFolderId = favFolderData.value.list!.first.id!; int favStatus = favFolderData.value.list!.first.favState!; print('favStatus: $favStatus'); var result = await VideoHttp.favVideo( aid: IdUtils.bv2av(bvid), addIds: favStatus == 0 ? '$defaultFolderId' : '', delIds: favStatus == 1 ? '$defaultFolderId' : '', ); if (result['status']) { if (result['data']['prompt']) { // 重新获取收藏状态 await queryHasFavVideo(); SmartDialog.showToast('✅ 操作成功'); } } return; } try { for (var i in favFolderData.value.list!.toList()) { if (i.favState == 1) { addMediaIdsNew.add(i.id); } else { delMediaIdsNew.add(i.id); } } } catch (e) { // ignore: avoid_print print(e); } SmartDialog.showLoading(msg: '请求中'); var result = await VideoHttp.favVideo( aid: IdUtils.bv2av(bvid), addIds: addMediaIdsNew.join(','), delIds: delMediaIdsNew.join(',')); SmartDialog.dismiss(); if (result['status']) { if (result['data']['prompt']) { addMediaIdsNew = []; delMediaIdsNew = []; Get.back(); // 重新获取收藏状态 await queryHasFavVideo(); SmartDialog.showToast('✅ 操作成功'); } } } // 分享视频 Future actionShareVideo() async { var result = await Share.share( '${videoDetail.value.title} UP主: ${videoDetail.value.owner!.name!} - ${HttpString.baseUrl}/video/$bvid') .whenComplete(() {}); return result; } Future queryVideoInFolder() async { var result = await VideoHttp.videoInFolder( mid: userInfo.mid, rid: IdUtils.bv2av(bvid)); if (result['status']) { favFolderData.value = result['data']; } return result; } // 选择文件夹 onChoose(bool checkValue, int index) { feedBack(); List datalist = favFolderData.value.list!; for (var i = 0; i < datalist.length; i++) { if (i == index) { datalist[i].favState = checkValue == true ? 1 : 0; datalist[i].mediaCount = checkValue == true ? datalist[i].mediaCount! + 1 : datalist[i].mediaCount! - 1; } } favFolderData.value.list = datalist; favFolderData.refresh(); } // 查询关注状态 Future queryFollowStatus() async { if (videoDetail.value.owner == null) { return; } var result = await VideoHttp.hasFollow(mid: videoDetail.value.owner!.mid!); if (result['status']) { followStatus.value = result['data']; } return result; } // 关注/取关up Future actionRelationMod() async { feedBack(); if (userInfo == null) { SmartDialog.showToast('账号未登录'); return; } final int currentStatus = followStatus['attribute']; int actionStatus = 0; switch (currentStatus) { case 0: actionStatus = 1; break; case 2: actionStatus = 2; break; default: actionStatus = 0; break; } SmartDialog.show( useSystem: true, animationType: SmartAnimationType.centerFade_otherSlide, builder: (BuildContext context) { return AlertDialog( title: const Text('提示'), content: Text(currentStatus == 0 ? '关注UP主?' : '取消关注UP主?'), actions: [ TextButton( onPressed: () => SmartDialog.dismiss(), child: Text( '点错了', style: TextStyle(color: Theme.of(context).colorScheme.outline), ), ), TextButton( onPressed: () async { var result = await VideoHttp.relationMod( mid: videoDetail.value.owner!.mid!, act: actionStatus, reSrc: 14, ); if (result['status']) { switch (currentStatus) { case 0: actionStatus = 2; break; case 2: actionStatus = 0; break; default: actionStatus = 0; break; } followStatus['attribute'] = actionStatus; followStatus.refresh(); if (actionStatus == 2) { if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: const Text('关注成功'), duration: const Duration(seconds: 2), action: SnackBarAction( label: '设置分组', onPressed: setFollowGroup, ), showCloseIcon: true, ), ); } } } SmartDialog.dismiss(); }, child: const Text('确认'), ) ], ); }, ); } // 修改分P或番剧分集 Future changeSeasonOrbangu(bvid, cid, aid) async { // 重新获取视频资源 final VideoDetailController videoDetailCtr = Get.find(tag: heroTag); videoDetailCtr.bvid = bvid; videoDetailCtr.cid.value = cid; videoDetailCtr.danmakuCid.value = cid; videoDetailCtr.queryVideoUrl(); // 重新请求评论 try { /// 未渲染回复组件时可能异常 final VideoReplyController videoReplyCtr = Get.find(tag: heroTag); videoReplyCtr.aid = aid; videoReplyCtr.queryReplyList(type: 'init'); } catch (_) {} this.bvid = bvid; lastPlayCid.value = cid; await queryVideoIntro(); } void startTimer() { const duration = Duration(seconds: 10); // 设置定时器间隔为10秒 timer = Timer.periodic(duration, (Timer timer) { if (!isPaused) { queryOnlineTotal(); // 定时器回调函数,发起请求 } }); } // 查看同时在看人数 Future queryOnlineTotal() async { var result = await VideoHttp.onlineTotal( aid: IdUtils.bv2av(bvid), bvid: bvid, cid: lastPlayCid.value, ); if (result['status']) { total.value = result['data']['total']; } } @override void onClose() { if (timer != null) { timer!.cancel(); // 销毁页面时取消定时器 } super.onClose(); } /// 列表循环或者顺序播放时,自动播放下一个 void nextPlay() { final List episodes = []; bool isPages = false; if (videoDetail.value.ugcSeason != null) { final UgcSeason ugcSeason = videoDetail.value.ugcSeason!; final List sections = ugcSeason.sections!; for (int i = 0; i < sections.length; i++) { final List episodesList = sections[i].episodes!; episodes.addAll(episodesList); } } else if (videoDetail.value.pages != null) { isPages = true; final List pages = videoDetail.value.pages!; episodes.addAll(pages); } final int currentIndex = episodes.indexWhere((e) => e.cid == lastPlayCid.value); int nextIndex = currentIndex + 1; final VideoDetailController videoDetailCtr = Get.find(tag: heroTag); final PlayRepeat platRepeat = videoDetailCtr.plPlayerController.playRepeat; // 列表循环 if (nextIndex >= episodes.length) { if (platRepeat == PlayRepeat.listCycle) { nextIndex = 0; } if (platRepeat == PlayRepeat.listOrder) { return; } } final int cid = episodes[nextIndex].cid!; final String rBvid = isPages ? bvid : episodes[nextIndex].bvid; final int rAid = isPages ? IdUtils.bv2av(bvid) : episodes[nextIndex].aid!; changeSeasonOrbangu(rBvid, cid, rAid); } // 设置关注分组 void setFollowGroup() { Get.bottomSheet( GroupPanel(mid: videoDetail.value.owner!.mid!), isScrollControlled: true, ); } // ai总结 Future aiConclusion() async { SmartDialog.showLoading(msg: '正在生产ai总结'); final res = await VideoHttp.aiConclusion( bvid: bvid, cid: lastPlayCid.value, upMid: videoDetail.value.owner!.mid!, ); SmartDialog.dismiss(); if (res['status']) { modelResult = res['data'].modelResult; } else { SmartDialog.showToast("当前视频可能暂不支持AI视频总结"); } return res; } }