mirror of
https://github.com/bggRGjQaUbCoE/PiliPlus.git
synced 2026-05-13 20:53:58 +08:00
Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
84cc65489f | ||
|
|
2b9cb54d91 | ||
|
|
54c7fef217 | ||
|
|
ba74cb8c01 | ||
|
|
675932aa69 | ||
|
|
d996e0a7dd | ||
|
|
b6279f702a | ||
|
|
695a89b91a |
@@ -58,7 +58,7 @@ class VideoCardHGrpc extends StatelessWidget {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
PiliScheme.routePush(Uri.parse(videoItem.smallCoverV5.base.uri));
|
||||
PiliScheme.routePushFromUrl(videoItem.smallCoverV5.base.uri);
|
||||
} catch (err) {
|
||||
SmartDialog.showToast(err.toString());
|
||||
}
|
||||
|
||||
@@ -604,21 +604,24 @@ class BangumiIntroController extends CommonController {
|
||||
RxInt followStatus = (-1).obs;
|
||||
|
||||
Future queryIsFollowed() async {
|
||||
dynamic result = await Request().get(
|
||||
'https://www.bilibili.com/bangumi/play/ss$seasonId',
|
||||
);
|
||||
dom.Document document = html_parser.parse(result.data);
|
||||
dom.Element? scriptElement = document.querySelector('script#__NEXT_DATA__');
|
||||
if (scriptElement != null) {
|
||||
dynamic scriptContent = jsonDecode(scriptElement.text);
|
||||
isFollowed.value =
|
||||
scriptContent['props']['pageProps']['followState']['isFollowed'];
|
||||
followStatus.value =
|
||||
scriptContent['props']['pageProps']['followState']['followStatus'];
|
||||
// int progress = scriptContent['props']['pageProps']['dehydratedState']
|
||||
// ['queries'][0]['state']['data']['result']
|
||||
// ['play_view_business_info']['user_status']['watch_progress']
|
||||
// ['current_watch_progress'];
|
||||
}
|
||||
try {
|
||||
dynamic result = await Request().get(
|
||||
'https://www.bilibili.com/bangumi/play/ss$seasonId',
|
||||
);
|
||||
dom.Document document = html_parser.parse(result.data);
|
||||
dom.Element? scriptElement =
|
||||
document.querySelector('script#__NEXT_DATA__');
|
||||
if (scriptElement != null) {
|
||||
dynamic scriptContent = jsonDecode(scriptElement.text);
|
||||
isFollowed.value =
|
||||
scriptContent['props']['pageProps']['followState']['isFollowed'];
|
||||
followStatus.value =
|
||||
scriptContent['props']['pageProps']['followState']['followStatus'];
|
||||
// int progress = scriptContent['props']['pageProps']['dehydratedState']
|
||||
// ['queries'][0]['state']['data']['result']
|
||||
// ['play_view_business_info']['user_status']['watch_progress']
|
||||
// ['current_watch_progress'];
|
||||
}
|
||||
} catch (_) {}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -104,13 +104,12 @@ InlineSpan? richNode(item, context) {
|
||||
return;
|
||||
}
|
||||
if (url.startsWith('//')) {
|
||||
url = url.replaceFirst('//', 'https://');
|
||||
PiliScheme.routePush(Uri.parse(url));
|
||||
PiliScheme.routePushFromUrl('https:$url');
|
||||
return;
|
||||
}
|
||||
Utils.handleWebview(url.startsWith('//')
|
||||
? "https://${url.split('//').last}"
|
||||
: url);
|
||||
Utils.handleWebview(
|
||||
url.startsWith('//') ? "https://$url" : url,
|
||||
);
|
||||
},
|
||||
child: Text(
|
||||
i.text ?? '',
|
||||
|
||||
@@ -3,7 +3,6 @@ import 'package:PiliPlus/common/widgets/video_progress_indicator.dart';
|
||||
import 'package:PiliPlus/models/user/history.dart';
|
||||
import 'package:PiliPlus/pages/common/multi_select_controller.dart';
|
||||
import 'package:PiliPlus/pages/fav_search/controller.dart';
|
||||
import 'package:PiliPlus/utils/app_scheme.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||
import 'package:get/get.dart';
|
||||
@@ -53,8 +52,15 @@ class HistoryItem extends StatelessWidget {
|
||||
// 'pageTitle': videoItem.title
|
||||
// },
|
||||
// );
|
||||
PiliScheme.routePush(Uri.parse(
|
||||
"https://www.bilibili.com/read/cv${videoItem.history.oid}"));
|
||||
Utils.toDupNamed(
|
||||
'/htmlRender',
|
||||
parameters: {
|
||||
'url': 'https://www.bilibili.com/read/cv${videoItem.history.oid}',
|
||||
'title': '',
|
||||
'id': 'cv${videoItem.history.oid}',
|
||||
'dynamicType': 'read'
|
||||
},
|
||||
);
|
||||
} else if (videoItem.history.business == 'live') {
|
||||
if (videoItem.liveStatus == 1) {
|
||||
// LiveItemModel liveItem = LiveItemModel.fromJson({
|
||||
|
||||
@@ -105,10 +105,12 @@ class HomeController extends GetxController with GetTickerProviderStateMixin {
|
||||
}
|
||||
|
||||
void querySearchDefault() async {
|
||||
var res = await Request().get(Api.searchDefault);
|
||||
if (res.data['code'] == 0) {
|
||||
defaultSearch.value = res.data['data']['name'];
|
||||
}
|
||||
try {
|
||||
var res = await Request().get(Api.searchDefault);
|
||||
if (res.data['code'] == 0) {
|
||||
defaultSearch.value = res.data['data']['name'];
|
||||
}
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
showUserInfoDialog(context) {
|
||||
|
||||
@@ -313,28 +313,30 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
||||
},
|
||||
),
|
||||
),
|
||||
PopScope(
|
||||
canPop: plPlayerController.isFullScreen.value != true,
|
||||
onPopInvokedWithResult: (bool didPop, Object? result) {
|
||||
if (plPlayerController.isFullScreen.value == true) {
|
||||
plPlayerController.triggerFullScreen(status: false);
|
||||
// if (MediaQuery.of(context).orientation ==
|
||||
// Orientation.landscape) {
|
||||
// verticalScreenForTwoSeconds();
|
||||
// }
|
||||
}
|
||||
},
|
||||
child: Listener(
|
||||
onPointerDown: (_) {
|
||||
_node.unfocus();
|
||||
Obx(
|
||||
() => PopScope(
|
||||
canPop: plPlayerController.isFullScreen.value != true,
|
||||
onPopInvokedWithResult: (bool didPop, Object? result) {
|
||||
if (plPlayerController.isFullScreen.value == true) {
|
||||
plPlayerController.triggerFullScreen(status: false);
|
||||
// if (MediaQuery.of(context).orientation ==
|
||||
// Orientation.landscape) {
|
||||
// verticalScreenForTwoSeconds();
|
||||
// }
|
||||
}
|
||||
},
|
||||
child: SizedBox(
|
||||
width: Get.size.width,
|
||||
height: MediaQuery.of(context).orientation ==
|
||||
Orientation.landscape
|
||||
? Get.size.height
|
||||
: Get.size.width * 9 / 16,
|
||||
child: videoPlayerPanel,
|
||||
child: Listener(
|
||||
onPointerDown: (_) {
|
||||
_node.unfocus();
|
||||
},
|
||||
child: SizedBox(
|
||||
width: Get.size.width,
|
||||
height: MediaQuery.of(context).orientation ==
|
||||
Orientation.landscape
|
||||
? Get.size.height
|
||||
: Get.size.width * 9 / 16,
|
||||
child: videoPlayerPanel,
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
@@ -124,37 +124,41 @@ class MainController extends GetxController {
|
||||
}
|
||||
|
||||
Future _queryPMUnread() async {
|
||||
dynamic res = await Request().get(Api.msgUnread);
|
||||
if (res.data['code'] == 0) {
|
||||
return {
|
||||
'status': true,
|
||||
'data': ((res.data['data']?['unfollow_unread'] as int?) ?? 0) +
|
||||
((res.data['data']?['follow_unread'] as int?) ?? 0),
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
'status': false,
|
||||
'msg': res.data['message'],
|
||||
};
|
||||
}
|
||||
try {
|
||||
dynamic res = await Request().get(Api.msgUnread);
|
||||
if (res.data['code'] == 0) {
|
||||
return {
|
||||
'status': true,
|
||||
'data': ((res.data['data']?['unfollow_unread'] as int?) ?? 0) +
|
||||
((res.data['data']?['follow_unread'] as int?) ?? 0),
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
'status': false,
|
||||
'msg': res.data['message'],
|
||||
};
|
||||
}
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
Future _queryMsgFeedUnread() async {
|
||||
if (isLogin.value.not) {
|
||||
return;
|
||||
}
|
||||
dynamic res = await Request().get(Api.msgFeedUnread);
|
||||
if (res.data['code'] == 0) {
|
||||
return {
|
||||
'status': true,
|
||||
'data': res.data['data'],
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
'status': false,
|
||||
'msg': res.data['message'],
|
||||
};
|
||||
}
|
||||
try {
|
||||
dynamic res = await Request().get(Api.msgFeedUnread);
|
||||
if (res.data['code'] == 0) {
|
||||
return {
|
||||
'status': true,
|
||||
'data': res.data['data'],
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
'status': false,
|
||||
'msg': res.data['message'],
|
||||
};
|
||||
}
|
||||
} catch (_) {}
|
||||
}
|
||||
|
||||
void getUnreadDynamic() async {
|
||||
|
||||
@@ -5,6 +5,7 @@ import 'package:PiliPlus/common/widgets/network_img_layer.dart';
|
||||
import 'package:PiliPlus/common/widgets/tabs.dart';
|
||||
import 'package:PiliPlus/grpc/grpc_client.dart';
|
||||
import 'package:PiliPlus/pages/mine/controller.dart';
|
||||
import 'package:PiliPlus/utils/app_scheme.dart';
|
||||
import 'package:PiliPlus/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
@@ -156,6 +157,7 @@ class _MainAppState extends State<MainApp>
|
||||
await GrpcClient.instance.shutdown();
|
||||
await GStorage.close();
|
||||
EventBus().off(EventName.loginEvent);
|
||||
PiliScheme.listener?.cancel();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
|
||||
@@ -60,7 +60,7 @@ class _MemberArticleState extends State<MemberArticle>
|
||||
return ListTile(
|
||||
dense: true,
|
||||
onTap: () {
|
||||
PiliScheme.routePush(Uri.parse(item.uri ?? ''));
|
||||
PiliScheme.routePushFromUrl(item.uri ?? '');
|
||||
},
|
||||
leading: item.originImageUrls?.isNotEmpty == true
|
||||
? Container(
|
||||
|
||||
@@ -142,7 +142,7 @@ class _MemberFavoriteState extends State<MemberFavorite>
|
||||
});
|
||||
}
|
||||
} else if (item1.type == 21) {
|
||||
PiliScheme.routePush(Uri.parse(item1.link ?? ''));
|
||||
PiliScheme.routePushFromUrl(item1.link ?? '');
|
||||
} else if (item1.type == 11) {
|
||||
Get.toNamed(
|
||||
'/subDetail',
|
||||
|
||||
@@ -118,9 +118,9 @@ class _MemberHomeState extends State<MemberHome>
|
||||
child: ListTile(
|
||||
dense: true,
|
||||
onTap: () {
|
||||
PiliScheme.routePush(Uri.parse(
|
||||
loadingState.response.article.item.first.uri ??
|
||||
''));
|
||||
PiliScheme.routePushFromUrl(
|
||||
loadingState.response.article.item.first.uri ?? '',
|
||||
);
|
||||
},
|
||||
leading: loadingState.response.article.item.first
|
||||
.originImageUrls?.isNotEmpty ==
|
||||
|
||||
@@ -73,7 +73,7 @@ class _AtMePageState extends State<AtMePage> {
|
||||
String? nativeUri =
|
||||
_atMeController.msgFeedAtMeList[i].item?.nativeUri;
|
||||
if (nativeUri != null) {
|
||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
||||
PiliScheme.routePushFromUrl(nativeUri);
|
||||
}
|
||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||
},
|
||||
|
||||
@@ -122,7 +122,7 @@ class LikeMeList extends StatelessWidget {
|
||||
onTap: () {
|
||||
String? nativeUri = msgFeedLikeMeList[i].item?.nativeUri;
|
||||
if (nativeUri != null) {
|
||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
||||
PiliScheme.routePushFromUrl(nativeUri);
|
||||
}
|
||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||
},
|
||||
|
||||
@@ -72,7 +72,7 @@ class _ReplyMePageState extends State<ReplyMePage> {
|
||||
String? nativeUri = _replyMeController
|
||||
.msgFeedReplyMeList[i].item?.nativeUri;
|
||||
if (nativeUri != null) {
|
||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
||||
PiliScheme.routePushFromUrl(nativeUri);
|
||||
}
|
||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||
},
|
||||
|
||||
@@ -184,8 +184,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
||||
recognizer: TapGestureRecognizer()
|
||||
..onTap = () {
|
||||
try {
|
||||
Uri uri = Uri.parse(match[2]!.replaceAll('"', ''));
|
||||
PiliScheme.routePush(uri);
|
||||
PiliScheme.routePushFromUrl(match[2]!.replaceAll('"', ''));
|
||||
} catch (err) {
|
||||
SmartDialog.showToast(err.toString());
|
||||
}
|
||||
@@ -209,8 +208,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
||||
recognizer: TapGestureRecognizer()
|
||||
..onTap = () {
|
||||
try {
|
||||
Uri uri = Uri.parse(match[3]!);
|
||||
PiliScheme.routePush(uri);
|
||||
PiliScheme.routePushFromUrl(match[3]!);
|
||||
} catch (err) {
|
||||
SmartDialog.showToast(err.toString());
|
||||
}
|
||||
@@ -231,8 +229,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
||||
recognizer: TapGestureRecognizer()
|
||||
..onTap = () {
|
||||
try {
|
||||
Uri uri = Uri.parse(match[0]!);
|
||||
PiliScheme.routePush(uri);
|
||||
PiliScheme.routePushFromUrl(match[0]!);
|
||||
} catch (err) {
|
||||
SmartDialog.showToast(err.toString());
|
||||
Utils.copyText(match[0] ?? '');
|
||||
|
||||
@@ -67,18 +67,22 @@ class SearchPanelController extends CommonController {
|
||||
void jump2Video() {
|
||||
if (RegExp(r'^av\d+$', caseSensitive: false).hasMatch(keyword)) {
|
||||
hasJump2Video = true;
|
||||
PiliScheme.videoPush(int.parse(keyword.substring(2)), null, false);
|
||||
PiliScheme.videoPush(
|
||||
int.parse(keyword.substring(2)),
|
||||
null,
|
||||
showDialog: false,
|
||||
);
|
||||
} else if (RegExp(r'^bv[a-z\d]{10}$', caseSensitive: false)
|
||||
.hasMatch(keyword)) {
|
||||
hasJump2Video = true;
|
||||
PiliScheme.videoPush(null, keyword, false);
|
||||
PiliScheme.videoPush(null, keyword, showDialog: false);
|
||||
}
|
||||
}
|
||||
|
||||
void onPushDetail(resultList) async {
|
||||
int? aid = int.tryParse(keyword);
|
||||
if (aid != null && resultList.first.aid == aid) {
|
||||
PiliScheme.videoPush(aid, null, false);
|
||||
PiliScheme.videoPush(aid, null, showDialog: false);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1053,6 +1053,7 @@ class VideoDetailController extends GetxController
|
||||
bvid: bvid,
|
||||
epid: epId,
|
||||
seasonId: seasonId,
|
||||
forcePgcApi: Get.arguments['pgcApi'] ?? false,
|
||||
);
|
||||
if (result['status']) {
|
||||
data = result['data'];
|
||||
|
||||
@@ -19,7 +19,6 @@ import 'package:PiliPlus/utils/feed_back.dart';
|
||||
import 'package:PiliPlus/utils/storage.dart';
|
||||
import 'package:PiliPlus/utils/url_utils.dart';
|
||||
import 'package:PiliPlus/utils/utils.dart';
|
||||
import '../../../../../utils/app_scheme.dart';
|
||||
import 'zan.dart';
|
||||
import 'package:html/parser.dart' show parse;
|
||||
|
||||
@@ -863,56 +862,12 @@ class ReplyItem extends StatelessWidget {
|
||||
});
|
||||
return;
|
||||
}
|
||||
final String redirectUrl =
|
||||
(await UrlUtils.parseRedirectUrl(matchStr)) ??
|
||||
matchStr;
|
||||
// if (redirectUrl == matchStr) {
|
||||
// Clipboard.setData(ClipboardData(text: matchStr));
|
||||
// SmartDialog.showToast('地址可能有误');
|
||||
// return;
|
||||
// }
|
||||
Uri uri = Uri.parse(redirectUrl);
|
||||
PiliScheme.routePush(uri);
|
||||
// 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(
|
||||
// '/webview',
|
||||
// parameters: {
|
||||
// 'url': redirectUrl,
|
||||
// 'type': 'url',
|
||||
// 'pageTitle': title
|
||||
// },
|
||||
// );
|
||||
// }
|
||||
Utils.handleWebview(matchStr);
|
||||
}
|
||||
} 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)) ??
|
||||
matchStr;
|
||||
final String pathSegment =
|
||||
Uri.parse(redirectUrl).path;
|
||||
final String lastPathSegment =
|
||||
pathSegment.split('/').last;
|
||||
if (lastPathSegment.startsWith('BV')) {
|
||||
UrlUtils.matchUrlPush(
|
||||
lastPathSegment,
|
||||
redirectUrl,
|
||||
);
|
||||
} else {
|
||||
Utils.handleWebview(redirectUrl);
|
||||
}
|
||||
} else {
|
||||
Utils.handleWebview(matchStr);
|
||||
}
|
||||
@@ -949,25 +904,8 @@ class ReplyItem extends StatelessWidget {
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
),
|
||||
recognizer: TapGestureRecognizer()
|
||||
..onTap = () async {
|
||||
if (matchStr.startsWith('https://b23.tv')) {
|
||||
final String redirectUrl =
|
||||
(await UrlUtils.parseRedirectUrl(matchStr)) ??
|
||||
matchStr;
|
||||
final String pathSegment = Uri.parse(redirectUrl).path;
|
||||
final String lastPathSegment =
|
||||
pathSegment.split('/').last;
|
||||
if (lastPathSegment.startsWith('BV')) {
|
||||
UrlUtils.matchUrlPush(
|
||||
lastPathSegment,
|
||||
redirectUrl,
|
||||
);
|
||||
} else {
|
||||
PiliScheme.routePush(Uri.parse(matchStr));
|
||||
}
|
||||
} else {
|
||||
PiliScheme.routePush(Uri.parse(matchStr));
|
||||
}
|
||||
..onTap = () {
|
||||
Utils.handleWebview(matchStr);
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
@@ -20,7 +20,6 @@ import 'package:PiliPlus/utils/feed_back.dart';
|
||||
import 'package:PiliPlus/utils/storage.dart';
|
||||
import 'package:PiliPlus/utils/url_utils.dart';
|
||||
import 'package:PiliPlus/utils/utils.dart';
|
||||
import '../../../../../utils/app_scheme.dart';
|
||||
import 'package:html/parser.dart' show parse;
|
||||
|
||||
class ReplyItemGrpc extends StatelessWidget {
|
||||
@@ -901,56 +900,12 @@ class ReplyItemGrpc extends StatelessWidget {
|
||||
});
|
||||
return;
|
||||
}
|
||||
final String redirectUrl =
|
||||
(await UrlUtils.parseRedirectUrl(matchStr)) ??
|
||||
matchStr;
|
||||
// if (redirectUrl == matchStr) {
|
||||
// Clipboard.setData(ClipboardData(text: matchStr));
|
||||
// SmartDialog.showToast('地址可能有误');
|
||||
// return;
|
||||
// }
|
||||
Uri uri = Uri.parse(redirectUrl);
|
||||
PiliScheme.routePush(uri);
|
||||
// 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(
|
||||
// '/webview',
|
||||
// parameters: {
|
||||
// 'url': redirectUrl,
|
||||
// 'type': 'url',
|
||||
// 'pageTitle': title
|
||||
// },
|
||||
// );
|
||||
// }
|
||||
Utils.handleWebview(matchStr);
|
||||
}
|
||||
} 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)) ??
|
||||
matchStr;
|
||||
final String pathSegment =
|
||||
Uri.parse(redirectUrl).path;
|
||||
final String lastPathSegment =
|
||||
pathSegment.split('/').last;
|
||||
if (lastPathSegment.startsWith('BV')) {
|
||||
UrlUtils.matchUrlPush(
|
||||
lastPathSegment,
|
||||
redirectUrl,
|
||||
);
|
||||
} else {
|
||||
Utils.handleWebview(redirectUrl);
|
||||
}
|
||||
} else {
|
||||
Utils.handleWebview(matchStr);
|
||||
}
|
||||
@@ -987,25 +942,8 @@ class ReplyItemGrpc extends StatelessWidget {
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
),
|
||||
recognizer: TapGestureRecognizer()
|
||||
..onTap = () async {
|
||||
if (matchStr.startsWith('https://b23.tv')) {
|
||||
final String redirectUrl =
|
||||
(await UrlUtils.parseRedirectUrl(matchStr)) ??
|
||||
matchStr;
|
||||
final String pathSegment = Uri.parse(redirectUrl).path;
|
||||
final String lastPathSegment =
|
||||
pathSegment.split('/').last;
|
||||
if (lastPathSegment.startsWith('BV')) {
|
||||
UrlUtils.matchUrlPush(
|
||||
lastPathSegment,
|
||||
redirectUrl,
|
||||
);
|
||||
} else {
|
||||
PiliScheme.routePush(Uri.parse(matchStr));
|
||||
}
|
||||
} else {
|
||||
PiliScheme.routePush(Uri.parse(matchStr));
|
||||
}
|
||||
..onTap = () {
|
||||
Utils.handleWebview(matchStr);
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
@@ -5,8 +5,6 @@ import 'package:PiliPlus/http/constants.dart';
|
||||
import 'package:PiliPlus/http/init.dart';
|
||||
import 'package:PiliPlus/utils/app_scheme.dart';
|
||||
import 'package:PiliPlus/utils/cache_manage.dart';
|
||||
import 'package:PiliPlus/utils/extension.dart';
|
||||
import 'package:PiliPlus/utils/id_utils.dart';
|
||||
import 'package:PiliPlus/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_inappwebview/flutter_inappwebview.dart';
|
||||
@@ -237,53 +235,20 @@ class _WebviewPageNewState extends State<WebviewPageNew> {
|
||||
}
|
||||
: null,
|
||||
shouldOverrideUrlLoading: (controller, navigationAction) async {
|
||||
final String? str =
|
||||
navigationAction.request.url!.pathSegments.getOrNull(0);
|
||||
if (str != null) {
|
||||
final Map matchRes = IdUtils.matchAvorBv(input: str);
|
||||
if (matchRes.isNotEmpty) {
|
||||
Get.back();
|
||||
PiliScheme.videoPush(matchRes['AV'], matchRes['BV']);
|
||||
return NavigationActionPolicy.CANCEL;
|
||||
}
|
||||
}
|
||||
|
||||
var url = navigationAction.request.url!.toString();
|
||||
|
||||
if (RegExp(
|
||||
r'^(https?://)?((www|m).)?(bilibili|b23).(com|tv)/video/BV[a-zA-Z\d]+')
|
||||
.hasMatch(url)) {
|
||||
try {
|
||||
String? bvid =
|
||||
RegExp(r'BV[a-zA-Z\d]+').firstMatch(url)?.group(0);
|
||||
if (bvid != null) {
|
||||
Get.back();
|
||||
PiliScheme.videoPush(null, bvid);
|
||||
return NavigationActionPolicy.CANCEL;
|
||||
}
|
||||
} catch (_) {}
|
||||
} else if (RegExp(
|
||||
r'^(https?://)?((www|m).)?(bilibili|b23).(com|tv)/playlist')
|
||||
.hasMatch(url)) {
|
||||
try {
|
||||
String? bvid =
|
||||
RegExp(r'bvid=(BV[a-zA-Z\d]+)').firstMatch(url)?.group(1);
|
||||
if (bvid != null) {
|
||||
PiliScheme.videoPush(null, bvid);
|
||||
return NavigationActionPolicy.CANCEL;
|
||||
}
|
||||
} catch (_) {}
|
||||
late String url = navigationAction.request.url.toString();
|
||||
bool hasMatch = await PiliScheme.routePush(
|
||||
navigationAction.request.url?.uriValue ?? Uri(),
|
||||
selfHandle: true,
|
||||
off: true,
|
||||
);
|
||||
// debugPrint('webview: [$url], [$hasMatch]');
|
||||
if (hasMatch) {
|
||||
_progressStream.add(1.0);
|
||||
return NavigationActionPolicy.CANCEL;
|
||||
} else if (RegExp(r'^(?!(https?://))\S+://', caseSensitive: false)
|
||||
.hasMatch(url)) {
|
||||
if (url.startsWith('bilibili://video/')) {
|
||||
String? str =
|
||||
navigationAction.request.url!.pathSegments.getOrNull(0);
|
||||
Get.offAndToNamed(
|
||||
'/searchResult',
|
||||
parameters: {'keyword': str ?? ''},
|
||||
);
|
||||
} else {
|
||||
var snackBar = SnackBar(
|
||||
if (context.mounted) {
|
||||
SnackBar snackBar = SnackBar(
|
||||
content: const Text('当前网页将要打开外部链接,是否打开'),
|
||||
showCloseIcon: true,
|
||||
action: SnackBarAction(
|
||||
|
||||
@@ -16,7 +16,6 @@ class WhisperDetailController extends GetxController {
|
||||
RxList<MessageItem> messageList = <MessageItem>[].obs;
|
||||
//表情转换图片规则
|
||||
List<dynamic>? eInfos;
|
||||
final TextEditingController replyContentController = TextEditingController();
|
||||
|
||||
@override
|
||||
void onInit() {
|
||||
@@ -67,10 +66,11 @@ class WhisperDetailController extends GetxController {
|
||||
}
|
||||
|
||||
Future sendMsg({
|
||||
required String message,
|
||||
dynamic picMsg,
|
||||
required VoidCallback onClearText,
|
||||
}) async {
|
||||
feedBack();
|
||||
String message = replyContentController.text;
|
||||
final userInfo = GStorage.userInfo.get('userInfoCache');
|
||||
if (userInfo == null) {
|
||||
SmartDialog.dismiss();
|
||||
@@ -96,7 +96,7 @@ class WhisperDetailController extends GetxController {
|
||||
if (result['status']) {
|
||||
// debugPrint(result['data']);
|
||||
querySessionMsg();
|
||||
replyContentController.text = "";
|
||||
onClearText();
|
||||
SmartDialog.dismiss();
|
||||
SmartDialog.showToast('发送成功');
|
||||
} else {
|
||||
@@ -104,10 +104,4 @@ class WhisperDetailController extends GetxController {
|
||||
SmartDialog.showToast(result['msg']);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
void onClose() {
|
||||
replyContentController.dispose();
|
||||
super.onClose();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -177,7 +177,7 @@ class _WhisperDetailPageState
|
||||
() => TextField(
|
||||
readOnly: readOnly.value,
|
||||
focusNode: focusNode,
|
||||
controller: _whisperDetailController.replyContentController,
|
||||
controller: editController,
|
||||
minLines: 1,
|
||||
maxLines: 4,
|
||||
onChanged: (value) {
|
||||
@@ -209,7 +209,10 @@ class _WhisperDetailPageState
|
||||
return IconButton(
|
||||
onPressed: () async {
|
||||
if (enablePublish.value) {
|
||||
_whisperDetailController.sendMsg();
|
||||
_whisperDetailController.sendMsg(
|
||||
message: editController.text,
|
||||
onClearText: editController.clear,
|
||||
);
|
||||
} else {
|
||||
try {
|
||||
XFile? pickedFile = await imagePicker.pickImage(
|
||||
@@ -238,7 +241,10 @@ class _WhisperDetailPageState
|
||||
};
|
||||
SmartDialog.showLoading(msg: '正在发送');
|
||||
await _whisperDetailController.sendMsg(
|
||||
picMsg: picMsg);
|
||||
picMsg: picMsg,
|
||||
message: editController.text,
|
||||
onClearText: editController.clear,
|
||||
);
|
||||
} else {
|
||||
SmartDialog.dismiss();
|
||||
SmartDialog.showToast(result['msg']);
|
||||
|
||||
@@ -3,13 +3,11 @@ enum BottomControlType {
|
||||
playOrPause,
|
||||
next,
|
||||
time,
|
||||
space,
|
||||
episode,
|
||||
fit,
|
||||
subtitle,
|
||||
speed,
|
||||
fullscreen,
|
||||
custom,
|
||||
viewPoints,
|
||||
superResolution,
|
||||
dmChart,
|
||||
|
||||
@@ -49,7 +49,6 @@ class PLVideoPlayer extends StatefulWidget {
|
||||
this.headerControl,
|
||||
this.bottomControl,
|
||||
this.danmuWidget,
|
||||
this.bottomList,
|
||||
this.customWidget,
|
||||
this.customWidgets,
|
||||
this.showEpisodes,
|
||||
@@ -63,7 +62,6 @@ class PLVideoPlayer extends StatefulWidget {
|
||||
final PreferredSizeWidget? headerControl;
|
||||
final PreferredSizeWidget? bottomControl;
|
||||
final Widget? danmuWidget;
|
||||
final List<BottomControlType>? bottomList;
|
||||
// List<Widget> or Widget
|
||||
|
||||
final Widget? customWidget;
|
||||
@@ -251,7 +249,7 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
||||
}
|
||||
|
||||
// 动态构建底部控制条
|
||||
List<Widget> buildBottomControl() {
|
||||
Widget buildBottomControl() {
|
||||
bool isSeason = videoIntroController?.videoDetail.value.ugcSeason != null;
|
||||
bool isPage = videoIntroController?.videoDetail.value.pages != null &&
|
||||
videoIntroController!.videoDetail.value.pages!.length > 1;
|
||||
@@ -321,7 +319,7 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
||||
|
||||
/// 时间进度
|
||||
BottomControlType.time: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.end,
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
// 播放时间
|
||||
Obx(() {
|
||||
@@ -354,9 +352,6 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
||||
],
|
||||
),
|
||||
|
||||
/// 空白占位
|
||||
BottomControlType.space: const Spacer(),
|
||||
|
||||
/// 高能进度条
|
||||
BottomControlType.dmChart: Obx(() => plPlayerController.dmTrend.isEmpty
|
||||
? const SizedBox.shrink()
|
||||
@@ -622,48 +617,61 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
||||
BottomControlType.fullscreen: SizedBox(
|
||||
width: widgetWidth,
|
||||
height: 30,
|
||||
child: Obx(() => ComBtn(
|
||||
icon: Icon(
|
||||
isFullScreen ? Icons.fullscreen_exit : Icons.fullscreen,
|
||||
semanticLabel: isFullScreen ? '退出全屏' : '全屏',
|
||||
size: 24,
|
||||
color: Colors.white,
|
||||
),
|
||||
fuc: () =>
|
||||
plPlayerController.triggerFullScreen(status: !isFullScreen),
|
||||
)),
|
||||
child: Obx(
|
||||
() => ComBtn(
|
||||
icon: Icon(
|
||||
isFullScreen ? Icons.fullscreen_exit : Icons.fullscreen,
|
||||
semanticLabel: isFullScreen ? '退出全屏' : '全屏',
|
||||
size: 24,
|
||||
color: Colors.white,
|
||||
),
|
||||
fuc: () =>
|
||||
plPlayerController.triggerFullScreen(status: !isFullScreen),
|
||||
),
|
||||
),
|
||||
),
|
||||
};
|
||||
final List<Widget> list = [];
|
||||
List<BottomControlType> userSpecifyItem = widget.bottomList ??
|
||||
[
|
||||
BottomControlType.playOrPause,
|
||||
BottomControlType.time,
|
||||
if (anySeason) BottomControlType.pre,
|
||||
if (anySeason) BottomControlType.next,
|
||||
BottomControlType.space,
|
||||
BottomControlType.dmChart,
|
||||
BottomControlType.superResolution,
|
||||
BottomControlType.viewPoints,
|
||||
if (anySeason) BottomControlType.episode,
|
||||
if (isFullScreen) BottomControlType.fit,
|
||||
BottomControlType.subtitle,
|
||||
BottomControlType.speed,
|
||||
BottomControlType.fullscreen,
|
||||
];
|
||||
for (var i = 0; i < userSpecifyItem.length; i++) {
|
||||
if (userSpecifyItem[i] == BottomControlType.custom) {
|
||||
if (widget.customWidget != null && widget.customWidget is Widget) {
|
||||
list.add(widget.customWidget!);
|
||||
}
|
||||
if (widget.customWidgets != null && widget.customWidgets!.isNotEmpty) {
|
||||
list.addAll(widget.customWidgets!);
|
||||
}
|
||||
} else {
|
||||
list.add(videoProgressWidgets[userSpecifyItem[i]]!);
|
||||
}
|
||||
}
|
||||
return list;
|
||||
|
||||
List<BottomControlType> userSpecifyItemLeft = [
|
||||
BottomControlType.playOrPause,
|
||||
BottomControlType.time,
|
||||
if (anySeason) BottomControlType.pre,
|
||||
if (anySeason) BottomControlType.next,
|
||||
];
|
||||
|
||||
List<BottomControlType> userSpecifyItemRight = [
|
||||
BottomControlType.dmChart,
|
||||
BottomControlType.superResolution,
|
||||
BottomControlType.viewPoints,
|
||||
if (anySeason) BottomControlType.episode,
|
||||
if (isFullScreen) BottomControlType.fit,
|
||||
BottomControlType.subtitle,
|
||||
BottomControlType.speed,
|
||||
BottomControlType.fullscreen,
|
||||
];
|
||||
|
||||
return Row(
|
||||
children: [
|
||||
...userSpecifyItemLeft.map((item) => videoProgressWidgets[item]!),
|
||||
Expanded(
|
||||
child: LayoutBuilder(
|
||||
builder: (context, constraints) => FittedBox(
|
||||
child: ConstrainedBox(
|
||||
constraints: BoxConstraints(
|
||||
minWidth: constraints.maxWidth,
|
||||
),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.end,
|
||||
children: userSpecifyItemRight
|
||||
.map((item) => videoProgressWidgets[item]!)
|
||||
.toList(),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
PlPlayerController get plPlayerController => widget.plPlayerController;
|
||||
@@ -1101,34 +1109,36 @@ class _PLVideoPlayerState extends State<PLVideoPlayer>
|
||||
|
||||
// 头部、底部控制条
|
||||
Obx(
|
||||
() => Column(
|
||||
children: [
|
||||
if (widget.headerControl != null ||
|
||||
plPlayerController.headerControl != null)
|
||||
ClipRect(
|
||||
child: AppBarAni(
|
||||
() => Positioned.fill(
|
||||
child: Column(
|
||||
children: [
|
||||
if (widget.headerControl != null ||
|
||||
plPlayerController.headerControl != null)
|
||||
ClipRect(
|
||||
child: AppBarAni(
|
||||
controller: animationController,
|
||||
visible: !plPlayerController.controlsLock.value &&
|
||||
plPlayerController.showControls.value,
|
||||
position: 'top',
|
||||
child: widget.headerControl ??
|
||||
plPlayerController.headerControl!,
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
if (plPlayerController.showControls.value)
|
||||
AppBarAni(
|
||||
controller: animationController,
|
||||
visible: !plPlayerController.controlsLock.value &&
|
||||
plPlayerController.showControls.value,
|
||||
position: 'top',
|
||||
child: widget.headerControl ??
|
||||
plPlayerController.headerControl!,
|
||||
position: 'bottom',
|
||||
child: widget.bottomControl ??
|
||||
BottomControl(
|
||||
controller: plPlayerController,
|
||||
buildBottomControl: buildBottomControl,
|
||||
),
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
if (plPlayerController.showControls.value)
|
||||
AppBarAni(
|
||||
controller: animationController,
|
||||
visible: !plPlayerController.controlsLock.value &&
|
||||
plPlayerController.showControls.value,
|
||||
position: 'bottom',
|
||||
child: widget.bottomControl ??
|
||||
BottomControl(
|
||||
controller: plPlayerController,
|
||||
buildBottomControl: buildBottomControl(),
|
||||
),
|
||||
),
|
||||
],
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
|
||||
@@ -17,11 +17,11 @@ import 'package:PiliPlus/utils/feed_back.dart';
|
||||
import '../../../common/widgets/audio_video_progress_bar.dart';
|
||||
|
||||
class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
final PlPlayerController? controller;
|
||||
final List<Widget>? buildBottomControl;
|
||||
final PlPlayerController controller;
|
||||
final Function buildBottomControl;
|
||||
const BottomControl({
|
||||
this.controller,
|
||||
this.buildBottomControl,
|
||||
required this.controller,
|
||||
required this.buildBottomControl,
|
||||
super.key,
|
||||
});
|
||||
|
||||
@@ -37,13 +37,13 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 10),
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.end,
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Obx(
|
||||
() {
|
||||
final int value = controller!.sliderPositionSeconds.value;
|
||||
final int max = controller!.durationSeconds.value.inSeconds;
|
||||
final int buffer = controller!.bufferedSeconds.value;
|
||||
final int value = controller.sliderPositionSeconds.value;
|
||||
final int max = controller.durationSeconds.value.inSeconds;
|
||||
final int buffer = controller.bufferedSeconds.value;
|
||||
if (value > max || max <= 0) {
|
||||
return nil;
|
||||
}
|
||||
@@ -57,12 +57,12 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
clipBehavior: Clip.none,
|
||||
alignment: Alignment.bottomCenter,
|
||||
children: [
|
||||
if (controller?.dmTrend.isNotEmpty == true &&
|
||||
controller?.showDmChart.value == true)
|
||||
buildDmChart(context, controller!, 4.5),
|
||||
if (controller?.viewPointList.isNotEmpty == true &&
|
||||
controller?.showVP.value == true)
|
||||
buildViewPointWidget(controller!, 8.75),
|
||||
if (controller.dmTrend.isNotEmpty &&
|
||||
controller.showDmChart.value)
|
||||
buildDmChart(context, controller, 4.5),
|
||||
if (controller.viewPointList.isNotEmpty &&
|
||||
controller.showVP.value)
|
||||
buildViewPointWidget(controller, 8.75),
|
||||
ProgressBar(
|
||||
progress: Duration(seconds: value),
|
||||
buffered: Duration(seconds: buffer),
|
||||
@@ -76,16 +76,16 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
thumbRadius: 7,
|
||||
onDragStart: (duration) {
|
||||
feedBack();
|
||||
controller!.onChangedSliderStart();
|
||||
controller.onChangedSliderStart();
|
||||
},
|
||||
onDragUpdate: (duration) {
|
||||
double newProgress =
|
||||
duration.timeStamp.inSeconds / max;
|
||||
if (controller!.showSeekPreview) {
|
||||
if (controller!.showPreview.value.not) {
|
||||
controller!.showPreview.value = true;
|
||||
if (controller.showSeekPreview) {
|
||||
if (controller.showPreview.value.not) {
|
||||
controller.showPreview.value = true;
|
||||
}
|
||||
controller!.previewDx.value =
|
||||
controller.previewDx.value =
|
||||
duration.localPosition.dx;
|
||||
}
|
||||
if ((newProgress - lastAnnouncedValue).abs() > 0.02) {
|
||||
@@ -98,17 +98,17 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
lastAnnouncedValue = newProgress;
|
||||
});
|
||||
}
|
||||
controller!
|
||||
controller
|
||||
.onUpdatedSliderProgress(duration.timeStamp);
|
||||
},
|
||||
onSeek: (duration) {
|
||||
if (controller!.showSeekPreview) {
|
||||
controller!.showPreview.value = false;
|
||||
if (controller.showSeekPreview) {
|
||||
controller.showPreview.value = false;
|
||||
}
|
||||
controller!.onChangedSliderEnd();
|
||||
controller!
|
||||
controller.onChangedSliderEnd();
|
||||
controller
|
||||
.onChangedSlider(duration.inSeconds.toDouble());
|
||||
controller!.seekTo(
|
||||
controller.seekTo(
|
||||
Duration(seconds: duration.inSeconds),
|
||||
type: 'slider');
|
||||
SemanticsService.announce(
|
||||
@@ -116,7 +116,7 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
TextDirection.ltr);
|
||||
},
|
||||
),
|
||||
if (controller?.segmentList.isNotEmpty == true)
|
||||
if (controller.segmentList.isNotEmpty)
|
||||
Positioned(
|
||||
left: 0,
|
||||
right: 0,
|
||||
@@ -125,13 +125,13 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
child: CustomPaint(
|
||||
size: Size(double.infinity, 3.5),
|
||||
painter: SegmentProgressBar(
|
||||
segmentColors: controller!.segmentList,
|
||||
segmentColors: controller.segmentList,
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
if (controller?.viewPointList.isNotEmpty == true &&
|
||||
controller?.showVP.value == true)
|
||||
if (controller.viewPointList.isNotEmpty &&
|
||||
controller.showVP.value)
|
||||
Positioned(
|
||||
left: 0,
|
||||
right: 0,
|
||||
@@ -140,17 +140,17 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
child: CustomPaint(
|
||||
size: Size(double.infinity, 3.5),
|
||||
painter: SegmentProgressBar(
|
||||
segmentColors: controller!.viewPointList,
|
||||
segmentColors: controller.viewPointList,
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
if (controller?.showSeekPreview == true)
|
||||
if (controller.showSeekPreview)
|
||||
Positioned(
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 18,
|
||||
child: buildSeekPreviewWidget(controller!),
|
||||
child: buildSeekPreviewWidget(controller),
|
||||
),
|
||||
],
|
||||
),
|
||||
@@ -158,9 +158,7 @@ class BottomControl extends StatelessWidget implements PreferredSizeWidget {
|
||||
);
|
||||
},
|
||||
),
|
||||
Row(
|
||||
children: [...buildBottomControl!],
|
||||
),
|
||||
buildBottomControl(),
|
||||
const SizedBox(height: 12),
|
||||
],
|
||||
),
|
||||
|
||||
@@ -15,251 +15,522 @@ import 'utils.dart';
|
||||
|
||||
class PiliScheme {
|
||||
static late AppLinks appLinks;
|
||||
static StreamSubscription? listener;
|
||||
|
||||
static Future<void> init() async {
|
||||
// Register our protocol only on Windows platform
|
||||
// registerProtocolHandler('bilibili');
|
||||
appLinks = AppLinks();
|
||||
|
||||
appLinks.uriLinkStream.listen((uri) {
|
||||
listener?.cancel();
|
||||
listener = appLinks.uriLinkStream.listen((uri) {
|
||||
debugPrint('onAppLink: $uri');
|
||||
routePush(uri);
|
||||
});
|
||||
}
|
||||
|
||||
/// 路由跳转
|
||||
static void routePush(Uri value) async {
|
||||
final String scheme = value.scheme;
|
||||
final String host = value.host;
|
||||
final String path = value.path;
|
||||
|
||||
if (scheme == 'bilibili') {
|
||||
debugPrint('$value');
|
||||
if (host == 'root') {
|
||||
Navigator.popUntil(
|
||||
Get.context!, (Route<dynamic> route) => route.isFirst);
|
||||
} else if (host == 'space') {
|
||||
final String mid = path.split('/').last;
|
||||
Utils.toDupNamed(
|
||||
'/member?mid=$mid',
|
||||
arguments: <String, dynamic>{'face': null},
|
||||
);
|
||||
} else if (host == 'video') {
|
||||
String pathQuery = path.split('/').last;
|
||||
if (value.queryParameters['comment_root_id'] != null) {
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往原视频',
|
||||
onPressed: () {
|
||||
String? enterUri = value.toString().split('?').first;
|
||||
routePush(Uri.parse(enterUri));
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: int.tryParse(pathQuery),
|
||||
rpid: int.tryParse(value.queryParameters['comment_root_id']!),
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.video,
|
||||
firstFloor: null,
|
||||
),
|
||||
),
|
||||
);
|
||||
return;
|
||||
}
|
||||
final numericRegex = RegExp(r'^[0-9]+$');
|
||||
if (numericRegex.hasMatch(pathQuery)) {
|
||||
pathQuery = 'AV$pathQuery';
|
||||
}
|
||||
Map map = IdUtils.matchAvorBv(input: pathQuery);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(map['AV'], map['BV']);
|
||||
} else {
|
||||
SmartDialog.showToast('投稿匹配失败');
|
||||
}
|
||||
} else if (host == 'live') {
|
||||
final String roomId = path.split('/').last;
|
||||
Utils.toDupNamed('/liveRoom?roomid=$roomId');
|
||||
} else if (host == 'bangumi') {
|
||||
if (path.startsWith('/season')) {
|
||||
final String seasonId = path.split('/').last;
|
||||
bangumiPush(int.parse(seasonId), null);
|
||||
}
|
||||
} else if (host == 'opus') {
|
||||
if (path.startsWith('/detail')) {
|
||||
var opusId = path.split('/').last;
|
||||
Utils.toDupNamed(
|
||||
'/webview',
|
||||
parameters: {
|
||||
'url': 'https://www.bilibili.com/opus/$opusId',
|
||||
'type': 'url',
|
||||
'pageTitle': '',
|
||||
},
|
||||
);
|
||||
}
|
||||
} else if (host == 'search') {
|
||||
Utils.toDupNamed('/searchResult', parameters: {'keyword': ''});
|
||||
} else if (host == 'article') {
|
||||
final String id = path.split('/').last.split('?').first;
|
||||
Utils.toDupNamed(
|
||||
'/htmlRender',
|
||||
parameters: {
|
||||
'url': 'www.bilibili.com/read/cv$id',
|
||||
'title': '',
|
||||
'id': 'cv$id',
|
||||
'dynamicType': 'read'
|
||||
},
|
||||
);
|
||||
} else if (host == 'comment' && path.startsWith("/detail/")) {
|
||||
//bilibili://comment/detail/17/832703053858603029/238686570016/?subType=0&anchor=238686628816&showEnter=1&extraIntentId=0&scene=1&enterName=%E6%9F%A5%E7%9C%8B%E5%8A%A8%E6%80%81%E8%AF%A6%E6%83%85&enterUri=bilibili://following/detail/832703053858603029
|
||||
//fmt.Sprintf("bilibili://comment/detail/%d/%d/%d/?subType=%d&anchor=%d&showEnter=1&extraIntentId=%d", rp.Type, rp.Oid, rootID, subType, rp.RpID, extraIntentID)
|
||||
debugPrint('${value.queryParameters}');
|
||||
List<String> pathParts = path.split('/');
|
||||
int type = int.parse(pathParts[2]);
|
||||
int oid = int.parse(pathParts[3]);
|
||||
int rootId = int.parse(pathParts[4]);
|
||||
// int subType = int.parse(value.queryParameters['subType'] ?? '0');
|
||||
// int rpID = int.parse(value.queryParameters['anchor'] ?? '0');
|
||||
// int extraIntentId =
|
||||
// int.parse(value.queryParameters['extraIntentId'] ?? '0');
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往',
|
||||
onPressed: () {
|
||||
String? enterUri = value.queryParameters['enterUri'];
|
||||
if (enterUri != null) {
|
||||
routePush(Uri.parse(enterUri));
|
||||
}
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: oid,
|
||||
rpid: rootId, // rpID,
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.values[type],
|
||||
firstFloor: null,
|
||||
),
|
||||
),
|
||||
);
|
||||
} else if (host == 'following' && path.startsWith("/detail/")) {
|
||||
void getToOpusWeb() async {
|
||||
String? id = RegExp(r'detail/(\d+)').firstMatch(path)?.group(1);
|
||||
if (id != null) {
|
||||
SmartDialog.showLoading();
|
||||
dynamic res = await DynamicsHttp.dynamicDetail(id: id);
|
||||
SmartDialog.dismiss();
|
||||
if (res['status']) {
|
||||
Get.toNamed('/dynamicDetail', arguments: {
|
||||
'item': res['data'],
|
||||
'floor': 1,
|
||||
'action': 'detail'
|
||||
});
|
||||
} else {
|
||||
SmartDialog.showToast(res['msg']);
|
||||
}
|
||||
} else {
|
||||
var opusId = path.split('/').last;
|
||||
Utils.toDupNamed(
|
||||
'/webview',
|
||||
parameters: {
|
||||
'url': 'https://m.bilibili.com/dynamic/$opusId',
|
||||
'type': 'url',
|
||||
'pageTitle': '',
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
if (value.queryParameters['comment_root_id'] != null) {
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往',
|
||||
onPressed: () {
|
||||
getToOpusWeb();
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: int.tryParse(path.split('/').last),
|
||||
rpid: int.tryParse(value.queryParameters['comment_root_id']!),
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.dynamics,
|
||||
firstFloor: null),
|
||||
),
|
||||
);
|
||||
} else {
|
||||
getToOpusWeb();
|
||||
}
|
||||
} else if (host == 'album') {
|
||||
String? rid =
|
||||
RegExp(r'album/(\d+)').firstMatch(value.toString())?.group(1);
|
||||
if (rid != null) {
|
||||
SmartDialog.showLoading();
|
||||
dynamic res = await DynamicsHttp.dynamicDetail(rid: rid, type: 2);
|
||||
SmartDialog.dismiss();
|
||||
if (res['status']) {
|
||||
Get.toNamed('/dynamicDetail', arguments: {
|
||||
'item': res['data'],
|
||||
'floor': 1,
|
||||
'action': 'detail'
|
||||
});
|
||||
} else {
|
||||
SmartDialog.showToast(res['msg']);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
debugPrint('$value');
|
||||
SmartDialog.showToast('未知路径:$value,请截图反馈给开发者');
|
||||
//Utils.toDupNamed(
|
||||
// '/webview',
|
||||
// parameters: {
|
||||
// 'url': ,
|
||||
// 'type': 'url',
|
||||
// 'pageTitle': ''
|
||||
// },
|
||||
// );
|
||||
}
|
||||
} else if (['http', 'https'].contains(scheme)) {
|
||||
fullPathPush(value);
|
||||
} else if (path.toLowerCase().startsWith('av')) {
|
||||
try {
|
||||
videoPush(int.parse(path.substring(2)), null);
|
||||
} catch (e) {
|
||||
debugPrint(e.toString());
|
||||
}
|
||||
} else if (path.toLowerCase().startsWith('bv')) {
|
||||
try {
|
||||
videoPush(null, path);
|
||||
} catch (e) {
|
||||
debugPrint(e.toString());
|
||||
static Future<bool> routePushFromUrl(
|
||||
String url, {
|
||||
bool selfHandle = false,
|
||||
bool off = false,
|
||||
}) async {
|
||||
try {
|
||||
if (url.startsWith('//')) {
|
||||
url = 'https:$url';
|
||||
} else if (RegExp(r'^\S+://').hasMatch(url).not) {
|
||||
url = 'https://$url';
|
||||
}
|
||||
return await routePush(Uri.parse(url), selfHandle: selfHandle, off: off);
|
||||
} catch (_) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/// 路由跳转
|
||||
static Future<bool> routePush(
|
||||
Uri uri, {
|
||||
bool selfHandle = false,
|
||||
bool off = false,
|
||||
}) async {
|
||||
final String scheme = uri.scheme;
|
||||
final String host = uri.host.toLowerCase();
|
||||
final String path = uri.path;
|
||||
|
||||
switch (scheme) {
|
||||
case 'bilibili':
|
||||
switch (host) {
|
||||
case 'root':
|
||||
Navigator.popUntil(
|
||||
Get.context!,
|
||||
(Route<dynamic> route) => route.isFirst,
|
||||
);
|
||||
return true;
|
||||
case 'pgc':
|
||||
// bilibili://pgc/season/ep/123456?h5_awaken_params=random
|
||||
String? id = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (id != null) {
|
||||
bool isEp = path.contains('/ep/');
|
||||
Utils.viewBangumi(
|
||||
seasonId: isEp ? null : id,
|
||||
epId: isEp ? id : null,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'space':
|
||||
// bilibili://space/12345678?frommodule=XX&h5awaken=random
|
||||
String? mid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (mid != null) {
|
||||
Utils.toDupNamed('/member?mid=$mid', off: off);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'video':
|
||||
if (uri.queryParameters['comment_root_id'] != null) {
|
||||
// to check
|
||||
// to video reply
|
||||
String? oid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (oid != null) {
|
||||
int? rpid =
|
||||
int.tryParse(uri.queryParameters['comment_root_id']!);
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往原视频',
|
||||
onPressed: () {
|
||||
String? enterUri =
|
||||
uri.toString().split('?').first; // to check
|
||||
routePush(Uri.parse(enterUri));
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: int.parse(oid),
|
||||
rpid: rpid,
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.video,
|
||||
firstFloor: null,
|
||||
),
|
||||
),
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// to video
|
||||
// bilibili://video/12345678?page=0&h5awaken=random
|
||||
String? aid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
String? bvid = RegExp(r'/(BV[a-z\d]{10})', caseSensitive: false)
|
||||
.firstMatch(path)
|
||||
?.group(1);
|
||||
if (aid != null || bvid != null) {
|
||||
videoPush(
|
||||
aid != null ? int.parse(aid) : null,
|
||||
bvid,
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'live':
|
||||
// bilibili://live/12345678?extra_jump_from=1&from=1&is_room_feed=1&h5awaken=random
|
||||
String? roomId = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (roomId != null) {
|
||||
Utils.toDupNamed('/liveRoom?roomid=$roomId', off: off);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'bangumi':
|
||||
// bilibili://bangumi/season/12345678?h5_awaken_params=random
|
||||
if (path.startsWith('/season')) {
|
||||
String? seasonId = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (seasonId != null) {
|
||||
Utils.viewBangumi(seasonId: seasonId, epId: null);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
case 'opus':
|
||||
// bilibili://opus/detail/12345678?h5awaken=random
|
||||
if (path.startsWith('/detail')) {
|
||||
bool hasMatch = await _onPushDynDetail(path, off);
|
||||
return hasMatch;
|
||||
}
|
||||
return false;
|
||||
case 'search':
|
||||
Utils.toDupNamed(
|
||||
'/searchResult',
|
||||
parameters: {'keyword': ''},
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
case 'article':
|
||||
// bilibili://article/40679479?jump_opus=1&jump_opus_type=1&opus_type=article&h5awaken=random
|
||||
String? id = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (id != null) {
|
||||
Utils.toDupNamed(
|
||||
'/htmlRender',
|
||||
parameters: {
|
||||
'url': 'www.bilibili.com/read/cv$id',
|
||||
'title': '',
|
||||
'id': 'cv$id',
|
||||
'dynamicType': 'read'
|
||||
},
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'comment':
|
||||
if (path.startsWith("/detail/")) {
|
||||
// bilibili://comment/detail/17/832703053858603029/238686570016/?subType=0&anchor=238686628816&showEnter=1&extraIntentId=0&scene=1&enterName=%E6%9F%A5%E7%9C%8B%E5%8A%A8%E6%80%81%E8%AF%A6%E6%83%85&enterUri=bilibili://following/detail/832703053858603029
|
||||
List<String> pathSegments = uri.pathSegments;
|
||||
int type = int.parse(pathSegments[1]);
|
||||
int oid = int.parse(pathSegments[2]);
|
||||
int rootId = int.parse(pathSegments[3]);
|
||||
// int subType = int.parse(value.queryParameters['subType'] ?? '0');
|
||||
// int rpID = int.parse(value.queryParameters['anchor'] ?? '0');
|
||||
// int extraIntentId =
|
||||
// int.parse(value.queryParameters['extraIntentId'] ?? '0');
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往',
|
||||
onPressed: () {
|
||||
String? enterUri = uri.queryParameters['enterUri'];
|
||||
if (enterUri != null) {
|
||||
routePush(Uri.parse(enterUri));
|
||||
}
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: oid,
|
||||
rpid: rootId,
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.values[type],
|
||||
firstFloor: null,
|
||||
),
|
||||
),
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
case 'following':
|
||||
if (path.startsWith("/detail/")) {
|
||||
if (uri.queryParameters['comment_root_id'] != null) {
|
||||
String? oid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (oid != null) {
|
||||
int? rpid =
|
||||
int.tryParse(uri.queryParameters['comment_root_id']!);
|
||||
Get.to(
|
||||
() => Scaffold(
|
||||
resizeToAvoidBottomInset: false,
|
||||
appBar: AppBar(
|
||||
title: const Text('评论详情'),
|
||||
actions: [
|
||||
IconButton(
|
||||
tooltip: '前往',
|
||||
onPressed: () {
|
||||
_onPushDynDetail(path, off);
|
||||
},
|
||||
icon: const Icon(Icons.open_in_new),
|
||||
),
|
||||
],
|
||||
),
|
||||
body: VideoReplyReplyPanel(
|
||||
oid: int.tryParse(oid),
|
||||
rpid: rpid,
|
||||
source: 'routePush',
|
||||
replyType: ReplyType.dynamics,
|
||||
firstFloor: null),
|
||||
),
|
||||
);
|
||||
}
|
||||
return true;
|
||||
} else {
|
||||
bool hasMatch = await _onPushDynDetail(path, off);
|
||||
return hasMatch;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
case 'album':
|
||||
String? rid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (rid != null) {
|
||||
SmartDialog.showLoading();
|
||||
dynamic res = await DynamicsHttp.dynamicDetail(rid: rid, type: 2);
|
||||
SmartDialog.dismiss();
|
||||
if (res['status']) {
|
||||
Utils.toDupNamed(
|
||||
'/dynamicDetail',
|
||||
arguments: {
|
||||
'item': res['data'],
|
||||
'floor': 1,
|
||||
'action': 'detail'
|
||||
},
|
||||
off: off,
|
||||
);
|
||||
} else {
|
||||
SmartDialog.showToast(res['msg']);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
default:
|
||||
if (selfHandle.not) {
|
||||
debugPrint('$uri');
|
||||
SmartDialog.showToast('未知路径:$uri,请截图反馈给开发者');
|
||||
}
|
||||
return false;
|
||||
}
|
||||
case 'http' || 'https':
|
||||
return await _fullPathPush(uri, selfHandle: selfHandle, off: off);
|
||||
default:
|
||||
String? aid = RegExp(r'^av(\d+)', caseSensitive: false)
|
||||
.firstMatch(path)
|
||||
?.group(1);
|
||||
String? bvid = RegExp(r'^BV[a-z\d]{10}', caseSensitive: false)
|
||||
.firstMatch(path)
|
||||
?.group(0);
|
||||
if (aid != null || bvid != null) {
|
||||
videoPush(
|
||||
aid != null ? int.parse(aid) : null,
|
||||
bvid,
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
if (selfHandle.not) {
|
||||
debugPrint('$uri');
|
||||
SmartDialog.showToast('未知路径:$uri,请截图反馈给开发者');
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
static Future<bool> _fullPathPush(
|
||||
Uri uri, {
|
||||
bool selfHandle = false,
|
||||
bool off = false,
|
||||
}) async {
|
||||
// https://m.bilibili.com/bangumi/play/ss39708
|
||||
// https | m.bilibili.com | /bangumi/play/ss39708
|
||||
|
||||
String host = uri.host.toLowerCase();
|
||||
|
||||
if (selfHandle &&
|
||||
host.contains('bilibili.com').not &&
|
||||
host.contains('b23.tv').not) {
|
||||
return false;
|
||||
}
|
||||
|
||||
void launchURL() {
|
||||
if (selfHandle.not) {
|
||||
_toWebview(uri.toString(), off);
|
||||
}
|
||||
}
|
||||
|
||||
// b23.tv
|
||||
// bilibili.com
|
||||
// m.bilibili.com
|
||||
// www.bilibili.com
|
||||
// space.bilibili.com
|
||||
// live.bilibili.com
|
||||
|
||||
// redirect
|
||||
if (host.contains('b23.tv')) {
|
||||
String? redirectUrl = await UrlUtils.parseRedirectUrl(uri.toString());
|
||||
if (redirectUrl != null) {
|
||||
uri = Uri.parse(redirectUrl);
|
||||
host = uri.host.toLowerCase();
|
||||
}
|
||||
if (host.contains('bilibili.com').not) {
|
||||
launchURL();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
final String path = uri.path;
|
||||
|
||||
if (host.contains('t.bilibili.com')) {
|
||||
bool hasMatch = await _onPushDynDetail(path, off);
|
||||
if (hasMatch.not) {
|
||||
launchURL();
|
||||
}
|
||||
return hasMatch;
|
||||
}
|
||||
|
||||
if (host.contains('live.bilibili.com')) {
|
||||
String? roomId = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (roomId != null) {
|
||||
Utils.toDupNamed('/liveRoom?roomid=$roomId', off: off);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
}
|
||||
|
||||
if (host.contains('space.bilibili.com')) {
|
||||
String? mid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (mid != null) {
|
||||
Utils.toDupNamed('/member?mid=$mid', off: off);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
}
|
||||
|
||||
List<String> pathSegments = uri.pathSegments;
|
||||
if (pathSegments.isEmpty) {
|
||||
launchURL();
|
||||
return false;
|
||||
}
|
||||
final String? area = pathSegments.first == 'mobile'
|
||||
? pathSegments.getOrNull(1)
|
||||
: pathSegments.first;
|
||||
switch (area) {
|
||||
case 'opus':
|
||||
bool hasMatch = await _onPushDynDetail(path, off);
|
||||
if (hasMatch.not) {
|
||||
launchURL();
|
||||
}
|
||||
return hasMatch;
|
||||
case 'playlist':
|
||||
String? bvid = uri.queryParameters['bvid'] ??
|
||||
RegExp(r'/(BV[a-z\d]{10})', caseSensitive: false)
|
||||
.firstMatch(path)
|
||||
?.group(1);
|
||||
if (bvid != null) {
|
||||
videoPush(null, bvid, off: false);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
case 'bangumi':
|
||||
debugPrint('番剧');
|
||||
String? id = RegExp(r'(ss|ep)\d+').firstMatch(path)?.group(0);
|
||||
if (id != null) {
|
||||
bool isSeason = id.startsWith('ss');
|
||||
id = id.substring(2);
|
||||
Utils.viewBangumi(
|
||||
seasonId: isSeason ? id : null,
|
||||
epId: isSeason ? null : id,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
case 'video':
|
||||
debugPrint('投稿');
|
||||
final Map<String, dynamic> map = IdUtils.matchAvorBv(input: path);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(
|
||||
map['AV'],
|
||||
map['BV'],
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
case 'read':
|
||||
debugPrint('专栏');
|
||||
String? id =
|
||||
RegExp(r'cv(\d+)', caseSensitive: false).firstMatch(path)?.group(1);
|
||||
if (id != null) {
|
||||
Utils.toDupNamed(
|
||||
'/htmlRender',
|
||||
parameters: {
|
||||
'url': 'https://www.bilibili.com/read/cv$id',
|
||||
'title': '',
|
||||
'id': 'cv$id',
|
||||
'dynamicType': 'read'
|
||||
},
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
case 'space':
|
||||
debugPrint('个人空间');
|
||||
String? mid = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (mid != null) {
|
||||
Utils.toDupNamed(
|
||||
'/member?mid=$mid',
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
default:
|
||||
Map map = IdUtils.matchAvorBv(input: area?.split('?').first);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(
|
||||
map['AV'],
|
||||
map['BV'],
|
||||
off: off,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
launchURL();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
static Future<bool> _onPushDynDetail(path, off) async {
|
||||
String? id = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||
if (id != null) {
|
||||
SmartDialog.showLoading();
|
||||
dynamic res = await DynamicsHttp.dynamicDetail(id: id);
|
||||
SmartDialog.dismiss();
|
||||
if (res['status']) {
|
||||
Utils.toDupNamed(
|
||||
'/dynamicDetail',
|
||||
arguments: {
|
||||
'item': res['data'],
|
||||
'floor': 1,
|
||||
'action': 'detail',
|
||||
},
|
||||
off: off,
|
||||
);
|
||||
} else {
|
||||
SmartDialog.showToast(res['msg']);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static void _toWebview(String url, bool off) {
|
||||
Utils.toDupNamed(
|
||||
'/webview',
|
||||
parameters: {'url': url},
|
||||
off: off,
|
||||
);
|
||||
}
|
||||
|
||||
// 投稿跳转
|
||||
static Future<void> videoPush(int? aid, String? bvid,
|
||||
[bool showDialog = true]) async {
|
||||
static Future<void> videoPush(
|
||||
int? aid,
|
||||
String? bvid, {
|
||||
bool showDialog = true,
|
||||
bool off = false,
|
||||
}) async {
|
||||
try {
|
||||
aid ??= IdUtils.bv2av(bvid!);
|
||||
bvid ??= IdUtils.av2bv(aid);
|
||||
@@ -272,187 +543,15 @@ class PiliScheme {
|
||||
}
|
||||
Utils.toDupNamed(
|
||||
'/video?bvid=$bvid&cid=$cid',
|
||||
arguments: <String, String?>{
|
||||
arguments: {
|
||||
'pic': null,
|
||||
'heroTag': Utils.makeHeroTag(aid),
|
||||
},
|
||||
off: off,
|
||||
);
|
||||
} catch (e) {
|
||||
SmartDialog.dismiss();
|
||||
SmartDialog.showToast('video获取失败: $e');
|
||||
}
|
||||
}
|
||||
|
||||
// 番剧跳转
|
||||
static Future<void> bangumiPush(int? seasonId, int? epId) async {
|
||||
debugPrint('seasonId: $seasonId, epId: $epId');
|
||||
// SmartDialog.showLoading<dynamic>(msg: '获取中...');
|
||||
try {
|
||||
Utils.viewBangumi(seasonId: seasonId, epId: epId);
|
||||
// var result = await SearchHttp.bangumiInfo(seasonId: seasonId, epId: epId);
|
||||
// if (result['status']) {
|
||||
// var bangumiDetail = result['data'];
|
||||
// EpisodeItem episode = result['data'].episodes.first;
|
||||
// int? epId = result['data'].userStatus?.progress?.lastEpId;
|
||||
// if (epId == null) {
|
||||
// epId = episode.epId;
|
||||
// } else {
|
||||
// for (var item in result['data'].episodes) {
|
||||
// if (item.epId == epId) {
|
||||
// episode = item;
|
||||
// break;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// String bvid = episode.bvid!;
|
||||
// int cid = episode.cid!;
|
||||
// dynamic pic = episode.cover;
|
||||
// final String heroTag = Utils.makeHeroTag(cid);
|
||||
// SmartDialog.dismiss().then(
|
||||
// (e) => Utils.toDupNamed(
|
||||
// '/video?bvid=$bvid&cid=$cid&seasonId=${bangumiDetail.seasonId}&epId=$epId',
|
||||
// arguments: <String, dynamic>{
|
||||
// 'pic': pic,
|
||||
// 'heroTag': heroTag,
|
||||
// 'videoType': SearchType.media_bangumi,
|
||||
// },
|
||||
// ),
|
||||
// );
|
||||
// } else {
|
||||
// SmartDialog.showToast(result['msg']);
|
||||
// }
|
||||
} catch (e) {
|
||||
SmartDialog.showToast('番剧获取失败:$e');
|
||||
}
|
||||
}
|
||||
|
||||
static Future<void> fullPathPush(Uri value) async {
|
||||
// https://m.bilibili.com/bangumi/play/ss39708
|
||||
// https | m.bilibili.com | /bangumi/play/ss39708
|
||||
// final String scheme = value.scheme!;
|
||||
final String host = value.host;
|
||||
final String path = value.path;
|
||||
Map<String, String> query = value.queryParameters;
|
||||
RegExp regExp = RegExp(r'^((www\.)|(m\.))?bilibili\.com$');
|
||||
if (regExp.hasMatch(host)) {
|
||||
debugPrint('bilibili.com');
|
||||
} else if (host.contains('live')) {
|
||||
int roomId = int.parse(path.split('/').last);
|
||||
Utils.toDupNamed('/liveRoom?roomid=$roomId');
|
||||
return;
|
||||
} else if (host.contains('space')) {
|
||||
var mid = path.split('/').last;
|
||||
Utils.toDupNamed('/member?mid=$mid', arguments: {'face': ''});
|
||||
return;
|
||||
} else if (host == 'b23.tv') {
|
||||
final String fullPath = 'https://$host$path';
|
||||
final String redirectUrl =
|
||||
(await UrlUtils.parseRedirectUrl(fullPath)) ?? fullPath;
|
||||
final String pathSegment = Uri.parse(redirectUrl).path;
|
||||
final String lastPathSegment = pathSegment.split('/').last;
|
||||
final RegExp avRegex = RegExp(r'^[aA][vV]\d+', caseSensitive: false);
|
||||
if (avRegex.hasMatch(lastPathSegment)) {
|
||||
final Map<String, dynamic> map =
|
||||
IdUtils.matchAvorBv(input: lastPathSegment);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(map['AV'], map['BV']);
|
||||
} else {
|
||||
SmartDialog.showToast('投稿匹配失败');
|
||||
}
|
||||
} else if (lastPathSegment.startsWith('ep')) {
|
||||
handleEpisodePath(lastPathSegment, redirectUrl);
|
||||
} else if (lastPathSegment.startsWith('ss')) {
|
||||
handleSeasonPath(lastPathSegment, redirectUrl);
|
||||
} else if (lastPathSegment.startsWith('BV')) {
|
||||
UrlUtils.matchUrlPush(
|
||||
lastPathSegment,
|
||||
redirectUrl,
|
||||
);
|
||||
} else {
|
||||
Utils.handleWebview(redirectUrl);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
List<String> pathPart = path.split('/');
|
||||
if (pathPart.length < 3) {
|
||||
Utils.handleWebview(value.toString());
|
||||
return;
|
||||
}
|
||||
final String area = pathPart[1] == 'mobile' ? pathPart[2] : pathPart[1];
|
||||
switch (area) {
|
||||
case 'bangumi':
|
||||
debugPrint('番剧');
|
||||
for (var pathSegment in pathPart) {
|
||||
if (pathSegment.startsWith('ss')) {
|
||||
bangumiPush(matchNum(pathSegment).first, null);
|
||||
return;
|
||||
} else if (pathSegment.startsWith('ep')) {
|
||||
bangumiPush(null, matchNum(pathSegment).first);
|
||||
return;
|
||||
}
|
||||
}
|
||||
Utils.handleWebview(value.toString());
|
||||
break;
|
||||
case 'video':
|
||||
debugPrint('投稿');
|
||||
final Map<String, dynamic> map = IdUtils.matchAvorBv(input: path);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(map['AV'], map['BV']);
|
||||
} else {
|
||||
SmartDialog.showToast('投稿匹配失败');
|
||||
}
|
||||
break;
|
||||
case 'read':
|
||||
debugPrint('专栏');
|
||||
late String id;
|
||||
if (query['id'] != null) {
|
||||
id = 'cv${matchNum(query['id']!).first}';
|
||||
} else {
|
||||
id = 'cv${matchNum(path).firstOrNull}';
|
||||
}
|
||||
Utils.toDupNamed('/htmlRender', parameters: {
|
||||
'url': value.toString(),
|
||||
'title': '',
|
||||
'id': id,
|
||||
'dynamicType': 'read'
|
||||
});
|
||||
break;
|
||||
case 'space':
|
||||
debugPrint('个人空间');
|
||||
Utils.toDupNamed(
|
||||
'/member?mid=${pathPart[1] == 'mobile' ? pathPart.getOrNull(3) : pathPart.getOrNull(2)}',
|
||||
arguments: {'face': ''});
|
||||
break;
|
||||
default:
|
||||
Map map = IdUtils.matchAvorBv(input: area.split('?').first);
|
||||
if (map.isNotEmpty) {
|
||||
videoPush(map['AV'], map['BV']);
|
||||
} else {
|
||||
// SmartDialog.showToast('未知路径或匹配错误:$value,先采用浏览器打开');
|
||||
Utils.handleWebview(value.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static List<int> matchNum(String str) {
|
||||
final RegExp regExp = RegExp(r'\d+');
|
||||
final Iterable<Match> matches = regExp.allMatches(str);
|
||||
|
||||
return matches.map((Match match) => int.parse(match.group(0)!)).toList();
|
||||
}
|
||||
|
||||
static void handleEpisodePath(String lastPathSegment, String redirectUrl) {
|
||||
final String seasonId = extractIdFromPath(lastPathSegment);
|
||||
bangumiPush(null, matchNum(seasonId).first);
|
||||
}
|
||||
|
||||
static void handleSeasonPath(String lastPathSegment, String redirectUrl) {
|
||||
final String seasonId = extractIdFromPath(lastPathSegment);
|
||||
bangumiPush(matchNum(seasonId).first, null);
|
||||
}
|
||||
|
||||
static String extractIdFromPath(String lastPathSegment) {
|
||||
return lastPathSegment.split('/').last;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||
import 'package:get/get.dart';
|
||||
|
||||
@@ -9,8 +10,10 @@ import 'utils.dart';
|
||||
|
||||
class UrlUtils {
|
||||
// 302重定向路由截取
|
||||
static Future<String?> parseRedirectUrl(String url,
|
||||
[bool returnOri = false]) async {
|
||||
static Future<String?> parseRedirectUrl(
|
||||
String url, [
|
||||
bool returnOri = false,
|
||||
]) async {
|
||||
try {
|
||||
final response = await Request().get(
|
||||
url,
|
||||
@@ -23,9 +26,10 @@ class UrlUtils {
|
||||
);
|
||||
if (response.statusCode == 302 || response.statusCode == 301) {
|
||||
String? redirectUrl = response.headers['location']?.first;
|
||||
debugPrint('redirectUrl: $redirectUrl');
|
||||
if (redirectUrl != null) {
|
||||
if (redirectUrl.startsWith('/')) {
|
||||
return url;
|
||||
return returnOri ? url : null;
|
||||
}
|
||||
if (redirectUrl.endsWith('/')) {
|
||||
redirectUrl = redirectUrl.substring(0, redirectUrl.length - 1);
|
||||
|
||||
@@ -370,46 +370,13 @@ class Utils {
|
||||
);
|
||||
}
|
||||
|
||||
static bool _handleInAppWebview(String url) {
|
||||
if (RegExp(
|
||||
r'^(https?://)?((www|m).)?(bilibili|b23).(com|tv)/video/BV[a-zA-Z\d]+')
|
||||
.hasMatch(url)) {
|
||||
try {
|
||||
String? bvid = RegExp(r'BV[a-zA-Z\d]+').firstMatch(url)?.group(0);
|
||||
if (bvid != null) {
|
||||
PiliScheme.videoPush(null, bvid);
|
||||
return true;
|
||||
}
|
||||
} catch (_) {}
|
||||
} else if (RegExp(
|
||||
r'^(https?://)?((www|m).)?(bilibili|b23).(com|tv)/playlist')
|
||||
.hasMatch(url)) {
|
||||
try {
|
||||
String? bvid =
|
||||
RegExp(r'bvid=(BV[a-zA-Z\d]+)').firstMatch(url)?.group(1);
|
||||
if (bvid != null) {
|
||||
PiliScheme.videoPush(null, bvid);
|
||||
return true;
|
||||
}
|
||||
} catch (_) {}
|
||||
} else if (RegExp(r'^(https?://)?((www|m).)?(bilibili|b23).(com|tv)')
|
||||
.hasMatch(url)) {
|
||||
toDupNamed(
|
||||
'/webview',
|
||||
parameters: {'url': url},
|
||||
);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static void handleWebview(
|
||||
String url, {
|
||||
bool off = false,
|
||||
bool inApp = false,
|
||||
}) {
|
||||
}) async {
|
||||
if (inApp.not && GStorage.openInBrowser) {
|
||||
if (_handleInAppWebview(url).not) {
|
||||
if ((await PiliScheme.routePushFromUrl(url, selfHandle: true)).not) {
|
||||
launchURL(url);
|
||||
}
|
||||
} else {
|
||||
@@ -419,12 +386,7 @@ class Utils {
|
||||
parameters: {'url': url},
|
||||
);
|
||||
} else {
|
||||
if (_handleInAppWebview(url).not) {
|
||||
toDupNamed(
|
||||
'/webview',
|
||||
parameters: {'url': url},
|
||||
);
|
||||
}
|
||||
PiliScheme.routePushFromUrl(url);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -757,8 +719,9 @@ class Utils {
|
||||
if (item.epId.toString() == epId.toString()) {
|
||||
// view as normal video
|
||||
Utils.toDupNamed(
|
||||
'/video?bvid=${item.bvid}&cid=${item.cid}',
|
||||
'/video?bvid=${item.bvid}&cid=${item.cid}&seasonId=${data.seasonId}&epId=${item.epId}',
|
||||
arguments: {
|
||||
'pgcApi': true,
|
||||
'pic': item.cover,
|
||||
'heroTag': Utils.makeHeroTag(item.cid),
|
||||
'videoType': SearchType.video,
|
||||
@@ -806,13 +769,23 @@ class Utils {
|
||||
String page, {
|
||||
dynamic arguments,
|
||||
Map<String, String>? parameters,
|
||||
bool off = false,
|
||||
}) {
|
||||
Get.toNamed(
|
||||
page,
|
||||
arguments: arguments,
|
||||
parameters: parameters,
|
||||
preventDuplicates: false,
|
||||
);
|
||||
if (off) {
|
||||
Get.offNamed(
|
||||
page,
|
||||
arguments: arguments,
|
||||
parameters: parameters,
|
||||
preventDuplicates: false,
|
||||
);
|
||||
} else {
|
||||
Get.toNamed(
|
||||
page,
|
||||
arguments: arguments,
|
||||
parameters: parameters,
|
||||
preventDuplicates: false,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
static Future copyText(
|
||||
|
||||
Reference in New Issue
Block a user