Compare commits
25 Commits
feature-ba
...
feature-fo
| Author | SHA1 | Date | |
|---|---|---|---|
| 569277572a | |||
| 19b84571c1 | |||
| 0812b8339e | |||
| b817a0c807 | |||
| 3da70d7e27 | |||
| 5e59db85be | |||
| 89026e671c | |||
| 1c8e7e53a5 | |||
| b264427be6 | |||
| d5134f972d | |||
| e2fd01a6d5 | |||
| 289cc99bc2 | |||
| 3d5ebe7e99 | |||
| d9964d37a4 | |||
| 5da39a9c52 | |||
| 44a162762c | |||
| d0f036ec35 | |||
| 10b928474b | |||
| 94f3b7c1e4 | |||
| fb8b2de115 | |||
| 0d5d33a365 | |||
| c39e91073b | |||
| d258474a5a | |||
| 083739e562 | |||
| 71ccb9c0e5 |
BIN
assets/images/live/default_bg.webp
Normal file
BIN
assets/images/live/default_bg.webp
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 16 KiB |
@ -214,6 +214,9 @@ class Api {
|
|||||||
// https://api.bilibili.com/x/relation/tags
|
// https://api.bilibili.com/x/relation/tags
|
||||||
static const String followingsClass = '/x/relation/tags';
|
static const String followingsClass = '/x/relation/tags';
|
||||||
|
|
||||||
|
// 搜索follow
|
||||||
|
static const followSearch = '/x/relation/followings/search';
|
||||||
|
|
||||||
// 粉丝
|
// 粉丝
|
||||||
// vmid 用户id pn 页码 ps 每页个数,最大50 order: desc
|
// vmid 用户id pn 页码 ps 每页个数,最大50 order: desc
|
||||||
// order_type 排序规则 最近访问传空,最常访问传 attention
|
// order_type 排序规则 最近访问传空,最常访问传 attention
|
||||||
@ -230,6 +233,10 @@ class Api {
|
|||||||
static const String liveRoomInfo =
|
static const String liveRoomInfo =
|
||||||
'${HttpString.liveBaseUrl}/xlive/web-room/v2/index/getRoomPlayInfo';
|
'${HttpString.liveBaseUrl}/xlive/web-room/v2/index/getRoomPlayInfo';
|
||||||
|
|
||||||
|
// 直播间详情 H5
|
||||||
|
static const String liveRoomInfoH5 =
|
||||||
|
'${HttpString.liveBaseUrl}/xlive/web-room/v1/index/getH5InfoByRoom';
|
||||||
|
|
||||||
// 用户信息 需要Wbi签名
|
// 用户信息 需要Wbi签名
|
||||||
// https://api.bilibili.com/x/space/wbi/acc/info?mid=503427686&token=&platform=web&web_location=1550101&w_rid=d709892496ce93e3d94d6d37c95bde91&wts=1689301482
|
// https://api.bilibili.com/x/space/wbi/acc/info?mid=503427686&token=&platform=web&web_location=1550101&w_rid=d709892496ce93e3d94d6d37c95bde91&wts=1689301482
|
||||||
static const String memberInfo = '/x/space/wbi/acc/info';
|
static const String memberInfo = '/x/space/wbi/acc/info';
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
import '../models/live/item.dart';
|
import '../models/live/item.dart';
|
||||||
import '../models/live/room_info.dart';
|
import '../models/live/room_info.dart';
|
||||||
|
import '../models/live/room_info_h5.dart';
|
||||||
import 'api.dart';
|
import 'api.dart';
|
||||||
import 'init.dart';
|
import 'init.dart';
|
||||||
|
|
||||||
@ -46,4 +47,22 @@ class LiveHttp {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Future liveRoomInfoH5({roomId, qn}) async {
|
||||||
|
var res = await Request().get(Api.liveRoomInfoH5, data: {
|
||||||
|
'room_id': roomId,
|
||||||
|
});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {
|
||||||
|
'status': true,
|
||||||
|
'data': RoomInfoH5Model.fromJson(res.data['data'])
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'data': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -461,4 +461,41 @@ class MemberHttp {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 搜索follow
|
||||||
|
static Future getfollowSearch({
|
||||||
|
required int mid,
|
||||||
|
required int ps,
|
||||||
|
required int pn,
|
||||||
|
required String name,
|
||||||
|
}) async {
|
||||||
|
Map<String, dynamic> data = {
|
||||||
|
'vmid': mid,
|
||||||
|
'pn': pn,
|
||||||
|
'ps': ps,
|
||||||
|
'order': 'desc',
|
||||||
|
'order_type': 'attention',
|
||||||
|
'gaia_source': 'main_web',
|
||||||
|
'name': name,
|
||||||
|
'web_location': 333.999,
|
||||||
|
};
|
||||||
|
Map params = await WbiSign().makSign(data);
|
||||||
|
var res = await Request().get(Api.followSearch, data: {
|
||||||
|
...data,
|
||||||
|
'w_rid': params['w_rid'],
|
||||||
|
'wts': params['wts'],
|
||||||
|
});
|
||||||
|
if (res.data['code'] == 0) {
|
||||||
|
return {
|
||||||
|
'status': true,
|
||||||
|
'data': FollowDataModel.fromJson(res.data['data'])
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'status': false,
|
||||||
|
'data': [],
|
||||||
|
'msg': res.data['message'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -51,7 +51,7 @@ class VideoHttp {
|
|||||||
(i['owner'] != null &&
|
(i['owner'] != null &&
|
||||||
!blackMidsList.contains(i['owner']['mid']))) {
|
!blackMidsList.contains(i['owner']['mid']))) {
|
||||||
RecVideoItemModel videoItem = RecVideoItemModel.fromJson(i);
|
RecVideoItemModel videoItem = RecVideoItemModel.fromJson(i);
|
||||||
if (!RecommendFilter.filter(videoItem)){
|
if (!RecommendFilter.filter(videoItem)) {
|
||||||
list.add(videoItem);
|
list.add(videoItem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -98,7 +98,7 @@ class VideoHttp {
|
|||||||
(i['args'] != null &&
|
(i['args'] != null &&
|
||||||
!blackMidsList.contains(i['args']['up_mid']))) {
|
!blackMidsList.contains(i['args']['up_mid']))) {
|
||||||
RecVideoItemAppModel videoItem = RecVideoItemAppModel.fromJson(i);
|
RecVideoItemAppModel videoItem = RecVideoItemAppModel.fromJson(i);
|
||||||
if (!RecommendFilter.filter(videoItem)){
|
if (!RecommendFilter.filter(videoItem)) {
|
||||||
list.add(videoItem);
|
list.add(videoItem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -217,7 +217,7 @@ class VideoHttp {
|
|||||||
List<HotVideoItemModel> list = [];
|
List<HotVideoItemModel> list = [];
|
||||||
for (var i in res.data['data']) {
|
for (var i in res.data['data']) {
|
||||||
HotVideoItemModel videoItem = HotVideoItemModel.fromJson(i);
|
HotVideoItemModel videoItem = HotVideoItemModel.fromJson(i);
|
||||||
if (!RecommendFilter.filter(videoItem, relatedVideos: true)){
|
if (!RecommendFilter.filter(videoItem, relatedVideos: true)) {
|
||||||
list.add(videoItem);
|
list.add(videoItem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -322,7 +322,7 @@ class VideoHttp {
|
|||||||
if (res.data['code'] == 0) {
|
if (res.data['code'] == 0) {
|
||||||
return {'status': true, 'data': res.data['data']};
|
return {'status': true, 'data': res.data['data']};
|
||||||
} else {
|
} else {
|
||||||
return {'status': false, 'data': []};
|
return {'status': false, 'data': [], 'msg': res.data['message']};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -25,7 +25,6 @@ import 'package:pilipala/utils/recommend_filter.dart';
|
|||||||
import 'package:catcher_2/catcher_2.dart';
|
import 'package:catcher_2/catcher_2.dart';
|
||||||
import './services/loggeer.dart';
|
import './services/loggeer.dart';
|
||||||
|
|
||||||
|
|
||||||
void main() async {
|
void main() async {
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
MediaKit.ensureInitialized();
|
MediaKit.ensureInitialized();
|
||||||
@ -63,7 +62,6 @@ void main() async {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
// 小白条、导航栏沉浸
|
// 小白条、导航栏沉浸
|
||||||
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
|
||||||
SystemChrome.setSystemUIOverlayStyle(const SystemUiOverlayStyle(
|
SystemChrome.setSystemUIOverlayStyle(const SystemUiOverlayStyle(
|
||||||
@ -72,7 +70,6 @@ void main() async {
|
|||||||
statusBarColor: Colors.transparent,
|
statusBarColor: Colors.transparent,
|
||||||
));
|
));
|
||||||
Data.init();
|
Data.init();
|
||||||
GStrorage.lazyInit();
|
|
||||||
PiliSchame.init();
|
PiliSchame.init();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
130
lib/models/live/room_info_h5.dart
Normal file
130
lib/models/live/room_info_h5.dart
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
class RoomInfoH5Model {
|
||||||
|
RoomInfoH5Model({
|
||||||
|
this.roomInfo,
|
||||||
|
this.anchorInfo,
|
||||||
|
this.isRoomFeed,
|
||||||
|
this.watchedShow,
|
||||||
|
this.likeInfoV3,
|
||||||
|
this.blockInfo,
|
||||||
|
});
|
||||||
|
|
||||||
|
RoomInfo? roomInfo;
|
||||||
|
AnchorInfo? anchorInfo;
|
||||||
|
int? isRoomFeed;
|
||||||
|
Map? watchedShow;
|
||||||
|
LikeInfoV3? likeInfoV3;
|
||||||
|
Map? blockInfo;
|
||||||
|
|
||||||
|
RoomInfoH5Model.fromJson(Map<String, dynamic> json) {
|
||||||
|
roomInfo = RoomInfo.fromJson(json['room_info']);
|
||||||
|
anchorInfo = AnchorInfo.fromJson(json['anchor_info']);
|
||||||
|
isRoomFeed = json['is_room_feed'];
|
||||||
|
watchedShow = json['watched_show'];
|
||||||
|
likeInfoV3 = LikeInfoV3.fromJson(json['like_info_v3']);
|
||||||
|
blockInfo = json['block_info'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RoomInfo {
|
||||||
|
RoomInfo({
|
||||||
|
this.uid,
|
||||||
|
this.roomId,
|
||||||
|
this.title,
|
||||||
|
this.cover,
|
||||||
|
this.description,
|
||||||
|
this.liveStatus,
|
||||||
|
this.liveStartTime,
|
||||||
|
this.areaId,
|
||||||
|
this.areaName,
|
||||||
|
this.parentAreaId,
|
||||||
|
this.parentAreaName,
|
||||||
|
this.online,
|
||||||
|
this.background,
|
||||||
|
this.appBackground,
|
||||||
|
this.liveId,
|
||||||
|
});
|
||||||
|
|
||||||
|
int? uid;
|
||||||
|
int? roomId;
|
||||||
|
String? title;
|
||||||
|
String? cover;
|
||||||
|
String? description;
|
||||||
|
int? liveStatus;
|
||||||
|
int? liveStartTime;
|
||||||
|
int? areaId;
|
||||||
|
String? areaName;
|
||||||
|
int? parentAreaId;
|
||||||
|
String? parentAreaName;
|
||||||
|
int? online;
|
||||||
|
String? background;
|
||||||
|
String? appBackground;
|
||||||
|
String? liveId;
|
||||||
|
|
||||||
|
RoomInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
uid = json['uid'];
|
||||||
|
roomId = json['room_id'];
|
||||||
|
title = json['title'];
|
||||||
|
cover = json['cover'];
|
||||||
|
description = json['description'];
|
||||||
|
liveStatus = json['liveS_satus'];
|
||||||
|
liveStartTime = json['live_start_time'];
|
||||||
|
areaId = json['area_id'];
|
||||||
|
areaName = json['area_name'];
|
||||||
|
parentAreaId = json['parent_area_id'];
|
||||||
|
parentAreaName = json['parent_area_name'];
|
||||||
|
online = json['online'];
|
||||||
|
background = json['background'];
|
||||||
|
appBackground = json['app_background'];
|
||||||
|
liveId = json['live_id'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class AnchorInfo {
|
||||||
|
AnchorInfo({
|
||||||
|
this.baseInfo,
|
||||||
|
this.relationInfo,
|
||||||
|
});
|
||||||
|
|
||||||
|
BaseInfo? baseInfo;
|
||||||
|
RelationInfo? relationInfo;
|
||||||
|
|
||||||
|
AnchorInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
baseInfo = BaseInfo.fromJson(json['base_info']);
|
||||||
|
relationInfo = RelationInfo.fromJson(json['relation_info']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class BaseInfo {
|
||||||
|
BaseInfo({
|
||||||
|
this.uname,
|
||||||
|
this.face,
|
||||||
|
});
|
||||||
|
|
||||||
|
String? uname;
|
||||||
|
String? face;
|
||||||
|
|
||||||
|
BaseInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
uname = json['uname'];
|
||||||
|
face = json['face'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RelationInfo {
|
||||||
|
RelationInfo({this.attention});
|
||||||
|
|
||||||
|
int? attention;
|
||||||
|
|
||||||
|
RelationInfo.fromJson(Map<String, dynamic> json) {
|
||||||
|
attention = json['attention'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class LikeInfoV3 {
|
||||||
|
LikeInfoV3({this.totalLikes});
|
||||||
|
|
||||||
|
int? totalLikes;
|
||||||
|
|
||||||
|
LikeInfoV3.fromJson(Map<String, dynamic> json) {
|
||||||
|
totalLikes = json['total_likes'];
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -7,6 +7,7 @@ import 'package:pilipala/http/index.dart';
|
|||||||
import 'package:pilipala/models/github/latest.dart';
|
import 'package:pilipala/models/github/latest.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
import '../../utils/cache_manage.dart';
|
||||||
|
|
||||||
class AboutPage extends StatefulWidget {
|
class AboutPage extends StatefulWidget {
|
||||||
const AboutPage({super.key});
|
const AboutPage({super.key});
|
||||||
@ -17,6 +18,19 @@ class AboutPage extends StatefulWidget {
|
|||||||
|
|
||||||
class _AboutPageState extends State<AboutPage> {
|
class _AboutPageState extends State<AboutPage> {
|
||||||
final AboutController _aboutController = Get.put(AboutController());
|
final AboutController _aboutController = Get.put(AboutController());
|
||||||
|
String cacheSize = '';
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
// 读取缓存占用
|
||||||
|
getCacheSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> getCacheSize() async {
|
||||||
|
final res = await CacheManage().loadApplicationCache();
|
||||||
|
setState(() => cacheSize = res);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@ -138,6 +152,17 @@ class _AboutPageState extends State<AboutPage> {
|
|||||||
title: const Text('错误日志'),
|
title: const Text('错误日志'),
|
||||||
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
onTap: () async {
|
||||||
|
var cleanStatus = await CacheManage().clearCacheAll();
|
||||||
|
if (cleanStatus) {
|
||||||
|
getCacheSize();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
title: const Text('清除缓存'),
|
||||||
|
subtitle: Text('图片及网络缓存 $cacheSize', style: subTitleStyle),
|
||||||
|
trailing: Icon(Icons.arrow_forward_ios, size: 16, color: outline),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@ -9,6 +9,7 @@ import 'package:pilipala/models/common/search_type.dart';
|
|||||||
import 'package:pilipala/utils/id_utils.dart';
|
import 'package:pilipala/utils/id_utils.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
|
import '../../../common/widgets/badge.dart';
|
||||||
|
|
||||||
// 收藏视频卡片 - 水平布局
|
// 收藏视频卡片 - 水平布局
|
||||||
class FavVideoCardH extends StatelessWidget {
|
class FavVideoCardH extends StatelessWidget {
|
||||||
@ -27,7 +28,9 @@ class FavVideoCardH extends StatelessWidget {
|
|||||||
onTap: () async {
|
onTap: () async {
|
||||||
// int? seasonId;
|
// int? seasonId;
|
||||||
String? epId;
|
String? epId;
|
||||||
if (videoItem.ogv != null && videoItem.ogv['type_name'] == '番剧') {
|
if (videoItem.ogv != null &&
|
||||||
|
(videoItem.ogv['type_name'] == '番剧' ||
|
||||||
|
videoItem.ogv['type_name'] == '国创')) {
|
||||||
videoItem.cid = await SearchHttp.ab2c(bvid: bvid);
|
videoItem.cid = await SearchHttp.ab2c(bvid: bvid);
|
||||||
// seasonId = videoItem.ogv['season_id'];
|
// seasonId = videoItem.ogv['season_id'];
|
||||||
epId = videoItem.epId;
|
epId = videoItem.epId;
|
||||||
@ -84,22 +87,21 @@ class FavVideoCardH extends StatelessWidget {
|
|||||||
height: maxHeight,
|
height: maxHeight,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Positioned(
|
PBadge(
|
||||||
right: 4,
|
text: Utils.timeFormat(videoItem.duration!),
|
||||||
bottom: 4,
|
right: 6.0,
|
||||||
child: Container(
|
bottom: 6.0,
|
||||||
padding: const EdgeInsets.symmetric(
|
type: 'gray',
|
||||||
vertical: 1, horizontal: 6),
|
),
|
||||||
decoration: BoxDecoration(
|
if (videoItem.ogv != null) ...[
|
||||||
borderRadius: BorderRadius.circular(4),
|
PBadge(
|
||||||
color: Colors.black54.withOpacity(0.4)),
|
text: videoItem.ogv['type_name'],
|
||||||
child: Text(
|
top: 6.0,
|
||||||
Utils.timeFormat(videoItem.duration!),
|
right: 6.0,
|
||||||
style: const TextStyle(
|
bottom: null,
|
||||||
fontSize: 11, color: Colors.white),
|
left: null,
|
||||||
),
|
|
||||||
),
|
),
|
||||||
)
|
],
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -128,86 +130,107 @@ class VideoContent extends StatelessWidget {
|
|||||||
return Expanded(
|
return Expanded(
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.fromLTRB(10, 2, 6, 0),
|
padding: const EdgeInsets.fromLTRB(10, 2, 6, 0),
|
||||||
child: Column(
|
child: Stack(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Column(
|
||||||
videoItem.title,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
textAlign: TextAlign.start,
|
|
||||||
style: const TextStyle(
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
letterSpacing: 0.3,
|
|
||||||
),
|
|
||||||
maxLines: 2,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
const Spacer(),
|
|
||||||
Text(
|
|
||||||
Utils.dateFormat(videoItem.ctime!),
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 11, color: Theme.of(context).colorScheme.outline),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
videoItem.owner.name,
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: Theme.of(context).textTheme.labelMedium!.fontSize,
|
|
||||||
color: Theme.of(context).colorScheme.outline,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Row(
|
|
||||||
children: [
|
children: [
|
||||||
StatView(
|
Text(
|
||||||
theme: 'gray',
|
videoItem.title,
|
||||||
view: videoItem.cntInfo['play'],
|
textAlign: TextAlign.start,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontWeight: FontWeight.w500,
|
||||||
|
letterSpacing: 0.3,
|
||||||
|
),
|
||||||
|
maxLines: 2,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
const SizedBox(width: 8),
|
if (videoItem.ogv != null) ...[
|
||||||
StatDanMu(theme: 'gray', danmu: videoItem.cntInfo['danmaku']),
|
Text(
|
||||||
const Spacer(),
|
videoItem.intro,
|
||||||
SizedBox(
|
style: TextStyle(
|
||||||
width: 26,
|
fontSize:
|
||||||
height: 26,
|
Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
child: IconButton(
|
|
||||||
style: ButtonStyle(
|
|
||||||
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
|
||||||
),
|
|
||||||
onPressed: () {
|
|
||||||
showDialog(
|
|
||||||
context: Get.context!,
|
|
||||||
builder: (context) {
|
|
||||||
return AlertDialog(
|
|
||||||
title: const Text('提示'),
|
|
||||||
content: const Text('要取消收藏吗?'),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Get.back(),
|
|
||||||
child: Text(
|
|
||||||
'取消',
|
|
||||||
style: TextStyle(
|
|
||||||
color: Theme.of(context)
|
|
||||||
.colorScheme
|
|
||||||
.outline),
|
|
||||||
)),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () async {
|
|
||||||
await callFn!();
|
|
||||||
Get.back();
|
|
||||||
},
|
|
||||||
child: const Text('确定取消'),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
Icons.clear_outlined,
|
|
||||||
color: Theme.of(context).colorScheme.outline,
|
color: Theme.of(context).colorScheme.outline,
|
||||||
size: 18,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
],
|
||||||
|
const Spacer(),
|
||||||
|
Text(
|
||||||
|
Utils.dateFormat(videoItem.favTime),
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 11,
|
||||||
|
color: Theme.of(context).colorScheme.outline),
|
||||||
|
),
|
||||||
|
if (videoItem.owner.name != '') ...[
|
||||||
|
Text(
|
||||||
|
videoItem.owner.name,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize:
|
||||||
|
Theme.of(context).textTheme.labelMedium!.fontSize,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 2),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
StatView(
|
||||||
|
theme: 'gray',
|
||||||
|
view: videoItem.cntInfo['play'],
|
||||||
|
),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
StatDanMu(
|
||||||
|
theme: 'gray', danmu: videoItem.cntInfo['danmaku']),
|
||||||
|
const Spacer(),
|
||||||
|
],
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
Positioned(
|
||||||
|
right: 0,
|
||||||
|
bottom: -4,
|
||||||
|
child: IconButton(
|
||||||
|
style: ButtonStyle(
|
||||||
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
showDialog(
|
||||||
|
context: Get.context!,
|
||||||
|
builder: (context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('提示'),
|
||||||
|
content: const Text('要取消收藏吗?'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Get.back(),
|
||||||
|
child: Text(
|
||||||
|
'取消',
|
||||||
|
style: TextStyle(
|
||||||
|
color:
|
||||||
|
Theme.of(context).colorScheme.outline),
|
||||||
|
)),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
await callFn!();
|
||||||
|
Get.back();
|
||||||
|
},
|
||||||
|
child: const Text('确定取消'),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
icon: Icon(
|
||||||
|
Icons.clear_outlined,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
size: 18,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@ -37,6 +37,29 @@ class _FollowPageState extends State<FollowPage> {
|
|||||||
: '${_followController.name}的关注',
|
: '${_followController.name}的关注',
|
||||||
style: Theme.of(context).textTheme.titleMedium,
|
style: Theme.of(context).textTheme.titleMedium,
|
||||||
),
|
),
|
||||||
|
actions: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: () => Get.toNamed('/followSearch?mid=$mid'),
|
||||||
|
icon: const Icon(Icons.search_outlined),
|
||||||
|
),
|
||||||
|
PopupMenuButton(
|
||||||
|
icon: const Icon(Icons.more_vert),
|
||||||
|
itemBuilder: (BuildContext context) => <PopupMenuEntry>[
|
||||||
|
PopupMenuItem(
|
||||||
|
onTap: () => Get.toNamed('/blackListPage'),
|
||||||
|
child: const Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
Icon(Icons.block, size: 19),
|
||||||
|
SizedBox(width: 10),
|
||||||
|
Text('黑名单管理'),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
body: Obx(
|
body: Obx(
|
||||||
() => !_followController.isOwner.value
|
() => !_followController.isOwner.value
|
||||||
@ -87,3 +110,22 @@ class _FollowPageState extends State<FollowPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _FakeAPI {
|
||||||
|
static const List<String> _kOptions = <String>[
|
||||||
|
'aardvark',
|
||||||
|
'bobcat',
|
||||||
|
'chameleon',
|
||||||
|
];
|
||||||
|
// Searches the options, but injects a fake "network" delay.
|
||||||
|
static Future<Iterable<String>> search(String query) async {
|
||||||
|
await Future<void>.delayed(
|
||||||
|
const Duration(seconds: 1)); // Fake 1 second delay.
|
||||||
|
if (query == '') {
|
||||||
|
return const Iterable<String>.empty();
|
||||||
|
}
|
||||||
|
return _kOptions.where((String option) {
|
||||||
|
return option.contains(query.toLowerCase());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -42,7 +42,7 @@ class FollowItem extends StatelessWidget {
|
|||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
dense: true,
|
dense: true,
|
||||||
trailing: ctr!.isOwner.value
|
trailing: ctr != null && ctr!.isOwner.value
|
||||||
? SizedBox(
|
? SizedBox(
|
||||||
height: 34,
|
height: 34,
|
||||||
child: TextButton(
|
child: TextButton(
|
||||||
|
|||||||
73
lib/pages/follow_search/controller.dart
Normal file
73
lib/pages/follow_search/controller.dart
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/http/member.dart';
|
||||||
|
|
||||||
|
import '../../models/follow/result.dart';
|
||||||
|
|
||||||
|
class FollowSearchController extends GetxController {
|
||||||
|
Rx<TextEditingController> controller = TextEditingController().obs;
|
||||||
|
final FocusNode searchFocusNode = FocusNode();
|
||||||
|
RxString searchKeyWord = ''.obs;
|
||||||
|
String hintText = '搜索';
|
||||||
|
RxString loadingStatus = 'init'.obs;
|
||||||
|
late int mid = 1;
|
||||||
|
RxString uname = ''.obs;
|
||||||
|
int ps = 20;
|
||||||
|
int pn = 1;
|
||||||
|
RxList<FollowItemModel> followList = <FollowItemModel>[].obs;
|
||||||
|
RxInt total = 0.obs;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onInit() {
|
||||||
|
super.onInit();
|
||||||
|
mid = int.parse(Get.parameters['mid']!);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 清空搜索
|
||||||
|
void onClear() {
|
||||||
|
if (searchKeyWord.value.isNotEmpty && controller.value.text != '') {
|
||||||
|
controller.value.clear();
|
||||||
|
searchKeyWord.value = '';
|
||||||
|
} else {
|
||||||
|
Get.back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void onChange(value) {
|
||||||
|
searchKeyWord.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 提交搜索内容
|
||||||
|
void submit() {
|
||||||
|
loadingStatus.value = 'loading';
|
||||||
|
searchFollow();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future searchFollow({type = 'init'}) async {
|
||||||
|
if (controller.value.text == '') {
|
||||||
|
return {'status': true, 'data': <FollowItemModel>[].obs};
|
||||||
|
}
|
||||||
|
if (type == 'init') {
|
||||||
|
ps = 1;
|
||||||
|
}
|
||||||
|
var res = await MemberHttp.getfollowSearch(
|
||||||
|
mid: mid,
|
||||||
|
ps: ps,
|
||||||
|
pn: pn,
|
||||||
|
name: controller.value.text,
|
||||||
|
);
|
||||||
|
if (res['status']) {
|
||||||
|
if (type == 'init') {
|
||||||
|
followList.value = res['data'].list;
|
||||||
|
} else {
|
||||||
|
followList.addAll(res['data'].list);
|
||||||
|
}
|
||||||
|
total.value = res['data'].total;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
void onLoad() {
|
||||||
|
searchFollow(type: 'onLoad');
|
||||||
|
}
|
||||||
|
}
|
||||||
4
lib/pages/follow_search/index.dart
Normal file
4
lib/pages/follow_search/index.dart
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
library follow_search;
|
||||||
|
|
||||||
|
export './controller.dart';
|
||||||
|
export './view.dart';
|
||||||
121
lib/pages/follow_search/view.dart
Normal file
121
lib/pages/follow_search/view.dart
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
import 'package:easy_debounce/easy_throttle.dart';
|
||||||
|
import 'package:flutter/cupertino.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
import 'package:pilipala/common/widgets/http_error.dart';
|
||||||
|
import 'package:pilipala/pages/follow_search/index.dart';
|
||||||
|
|
||||||
|
import '../follow/widgets/follow_item.dart';
|
||||||
|
|
||||||
|
class FollowSearchPage extends StatefulWidget {
|
||||||
|
const FollowSearchPage({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<FollowSearchPage> createState() => _FollowSearchPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _FollowSearchPageState extends State<FollowSearchPage> {
|
||||||
|
final FollowSearchController _followSearchController =
|
||||||
|
Get.put(FollowSearchController());
|
||||||
|
late Future? _futureBuilder;
|
||||||
|
final ScrollController scrollController = ScrollController();
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_futureBuilder = _followSearchController.searchFollow();
|
||||||
|
scrollController.addListener(
|
||||||
|
() {
|
||||||
|
if (scrollController.position.pixels >=
|
||||||
|
scrollController.position.maxScrollExtent - 200) {
|
||||||
|
EasyThrottle.throttle(
|
||||||
|
'my-throttler', const Duration(milliseconds: 500), () {
|
||||||
|
_followSearchController.onLoad();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void reRequest() {
|
||||||
|
setState(() {
|
||||||
|
_futureBuilder = _followSearchController.searchFollow();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
titleSpacing: 0,
|
||||||
|
actions: [
|
||||||
|
IconButton(
|
||||||
|
onPressed: reRequest,
|
||||||
|
icon: const Icon(CupertinoIcons.search, size: 22),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 6),
|
||||||
|
],
|
||||||
|
title: TextField(
|
||||||
|
autofocus: true,
|
||||||
|
focusNode: _followSearchController.searchFocusNode,
|
||||||
|
controller: _followSearchController.controller.value,
|
||||||
|
textInputAction: TextInputAction.search,
|
||||||
|
onChanged: (value) => _followSearchController.onChange(value),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
hintText: _followSearchController.hintText,
|
||||||
|
border: InputBorder.none,
|
||||||
|
suffixIcon: IconButton(
|
||||||
|
icon: Icon(
|
||||||
|
Icons.clear,
|
||||||
|
size: 22,
|
||||||
|
color: Theme.of(context).colorScheme.outline,
|
||||||
|
),
|
||||||
|
onPressed: () => _followSearchController.onClear(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSubmitted: (String value) => reRequest(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: FutureBuilder(
|
||||||
|
future: _futureBuilder,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
|
var data = snapshot.data;
|
||||||
|
if (data == null) {
|
||||||
|
return CustomScrollView(
|
||||||
|
slivers: [
|
||||||
|
HttpError(errMsg: snapshot.data['msg'], fn: reRequest)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (data['status']) {
|
||||||
|
RxList followList = _followSearchController.followList;
|
||||||
|
return Obx(
|
||||||
|
() => followList.isNotEmpty
|
||||||
|
? ListView.builder(
|
||||||
|
controller: scrollController,
|
||||||
|
itemCount: followList.length,
|
||||||
|
itemBuilder: ((context, index) {
|
||||||
|
return FollowItem(
|
||||||
|
item: followList[index],
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
: CustomScrollView(
|
||||||
|
slivers: [HttpError(errMsg: '未搜索到结果', fn: reRequest)],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return CustomScrollView(
|
||||||
|
slivers: [
|
||||||
|
HttpError(errMsg: snapshot.data['msg'], fn: reRequest)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -184,18 +184,32 @@ class VideoStat extends StatelessWidget {
|
|||||||
tileMode: TileMode.mirror,
|
tileMode: TileMode.mirror,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
child: RichText(
|
child: Row(
|
||||||
maxLines: 1,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
textAlign: TextAlign.justify,
|
children: [
|
||||||
softWrap: false,
|
Text(
|
||||||
text: TextSpan(
|
liveItem!.areaName!,
|
||||||
style: const TextStyle(fontSize: 11, color: Colors.white),
|
style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
children: [
|
),
|
||||||
TextSpan(text: liveItem!.areaName!),
|
Text(
|
||||||
TextSpan(text: liveItem!.watchedShow!['text_small']),
|
liveItem!.watchedShow!['text_small'],
|
||||||
],
|
style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
),
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// child: RichText(
|
||||||
|
// maxLines: 1,
|
||||||
|
// textAlign: TextAlign.justify,
|
||||||
|
// softWrap: false,
|
||||||
|
// text: TextSpan(
|
||||||
|
// style: const TextStyle(fontSize: 11, color: Colors.white),
|
||||||
|
// children: [
|
||||||
|
// TextSpan(text: liveItem!.areaName!),
|
||||||
|
// TextSpan(text: liveItem!.watchedShow!['text_small']),
|
||||||
|
// ],
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -4,6 +4,8 @@ import 'package:pilipala/http/live.dart';
|
|||||||
import 'package:pilipala/models/live/room_info.dart';
|
import 'package:pilipala/models/live/room_info.dart';
|
||||||
import 'package:pilipala/plugin/pl_player/index.dart';
|
import 'package:pilipala/plugin/pl_player/index.dart';
|
||||||
|
|
||||||
|
import '../../models/live/room_info_h5.dart';
|
||||||
|
|
||||||
class LiveRoomController extends GetxController {
|
class LiveRoomController extends GetxController {
|
||||||
String cover = '';
|
String cover = '';
|
||||||
late int roomId;
|
late int roomId;
|
||||||
@ -21,6 +23,7 @@ class LiveRoomController extends GetxController {
|
|||||||
// controlsStyle: ControlsStyle.live,
|
// controlsStyle: ControlsStyle.live,
|
||||||
// enabledButtons: const EnabledButtons(pip: true),
|
// enabledButtons: const EnabledButtons(pip: true),
|
||||||
// );
|
// );
|
||||||
|
Rx<RoomInfoH5Model> roomInfoH5 = RoomInfoH5Model().obs;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -37,10 +40,11 @@ class LiveRoomController extends GetxController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
queryLiveInfo();
|
queryLiveInfo();
|
||||||
|
queryLiveInfoH5();
|
||||||
}
|
}
|
||||||
|
|
||||||
playerInit(source) {
|
playerInit(source) async {
|
||||||
plPlayerController.setDataSource(
|
await plPlayerController.setDataSource(
|
||||||
DataSource(
|
DataSource(
|
||||||
videoSource: source,
|
videoSource: source,
|
||||||
audioSource: null,
|
audioSource: null,
|
||||||
@ -66,7 +70,8 @@ class LiveRoomController extends GetxController {
|
|||||||
String videoUrl = (item.urlInfo?.first.host)! +
|
String videoUrl = (item.urlInfo?.first.host)! +
|
||||||
item.baseUrl! +
|
item.baseUrl! +
|
||||||
item.urlInfo!.first.extra!;
|
item.urlInfo!.first.extra!;
|
||||||
playerInit(videoUrl);
|
await playerInit(videoUrl);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -80,4 +85,12 @@ class LiveRoomController extends GetxController {
|
|||||||
volumeOff.value = true;
|
volumeOff.value = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future queryLiveInfoH5() async {
|
||||||
|
var res = await LiveHttp.liveRoomInfoH5(roomId: roomId);
|
||||||
|
if (res['status']) {
|
||||||
|
roomInfoH5.value = res['data'];
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -19,6 +19,8 @@ class LiveRoomPage extends StatefulWidget {
|
|||||||
class _LiveRoomPageState extends State<LiveRoomPage> {
|
class _LiveRoomPageState extends State<LiveRoomPage> {
|
||||||
final LiveRoomController _liveRoomController = Get.put(LiveRoomController());
|
final LiveRoomController _liveRoomController = Get.put(LiveRoomController());
|
||||||
PlPlayerController? plPlayerController;
|
PlPlayerController? plPlayerController;
|
||||||
|
late Future? _futureBuilder;
|
||||||
|
late Future? _futureBuilderFuture;
|
||||||
|
|
||||||
bool isShowCover = true;
|
bool isShowCover = true;
|
||||||
bool isPlay = true;
|
bool isPlay = true;
|
||||||
@ -39,6 +41,8 @@ class _LiveRoomPageState extends State<LiveRoomPage> {
|
|||||||
if (Platform.isAndroid) {
|
if (Platform.isAndroid) {
|
||||||
floating = Floating();
|
floating = Floating();
|
||||||
}
|
}
|
||||||
|
_futureBuilder = _liveRoomController.queryLiveInfoH5();
|
||||||
|
_futureBuilderFuture = _liveRoomController.queryLiveInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -52,57 +56,123 @@ class _LiveRoomPageState extends State<LiveRoomPage> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
Widget videoPlayerPanel = FutureBuilder(
|
||||||
|
future: _futureBuilderFuture,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.hasData && snapshot.data['status']) {
|
||||||
|
return PLVideoPlayer(
|
||||||
|
controller: plPlayerController!,
|
||||||
|
bottomControl: BottomControl(
|
||||||
|
controller: plPlayerController,
|
||||||
|
liveRoomCtr: _liveRoomController,
|
||||||
|
floating: floating,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
Widget childWhenDisabled = Scaffold(
|
Widget childWhenDisabled = Scaffold(
|
||||||
primary: true,
|
primary: true,
|
||||||
appBar: PreferredSize(
|
backgroundColor: Colors.black,
|
||||||
preferredSize: Size.fromHeight(
|
body: Stack(
|
||||||
MediaQuery.of(context).orientation == Orientation.portrait ? 56 : 0,
|
|
||||||
),
|
|
||||||
child: AppBar(
|
|
||||||
centerTitle: false,
|
|
||||||
titleSpacing: 0,
|
|
||||||
title: _liveRoomController.liveItem != null
|
|
||||||
? Row(
|
|
||||||
children: [
|
|
||||||
NetworkImgLayer(
|
|
||||||
width: 34,
|
|
||||||
height: 34,
|
|
||||||
type: 'avatar',
|
|
||||||
src: _liveRoomController.liveItem.face,
|
|
||||||
),
|
|
||||||
const SizedBox(width: 10),
|
|
||||||
Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
_liveRoomController.liveItem.uname,
|
|
||||||
style: const TextStyle(fontSize: 14),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 1),
|
|
||||||
if (_liveRoomController.liveItem.watchedShow != null)
|
|
||||||
Text(
|
|
||||||
_liveRoomController
|
|
||||||
.liveItem.watchedShow['text_large'] ??
|
|
||||||
'',
|
|
||||||
style: const TextStyle(fontSize: 12)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
// actions: [
|
|
||||||
// SizedBox(
|
|
||||||
// height: 34,
|
|
||||||
// child: ElevatedButton(onPressed: () {}, child: const Text('关注')),
|
|
||||||
// ),
|
|
||||||
// const SizedBox(width: 12),
|
|
||||||
// ],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
body: Column(
|
|
||||||
children: [
|
children: [
|
||||||
Stack(
|
// Obx(
|
||||||
|
// () => Positioned.fill(
|
||||||
|
// child: Opacity(
|
||||||
|
// opacity: 0.8,
|
||||||
|
// child: _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground !=
|
||||||
|
// '' &&
|
||||||
|
// _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground !=
|
||||||
|
// null
|
||||||
|
// ? NetworkImgLayer(
|
||||||
|
// width: Get.width,
|
||||||
|
// height: Get.height,
|
||||||
|
// src: _liveRoomController
|
||||||
|
// .roomInfoH5.value.roomInfo?.appBackground ??
|
||||||
|
// '',
|
||||||
|
// )
|
||||||
|
// : Image.asset(
|
||||||
|
// 'assets/images/live/default_bg.webp',
|
||||||
|
// width: Get.width,
|
||||||
|
// height: Get.height,
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
Positioned.fill(
|
||||||
|
child: Opacity(
|
||||||
|
opacity: 0.8,
|
||||||
|
child: Image.asset(
|
||||||
|
'assets/images/live/default_bg.webp',
|
||||||
|
width: Get.width,
|
||||||
|
height: Get.height,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Column(
|
||||||
children: [
|
children: [
|
||||||
|
AppBar(
|
||||||
|
centerTitle: false,
|
||||||
|
titleSpacing: 0,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
foregroundColor: Colors.white,
|
||||||
|
toolbarHeight:
|
||||||
|
MediaQuery.of(context).orientation == Orientation.portrait
|
||||||
|
? 56
|
||||||
|
: 0,
|
||||||
|
title: FutureBuilder(
|
||||||
|
future: _futureBuilder,
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
if (snapshot.data == null) {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
Map data = snapshot.data as Map;
|
||||||
|
if (data['status']) {
|
||||||
|
return Obx(
|
||||||
|
() => Row(
|
||||||
|
children: [
|
||||||
|
NetworkImgLayer(
|
||||||
|
width: 34,
|
||||||
|
height: 34,
|
||||||
|
type: 'avatar',
|
||||||
|
src: _liveRoomController
|
||||||
|
.roomInfoH5.value.anchorInfo!.baseInfo!.face,
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
_liveRoomController.roomInfoH5.value
|
||||||
|
.anchorInfo!.baseInfo!.uname!,
|
||||||
|
style: const TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 1),
|
||||||
|
if (_liveRoomController
|
||||||
|
.roomInfoH5.value.watchedShow !=
|
||||||
|
null)
|
||||||
|
Text(
|
||||||
|
_liveRoomController.roomInfoH5.value
|
||||||
|
.watchedShow!['text_large'] ??
|
||||||
|
'',
|
||||||
|
style: const TextStyle(fontSize: 12),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
PopScope(
|
PopScope(
|
||||||
canPop: plPlayerController?.isFullScreen.value != true,
|
canPop: plPlayerController?.isFullScreen.value != true,
|
||||||
onPopInvoked: (bool didPop) {
|
onPopInvoked: (bool didPop) {
|
||||||
@ -120,55 +190,19 @@ class _LiveRoomPageState extends State<LiveRoomPage> {
|
|||||||
Orientation.landscape
|
Orientation.landscape
|
||||||
? Get.size.height
|
? Get.size.height
|
||||||
: Get.size.width * 9 / 16,
|
: Get.size.width * 9 / 16,
|
||||||
child: plPlayerController!.videoPlayerController != null
|
child: videoPlayerPanel,
|
||||||
? PLVideoPlayer(
|
|
||||||
controller: plPlayerController!,
|
|
||||||
bottomControl: BottomControl(
|
|
||||||
controller: plPlayerController,
|
|
||||||
liveRoomCtr: _liveRoomController,
|
|
||||||
floating: floating,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
// if (_liveRoomController.liveItem != null &&
|
|
||||||
// _liveRoomController.liveItem.cover != null)
|
|
||||||
// Visibility(
|
|
||||||
// visible: isShowCover,
|
|
||||||
// child: Positioned(
|
|
||||||
// top: 0,
|
|
||||||
// left: 0,
|
|
||||||
// right: 0,
|
|
||||||
// child: NetworkImgLayer(
|
|
||||||
// type: 'emote',
|
|
||||||
// src: _liveRoomController.liveItem.cover,
|
|
||||||
// width: Get.size.width,
|
|
||||||
// height: videoHeight,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
Widget childWhenEnabled = AspectRatio(
|
|
||||||
aspectRatio: 16 / 9,
|
|
||||||
child: plPlayerController!.videoPlayerController != null
|
|
||||||
? PLVideoPlayer(
|
|
||||||
controller: plPlayerController!,
|
|
||||||
bottomControl: BottomControl(
|
|
||||||
controller: plPlayerController,
|
|
||||||
liveRoomCtr: _liveRoomController,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: const SizedBox(),
|
|
||||||
);
|
|
||||||
if (Platform.isAndroid) {
|
if (Platform.isAndroid) {
|
||||||
return PiPSwitcher(
|
return PiPSwitcher(
|
||||||
childWhenDisabled: childWhenDisabled,
|
childWhenDisabled: childWhenDisabled,
|
||||||
childWhenEnabled: childWhenEnabled,
|
childWhenEnabled: videoPlayerPanel,
|
||||||
|
floating: floating,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
return childWhenDisabled;
|
return childWhenDisabled;
|
||||||
|
|||||||
@ -25,7 +25,7 @@ class MemberArchiveController extends GetxController {
|
|||||||
|
|
||||||
// 获取用户投稿
|
// 获取用户投稿
|
||||||
Future getMemberArchive(type) async {
|
Future getMemberArchive(type) async {
|
||||||
if (type == 'onRefresh') {
|
if (type == 'init') {
|
||||||
pn = 1;
|
pn = 1;
|
||||||
}
|
}
|
||||||
var res = await MemberHttp.memberArchive(
|
var res = await MemberHttp.memberArchive(
|
||||||
@ -34,7 +34,12 @@ class MemberArchiveController extends GetxController {
|
|||||||
order: currentOrder['type']!,
|
order: currentOrder['type']!,
|
||||||
);
|
);
|
||||||
if (res['status']) {
|
if (res['status']) {
|
||||||
archivesList.addAll(res['data'].list.vlist);
|
if (type == 'init') {
|
||||||
|
archivesList.value = res['data'].list.vlist;
|
||||||
|
}
|
||||||
|
if (type == 'onLoad') {
|
||||||
|
archivesList.addAll(res['data'].list.vlist);
|
||||||
|
}
|
||||||
count = res['data'].page['count'];
|
count = res['data'].page['count'];
|
||||||
pn += 1;
|
pn += 1;
|
||||||
}
|
}
|
||||||
@ -42,13 +47,14 @@ class MemberArchiveController extends GetxController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
toggleSort() async {
|
toggleSort() async {
|
||||||
pn = 1;
|
List<String> typeList = orderList.map((e) => e['type']!).toList();
|
||||||
int index = orderList.indexOf(currentOrder);
|
int index = typeList.indexOf(currentOrder['type']!);
|
||||||
if (index == orderList.length - 1) {
|
if (index == orderList.length - 1) {
|
||||||
currentOrder.value = orderList.first;
|
currentOrder.value = orderList.first;
|
||||||
} else {
|
} else {
|
||||||
currentOrder.value = orderList[index + 1];
|
currentOrder.value = orderList[index + 1];
|
||||||
}
|
}
|
||||||
|
getMemberArchive('init');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 上拉加载
|
// 上拉加载
|
||||||
|
|||||||
@ -25,8 +25,7 @@ class _MemberArchivePageState extends State<MemberArchivePage> {
|
|||||||
final String heroTag = Utils.makeHeroTag(mid);
|
final String heroTag = Utils.makeHeroTag(mid);
|
||||||
_memberArchivesController =
|
_memberArchivesController =
|
||||||
Get.put(MemberArchiveController(), tag: heroTag);
|
Get.put(MemberArchiveController(), tag: heroTag);
|
||||||
_futureBuilderFuture =
|
_futureBuilderFuture = _memberArchivesController.getMemberArchive('init');
|
||||||
_memberArchivesController.getMemberArchive('onRefresh');
|
|
||||||
scrollController = _memberArchivesController.scrollController;
|
scrollController = _memberArchivesController.scrollController;
|
||||||
scrollController.addListener(
|
scrollController.addListener(
|
||||||
() {
|
() {
|
||||||
@ -48,39 +47,16 @@ class _MemberArchivePageState extends State<MemberArchivePage> {
|
|||||||
titleSpacing: 0,
|
titleSpacing: 0,
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
title: Text('他的投稿', style: Theme.of(context).textTheme.titleMedium),
|
title: Text('他的投稿', style: Theme.of(context).textTheme.titleMedium),
|
||||||
// actions: [
|
actions: [
|
||||||
// Obx(
|
Obx(
|
||||||
// () => PopupMenuButton<String>(
|
() => TextButton.icon(
|
||||||
// padding: EdgeInsets.zero,
|
icon: const Icon(Icons.sort, size: 20),
|
||||||
// tooltip: '投稿排序',
|
onPressed: _memberArchivesController.toggleSort,
|
||||||
// icon: Icon(
|
label: Text(_memberArchivesController.currentOrder['label']!),
|
||||||
// Icons.more_vert_outlined,
|
),
|
||||||
// color: Theme.of(context).colorScheme.outline,
|
),
|
||||||
// ),
|
const SizedBox(width: 6),
|
||||||
// position: PopupMenuPosition.under,
|
],
|
||||||
// onSelected: (String type) {},
|
|
||||||
// itemBuilder: (BuildContext context) => <PopupMenuEntry<String>>[
|
|
||||||
// for (var i in _memberArchivesController.orderList) ...[
|
|
||||||
// PopupMenuItem<String>(
|
|
||||||
// onTap: () {},
|
|
||||||
// value: _memberArchivesController.currentOrder['label'],
|
|
||||||
// child: Row(
|
|
||||||
// mainAxisSize: MainAxisSize.min,
|
|
||||||
// children: [
|
|
||||||
// Text(i['label']!),
|
|
||||||
// if (_memberArchivesController.currentOrder['label'] ==
|
|
||||||
// i['label']) ...[
|
|
||||||
// const SizedBox(width: 10),
|
|
||||||
// const Icon(Icons.done, size: 20),
|
|
||||||
// ],
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ]
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
// ],
|
|
||||||
),
|
),
|
||||||
body: CustomScrollView(
|
body: CustomScrollView(
|
||||||
controller: _memberArchivesController.scrollController,
|
controller: _memberArchivesController.scrollController,
|
||||||
|
|||||||
@ -119,7 +119,7 @@ class MineController extends GetxController {
|
|||||||
SmartDialog.showToast('账号未登录');
|
SmartDialog.showToast('账号未登录');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Get.toNamed('/follow?mid=${userInfo.value.mid}');
|
Get.toNamed('/follow?mid=${userInfo.value.mid}', preventDuplicates: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
pushFans() {
|
pushFans() {
|
||||||
@ -127,7 +127,7 @@ class MineController extends GetxController {
|
|||||||
SmartDialog.showToast('账号未登录');
|
SmartDialog.showToast('账号未登录');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Get.toNamed('/fan?mid=${userInfo.value.mid}');
|
Get.toNamed('/fan?mid=${userInfo.value.mid}', preventDuplicates: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
pushDynamic() {
|
pushDynamic() {
|
||||||
@ -135,6 +135,7 @@ class MineController extends GetxController {
|
|||||||
SmartDialog.showToast('账号未登录');
|
SmartDialog.showToast('账号未登录');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Get.toNamed('/memberDynamics?mid=${userInfo.value.mid}');
|
Get.toNamed('/memberDynamics?mid=${userInfo.value.mid}',
|
||||||
|
preventDuplicates: false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -19,6 +19,7 @@ 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 '../../../utils/id_utils.dart';
|
||||||
import 'widgets/header_control.dart';
|
import 'widgets/header_control.dart';
|
||||||
|
|
||||||
class VideoDetailController extends GetxController
|
class VideoDetailController extends GetxController
|
||||||
@ -61,7 +62,7 @@ class VideoDetailController extends GetxController
|
|||||||
Box localCache = GStrorage.localCache;
|
Box localCache = GStrorage.localCache;
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
|
|
||||||
int oid = 0;
|
RxInt oid = 0.obs;
|
||||||
// 评论id 请求楼中楼评论使用
|
// 评论id 请求楼中楼评论使用
|
||||||
int fRpid = 0;
|
int fRpid = 0;
|
||||||
|
|
||||||
@ -135,13 +136,14 @@ class VideoDetailController extends GetxController
|
|||||||
defaultValue: VideoDecodeFormats.values.last.code);
|
defaultValue: VideoDecodeFormats.values.last.code);
|
||||||
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']!);
|
||||||
}
|
}
|
||||||
|
|
||||||
showReplyReplyPanel() {
|
showReplyReplyPanel() {
|
||||||
PersistentBottomSheetController? ctr =
|
PersistentBottomSheetController? ctr =
|
||||||
scaffoldKey.currentState?.showBottomSheet((BuildContext context) {
|
scaffoldKey.currentState?.showBottomSheet((BuildContext context) {
|
||||||
return VideoReplyReplyPanel(
|
return VideoReplyReplyPanel(
|
||||||
oid: oid,
|
oid: oid.value,
|
||||||
rpid: fRpid,
|
rpid: fRpid,
|
||||||
closePanel: () => {
|
closePanel: () => {
|
||||||
fRpid = 0,
|
fRpid = 0,
|
||||||
|
|||||||
@ -298,7 +298,6 @@ class VideoIntroController extends GetxController {
|
|||||||
await queryVideoInFolder();
|
await queryVideoInFolder();
|
||||||
int defaultFolderId = favFolderData.value.list!.first.id!;
|
int defaultFolderId = favFolderData.value.list!.first.id!;
|
||||||
int favStatus = favFolderData.value.list!.first.favState!;
|
int favStatus = favFolderData.value.list!.first.favState!;
|
||||||
print('favStatus: $favStatus');
|
|
||||||
var result = await VideoHttp.favVideo(
|
var result = await VideoHttp.favVideo(
|
||||||
aid: IdUtils.bv2av(bvid),
|
aid: IdUtils.bv2av(bvid),
|
||||||
addIds: favStatus == 0 ? '$defaultFolderId' : '',
|
addIds: favStatus == 0 ? '$defaultFolderId' : '',
|
||||||
@ -310,6 +309,8 @@ class VideoIntroController extends GetxController {
|
|||||||
await queryHasFavVideo();
|
await queryHasFavVideo();
|
||||||
SmartDialog.showToast('✅ 操作成功');
|
SmartDialog.showToast('✅ 操作成功');
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
SmartDialog.showToast(result['msg']);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -340,6 +341,8 @@ class VideoIntroController extends GetxController {
|
|||||||
await queryHasFavVideo();
|
await queryHasFavVideo();
|
||||||
SmartDialog.showToast('✅ 操作成功');
|
SmartDialog.showToast('✅ 操作成功');
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
SmartDialog.showToast(result['msg']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -476,6 +479,7 @@ class VideoIntroController extends GetxController {
|
|||||||
final VideoDetailController videoDetailCtr =
|
final VideoDetailController videoDetailCtr =
|
||||||
Get.find<VideoDetailController>(tag: heroTag);
|
Get.find<VideoDetailController>(tag: heroTag);
|
||||||
videoDetailCtr.bvid = bvid;
|
videoDetailCtr.bvid = bvid;
|
||||||
|
videoDetailCtr.oid.value = aid;
|
||||||
videoDetailCtr.cid.value = cid;
|
videoDetailCtr.cid.value = cid;
|
||||||
videoDetailCtr.danmakuCid.value = cid;
|
videoDetailCtr.danmakuCid.value = cid;
|
||||||
videoDetailCtr.queryVideoUrl();
|
videoDetailCtr.queryVideoUrl();
|
||||||
|
|||||||
@ -53,9 +53,13 @@ class VideoReplyController extends GetxController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future queryReplyList({type = 'init'}) async {
|
Future queryReplyList({type = 'init'}) async {
|
||||||
|
if (isLoadingMore) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
isLoadingMore = true;
|
isLoadingMore = true;
|
||||||
if (type == 'init') {
|
if (type == 'init') {
|
||||||
currentPage = 0;
|
currentPage = 0;
|
||||||
|
noMore.value = '';
|
||||||
}
|
}
|
||||||
if (noMore.value == '没有更多了') {
|
if (noMore.value == '没有更多了') {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@ -16,11 +16,13 @@ import 'widgets/reply_item.dart';
|
|||||||
|
|
||||||
class VideoReplyPanel extends StatefulWidget {
|
class VideoReplyPanel extends StatefulWidget {
|
||||||
final String? bvid;
|
final String? bvid;
|
||||||
|
final int? oid;
|
||||||
final int rpid;
|
final int rpid;
|
||||||
final String? replyLevel;
|
final String? replyLevel;
|
||||||
|
|
||||||
const VideoReplyPanel({
|
const VideoReplyPanel({
|
||||||
this.bvid,
|
this.bvid,
|
||||||
|
this.oid,
|
||||||
this.rpid = 0,
|
this.rpid = 0,
|
||||||
this.replyLevel,
|
this.replyLevel,
|
||||||
super.key,
|
super.key,
|
||||||
@ -48,16 +50,17 @@ class _VideoReplyPanelState extends State<VideoReplyPanel>
|
|||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
int oid = widget.bvid != null ? IdUtils.bv2av(widget.bvid!) : 0;
|
// int oid = widget.bvid != null ? IdUtils.bv2av(widget.bvid!) : 0;
|
||||||
heroTag = Get.arguments['heroTag'];
|
heroTag = Get.arguments['heroTag'];
|
||||||
replyLevel = widget.replyLevel ?? '1';
|
replyLevel = widget.replyLevel ?? '1';
|
||||||
if (replyLevel == '2') {
|
if (replyLevel == '2') {
|
||||||
_videoReplyController = Get.put(
|
_videoReplyController = Get.put(
|
||||||
VideoReplyController(oid, widget.rpid.toString(), replyLevel),
|
VideoReplyController(widget.oid, widget.rpid.toString(), replyLevel),
|
||||||
tag: widget.rpid.toString());
|
tag: widget.rpid.toString());
|
||||||
} else {
|
} else {
|
||||||
_videoReplyController =
|
_videoReplyController = Get.put(
|
||||||
Get.put(VideoReplyController(oid, '', replyLevel), tag: heroTag);
|
VideoReplyController(widget.oid, '', replyLevel),
|
||||||
|
tag: heroTag);
|
||||||
}
|
}
|
||||||
|
|
||||||
fabAnimationCtr = AnimationController(
|
fabAnimationCtr = AnimationController(
|
||||||
@ -75,7 +78,8 @@ class _VideoReplyPanelState extends State<VideoReplyPanel>
|
|||||||
() {
|
() {
|
||||||
if (scrollController.position.pixels >=
|
if (scrollController.position.pixels >=
|
||||||
scrollController.position.maxScrollExtent - 300) {
|
scrollController.position.maxScrollExtent - 300) {
|
||||||
EasyThrottle.throttle('replylist', const Duration(seconds: 2), () {
|
EasyThrottle.throttle('replylist', const Duration(milliseconds: 200),
|
||||||
|
() {
|
||||||
_videoReplyController.onLoad();
|
_videoReplyController.onLoad();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -110,7 +114,7 @@ class _VideoReplyPanelState extends State<VideoReplyPanel>
|
|||||||
final VideoDetailController videoDetailCtr =
|
final VideoDetailController videoDetailCtr =
|
||||||
Get.find<VideoDetailController>(tag: heroTag);
|
Get.find<VideoDetailController>(tag: heroTag);
|
||||||
if (replyItem != null) {
|
if (replyItem != null) {
|
||||||
videoDetailCtr.oid = replyItem.oid;
|
videoDetailCtr.oid.value = replyItem.oid;
|
||||||
videoDetailCtr.fRpid = replyItem.rpid!;
|
videoDetailCtr.fRpid = replyItem.rpid!;
|
||||||
videoDetailCtr.firstFloor = replyItem;
|
videoDetailCtr.firstFloor = replyItem;
|
||||||
videoDetailCtr.showReplyReplyPanel();
|
videoDetailCtr.showReplyReplyPanel();
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
import 'package:flutter/gestures.dart';
|
import 'package:flutter/gestures.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/common/widgets/badge.dart';
|
import 'package:pilipala/common/widgets/badge.dart';
|
||||||
@ -12,10 +11,9 @@ import 'package:pilipala/pages/preview/index.dart';
|
|||||||
import 'package:pilipala/pages/video/detail/index.dart';
|
import 'package:pilipala/pages/video/detail/index.dart';
|
||||||
import 'package:pilipala/pages/video/detail/reply_new/index.dart';
|
import 'package:pilipala/pages/video/detail/reply_new/index.dart';
|
||||||
import 'package:pilipala/utils/feed_back.dart';
|
import 'package:pilipala/utils/feed_back.dart';
|
||||||
import 'package:pilipala/utils/id_utils.dart';
|
|
||||||
import 'package:pilipala/utils/storage.dart';
|
import 'package:pilipala/utils/storage.dart';
|
||||||
|
import 'package:pilipala/utils/url_utils.dart';
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
|
|
||||||
import 'zan.dart';
|
import 'zan.dart';
|
||||||
|
|
||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
@ -541,7 +539,6 @@ InlineSpan buildContent(
|
|||||||
// fReplyItem 父级回复内容,用作二楼回复(回复详情)展示
|
// fReplyItem 父级回复内容,用作二楼回复(回复详情)展示
|
||||||
final content = replyItem.content;
|
final content = replyItem.content;
|
||||||
final List<InlineSpan> spanChilds = <InlineSpan>[];
|
final List<InlineSpan> spanChilds = <InlineSpan>[];
|
||||||
bool hasMatchMember = false;
|
|
||||||
|
|
||||||
// 投票
|
// 投票
|
||||||
if (content.vote.isNotEmpty) {
|
if (content.vote.isNotEmpty) {
|
||||||
@ -591,7 +588,7 @@ InlineSpan buildContent(
|
|||||||
if (patternStr.isNotEmpty) {
|
if (patternStr.isNotEmpty) {
|
||||||
patternStr += "|";
|
patternStr += "|";
|
||||||
}
|
}
|
||||||
patternStr += r'(\b\d{1,2}[::]\d{2}\b)';
|
patternStr += r'(\b(?:\d+[::])?[0-5]?[0-9][::][0-5]?[0-9]\b)';
|
||||||
final RegExp pattern = RegExp(patternStr);
|
final RegExp pattern = RegExp(patternStr);
|
||||||
List<String> matchedStrs = [];
|
List<String> matchedStrs = [];
|
||||||
void addPlainTextSpan(str) {
|
void addPlainTextSpan(str) {
|
||||||
@ -639,7 +636,9 @@ InlineSpan buildContent(
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else if (RegExp(r'^\b[0-9]{1,2}[::][0-9]{2}\b$').hasMatch(matchStr)) {
|
|
||||||
|
} else if (RegExp(r'^\b(?:\d+[::])?[0-5]?[0-9][::][0-5]?[0-9]\b$').hasMatch(matchStr)) {
|
||||||
|
matchStr = matchStr.replaceAll(':', ':');
|
||||||
spanChilds.add(
|
spanChilds.add(
|
||||||
TextSpan(
|
TextSpan(
|
||||||
text: ' $matchStr ',
|
text: ' $matchStr ',
|
||||||
@ -650,7 +649,6 @@ InlineSpan buildContent(
|
|||||||
..onTap = () {
|
..onTap = () {
|
||||||
// 跳转到指定位置
|
// 跳转到指定位置
|
||||||
try {
|
try {
|
||||||
matchStr = matchStr.replaceAll(':', ':');
|
|
||||||
SmartDialog.showToast('跳转至:$matchStr');
|
SmartDialog.showToast('跳转至:$matchStr');
|
||||||
Get.find<VideoDetailController>(tag: Get.arguments['heroTag'])
|
Get.find<VideoDetailController>(tag: Get.arguments['heroTag'])
|
||||||
.plPlayerController
|
.plPlayerController
|
||||||
@ -692,16 +690,54 @@ InlineSpan buildContent(
|
|||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
),
|
),
|
||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () {
|
..onTap = () async {
|
||||||
|
final String title = content.jumpUrl[matchStr]['title'];
|
||||||
if (appUrlSchema == '') {
|
if (appUrlSchema == '') {
|
||||||
final String str = Uri.parse(matchStr).pathSegments[0];
|
final String redirectUrl =
|
||||||
final Map matchRes = IdUtils.matchAvorBv(input: str);
|
await UrlUtils.parseRedirectUrl(matchStr);
|
||||||
final List matchKeys = matchRes.keys.toList();
|
final String pathSegment = Uri.parse(redirectUrl).path;
|
||||||
if (matchKeys.isNotEmpty) {
|
final String lastPathSegment =
|
||||||
if (matchKeys.first == 'BV') {
|
pathSegment.split('/').last;
|
||||||
|
if (lastPathSegment.startsWith('BV')) {
|
||||||
|
UrlUtils.matchUrlPush(
|
||||||
|
lastPathSegment,
|
||||||
|
title,
|
||||||
|
redirectUrl,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Get.toNamed(
|
||||||
|
'/webview',
|
||||||
|
parameters: {
|
||||||
|
'url': redirectUrl,
|
||||||
|
'type': 'url',
|
||||||
|
'pageTitle': title
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (appUrlSchema.startsWith('bilibili://search')) {
|
||||||
|
Get.toNamed('/searchResult',
|
||||||
|
parameters: {'keyword': title});
|
||||||
|
} else if (matchStr.startsWith('https://b23.tv')) {
|
||||||
|
final String redirectUrl =
|
||||||
|
await UrlUtils.parseRedirectUrl(matchStr);
|
||||||
|
final String pathSegment = Uri.parse(redirectUrl).path;
|
||||||
|
final String lastPathSegment =
|
||||||
|
pathSegment.split('/').last;
|
||||||
|
if (lastPathSegment.startsWith('BV')) {
|
||||||
|
UrlUtils.matchUrlPush(
|
||||||
|
lastPathSegment,
|
||||||
|
title,
|
||||||
|
redirectUrl,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
Get.toNamed(
|
Get.toNamed(
|
||||||
'/searchResult',
|
'/webview',
|
||||||
parameters: {'keyword': matchRes['BV']},
|
parameters: {
|
||||||
|
'url': redirectUrl,
|
||||||
|
'type': 'url',
|
||||||
|
'pageTitle': title
|
||||||
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -710,16 +746,10 @@ InlineSpan buildContent(
|
|||||||
parameters: {
|
parameters: {
|
||||||
'url': matchStr,
|
'url': matchStr,
|
||||||
'type': 'url',
|
'type': 'url',
|
||||||
'pageTitle': ''
|
'pageTitle': title
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
if (appUrlSchema.startsWith('bilibili://search')) {
|
|
||||||
Get.toNamed('/searchResult', parameters: {
|
|
||||||
'keyword': content.jumpUrl[matchStr]['title']
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@ -739,6 +769,47 @@ InlineSpan buildContent(
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (content.jumpUrl.keys.isNotEmpty) {
|
||||||
|
List<String> unmatchedItems = content.jumpUrl.keys
|
||||||
|
.toList()
|
||||||
|
.where((item) => !content.message.contains(item))
|
||||||
|
.toList();
|
||||||
|
if (unmatchedItems.isNotEmpty) {
|
||||||
|
for (int i = 0; i < unmatchedItems.length; i++) {
|
||||||
|
String patternStr = unmatchedItems[i];
|
||||||
|
spanChilds.addAll(
|
||||||
|
[
|
||||||
|
if (content.jumpUrl[patternStr]?['prefix_icon'] != null) ...[
|
||||||
|
WidgetSpan(
|
||||||
|
child: Image.network(
|
||||||
|
content.jumpUrl[patternStr]['prefix_icon'],
|
||||||
|
height: 19,
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
TextSpan(
|
||||||
|
text: content.jumpUrl[patternStr]['title'],
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
),
|
||||||
|
recognizer: TapGestureRecognizer()
|
||||||
|
..onTap = () {
|
||||||
|
Get.toNamed(
|
||||||
|
'/webview',
|
||||||
|
parameters: {
|
||||||
|
'url': patternStr,
|
||||||
|
'type': 'url',
|
||||||
|
'pageTitle': content.jumpUrl[patternStr]['title']
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
// 图片渲染
|
// 图片渲染
|
||||||
if (content.pictures.isNotEmpty) {
|
if (content.pictures.isNotEmpty) {
|
||||||
final List<String> picList = <String>[];
|
final List<String> picList = <String>[];
|
||||||
@ -753,11 +824,15 @@ InlineSpan buildContent(
|
|||||||
builder: (BuildContext context, BoxConstraints box) {
|
builder: (BuildContext context, BoxConstraints box) {
|
||||||
double maxHeight = box.maxWidth * 0.6; // 设置最大高度
|
double maxHeight = box.maxWidth * 0.6; // 设置最大高度
|
||||||
// double width = (box.maxWidth / 2).truncateToDouble();
|
// double width = (box.maxWidth / 2).truncateToDouble();
|
||||||
double height = ((box.maxWidth /
|
double height = 100;
|
||||||
2 *
|
try {
|
||||||
pictureItem['img_height'] /
|
height = ((box.maxWidth /
|
||||||
pictureItem['img_width']))
|
2 *
|
||||||
.truncateToDouble();
|
pictureItem['img_height'] /
|
||||||
|
pictureItem['img_width']))
|
||||||
|
.truncateToDouble();
|
||||||
|
} catch (_) {}
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
showDialog(
|
showDialog(
|
||||||
@ -797,8 +872,7 @@ InlineSpan buildContent(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
} else if (len > 1) {
|
||||||
if (len > 1) {
|
|
||||||
List<Widget> list = [];
|
List<Widget> list = [];
|
||||||
for (var i = 0; i < len; i++) {
|
for (var i = 0; i < len; i++) {
|
||||||
picList.add(content.pictures[i]['img_src']);
|
picList.add(content.pictures[i]['img_src']);
|
||||||
@ -816,10 +890,11 @@ InlineSpan buildContent(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
child: NetworkImgLayer(
|
child: NetworkImgLayer(
|
||||||
src: content.pictures[i]['img_src'],
|
src: content.pictures[i]['img_src'],
|
||||||
width: box.maxWidth,
|
width: box.maxWidth,
|
||||||
height: box.maxWidth,
|
height: box.maxWidth,
|
||||||
),
|
origAspectRatio: content.pictures[i]['img_width'] /
|
||||||
|
content.pictures[i]['img_height']),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@ -570,8 +570,11 @@ class _VideoDetailPageState extends State<VideoDetailPage>
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
VideoReplyPanel(
|
Obx(
|
||||||
bvid: videoDetailController.bvid,
|
() => VideoReplyPanel(
|
||||||
|
bvid: videoDetailController.bvid,
|
||||||
|
oid: videoDetailController.oid.value,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@ -586,6 +586,7 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
|||||||
),
|
),
|
||||||
|
|
||||||
/// 进度条 live模式下禁用
|
/// 进度条 live模式下禁用
|
||||||
|
|
||||||
Obx(
|
Obx(
|
||||||
() {
|
() {
|
||||||
final int value = _.sliderPositionSeconds.value;
|
final int value = _.sliderPositionSeconds.value;
|
||||||
@ -609,7 +610,7 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (_.videoType.value == 'live') {
|
if (_.videoType.value == 'live') {
|
||||||
return nil;
|
return const SizedBox();
|
||||||
}
|
}
|
||||||
if (value > max || max <= 0) {
|
if (value > max || max <= 0) {
|
||||||
return nil;
|
return nil;
|
||||||
|
|||||||
@ -3,6 +3,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
|
import 'package:pilipala/pages/follow_search/view.dart';
|
||||||
import 'package:pilipala/pages/setting/pages/logs.dart';
|
import 'package:pilipala/pages/setting/pages/logs.dart';
|
||||||
|
|
||||||
import '../pages/about/index.dart';
|
import '../pages/about/index.dart';
|
||||||
@ -104,7 +105,8 @@ class Routes {
|
|||||||
CustomGetPage(
|
CustomGetPage(
|
||||||
name: '/replyReply', page: () => const VideoReplyReplyPanel()),
|
name: '/replyReply', page: () => const VideoReplyReplyPanel()),
|
||||||
// 推荐设置
|
// 推荐设置
|
||||||
CustomGetPage(name: '/recommendSetting', page: () => const RecommendSetting()),
|
CustomGetPage(
|
||||||
|
name: '/recommendSetting', page: () => const RecommendSetting()),
|
||||||
// 播放设置
|
// 播放设置
|
||||||
CustomGetPage(name: '/playSetting', page: () => const PlaySetting()),
|
CustomGetPage(name: '/playSetting', page: () => const PlaySetting()),
|
||||||
// 外观设置
|
// 外观设置
|
||||||
@ -156,6 +158,8 @@ class Routes {
|
|||||||
name: '/memberSeasons', page: () => const MemberSeasonsPage()),
|
name: '/memberSeasons', page: () => const MemberSeasonsPage()),
|
||||||
// 日志
|
// 日志
|
||||||
CustomGetPage(name: '/logs', page: () => const LogsPage()),
|
CustomGetPage(name: '/logs', page: () => const LogsPage()),
|
||||||
|
// 搜索关注
|
||||||
|
CustomGetPage(name: '/followSearch', page: () => const FollowSearchPage()),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
154
lib/utils/cache_manage.dart
Normal file
154
lib/utils/cache_manage.dart
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
import 'dart:io';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
|
|
||||||
|
class CacheManage {
|
||||||
|
CacheManage._internal();
|
||||||
|
|
||||||
|
static final CacheManage cacheManage = CacheManage._internal();
|
||||||
|
|
||||||
|
factory CacheManage() => cacheManage;
|
||||||
|
|
||||||
|
// 获取缓存目录
|
||||||
|
Future<String> loadApplicationCache() async {
|
||||||
|
/// clear all of image in memory
|
||||||
|
// clearMemoryImageCache();
|
||||||
|
/// get ImageCache
|
||||||
|
// var res = getMemoryImageCache();
|
||||||
|
|
||||||
|
// 缓存大小
|
||||||
|
double cacheSize = 0;
|
||||||
|
// cached_network_image directory
|
||||||
|
Directory tempDirectory = await getTemporaryDirectory();
|
||||||
|
// get_storage directory
|
||||||
|
Directory docDirectory = await getApplicationDocumentsDirectory();
|
||||||
|
|
||||||
|
// 获取缓存大小
|
||||||
|
if (tempDirectory.existsSync()) {
|
||||||
|
double value = await getTotalSizeOfFilesInDir(tempDirectory);
|
||||||
|
cacheSize += value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 获取缓存大小 dioCache
|
||||||
|
if (docDirectory.existsSync()) {
|
||||||
|
double value = 0;
|
||||||
|
String dioCacheFileName =
|
||||||
|
'${docDirectory.path}${Platform.pathSeparator}DioCache.db';
|
||||||
|
var dioCacheFile = File(dioCacheFileName);
|
||||||
|
if (dioCacheFile.existsSync()) {
|
||||||
|
value = await getTotalSizeOfFilesInDir(dioCacheFile);
|
||||||
|
}
|
||||||
|
cacheSize += value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatSize(cacheSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 循环计算文件的大小(递归)
|
||||||
|
Future<double> getTotalSizeOfFilesInDir(final FileSystemEntity file) async {
|
||||||
|
if (file is File) {
|
||||||
|
int length = await file.length();
|
||||||
|
return double.parse(length.toString());
|
||||||
|
}
|
||||||
|
if (file is Directory) {
|
||||||
|
final List<FileSystemEntity> children = file.listSync();
|
||||||
|
double total = 0;
|
||||||
|
for (final FileSystemEntity child in children) {
|
||||||
|
total += await getTotalSizeOfFilesInDir(child);
|
||||||
|
}
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 缓存大小格式转换
|
||||||
|
String formatSize(double value) {
|
||||||
|
List<String> unitArr = ['B', 'K', 'M', 'G'];
|
||||||
|
int index = 0;
|
||||||
|
while (value > 1024) {
|
||||||
|
index++;
|
||||||
|
value = value / 1024;
|
||||||
|
}
|
||||||
|
String size = value.toStringAsFixed(2);
|
||||||
|
return size + unitArr[index];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 清除缓存
|
||||||
|
Future<bool> clearCacheAll() async {
|
||||||
|
bool cleanStatus = await SmartDialog.show(
|
||||||
|
useSystem: true,
|
||||||
|
animationType: SmartAnimationType.centerFade_otherSlide,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('提示'),
|
||||||
|
content: const Text('该操作将清除图片及网络请求缓存数据,确认清除?'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: (() => {SmartDialog.dismiss()}),
|
||||||
|
child: Text(
|
||||||
|
'取消',
|
||||||
|
style: TextStyle(color: Theme.of(context).colorScheme.outline),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () async {
|
||||||
|
SmartDialog.dismiss();
|
||||||
|
SmartDialog.showLoading(msg: '正在清除...');
|
||||||
|
try {
|
||||||
|
// 清除缓存 图片缓存
|
||||||
|
await clearLibraryCache();
|
||||||
|
Timer(const Duration(milliseconds: 500), () {
|
||||||
|
SmartDialog.dismiss().then((res) {
|
||||||
|
SmartDialog.showToast('清除完成');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
SmartDialog.dismiss();
|
||||||
|
SmartDialog.showToast(err.toString());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: const Text('确认'),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
).then((res) {
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
return cleanStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 清除 Documents 目录下的 DioCache.db
|
||||||
|
Future clearApplicationCache() async {
|
||||||
|
Directory directory = await getApplicationDocumentsDirectory();
|
||||||
|
if (directory.existsSync()) {
|
||||||
|
String dioCacheFileName =
|
||||||
|
'${directory.path}${Platform.pathSeparator}DioCache.db';
|
||||||
|
var dioCacheFile = File(dioCacheFileName);
|
||||||
|
if (dioCacheFile.existsSync()) {
|
||||||
|
dioCacheFile.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 清除 Library/Caches 目录及文件缓存
|
||||||
|
Future clearLibraryCache() async {
|
||||||
|
var appDocDir = await getTemporaryDirectory();
|
||||||
|
if (appDocDir.existsSync()) {
|
||||||
|
await appDocDir.delete(recursive: true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 递归方式删除目录及文件
|
||||||
|
Future deleteDirectory(FileSystemEntity file) async {
|
||||||
|
if (file is Directory) {
|
||||||
|
final List<FileSystemEntity> children = file.listSync();
|
||||||
|
for (final FileSystemEntity child in children) {
|
||||||
|
await deleteDirectory(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await file.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -42,6 +42,8 @@ class GStrorage {
|
|||||||
return deletedEntries > 10;
|
return deletedEntries > 10;
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
// 视频设置
|
||||||
|
video = await Hive.openBox('video');
|
||||||
}
|
}
|
||||||
|
|
||||||
static void regAdapter() {
|
static void regAdapter() {
|
||||||
@ -52,11 +54,6 @@ class GStrorage {
|
|||||||
Hive.registerAdapter(HotSearchItemAdapter());
|
Hive.registerAdapter(HotSearchItemAdapter());
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future<void> lazyInit() async {
|
|
||||||
// 视频设置
|
|
||||||
video = await Hive.openBox('video');
|
|
||||||
}
|
|
||||||
|
|
||||||
static Future<void> close() async {
|
static Future<void> close() async {
|
||||||
// user.compact();
|
// user.compact();
|
||||||
// user.close();
|
// user.close();
|
||||||
|
|||||||
61
lib/utils/url_utils.dart
Normal file
61
lib/utils/url_utils.dart
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
import 'package:dio/dio.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
|
|
||||||
|
import '../http/search.dart';
|
||||||
|
import 'id_utils.dart';
|
||||||
|
import 'utils.dart';
|
||||||
|
|
||||||
|
class UrlUtils {
|
||||||
|
// 302重定向路由截取
|
||||||
|
static Future<String> parseRedirectUrl(String url) async {
|
||||||
|
late String redirectUrl;
|
||||||
|
final dio = Dio();
|
||||||
|
dio.options.followRedirects = false;
|
||||||
|
dio.options.validateStatus = (status) {
|
||||||
|
return status == 200 || status == 301 || status == 302;
|
||||||
|
};
|
||||||
|
final response = await dio.get(url);
|
||||||
|
if (response.statusCode == 302) {
|
||||||
|
redirectUrl = response.headers['location']?.first as String;
|
||||||
|
if (redirectUrl.endsWith('/')) {
|
||||||
|
redirectUrl = redirectUrl.substring(0, redirectUrl.length - 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (url.endsWith('/')) {
|
||||||
|
url = url.substring(0, url.length - 1);
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
return redirectUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 匹配url路由跳转
|
||||||
|
static matchUrlPush(
|
||||||
|
String pathSegment,
|
||||||
|
String title,
|
||||||
|
String redirectUrl,
|
||||||
|
) async {
|
||||||
|
final Map matchRes = IdUtils.matchAvorBv(input: pathSegment);
|
||||||
|
if (matchRes.containsKey('BV')) {
|
||||||
|
final String bv = matchRes['BV'];
|
||||||
|
final int cid = await SearchHttp.ab2c(bvid: bv);
|
||||||
|
final String heroTag = Utils.makeHeroTag(bv);
|
||||||
|
await Get.toNamed(
|
||||||
|
'/video?bvid=$bv&cid=$cid',
|
||||||
|
arguments: <String, String?>{
|
||||||
|
'pic': '',
|
||||||
|
'heroTag': heroTag,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
await Get.toNamed(
|
||||||
|
'/webview',
|
||||||
|
parameters: {
|
||||||
|
'url': redirectUrl,
|
||||||
|
'type': 'url',
|
||||||
|
'pageTitle': title,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -188,6 +188,7 @@ flutter:
|
|||||||
- assets/images/
|
- assets/images/
|
||||||
- assets/images/lv/
|
- assets/images/lv/
|
||||||
- assets/images/logo/
|
- assets/images/logo/
|
||||||
|
- assets/images/live/
|
||||||
# An image asset can refer to one or more resolution-specific "variants", see
|
# An image asset can refer to one or more resolution-specific "variants", see
|
||||||
# https://flutter.dev/assets-and-images/#resolution-aware
|
# https://flutter.dev/assets-and-images/#resolution-aware
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user