mirror of
https://github.com/HChaZZY/PiliPlus.git
synced 2025-12-06 09:13:48 +08:00
opt: jump url
Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
This commit is contained in:
@@ -58,7 +58,7 @@ class VideoCardHGrpc extends StatelessWidget {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
PiliScheme.routePush(Uri.parse(videoItem.smallCoverV5.base.uri));
|
PiliScheme.routePushFromUrl(videoItem.smallCoverV5.base.uri);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
SmartDialog.showToast(err.toString());
|
SmartDialog.showToast(err.toString());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -104,13 +104,12 @@ InlineSpan? richNode(item, context) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (url.startsWith('//')) {
|
if (url.startsWith('//')) {
|
||||||
url = url.replaceFirst('//', 'https://');
|
PiliScheme.routePushFromUrl('https:$url');
|
||||||
PiliScheme.routePush(Uri.parse(url));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Utils.handleWebview(url.startsWith('//')
|
Utils.handleWebview(
|
||||||
? "https://${url.split('//').last}"
|
url.startsWith('//') ? "https://$url" : url,
|
||||||
: url);
|
);
|
||||||
},
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
i.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/models/user/history.dart';
|
||||||
import 'package:PiliPlus/pages/common/multi_select_controller.dart';
|
import 'package:PiliPlus/pages/common/multi_select_controller.dart';
|
||||||
import 'package:PiliPlus/pages/fav_search/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/material.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
@@ -53,8 +52,15 @@ class HistoryItem extends StatelessWidget {
|
|||||||
// 'pageTitle': videoItem.title
|
// 'pageTitle': videoItem.title
|
||||||
// },
|
// },
|
||||||
// );
|
// );
|
||||||
PiliScheme.routePush(Uri.parse(
|
Utils.toDupNamed(
|
||||||
"https://www.bilibili.com/read/cv${videoItem.history.oid}"));
|
'/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') {
|
} else if (videoItem.history.business == 'live') {
|
||||||
if (videoItem.liveStatus == 1) {
|
if (videoItem.liveStatus == 1) {
|
||||||
// LiveItemModel liveItem = LiveItemModel.fromJson({
|
// LiveItemModel liveItem = LiveItemModel.fromJson({
|
||||||
|
|||||||
@@ -60,7 +60,7 @@ class _MemberArticleState extends State<MemberArticle>
|
|||||||
return ListTile(
|
return ListTile(
|
||||||
dense: true,
|
dense: true,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
PiliScheme.routePush(Uri.parse(item.uri ?? ''));
|
PiliScheme.routePushFromUrl(item.uri ?? '');
|
||||||
},
|
},
|
||||||
leading: item.originImageUrls?.isNotEmpty == true
|
leading: item.originImageUrls?.isNotEmpty == true
|
||||||
? Container(
|
? Container(
|
||||||
|
|||||||
@@ -142,7 +142,7 @@ class _MemberFavoriteState extends State<MemberFavorite>
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
} else if (item1.type == 21) {
|
} else if (item1.type == 21) {
|
||||||
PiliScheme.routePush(Uri.parse(item1.link ?? ''));
|
PiliScheme.routePushFromUrl(item1.link ?? '');
|
||||||
} else if (item1.type == 11) {
|
} else if (item1.type == 11) {
|
||||||
Get.toNamed(
|
Get.toNamed(
|
||||||
'/subDetail',
|
'/subDetail',
|
||||||
|
|||||||
@@ -118,9 +118,9 @@ class _MemberHomeState extends State<MemberHome>
|
|||||||
child: ListTile(
|
child: ListTile(
|
||||||
dense: true,
|
dense: true,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
PiliScheme.routePush(Uri.parse(
|
PiliScheme.routePushFromUrl(
|
||||||
loadingState.response.article.item.first.uri ??
|
loadingState.response.article.item.first.uri ?? '',
|
||||||
''));
|
);
|
||||||
},
|
},
|
||||||
leading: loadingState.response.article.item.first
|
leading: loadingState.response.article.item.first
|
||||||
.originImageUrls?.isNotEmpty ==
|
.originImageUrls?.isNotEmpty ==
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ class _AtMePageState extends State<AtMePage> {
|
|||||||
String? nativeUri =
|
String? nativeUri =
|
||||||
_atMeController.msgFeedAtMeList[i].item?.nativeUri;
|
_atMeController.msgFeedAtMeList[i].item?.nativeUri;
|
||||||
if (nativeUri != null) {
|
if (nativeUri != null) {
|
||||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
PiliScheme.routePushFromUrl(nativeUri);
|
||||||
}
|
}
|
||||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -122,7 +122,7 @@ class LikeMeList extends StatelessWidget {
|
|||||||
onTap: () {
|
onTap: () {
|
||||||
String? nativeUri = msgFeedLikeMeList[i].item?.nativeUri;
|
String? nativeUri = msgFeedLikeMeList[i].item?.nativeUri;
|
||||||
if (nativeUri != null) {
|
if (nativeUri != null) {
|
||||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
PiliScheme.routePushFromUrl(nativeUri);
|
||||||
}
|
}
|
||||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -72,7 +72,7 @@ class _ReplyMePageState extends State<ReplyMePage> {
|
|||||||
String? nativeUri = _replyMeController
|
String? nativeUri = _replyMeController
|
||||||
.msgFeedReplyMeList[i].item?.nativeUri;
|
.msgFeedReplyMeList[i].item?.nativeUri;
|
||||||
if (nativeUri != null) {
|
if (nativeUri != null) {
|
||||||
PiliScheme.routePush(Uri.parse(nativeUri));
|
PiliScheme.routePushFromUrl(nativeUri);
|
||||||
}
|
}
|
||||||
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
// SmartDialog.showToast("跳转至:$nativeUri(暂未实现)");
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -184,8 +184,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
|||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () {
|
..onTap = () {
|
||||||
try {
|
try {
|
||||||
Uri uri = Uri.parse(match[2]!.replaceAll('"', ''));
|
PiliScheme.routePushFromUrl(match[2]!.replaceAll('"', ''));
|
||||||
PiliScheme.routePush(uri);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
SmartDialog.showToast(err.toString());
|
SmartDialog.showToast(err.toString());
|
||||||
}
|
}
|
||||||
@@ -209,8 +208,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
|||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () {
|
..onTap = () {
|
||||||
try {
|
try {
|
||||||
Uri uri = Uri.parse(match[3]!);
|
PiliScheme.routePushFromUrl(match[3]!);
|
||||||
PiliScheme.routePush(uri);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
SmartDialog.showToast(err.toString());
|
SmartDialog.showToast(err.toString());
|
||||||
}
|
}
|
||||||
@@ -231,8 +229,7 @@ class _SysMsgPageState extends State<SysMsgPage> {
|
|||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () {
|
..onTap = () {
|
||||||
try {
|
try {
|
||||||
Uri uri = Uri.parse(match[0]!);
|
PiliScheme.routePushFromUrl(match[0]!);
|
||||||
PiliScheme.routePush(uri);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
SmartDialog.showToast(err.toString());
|
SmartDialog.showToast(err.toString());
|
||||||
Utils.copyText(match[0] ?? '');
|
Utils.copyText(match[0] ?? '');
|
||||||
|
|||||||
@@ -67,18 +67,22 @@ class SearchPanelController extends CommonController {
|
|||||||
void jump2Video() {
|
void jump2Video() {
|
||||||
if (RegExp(r'^av\d+$', caseSensitive: false).hasMatch(keyword)) {
|
if (RegExp(r'^av\d+$', caseSensitive: false).hasMatch(keyword)) {
|
||||||
hasJump2Video = true;
|
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)
|
} else if (RegExp(r'^bv[a-z\d]{10}$', caseSensitive: false)
|
||||||
.hasMatch(keyword)) {
|
.hasMatch(keyword)) {
|
||||||
hasJump2Video = true;
|
hasJump2Video = true;
|
||||||
PiliScheme.videoPush(null, keyword, false);
|
PiliScheme.videoPush(null, keyword, showDialog: false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void onPushDetail(resultList) async {
|
void onPushDetail(resultList) async {
|
||||||
int? aid = int.tryParse(keyword);
|
int? aid = int.tryParse(keyword);
|
||||||
if (aid != null && resultList.first.aid == aid) {
|
if (aid != null && resultList.first.aid == aid) {
|
||||||
PiliScheme.videoPush(aid, null, false);
|
PiliScheme.videoPush(aid, null, showDialog: false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,6 @@ import 'package:PiliPlus/utils/feed_back.dart';
|
|||||||
import 'package:PiliPlus/utils/storage.dart';
|
import 'package:PiliPlus/utils/storage.dart';
|
||||||
import 'package:PiliPlus/utils/url_utils.dart';
|
import 'package:PiliPlus/utils/url_utils.dart';
|
||||||
import 'package:PiliPlus/utils/utils.dart';
|
import 'package:PiliPlus/utils/utils.dart';
|
||||||
import '../../../../../utils/app_scheme.dart';
|
|
||||||
import 'zan.dart';
|
import 'zan.dart';
|
||||||
import 'package:html/parser.dart' show parse;
|
import 'package:html/parser.dart' show parse;
|
||||||
|
|
||||||
@@ -863,56 +862,12 @@ class ReplyItem extends StatelessWidget {
|
|||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final String redirectUrl =
|
Utils.handleWebview(matchStr);
|
||||||
(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
|
|
||||||
// },
|
|
||||||
// );
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (appUrlSchema.startsWith('bilibili://search')) {
|
if (appUrlSchema.startsWith('bilibili://search')) {
|
||||||
Get.toNamed('/searchResult',
|
Get.toNamed('/searchResult',
|
||||||
parameters: {'keyword': title});
|
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 {
|
} else {
|
||||||
Utils.handleWebview(matchStr);
|
Utils.handleWebview(matchStr);
|
||||||
}
|
}
|
||||||
@@ -949,25 +904,8 @@ class ReplyItem extends StatelessWidget {
|
|||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
),
|
),
|
||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () async {
|
..onTap = () {
|
||||||
if (matchStr.startsWith('https://b23.tv')) {
|
Utils.handleWebview(matchStr);
|
||||||
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));
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -20,7 +20,6 @@ import 'package:PiliPlus/utils/feed_back.dart';
|
|||||||
import 'package:PiliPlus/utils/storage.dart';
|
import 'package:PiliPlus/utils/storage.dart';
|
||||||
import 'package:PiliPlus/utils/url_utils.dart';
|
import 'package:PiliPlus/utils/url_utils.dart';
|
||||||
import 'package:PiliPlus/utils/utils.dart';
|
import 'package:PiliPlus/utils/utils.dart';
|
||||||
import '../../../../../utils/app_scheme.dart';
|
|
||||||
import 'package:html/parser.dart' show parse;
|
import 'package:html/parser.dart' show parse;
|
||||||
|
|
||||||
class ReplyItemGrpc extends StatelessWidget {
|
class ReplyItemGrpc extends StatelessWidget {
|
||||||
@@ -901,56 +900,12 @@ class ReplyItemGrpc extends StatelessWidget {
|
|||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final String redirectUrl =
|
Utils.handleWebview(matchStr);
|
||||||
(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
|
|
||||||
// },
|
|
||||||
// );
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (appUrlSchema.startsWith('bilibili://search')) {
|
if (appUrlSchema.startsWith('bilibili://search')) {
|
||||||
Get.toNamed('/searchResult',
|
Get.toNamed('/searchResult',
|
||||||
parameters: {'keyword': title});
|
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 {
|
} else {
|
||||||
Utils.handleWebview(matchStr);
|
Utils.handleWebview(matchStr);
|
||||||
}
|
}
|
||||||
@@ -987,25 +942,8 @@ class ReplyItemGrpc extends StatelessWidget {
|
|||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
),
|
),
|
||||||
recognizer: TapGestureRecognizer()
|
recognizer: TapGestureRecognizer()
|
||||||
..onTap = () async {
|
..onTap = () {
|
||||||
if (matchStr.startsWith('https://b23.tv')) {
|
Utils.handleWebview(matchStr);
|
||||||
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));
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,8 +5,6 @@ import 'package:PiliPlus/http/constants.dart';
|
|||||||
import 'package:PiliPlus/http/init.dart';
|
import 'package:PiliPlus/http/init.dart';
|
||||||
import 'package:PiliPlus/utils/app_scheme.dart';
|
import 'package:PiliPlus/utils/app_scheme.dart';
|
||||||
import 'package:PiliPlus/utils/cache_manage.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:PiliPlus/utils/utils.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_inappwebview/flutter_inappwebview.dart';
|
import 'package:flutter_inappwebview/flutter_inappwebview.dart';
|
||||||
@@ -237,53 +235,20 @@ class _WebviewPageNewState extends State<WebviewPageNew> {
|
|||||||
}
|
}
|
||||||
: null,
|
: null,
|
||||||
shouldOverrideUrlLoading: (controller, navigationAction) async {
|
shouldOverrideUrlLoading: (controller, navigationAction) async {
|
||||||
final String? str =
|
late String url = navigationAction.request.url.toString();
|
||||||
navigationAction.request.url!.pathSegments.getOrNull(0);
|
bool hasMatch = await PiliScheme.routePush(
|
||||||
if (str != null) {
|
navigationAction.request.url?.uriValue ?? Uri(),
|
||||||
final Map matchRes = IdUtils.matchAvorBv(input: str);
|
selfHandle: true,
|
||||||
if (matchRes.isNotEmpty) {
|
off: true,
|
||||||
Get.back();
|
);
|
||||||
PiliScheme.videoPush(matchRes['AV'], matchRes['BV']);
|
// debugPrint('webview: [$url], [$hasMatch]');
|
||||||
return NavigationActionPolicy.CANCEL;
|
if (hasMatch) {
|
||||||
}
|
_progressStream.add(1.0);
|
||||||
}
|
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 (_) {}
|
|
||||||
} else if (RegExp(r'^(?!(https?://))\S+://', caseSensitive: false)
|
} else if (RegExp(r'^(?!(https?://))\S+://', caseSensitive: false)
|
||||||
.hasMatch(url)) {
|
.hasMatch(url)) {
|
||||||
if (url.startsWith('bilibili://video/')) {
|
if (context.mounted) {
|
||||||
String? str =
|
SnackBar snackBar = SnackBar(
|
||||||
navigationAction.request.url!.pathSegments.getOrNull(0);
|
|
||||||
Get.offAndToNamed(
|
|
||||||
'/searchResult',
|
|
||||||
parameters: {'keyword': str ?? ''},
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
var snackBar = SnackBar(
|
|
||||||
content: const Text('当前网页将要打开外部链接,是否打开'),
|
content: const Text('当前网页将要打开外部链接,是否打开'),
|
||||||
showCloseIcon: true,
|
showCloseIcon: true,
|
||||||
action: SnackBarAction(
|
action: SnackBarAction(
|
||||||
|
|||||||
@@ -26,240 +26,527 @@ class PiliScheme {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 路由跳转
|
static Future<bool> routePushFromUrl(
|
||||||
static void routePush(Uri value) async {
|
String url, {
|
||||||
final String scheme = value.scheme;
|
bool selfHandle = false,
|
||||||
final String host = value.host;
|
bool off = false,
|
||||||
final String path = value.path;
|
}) async {
|
||||||
|
try {
|
||||||
if (scheme == 'bilibili') {
|
if (url.startsWith('//')) {
|
||||||
debugPrint('$value');
|
url = 'https:$url';
|
||||||
if (host == 'root') {
|
} else if (RegExp(r'^\S+://').hasMatch(url).not) {
|
||||||
Navigator.popUntil(
|
url = 'https://$url';
|
||||||
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());
|
|
||||||
}
|
}
|
||||||
|
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;
|
||||||
|
|
||||||
|
void launchURL() {
|
||||||
|
if (selfHandle.not) {
|
||||||
|
Utils.launchURL(uri.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (scheme) {
|
||||||
|
case 'bilibili':
|
||||||
|
switch (host) {
|
||||||
|
case 'root':
|
||||||
|
Navigator.popUntil(
|
||||||
|
Get.context!,
|
||||||
|
(Route<dynamic> route) => route.isFirst,
|
||||||
|
);
|
||||||
|
return true;
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
return false;
|
||||||
|
case 'bangumi':
|
||||||
|
// to check
|
||||||
|
if (path.startsWith('/season')) {
|
||||||
|
String? seasonId = RegExp(r'/(\d+)').firstMatch(path)?.group(1);
|
||||||
|
if (seasonId != null) {
|
||||||
|
Utils.viewBangumi(seasonId: seasonId, epId: null);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
return false;
|
||||||
|
case 'opus':
|
||||||
|
// bilibili://opus/detail/12345678?h5awaken=random
|
||||||
|
if (path.startsWith('/detail')) {
|
||||||
|
bool hasMatch = await _onPushDynDetail(path, off);
|
||||||
|
if (hasMatch.not) {
|
||||||
|
launchURL();
|
||||||
|
}
|
||||||
|
return hasMatch;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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: () async {
|
||||||
|
bool hasMatch = await _onPushDynDetail(path, off);
|
||||||
|
if (hasMatch.not) {
|
||||||
|
launchURL();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
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);
|
||||||
|
if (hasMatch.not) {
|
||||||
|
launchURL();
|
||||||
|
}
|
||||||
|
return hasMatch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
if (selfHandle.not) {
|
||||||
|
debugPrint('$uri');
|
||||||
|
SmartDialog.showToast('未知路径:$uri,请截图反馈给开发者');
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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,请截图反馈给开发者');
|
||||||
|
}
|
||||||
|
launchURL();
|
||||||
|
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: off,
|
||||||
|
);
|
||||||
|
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,
|
static Future<void> videoPush(
|
||||||
[bool showDialog = true]) async {
|
int? aid,
|
||||||
|
String? bvid, {
|
||||||
|
bool showDialog = true,
|
||||||
|
bool off = false,
|
||||||
|
}) async {
|
||||||
try {
|
try {
|
||||||
aid ??= IdUtils.bv2av(bvid!);
|
aid ??= IdUtils.bv2av(bvid!);
|
||||||
bvid ??= IdUtils.av2bv(aid);
|
bvid ??= IdUtils.av2bv(aid);
|
||||||
@@ -272,187 +559,15 @@ class PiliScheme {
|
|||||||
}
|
}
|
||||||
Utils.toDupNamed(
|
Utils.toDupNamed(
|
||||||
'/video?bvid=$bvid&cid=$cid',
|
'/video?bvid=$bvid&cid=$cid',
|
||||||
arguments: <String, String?>{
|
arguments: {
|
||||||
'pic': null,
|
'pic': null,
|
||||||
'heroTag': Utils.makeHeroTag(aid),
|
'heroTag': Utils.makeHeroTag(aid),
|
||||||
},
|
},
|
||||||
|
off: off,
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
SmartDialog.dismiss();
|
SmartDialog.dismiss();
|
||||||
SmartDialog.showToast('video获取失败: $e');
|
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:dio/dio.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:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
|
|
||||||
@@ -9,8 +10,10 @@ import 'utils.dart';
|
|||||||
|
|
||||||
class UrlUtils {
|
class UrlUtils {
|
||||||
// 302重定向路由截取
|
// 302重定向路由截取
|
||||||
static Future<String?> parseRedirectUrl(String url,
|
static Future<String?> parseRedirectUrl(
|
||||||
[bool returnOri = false]) async {
|
String url, [
|
||||||
|
bool returnOri = false,
|
||||||
|
]) async {
|
||||||
try {
|
try {
|
||||||
final response = await Request().get(
|
final response = await Request().get(
|
||||||
url,
|
url,
|
||||||
@@ -23,9 +26,10 @@ class UrlUtils {
|
|||||||
);
|
);
|
||||||
if (response.statusCode == 302 || response.statusCode == 301) {
|
if (response.statusCode == 302 || response.statusCode == 301) {
|
||||||
String? redirectUrl = response.headers['location']?.first;
|
String? redirectUrl = response.headers['location']?.first;
|
||||||
|
debugPrint('redirectUrl: $redirectUrl');
|
||||||
if (redirectUrl != null) {
|
if (redirectUrl != null) {
|
||||||
if (redirectUrl.startsWith('/')) {
|
if (redirectUrl.startsWith('/')) {
|
||||||
return url;
|
return returnOri ? url : null;
|
||||||
}
|
}
|
||||||
if (redirectUrl.endsWith('/')) {
|
if (redirectUrl.endsWith('/')) {
|
||||||
redirectUrl = redirectUrl.substring(0, redirectUrl.length - 1);
|
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(
|
static void handleWebview(
|
||||||
String url, {
|
String url, {
|
||||||
bool off = false,
|
bool off = false,
|
||||||
bool inApp = false,
|
bool inApp = false,
|
||||||
}) {
|
}) async {
|
||||||
if (inApp.not && GStorage.openInBrowser) {
|
if (inApp.not && GStorage.openInBrowser) {
|
||||||
if (_handleInAppWebview(url).not) {
|
if ((await PiliScheme.routePushFromUrl(url, selfHandle: true)).not) {
|
||||||
launchURL(url);
|
launchURL(url);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -419,12 +386,7 @@ class Utils {
|
|||||||
parameters: {'url': url},
|
parameters: {'url': url},
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
if (_handleInAppWebview(url).not) {
|
PiliScheme.routePushFromUrl(url);
|
||||||
toDupNamed(
|
|
||||||
'/webview',
|
|
||||||
parameters: {'url': url},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -807,13 +769,23 @@ class Utils {
|
|||||||
String page, {
|
String page, {
|
||||||
dynamic arguments,
|
dynamic arguments,
|
||||||
Map<String, String>? parameters,
|
Map<String, String>? parameters,
|
||||||
|
bool off = false,
|
||||||
}) {
|
}) {
|
||||||
Get.toNamed(
|
if (off) {
|
||||||
page,
|
Get.offNamed(
|
||||||
arguments: arguments,
|
page,
|
||||||
parameters: parameters,
|
arguments: arguments,
|
||||||
preventDuplicates: false,
|
parameters: parameters,
|
||||||
);
|
preventDuplicates: false,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Get.toNamed(
|
||||||
|
page,
|
||||||
|
arguments: arguments,
|
||||||
|
parameters: parameters,
|
||||||
|
preventDuplicates: false,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future copyText(
|
static Future copyText(
|
||||||
|
|||||||
Reference in New Issue
Block a user