Compare commits
2 Commits
feature-ad
...
feature-ap
| Author | SHA1 | Date | |
|---|---|---|---|
| 53b72bec25 | |||
| 20745a4541 |
@ -36,7 +36,7 @@ class NetworkImgLayer extends StatelessWidget {
|
|||||||
final int defaultImgQuality = GlobalData().imgQuality;
|
final int defaultImgQuality = GlobalData().imgQuality;
|
||||||
final String imageUrl =
|
final String imageUrl =
|
||||||
'${src!.startsWith('//') ? 'https:${src!}' : src!}@${quality ?? defaultImgQuality}q.webp';
|
'${src!.startsWith('//') ? 'https:${src!}' : src!}@${quality ?? defaultImgQuality}q.webp';
|
||||||
print(imageUrl);
|
// print(imageUrl);
|
||||||
int? memCacheWidth, memCacheHeight;
|
int? memCacheWidth, memCacheHeight;
|
||||||
double aspectRatio = (width / height).toDouble();
|
double aspectRatio = (width / height).toDouble();
|
||||||
|
|
||||||
|
|||||||
@ -484,9 +484,6 @@ class Api {
|
|||||||
/// 激活buvid3
|
/// 激活buvid3
|
||||||
static const activateBuvidApi = '/x/internal/gaia-gateway/ExClimbWuzhi';
|
static const activateBuvidApi = '/x/internal/gaia-gateway/ExClimbWuzhi';
|
||||||
|
|
||||||
/// 获取字幕配置
|
|
||||||
static const getSubtitleConfig = '/x/player/v2';
|
|
||||||
|
|
||||||
/// 我的订阅
|
/// 我的订阅
|
||||||
static const userSubFolder = '/x/v3/fav/folder/collected/list';
|
static const userSubFolder = '/x/v3/fav/folder/collected/list';
|
||||||
|
|
||||||
|
|||||||
@ -8,11 +8,9 @@ import '../models/model_rec_video_item.dart';
|
|||||||
import '../models/user/fav_folder.dart';
|
import '../models/user/fav_folder.dart';
|
||||||
import '../models/video/ai.dart';
|
import '../models/video/ai.dart';
|
||||||
import '../models/video/play/url.dart';
|
import '../models/video/play/url.dart';
|
||||||
import '../models/video/subTitile/result.dart';
|
|
||||||
import '../models/video_detail_res.dart';
|
import '../models/video_detail_res.dart';
|
||||||
import '../utils/recommend_filter.dart';
|
import '../utils/recommend_filter.dart';
|
||||||
import '../utils/storage.dart';
|
import '../utils/storage.dart';
|
||||||
import '../utils/subtitle.dart';
|
|
||||||
import '../utils/wbi_sign.dart';
|
import '../utils/wbi_sign.dart';
|
||||||
import 'api.dart';
|
import 'api.dart';
|
||||||
import 'init.dart';
|
import 'init.dart';
|
||||||
@ -478,25 +476,6 @@ class VideoHttp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future getSubtitle({int? cid, String? bvid}) async {
|
|
||||||
var res = await Request().get(Api.getSubtitleConfig, data: {
|
|
||||||
'cid': cid,
|
|
||||||
'bvid': bvid,
|
|
||||||
});
|
|
||||||
try {
|
|
||||||
if (res.data['code'] == 0) {
|
|
||||||
return {
|
|
||||||
'status': true,
|
|
||||||
'data': SubTitlteModel.fromJson(res.data['data']),
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
return {'status': false, 'data': [], 'msg': res.data['msg']};
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
print(err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 视频排行
|
// 视频排行
|
||||||
static Future getRankVideoList(int rid) async {
|
static Future getRankVideoList(int rid) async {
|
||||||
try {
|
try {
|
||||||
@ -519,12 +498,4 @@ class VideoHttp {
|
|||||||
return {'status': false, 'data': [], 'msg': err};
|
return {'status': false, 'data': [], 'msg': err};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 获取字幕内容
|
|
||||||
static Future<Map<String, dynamic>> getSubtitleContent(url) async {
|
|
||||||
var res = await Request().get('https:$url');
|
|
||||||
final String content = SubTitleUtils.convertToWebVTT(res.data['body']);
|
|
||||||
final List body = res.data['body'];
|
|
||||||
return {'content': content, 'body': body};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,47 +0,0 @@
|
|||||||
enum SubtitleType {
|
|
||||||
// 中文(中国)
|
|
||||||
zhCN,
|
|
||||||
// 中文(自动翻译)
|
|
||||||
aizh,
|
|
||||||
// 英语(自动生成)
|
|
||||||
aien,
|
|
||||||
}
|
|
||||||
|
|
||||||
extension SubtitleTypeExtension on SubtitleType {
|
|
||||||
String get description {
|
|
||||||
switch (this) {
|
|
||||||
case SubtitleType.zhCN:
|
|
||||||
return '中文(中国)';
|
|
||||||
case SubtitleType.aizh:
|
|
||||||
return '中文(自动翻译)';
|
|
||||||
case SubtitleType.aien:
|
|
||||||
return '英语(自动生成)';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension SubtitleIdExtension on SubtitleType {
|
|
||||||
String get id {
|
|
||||||
switch (this) {
|
|
||||||
case SubtitleType.zhCN:
|
|
||||||
return 'zh-CN';
|
|
||||||
case SubtitleType.aizh:
|
|
||||||
return 'ai-zh';
|
|
||||||
case SubtitleType.aien:
|
|
||||||
return 'ai-en';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
extension SubtitleCodeExtension on SubtitleType {
|
|
||||||
int get code {
|
|
||||||
switch (this) {
|
|
||||||
case SubtitleType.zhCN:
|
|
||||||
return 1;
|
|
||||||
case SubtitleType.aizh:
|
|
||||||
return 2;
|
|
||||||
case SubtitleType.aien:
|
|
||||||
return 3;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,20 +0,0 @@
|
|||||||
class SubTitileContentModel {
|
|
||||||
double? from;
|
|
||||||
double? to;
|
|
||||||
int? location;
|
|
||||||
String? content;
|
|
||||||
|
|
||||||
SubTitileContentModel({
|
|
||||||
this.from,
|
|
||||||
this.to,
|
|
||||||
this.location,
|
|
||||||
this.content,
|
|
||||||
});
|
|
||||||
|
|
||||||
SubTitileContentModel.fromJson(Map<String, dynamic> json) {
|
|
||||||
from = json['from'];
|
|
||||||
to = json['to'];
|
|
||||||
location = json['location'];
|
|
||||||
content = json['content'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,89 +0,0 @@
|
|||||||
import 'package:get/get.dart';
|
|
||||||
import '../../common/subtitle_type.dart';
|
|
||||||
|
|
||||||
class SubTitlteModel {
|
|
||||||
SubTitlteModel({
|
|
||||||
this.aid,
|
|
||||||
this.bvid,
|
|
||||||
this.cid,
|
|
||||||
this.loginMid,
|
|
||||||
this.loginMidHash,
|
|
||||||
this.isOwner,
|
|
||||||
this.name,
|
|
||||||
this.subtitles,
|
|
||||||
});
|
|
||||||
|
|
||||||
int? aid;
|
|
||||||
String? bvid;
|
|
||||||
int? cid;
|
|
||||||
int? loginMid;
|
|
||||||
String? loginMidHash;
|
|
||||||
bool? isOwner;
|
|
||||||
String? name;
|
|
||||||
List<SubTitlteItemModel>? subtitles;
|
|
||||||
|
|
||||||
factory SubTitlteModel.fromJson(Map<String, dynamic> json) => SubTitlteModel(
|
|
||||||
aid: json["aid"],
|
|
||||||
bvid: json["bvid"],
|
|
||||||
cid: json["cid"],
|
|
||||||
loginMid: json["login_mid"],
|
|
||||||
loginMidHash: json["login_mid_hash"],
|
|
||||||
isOwner: json["is_owner"],
|
|
||||||
name: json["name"],
|
|
||||||
subtitles: json["subtitle"] != null
|
|
||||||
? json["subtitle"]["subtitles"]
|
|
||||||
.map<SubTitlteItemModel>((x) => SubTitlteItemModel.fromJson(x))
|
|
||||||
.toList()
|
|
||||||
: [],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
class SubTitlteItemModel {
|
|
||||||
SubTitlteItemModel({
|
|
||||||
this.id,
|
|
||||||
this.lan,
|
|
||||||
this.lanDoc,
|
|
||||||
this.isLock,
|
|
||||||
this.subtitleUrl,
|
|
||||||
this.type,
|
|
||||||
this.aiType,
|
|
||||||
this.aiStatus,
|
|
||||||
this.title,
|
|
||||||
this.code,
|
|
||||||
this.content,
|
|
||||||
this.body,
|
|
||||||
});
|
|
||||||
|
|
||||||
int? id;
|
|
||||||
String? lan;
|
|
||||||
String? lanDoc;
|
|
||||||
bool? isLock;
|
|
||||||
String? subtitleUrl;
|
|
||||||
int? type;
|
|
||||||
int? aiType;
|
|
||||||
int? aiStatus;
|
|
||||||
String? title;
|
|
||||||
int? code;
|
|
||||||
String? content;
|
|
||||||
List? body;
|
|
||||||
|
|
||||||
factory SubTitlteItemModel.fromJson(Map<String, dynamic> json) =>
|
|
||||||
SubTitlteItemModel(
|
|
||||||
id: json["id"],
|
|
||||||
lan: json["lan"].replaceAll('-', ''),
|
|
||||||
lanDoc: json["lan_doc"],
|
|
||||||
isLock: json["is_lock"],
|
|
||||||
subtitleUrl: json["subtitle_url"],
|
|
||||||
type: json["type"],
|
|
||||||
aiType: json["ai_type"],
|
|
||||||
aiStatus: json["ai_status"],
|
|
||||||
title: json["lan_doc"],
|
|
||||||
code: SubtitleType.values
|
|
||||||
.firstWhereOrNull(
|
|
||||||
(element) => element.id.toString() == json["lan"])
|
|
||||||
?.index ??
|
|
||||||
-1,
|
|
||||||
content: '',
|
|
||||||
body: [],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@ -9,7 +9,7 @@ import 'package:pilipala/utils/storage.dart';
|
|||||||
class RankController extends GetxController with GetTickerProviderStateMixin {
|
class RankController extends GetxController with GetTickerProviderStateMixin {
|
||||||
bool flag = false;
|
bool flag = false;
|
||||||
late RxList tabs = [].obs;
|
late RxList tabs = [].obs;
|
||||||
RxInt initialIndex = 0.obs;
|
RxInt initialIndex = 1.obs;
|
||||||
late TabController tabController;
|
late TabController tabController;
|
||||||
late List tabsCtrList;
|
late List tabsCtrList;
|
||||||
late List<Widget> tabsPageList;
|
late List<Widget> tabsPageList;
|
||||||
@ -50,5 +50,21 @@ class RankController extends GetxController with GetTickerProviderStateMixin {
|
|||||||
length: tabs.length,
|
length: tabs.length,
|
||||||
vsync: this,
|
vsync: this,
|
||||||
);
|
);
|
||||||
|
// 监听 tabController 切换
|
||||||
|
if (enableGradientBg) {
|
||||||
|
tabController.animation!.addListener(() {
|
||||||
|
if (tabController.indexIsChanging) {
|
||||||
|
if (initialIndex.value != tabController.index) {
|
||||||
|
initialIndex.value = tabController.index;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final int temp = tabController.animation!.value.round();
|
||||||
|
if (initialIndex.value != temp) {
|
||||||
|
initialIndex.value = temp;
|
||||||
|
tabController.index = initialIndex.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -22,20 +22,15 @@ class ZonePage extends StatefulWidget {
|
|||||||
State<ZonePage> createState() => _ZonePageState();
|
State<ZonePage> createState() => _ZonePageState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _ZonePageState extends State<ZonePage>
|
class _ZonePageState extends State<ZonePage> {
|
||||||
with AutomaticKeepAliveClientMixin {
|
final ZoneController _zoneController = Get.put(ZoneController());
|
||||||
late ZoneController _zoneController;
|
|
||||||
List videoList = [];
|
List videoList = [];
|
||||||
Future? _futureBuilderFuture;
|
Future? _futureBuilderFuture;
|
||||||
late ScrollController scrollController;
|
late ScrollController scrollController;
|
||||||
|
|
||||||
@override
|
|
||||||
bool get wantKeepAlive => true;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_zoneController = Get.put(ZoneController(), tag: widget.rid.toString());
|
|
||||||
_futureBuilderFuture = _zoneController.queryRankFeed('init', widget.rid);
|
_futureBuilderFuture = _zoneController.queryRankFeed('init', widget.rid);
|
||||||
scrollController = _zoneController.scrollController;
|
scrollController = _zoneController.scrollController;
|
||||||
StreamController<bool> mainStream =
|
StreamController<bool> mainStream =
|
||||||
@ -73,7 +68,6 @@ class _ZonePageState extends State<ZonePage>
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
|
||||||
return RefreshIndicator(
|
return RefreshIndicator(
|
||||||
onRefresh: () async {
|
onRefresh: () async {
|
||||||
return await _zoneController.onRefresh();
|
return await _zoneController.onRefresh();
|
||||||
|
|||||||
@ -20,7 +20,6 @@ import 'package:pilipala/utils/utils.dart';
|
|||||||
import 'package:pilipala/utils/video_utils.dart';
|
import 'package:pilipala/utils/video_utils.dart';
|
||||||
import 'package:screen_brightness/screen_brightness.dart';
|
import 'package:screen_brightness/screen_brightness.dart';
|
||||||
|
|
||||||
import '../../../models/video/subTitile/content.dart';
|
|
||||||
import '../../../http/danmaku.dart';
|
import '../../../http/danmaku.dart';
|
||||||
import '../../../utils/id_utils.dart';
|
import '../../../utils/id_utils.dart';
|
||||||
import 'widgets/header_control.dart';
|
import 'widgets/header_control.dart';
|
||||||
@ -94,10 +93,7 @@ class VideoDetailController extends GetxController
|
|||||||
late int cacheAudioQa;
|
late int cacheAudioQa;
|
||||||
|
|
||||||
PersistentBottomSheetController? replyReplyBottomSheetCtr;
|
PersistentBottomSheetController? replyReplyBottomSheetCtr;
|
||||||
RxList<SubTitileContentModel> subtitleContents =
|
|
||||||
<SubTitileContentModel>[].obs;
|
|
||||||
late bool enableRelatedVideo;
|
late bool enableRelatedVideo;
|
||||||
List subtitles = [];
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -149,7 +145,6 @@ class VideoDetailController extends GetxController
|
|||||||
cacheAudioQa = setting.get(SettingBoxKey.defaultAudioQa,
|
cacheAudioQa = setting.get(SettingBoxKey.defaultAudioQa,
|
||||||
defaultValue: AudioQuality.hiRes.code);
|
defaultValue: AudioQuality.hiRes.code);
|
||||||
oid.value = IdUtils.bv2av(Get.parameters['bvid']!);
|
oid.value = IdUtils.bv2av(Get.parameters['bvid']!);
|
||||||
getSubtitle();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
showReplyReplyPanel() {
|
showReplyReplyPanel() {
|
||||||
@ -256,8 +251,6 @@ class VideoDetailController extends GetxController
|
|||||||
|
|
||||||
/// 开启自动全屏时,在player初始化完成后立即传入headerControl
|
/// 开启自动全屏时,在player初始化完成后立即传入headerControl
|
||||||
plPlayerController.headerControl = headerControl;
|
plPlayerController.headerControl = headerControl;
|
||||||
|
|
||||||
plPlayerController.subtitles.value = subtitles;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 视频链接
|
// 视频链接
|
||||||
@ -395,45 +388,6 @@ class VideoDetailController extends GetxController
|
|||||||
: print('replyReplyBottomSheetCtr is null');
|
: print('replyReplyBottomSheetCtr is null');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 获取字幕配置
|
|
||||||
Future getSubtitle() async {
|
|
||||||
var result = await VideoHttp.getSubtitle(bvid: bvid, cid: cid.value);
|
|
||||||
if (result['status']) {
|
|
||||||
if (result['data'].subtitles.isNotEmpty) {
|
|
||||||
subtitles = result['data'].subtitles;
|
|
||||||
if (subtitles.isNotEmpty) {
|
|
||||||
for (var i in subtitles) {
|
|
||||||
final Map<String, dynamic> res = await VideoHttp.getSubtitleContent(
|
|
||||||
i.subtitleUrl,
|
|
||||||
);
|
|
||||||
i.content = res['content'];
|
|
||||||
i.body = res['body'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result['data'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 获取字幕内容
|
|
||||||
// Future getSubtitleContent(String url) async {
|
|
||||||
// var res = await Request().get('https:$url');
|
|
||||||
// subtitleContents.value = res.data['body'].map<SubTitileContentModel>((e) {
|
|
||||||
// return SubTitileContentModel.fromJson(e);
|
|
||||||
// }).toList();
|
|
||||||
// setSubtitleContent();
|
|
||||||
// }
|
|
||||||
|
|
||||||
setSubtitleContent() {
|
|
||||||
plPlayerController.subtitleContent.value = '';
|
|
||||||
plPlayerController.subtitles.value = subtitles;
|
|
||||||
}
|
|
||||||
|
|
||||||
clearSubtitleContent() {
|
|
||||||
plPlayerController.subtitleContent.value = '';
|
|
||||||
plPlayerController.subtitles.value = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
/// 发送弹幕
|
/// 发送弹幕
|
||||||
void showShootDanmakuSheet() {
|
void showShootDanmakuSheet() {
|
||||||
final TextEditingController textController = TextEditingController();
|
final TextEditingController textController = TextEditingController();
|
||||||
|
|||||||
@ -15,8 +15,6 @@ import 'package:pilipala/pages/video/detail/widgets/ai_detail.dart';
|
|||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
import '../../../../http/user.dart';
|
|
||||||
import '../widgets/expandable_section.dart';
|
|
||||||
import 'widgets/action_item.dart';
|
import 'widgets/action_item.dart';
|
||||||
import 'widgets/fav_panel.dart';
|
import 'widgets/fav_panel.dart';
|
||||||
import 'widgets/intro_detail.dart';
|
import 'widgets/intro_detail.dart';
|
||||||
@ -139,7 +137,6 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
late String memberHeroTag;
|
late String memberHeroTag;
|
||||||
late bool enableAi;
|
late bool enableAi;
|
||||||
bool isProcessing = false;
|
bool isProcessing = false;
|
||||||
RxBool isExpand = false.obs;
|
|
||||||
void Function()? handleState(Future Function() action) {
|
void Function()? handleState(Future Function() action) {
|
||||||
return isProcessing
|
return isProcessing
|
||||||
? null
|
? null
|
||||||
@ -215,7 +212,13 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
// 视频介绍
|
// 视频介绍
|
||||||
showIntroDetail() {
|
showIntroDetail() {
|
||||||
feedBack();
|
feedBack();
|
||||||
isExpand.value = !(isExpand.value);
|
showBottomSheet(
|
||||||
|
context: context,
|
||||||
|
enableDrag: true,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return IntroDetail(videoDetail: widget.videoDetail!);
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 用户主页
|
// 用户主页
|
||||||
@ -327,16 +330,6 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
/// 视频简介
|
|
||||||
Obx(
|
|
||||||
() => ExpandedSection(
|
|
||||||
expand: isExpand.value,
|
|
||||||
begin: 0,
|
|
||||||
end: 1,
|
|
||||||
child: IntroDetail(videoDetail: widget.videoDetail!),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
/// 点赞收藏转发
|
/// 点赞收藏转发
|
||||||
actionGrid(context, videoIntroController),
|
actionGrid(context, videoIntroController),
|
||||||
// 合集
|
// 合集
|
||||||
@ -445,7 +438,6 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
margin: const EdgeInsets.only(top: 6, bottom: 4),
|
margin: const EdgeInsets.only(top: 6, bottom: 4),
|
||||||
height: constraints.maxWidth / 5 * 0.8,
|
height: constraints.maxWidth / 5 * 0.8,
|
||||||
child: GridView.count(
|
child: GridView.count(
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
|
||||||
primary: false,
|
primary: false,
|
||||||
padding: EdgeInsets.zero,
|
padding: EdgeInsets.zero,
|
||||||
crossAxisCount: 5,
|
crossAxisCount: 5,
|
||||||
@ -459,6 +451,12 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
selectStatus: videoIntroController.hasLike.value,
|
selectStatus: videoIntroController.hasLike.value,
|
||||||
text: widget.videoDetail!.stat!.like!.toString()),
|
text: widget.videoDetail!.stat!.like!.toString()),
|
||||||
),
|
),
|
||||||
|
// ActionItem(
|
||||||
|
// icon: const Icon(FontAwesomeIcons.clock),
|
||||||
|
// onTap: () => videoIntroController.actionShareVideo(),
|
||||||
|
// selectStatus: false,
|
||||||
|
// loadingStatus: loadingStatus,
|
||||||
|
// text: '稍后再看'),
|
||||||
Obx(
|
Obx(
|
||||||
() => ActionItem(
|
() => ActionItem(
|
||||||
icon: const Icon(FontAwesomeIcons.b),
|
icon: const Icon(FontAwesomeIcons.b),
|
||||||
@ -479,14 +477,10 @@ class _VideoInfoState extends State<VideoInfo> with TickerProviderStateMixin {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
ActionItem(
|
ActionItem(
|
||||||
icon: const Icon(FontAwesomeIcons.clock),
|
icon: const Icon(FontAwesomeIcons.comment),
|
||||||
onTap: () async {
|
onTap: () => videoDetailCtr.tabCtr.animateTo(1),
|
||||||
final res =
|
|
||||||
await UserHttp.toViewLater(bvid: widget.videoDetail!.bvid);
|
|
||||||
SmartDialog.showToast(res['msg']);
|
|
||||||
},
|
|
||||||
selectStatus: false,
|
selectStatus: false,
|
||||||
text: '稍后看',
|
text: widget.videoDetail!.stat!.reply!.toString(),
|
||||||
),
|
),
|
||||||
ActionItem(
|
ActionItem(
|
||||||
icon: const Icon(FontAwesomeIcons.shareFromSquare),
|
icon: const Icon(FontAwesomeIcons.shareFromSquare),
|
||||||
|
|||||||
@ -1,10 +1,16 @@
|
|||||||
import 'package:flutter/gestures.dart';
|
import 'package:flutter/gestures.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
import 'package:hive/hive.dart';
|
||||||
|
import 'package:pilipala/common/widgets/stat/danmu.dart';
|
||||||
|
import 'package:pilipala/common/widgets/stat/view.dart';
|
||||||
|
import 'package:pilipala/utils/storage.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
|
|
||||||
|
Box localCache = GStrorage.localCache;
|
||||||
|
late double sheetHeight;
|
||||||
|
|
||||||
class IntroDetail extends StatelessWidget {
|
class IntroDetail extends StatelessWidget {
|
||||||
const IntroDetail({
|
const IntroDetail({
|
||||||
super.key,
|
super.key,
|
||||||
@ -14,39 +20,105 @@ class IntroDetail extends StatelessWidget {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return SizedBox(
|
sheetHeight = localCache.get('sheetHeight');
|
||||||
width: double.infinity,
|
return Container(
|
||||||
child: SelectableRegion(
|
color: Theme.of(context).colorScheme.background,
|
||||||
focusNode: FocusNode(),
|
padding: EdgeInsets.only(
|
||||||
selectionControls: MaterialTextSelectionControls(),
|
left: 14,
|
||||||
|
right: 14,
|
||||||
|
bottom: MediaQuery.of(context).padding.bottom + 20),
|
||||||
|
height: sheetHeight,
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
children: [
|
||||||
children: <Widget>[
|
InkWell(
|
||||||
const SizedBox(height: 4),
|
onTap: () => Get.back(),
|
||||||
GestureDetector(
|
child: Container(
|
||||||
onTap: () {
|
height: 35,
|
||||||
Clipboard.setData(ClipboardData(text: videoDetail!.bvid!));
|
padding: const EdgeInsets.only(bottom: 2),
|
||||||
SmartDialog.showToast('已复制');
|
child: Center(
|
||||||
},
|
child: Container(
|
||||||
child: Text(
|
width: 32,
|
||||||
videoDetail!.bvid!,
|
height: 3,
|
||||||
style: TextStyle(
|
decoration: BoxDecoration(
|
||||||
fontSize: 13, color: Theme.of(context).colorScheme.primary),
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
borderRadius:
|
||||||
|
const BorderRadius.all(Radius.circular(3))),
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 4),
|
Expanded(
|
||||||
Text.rich(
|
child: SingleChildScrollView(
|
||||||
style: const TextStyle(height: 1.4),
|
child: Column(
|
||||||
TextSpan(
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
buildContent(context, videoDetail!),
|
Text(
|
||||||
],
|
videoDetail!.title,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: FontWeight.w500,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
StatView(
|
||||||
|
theme: 'gray',
|
||||||
|
view: videoDetail!.stat!.view,
|
||||||
|
size: 'medium',
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
StatDanMu(
|
||||||
|
theme: 'gray',
|
||||||
|
danmu: videoDetail!.stat!.danmaku,
|
||||||
|
size: 'medium',
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Text(
|
||||||
|
Utils.dateFormat(videoDetail!.pubdate,
|
||||||
|
formatType: 'detail'),
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 12,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
SizedBox(
|
||||||
|
width: double.infinity,
|
||||||
|
child: SelectableRegion(
|
||||||
|
focusNode: FocusNode(),
|
||||||
|
selectionControls: MaterialTextSelectionControls(),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: <Widget>[
|
||||||
|
Text(
|
||||||
|
videoDetail!.bvid!,
|
||||||
|
style: const TextStyle(fontSize: 13),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text.rich(
|
||||||
|
style: const TextStyle(
|
||||||
|
height: 1.4,
|
||||||
|
// fontSize: 13,
|
||||||
|
),
|
||||||
|
TextSpan(
|
||||||
|
children: [
|
||||||
|
buildContent(context, videoDetail!),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
)
|
||||||
],
|
],
|
||||||
),
|
));
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
InlineSpan buildContent(BuildContext context, content) {
|
InlineSpan buildContent(BuildContext context, content) {
|
||||||
|
|||||||
@ -212,7 +212,6 @@ class _VideoDetailPageState extends State<VideoDetailPage>
|
|||||||
videoIntroController.isPaused = true;
|
videoIntroController.isPaused = true;
|
||||||
plPlayerController!.removeStatusLister(playerListener);
|
plPlayerController!.removeStatusLister(playerListener);
|
||||||
plPlayerController!.pause();
|
plPlayerController!.pause();
|
||||||
vdCtr.clearSubtitleContent();
|
|
||||||
}
|
}
|
||||||
setState(() => isShowing = false);
|
setState(() => isShowing = false);
|
||||||
super.didPushNext();
|
super.didPushNext();
|
||||||
@ -223,10 +222,7 @@ class _VideoDetailPageState extends State<VideoDetailPage>
|
|||||||
void didPopNext() async {
|
void didPopNext() async {
|
||||||
if (plPlayerController != null &&
|
if (plPlayerController != null &&
|
||||||
plPlayerController!.videoPlayerController != null) {
|
plPlayerController!.videoPlayerController != null) {
|
||||||
setState(() {
|
setState(() => isShowing = true);
|
||||||
vdCtr.setSubtitleContent();
|
|
||||||
isShowing = true;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
vdCtr.isFirstTime = false;
|
vdCtr.isFirstTime = false;
|
||||||
final bool autoplay = autoPlayEnable;
|
final bool autoplay = autoPlayEnable;
|
||||||
@ -376,9 +372,6 @@ class _VideoDetailPageState extends State<VideoDetailPage>
|
|||||||
false)
|
false)
|
||||||
? SvgPicture.asset(
|
? SvgPicture.asset(
|
||||||
'assets/images/video/danmu_close.svg',
|
'assets/images/video/danmu_close.svg',
|
||||||
// ignore: deprecated_member_use
|
|
||||||
color:
|
|
||||||
Theme.of(context).colorScheme.outline,
|
|
||||||
)
|
)
|
||||||
: SvgPicture.asset(
|
: SvgPicture.asset(
|
||||||
'assets/images/video/danmu_open.svg',
|
'assets/images/video/danmu_open.svg',
|
||||||
|
|||||||
@ -17,16 +17,12 @@ class ScrollAppBar extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final double statusBarHeight = MediaQuery.of(context).padding.top;
|
final double statusBarHeight = MediaQuery.of(context).padding.top;
|
||||||
final videoHeight = MediaQuery.sizeOf(context).width * 9 / 16;
|
final videoHeight = MediaQuery.sizeOf(context).width * 9 / 16;
|
||||||
double scrollDistance = scrollVal;
|
|
||||||
if (scrollVal > videoHeight - kToolbarHeight) {
|
|
||||||
scrollDistance = videoHeight - kToolbarHeight;
|
|
||||||
}
|
|
||||||
return Positioned(
|
return Positioned(
|
||||||
top: -videoHeight + scrollDistance + kToolbarHeight + 0.5,
|
top: -videoHeight + scrollVal + kToolbarHeight + 0.5,
|
||||||
left: 0,
|
left: 0,
|
||||||
right: 0,
|
right: 0,
|
||||||
child: Opacity(
|
child: Opacity(
|
||||||
opacity: scrollDistance / (videoHeight - kToolbarHeight),
|
opacity: scrollVal / (videoHeight - kToolbarHeight),
|
||||||
child: Container(
|
child: Container(
|
||||||
height: statusBarHeight + kToolbarHeight,
|
height: statusBarHeight + kToolbarHeight,
|
||||||
color: Theme.of(context).colorScheme.background,
|
color: Theme.of(context).colorScheme.background,
|
||||||
|
|||||||
@ -32,14 +32,28 @@ class _ExpandedSectionState extends State<ExpandedSection>
|
|||||||
_runExpandCheck();
|
_runExpandCheck();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///Setting up the animation
|
||||||
|
// void prepareAnimations() {
|
||||||
|
// expandController = AnimationController(
|
||||||
|
// vsync: this, duration: const Duration(milliseconds: 500));
|
||||||
|
// animation = CurvedAnimation(
|
||||||
|
// parent: expandController,
|
||||||
|
// curve: Curves.fastOutSlowIn,
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
void prepareAnimations() {
|
void prepareAnimations() {
|
||||||
expandController = AnimationController(
|
expandController = AnimationController(
|
||||||
vsync: this, duration: const Duration(milliseconds: 400));
|
vsync: this, duration: const Duration(milliseconds: 400));
|
||||||
Animation<double> curve = CurvedAnimation(
|
Animation<double> curve = CurvedAnimation(
|
||||||
parent: expandController,
|
parent: expandController,
|
||||||
curve: Curves.linear,
|
curve: Curves.fastOutSlowIn,
|
||||||
);
|
);
|
||||||
animation = Tween(begin: widget.begin, end: widget.end).animate(curve);
|
animation = Tween(begin: widget.begin, end: widget.end).animate(curve);
|
||||||
|
// animation = CurvedAnimation(
|
||||||
|
// parent: expandController,
|
||||||
|
// curve: Curves.fastOutSlowIn,
|
||||||
|
// );
|
||||||
}
|
}
|
||||||
|
|
||||||
void _runExpandCheck() {
|
void _runExpandCheck() {
|
||||||
@ -53,9 +67,7 @@ class _ExpandedSectionState extends State<ExpandedSection>
|
|||||||
@override
|
@override
|
||||||
void didUpdateWidget(ExpandedSection oldWidget) {
|
void didUpdateWidget(ExpandedSection oldWidget) {
|
||||||
super.didUpdateWidget(oldWidget);
|
super.didUpdateWidget(oldWidget);
|
||||||
if (widget.expand != oldWidget.expand) {
|
_runExpandCheck();
|
||||||
_runExpandCheck();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
|||||||
@ -344,56 +344,6 @@ class _HeaderControlState extends State<HeaderControl> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 选择字幕
|
|
||||||
void showSubtitleDialog() async {
|
|
||||||
int tempThemeValue = widget.controller!.subTitleCode.value;
|
|
||||||
int len = widget.videoDetailCtr!.subtitles.length;
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (BuildContext context) {
|
|
||||||
return AlertDialog(
|
|
||||||
title: const Text('选择字幕'),
|
|
||||||
contentPadding: const EdgeInsets.fromLTRB(0, 12, 0, 18),
|
|
||||||
content: StatefulBuilder(builder: (context, StateSetter setState) {
|
|
||||||
return len == 0
|
|
||||||
? const SizedBox(
|
|
||||||
height: 60,
|
|
||||||
child: Center(
|
|
||||||
child: Text('没有字幕'),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
RadioListTile(
|
|
||||||
value: -1,
|
|
||||||
title: const Text('关闭弹幕'),
|
|
||||||
groupValue: tempThemeValue,
|
|
||||||
onChanged: (value) {
|
|
||||||
tempThemeValue = value!;
|
|
||||||
widget.controller?.toggleSubtitle(value);
|
|
||||||
Get.back();
|
|
||||||
},
|
|
||||||
),
|
|
||||||
...widget.videoDetailCtr!.subtitles
|
|
||||||
.map((e) => RadioListTile(
|
|
||||||
value: e.code,
|
|
||||||
title: Text(e.title),
|
|
||||||
groupValue: tempThemeValue,
|
|
||||||
onChanged: (value) {
|
|
||||||
tempThemeValue = value!;
|
|
||||||
widget.controller?.toggleSubtitle(value);
|
|
||||||
Get.back();
|
|
||||||
},
|
|
||||||
))
|
|
||||||
.toList(),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/// 选择倍速
|
/// 选择倍速
|
||||||
void showSetSpeedSheet() {
|
void showSetSpeedSheet() {
|
||||||
final double currentSpeed = widget.controller!.playbackSpeed;
|
final double currentSpeed = widget.controller!.playbackSpeed;
|
||||||
@ -1165,31 +1115,6 @@ class _HeaderControlState extends State<HeaderControl> {
|
|||||||
),
|
),
|
||||||
SizedBox(width: buttonSpace),
|
SizedBox(width: buttonSpace),
|
||||||
],
|
],
|
||||||
|
|
||||||
/// 字幕
|
|
||||||
// SizedBox(
|
|
||||||
// width: 34,
|
|
||||||
// height: 34,
|
|
||||||
// child: IconButton(
|
|
||||||
// style: ButtonStyle(
|
|
||||||
// padding: MaterialStateProperty.all(EdgeInsets.zero),
|
|
||||||
// ),
|
|
||||||
// onPressed: () => showSubtitleDialog(),
|
|
||||||
// icon: const Icon(
|
|
||||||
// Icons.closed_caption_off,
|
|
||||||
// size: 22,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
ComBtn(
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.closed_caption_off,
|
|
||||||
size: 22,
|
|
||||||
color: Colors.white,
|
|
||||||
),
|
|
||||||
fuc: () => showSubtitleDialog(),
|
|
||||||
),
|
|
||||||
SizedBox(width: buttonSpace),
|
|
||||||
Obx(
|
Obx(
|
||||||
() => SizedBox(
|
() => SizedBox(
|
||||||
width: 45,
|
width: 45,
|
||||||
|
|||||||
@ -21,8 +21,6 @@ import 'package:pilipala/utils/storage.dart';
|
|||||||
import 'package:screen_brightness/screen_brightness.dart';
|
import 'package:screen_brightness/screen_brightness.dart';
|
||||||
import 'package:status_bar_control/status_bar_control.dart';
|
import 'package:status_bar_control/status_bar_control.dart';
|
||||||
import 'package:universal_platform/universal_platform.dart';
|
import 'package:universal_platform/universal_platform.dart';
|
||||||
import '../../models/video/subTitile/content.dart';
|
|
||||||
import '../../models/video/subTitile/result.dart';
|
|
||||||
// import 'package:wakelock_plus/wakelock_plus.dart';
|
// import 'package:wakelock_plus/wakelock_plus.dart';
|
||||||
|
|
||||||
Box videoStorage = GStrorage.video;
|
Box videoStorage = GStrorage.video;
|
||||||
@ -75,8 +73,6 @@ class PlPlayerController {
|
|||||||
final Rx<bool> _doubleSpeedStatus = false.obs;
|
final Rx<bool> _doubleSpeedStatus = false.obs;
|
||||||
final Rx<bool> _controlsLock = false.obs;
|
final Rx<bool> _controlsLock = false.obs;
|
||||||
final Rx<bool> _isFullScreen = false.obs;
|
final Rx<bool> _isFullScreen = false.obs;
|
||||||
final Rx<bool> _subTitleOpen = false.obs;
|
|
||||||
final Rx<int> _subTitleCode = (-1).obs;
|
|
||||||
// 默认投稿视频格式
|
// 默认投稿视频格式
|
||||||
static Rx<String> _videoType = 'archive'.obs;
|
static Rx<String> _videoType = 'archive'.obs;
|
||||||
|
|
||||||
@ -122,7 +118,6 @@ class PlPlayerController {
|
|||||||
PreferredSizeWidget? headerControl;
|
PreferredSizeWidget? headerControl;
|
||||||
PreferredSizeWidget? bottomControl;
|
PreferredSizeWidget? bottomControl;
|
||||||
Widget? danmuWidget;
|
Widget? danmuWidget;
|
||||||
late RxList subtitles;
|
|
||||||
|
|
||||||
/// 数据加载监听
|
/// 数据加载监听
|
||||||
Stream<DataStatus> get onDataStatusChanged => dataStatus.status.stream;
|
Stream<DataStatus> get onDataStatusChanged => dataStatus.status.stream;
|
||||||
@ -152,11 +147,6 @@ class PlPlayerController {
|
|||||||
Rx<bool> get mute => _mute;
|
Rx<bool> get mute => _mute;
|
||||||
Stream<bool> get onMuteChanged => _mute.stream;
|
Stream<bool> get onMuteChanged => _mute.stream;
|
||||||
|
|
||||||
/// 字幕开启状态
|
|
||||||
Rx<bool> get subTitleOpen => _subTitleOpen;
|
|
||||||
Rx<int> get subTitleCode => _subTitleCode;
|
|
||||||
// Stream<bool> get onSubTitleOpenChanged => _subTitleOpen.stream;
|
|
||||||
|
|
||||||
/// [videoPlayerController] instace of Player
|
/// [videoPlayerController] instace of Player
|
||||||
Player? get videoPlayerController => _videoPlayerController;
|
Player? get videoPlayerController => _videoPlayerController;
|
||||||
|
|
||||||
@ -241,10 +231,6 @@ class PlPlayerController {
|
|||||||
// 播放顺序相关
|
// 播放顺序相关
|
||||||
PlayRepeat playRepeat = PlayRepeat.pause;
|
PlayRepeat playRepeat = PlayRepeat.pause;
|
||||||
|
|
||||||
RxList<SubTitileContentModel> subtitleContents =
|
|
||||||
<SubTitileContentModel>[].obs;
|
|
||||||
RxString subtitleContent = ''.obs;
|
|
||||||
|
|
||||||
void updateSliderPositionSecond() {
|
void updateSliderPositionSecond() {
|
||||||
int newSecond = _sliderPosition.value.inSeconds;
|
int newSecond = _sliderPosition.value.inSeconds;
|
||||||
if (sliderPositionSeconds.value != newSecond) {
|
if (sliderPositionSeconds.value != newSecond) {
|
||||||
@ -364,8 +350,6 @@ class PlPlayerController {
|
|||||||
bool enableHeart = true,
|
bool enableHeart = true,
|
||||||
// 是否首次加载
|
// 是否首次加载
|
||||||
bool isFirstTime = true,
|
bool isFirstTime = true,
|
||||||
// 是否开启字幕
|
|
||||||
bool enableSubTitle = false,
|
|
||||||
}) async {
|
}) async {
|
||||||
try {
|
try {
|
||||||
_autoPlay = autoplay;
|
_autoPlay = autoplay;
|
||||||
@ -380,9 +364,7 @@ class PlPlayerController {
|
|||||||
_cid = cid;
|
_cid = cid;
|
||||||
_enableHeart = enableHeart;
|
_enableHeart = enableHeart;
|
||||||
_isFirstTime = isFirstTime;
|
_isFirstTime = isFirstTime;
|
||||||
_subTitleOpen.value = enableSubTitle;
|
|
||||||
subtitles = [].obs;
|
|
||||||
subtitleContent.value = '';
|
|
||||||
if (_videoPlayerController != null &&
|
if (_videoPlayerController != null &&
|
||||||
_videoPlayerController!.state.playing) {
|
_videoPlayerController!.state.playing) {
|
||||||
await pause(notify: false);
|
await pause(notify: false);
|
||||||
@ -593,8 +575,6 @@ class PlPlayerController {
|
|||||||
_sliderPosition.value = event;
|
_sliderPosition.value = event;
|
||||||
updateSliderPositionSecond();
|
updateSliderPositionSecond();
|
||||||
}
|
}
|
||||||
querySubtitleContent(
|
|
||||||
videoPlayerController!.state.position.inSeconds.toDouble());
|
|
||||||
|
|
||||||
/// 触发回调事件
|
/// 触发回调事件
|
||||||
for (var element in _positionListeners) {
|
for (var element in _positionListeners) {
|
||||||
@ -629,10 +609,6 @@ class PlPlayerController {
|
|||||||
const Duration(seconds: 1),
|
const Duration(seconds: 1),
|
||||||
() => videoPlayerServiceHandler.onPositionChange(event));
|
() => videoPlayerServiceHandler.onPositionChange(event));
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// onSubTitleOpenChanged.listen((bool event) {
|
|
||||||
// toggleSubtitle(event ? subTitleCode.value : -1);
|
|
||||||
// })
|
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -1071,61 +1047,12 @@ class PlPlayerController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 字幕
|
|
||||||
void toggleSubtitle(int code) {
|
|
||||||
_subTitleOpen.value = code != -1;
|
|
||||||
_subTitleCode.value = code;
|
|
||||||
// if (code == -1) {
|
|
||||||
// // 关闭字幕
|
|
||||||
// _subTitleOpen.value = false;
|
|
||||||
// _subTitleCode.value = code;
|
|
||||||
// _videoPlayerController?.setSubtitleTrack(SubtitleTrack.no());
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
// final SubTitlteItemModel? subtitle = subtitles?.firstWhereOrNull(
|
|
||||||
// (element) => element.code == code,
|
|
||||||
// );
|
|
||||||
// _subTitleOpen.value = true;
|
|
||||||
// _subTitleCode.value = code;
|
|
||||||
// _videoPlayerController?.setSubtitleTrack(
|
|
||||||
// SubtitleTrack.data(
|
|
||||||
// subtitle!.content!,
|
|
||||||
// title: subtitle.title,
|
|
||||||
// language: subtitle.lan,
|
|
||||||
// ),
|
|
||||||
// );
|
|
||||||
}
|
|
||||||
|
|
||||||
void querySubtitleContent(double progress) {
|
|
||||||
if (subTitleCode.value == -1) {
|
|
||||||
subtitleContent.value = '';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (subtitles.isEmpty) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
final SubTitlteItemModel? subtitle = subtitles.firstWhereOrNull(
|
|
||||||
(element) => element.code == subTitleCode.value,
|
|
||||||
);
|
|
||||||
if (subtitle != null && subtitle.body!.isNotEmpty) {
|
|
||||||
for (var content in subtitle.body!) {
|
|
||||||
if (progress >= content['from']! && progress <= content['to']!) {
|
|
||||||
subtitleContent.value = content['content']!;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
setPlayRepeat(PlayRepeat type) {
|
setPlayRepeat(PlayRepeat type) {
|
||||||
playRepeat = type;
|
playRepeat = type;
|
||||||
videoStorage.put(VideoBoxKey.playRepeat, type.value);
|
videoStorage.put(VideoBoxKey.playRepeat, type.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> dispose({String type = 'single'}) async {
|
Future<void> dispose({String type = 'single'}) async {
|
||||||
print('dispose');
|
|
||||||
print('dispose: ${playerCount.value}');
|
|
||||||
|
|
||||||
// 每次减1,最后销毁
|
// 每次减1,最后销毁
|
||||||
if (type == 'single' && playerCount.value > 1) {
|
if (type == 'single' && playerCount.value > 1) {
|
||||||
_playerCount.value -= 1;
|
_playerCount.value -= 1;
|
||||||
@ -1135,7 +1062,6 @@ class PlPlayerController {
|
|||||||
}
|
}
|
||||||
_playerCount.value = 0;
|
_playerCount.value = 0;
|
||||||
try {
|
try {
|
||||||
print('dispose dispose ---------');
|
|
||||||
_timer?.cancel();
|
_timer?.cancel();
|
||||||
_timerForVolume?.cancel();
|
_timerForVolume?.cancel();
|
||||||
_timerForGettingVolume?.cancel();
|
_timerForGettingVolume?.cancel();
|
||||||
|
|||||||
@ -580,45 +580,6 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
|||||||
if (widget.danmuWidget != null)
|
if (widget.danmuWidget != null)
|
||||||
Positioned.fill(top: 4, child: widget.danmuWidget!),
|
Positioned.fill(top: 4, child: widget.danmuWidget!),
|
||||||
|
|
||||||
/// 开启且有字幕时展示
|
|
||||||
Stack(
|
|
||||||
children: [
|
|
||||||
Positioned(
|
|
||||||
left: 0,
|
|
||||||
right: 0,
|
|
||||||
bottom: 30,
|
|
||||||
child: Align(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: Obx(
|
|
||||||
() => Visibility(
|
|
||||||
visible: widget.controller.subTitleCode.value != -1,
|
|
||||||
child: Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
borderRadius: BorderRadius.circular(4),
|
|
||||||
color: widget.controller.subtitleContent.value != ''
|
|
||||||
? Colors.black.withOpacity(0.6)
|
|
||||||
: Colors.transparent,
|
|
||||||
),
|
|
||||||
padding: widget.controller.subTitleCode.value != -1
|
|
||||||
? const EdgeInsets.symmetric(
|
|
||||||
horizontal: 10,
|
|
||||||
vertical: 4,
|
|
||||||
)
|
|
||||||
: EdgeInsets.zero,
|
|
||||||
child: Text(
|
|
||||||
widget.controller.subtitleContent.value,
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 12,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
|
|
||||||
/// 手势
|
/// 手势
|
||||||
Positioned.fill(
|
Positioned.fill(
|
||||||
left: 16,
|
left: 16,
|
||||||
|
|||||||
@ -20,7 +20,7 @@ class PiliSchame {
|
|||||||
/// 完整链接进入 b23.无效
|
/// 完整链接进入 b23.无效
|
||||||
appScheme.getLatestScheme().then((SchemeEntity? value) {
|
appScheme.getLatestScheme().then((SchemeEntity? value) {
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
_fullPathPush(value);
|
_routePush(value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -37,7 +37,6 @@ class PiliSchame {
|
|||||||
final String scheme = value.scheme;
|
final String scheme = value.scheme;
|
||||||
final String host = value.host;
|
final String host = value.host;
|
||||||
final String path = value.path;
|
final String path = value.path;
|
||||||
|
|
||||||
if (scheme == 'bilibili') {
|
if (scheme == 'bilibili') {
|
||||||
if (host == 'root') {
|
if (host == 'root') {
|
||||||
Navigator.popUntil(
|
Navigator.popUntil(
|
||||||
@ -85,6 +84,14 @@ class PiliSchame {
|
|||||||
}
|
}
|
||||||
} else if (host == 'search') {
|
} else if (host == 'search') {
|
||||||
Get.toNamed('/searchResult', parameters: {'keyword': ''});
|
Get.toNamed('/searchResult', parameters: {'keyword': ''});
|
||||||
|
} else if (host == 'article') {
|
||||||
|
final String id = path.split('/').last.split('?').first;
|
||||||
|
Get.toNamed('/htmlRender', parameters: {
|
||||||
|
'url': 'https://www.bilibili.com/read/cv$id',
|
||||||
|
'title': 'cv$id',
|
||||||
|
'id': 'cv$id',
|
||||||
|
'dynamicType': 'read'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (scheme == 'https') {
|
if (scheme == 'https') {
|
||||||
@ -226,6 +233,13 @@ class PiliSchame {
|
|||||||
break;
|
break;
|
||||||
case 'read':
|
case 'read':
|
||||||
print('专栏');
|
print('专栏');
|
||||||
|
String id = 'cv${matchNum(query!['id']!).first}';
|
||||||
|
Get.toNamed('/htmlRender', parameters: {
|
||||||
|
'url': value.dataString!,
|
||||||
|
'title': '',
|
||||||
|
'id': id,
|
||||||
|
'dynamicType': 'read'
|
||||||
|
});
|
||||||
break;
|
break;
|
||||||
case 'space':
|
case 'space':
|
||||||
print('个人空间');
|
print('个人空间');
|
||||||
|
|||||||
@ -1,32 +0,0 @@
|
|||||||
class SubTitleUtils {
|
|
||||||
// 格式整理
|
|
||||||
static String convertToWebVTT(List jsonData) {
|
|
||||||
String webVTTContent = 'WEBVTT FILE\n\n';
|
|
||||||
|
|
||||||
for (int i = 0; i < jsonData.length; i++) {
|
|
||||||
final item = jsonData[i];
|
|
||||||
double from = item['from'] as double;
|
|
||||||
double to = item['to'] as double;
|
|
||||||
int sid = (item['sid'] ?? 0) as int;
|
|
||||||
String content = item['content'] as String;
|
|
||||||
|
|
||||||
webVTTContent += '$sid\n';
|
|
||||||
webVTTContent += '${formatTime(from)} --> ${formatTime(to)}\n';
|
|
||||||
webVTTContent += '$content\n\n';
|
|
||||||
}
|
|
||||||
|
|
||||||
return webVTTContent;
|
|
||||||
}
|
|
||||||
|
|
||||||
static String formatTime(num seconds) {
|
|
||||||
final String h = (seconds / 3600).floor().toString().padLeft(2, '0');
|
|
||||||
final String m = (seconds % 3600 / 60).floor().toString().padLeft(2, '0');
|
|
||||||
final String s = (seconds % 60).floor().toString().padLeft(2, '0');
|
|
||||||
final String ms =
|
|
||||||
(seconds * 1000 % 1000).floor().toString().padLeft(3, '0');
|
|
||||||
if (h == '00') {
|
|
||||||
return "$m:$s.$ms";
|
|
||||||
}
|
|
||||||
return "$h:$m:$s.$ms";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user