350 lines
13 KiB
JavaScript
350 lines
13 KiB
JavaScript
// =====================
|
|
// B站 xptv 插件 (v4.1 最终稳定版)
|
|
// =====================
|
|
|
|
const $configObj = argsify($config_str)
|
|
// 请用您刚复制的完整、新鲜的 Cookie 字符串替换下面这个值
|
|
const BILI_COOKIE = "" //哔哩哔哩 Cookie
|
|
const UA = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36"
|
|
|
|
// 配置
|
|
async function getConfig() {
|
|
// 启动检查:确保关键令牌存在
|
|
if (!BILI_COOKIE || !BILI_COOKIE.includes("SESSDATA") || !BILI_COOKIE.includes("bili_jct")) {
|
|
$utils.toastError("BILI_COOKIE 缺失 SESSDATA 或 bili_jct,请重新登录并复制完整 Cookie");
|
|
return jsonify({ ver: 4, title: "哔哩哔哩 (Cookie失效)", site: "https://www.bilibili.com/", tabs: [] })
|
|
}
|
|
|
|
$utils.toastInfo("加载 Bilibili v4.1 配置...")
|
|
|
|
const appConfig = {
|
|
ver: 4,
|
|
title: "哔哩哔哩 (稳定版)",
|
|
site: "https://www.bilibili.com/",
|
|
tabs: [
|
|
// === 核心推荐 ===
|
|
{ name: "个性推荐", ext: { id: "recommend" } },
|
|
{ name: "全站热门", ext: { id: "hot" } },
|
|
|
|
// === 影视分区 (最新/热门) ===
|
|
{ name: "电视剧(最新)", ext: { id: "category", rid: 11, type: "new" } },
|
|
{ name: "电视剧(热门)", ext: { id: "category", rid: 11, type: "rank" } },
|
|
{ name: "电影(最新)", ext: { id: "category", rid: 23, type: "new" } },
|
|
{ name: "电影(热门)", ext: { id: "category", rid: 23, type: "rank" } },
|
|
{ name: "番剧(最新)", ext: { id: "category", rid: 13, type: "new" } },
|
|
{ name: "纪录片(最新)", ext: { id: "category", rid: 177, type: "new" } },
|
|
|
|
// === 热门分区 ===
|
|
{ name: "综艺", ext: { id: "category", rid: 5, type: "rank" } },
|
|
{ name: "动画", ext: { id: "category", rid: 1, type: "rank" } },
|
|
{ name: "国创", ext: { id: "category", rid: 168, type: "rank" } },
|
|
{ name: "音乐", ext: { id: "category", rid: 3, type: "rank" } },
|
|
{ name: "舞蹈", ext: { id: "category", rid: 129, type: "rank" } },
|
|
{ name: "游戏", ext: { id: "category", rid: 4, type: "rank" } },
|
|
{ name: "知识", ext: { id: "category", rid: 36, type: "rank" } },
|
|
{ name: "科技", ext: { id: "category", rid: 188, type: "rank" } },
|
|
{ name: "体育", ext: { id: "category", rid: 234, type: "rank" } },
|
|
{ name: "汽车", ext: { id: "category", rid: 223, type: "rank" } },
|
|
{ name: "生活", ext: { id: "category", rid: 160, type: "rank" } },
|
|
{ name: "美食", ext: { id: "category", rid: 211, type: "rank" } },
|
|
{ name: "鬼畜", ext: { id: "category", rid: 119, type: "rank" } },
|
|
{ name: "时尚", ext: { id: "category", rid: 155, type: "rank" } },
|
|
{ name: "动物圈", ext: { id: "category", rid: 217, type: "rank" } }
|
|
]
|
|
}
|
|
|
|
return jsonify(appConfig)
|
|
}
|
|
|
|
// 获取卡片列表
|
|
async function getCards(ext) {
|
|
ext = argsify(ext)
|
|
const { page = 1, id, rid, type = "rank", wd = "" } = ext
|
|
let cards = []
|
|
|
|
try {
|
|
let apiUrl = ""
|
|
let sortText = ""
|
|
|
|
if (id === "recommend") {
|
|
apiUrl = `https://api.bilibili.com/x/web-interface/index/top/feed/rcmd?ps=20&fresh_idx=${page}&feed_version=V1`
|
|
sortText = "个性推荐"
|
|
} else if (id === "hot") {
|
|
apiUrl = `https://api.bilibili.com/x/web-interface/popular?pn=${page}&ps=20`
|
|
sortText = "全站热门"
|
|
} else if (id === "category") {
|
|
sortText = type === "new" ? "最新发布" : "热门排行"
|
|
if (type === "new") {
|
|
apiUrl = `https://api.bilibili.com/x/web-interface/dynamic/region?rid=${rid}&pn=${page}&ps=20`
|
|
} else {
|
|
apiUrl = `https://api.bilibili.com/x/web-interface/ranking/v2?rid=${rid}&type=all&pn=${page}&ps=20`
|
|
}
|
|
}
|
|
|
|
if (apiUrl) {
|
|
$utils.toastInfo(`加载: ${sortText} (第${page}页)`)
|
|
|
|
const { data } = await $fetch.get(apiUrl, {
|
|
headers: { "User-Agent": UA, "Cookie": BILI_COOKIE, "Referer": "https://www.bilibili.com/" }
|
|
})
|
|
const json = argsify(data)
|
|
|
|
// API 错误码检查
|
|
if (json.code !== 0) {
|
|
const errMsg = json.msg || json.message || "未知错误";
|
|
$print(`BiliBili API Error Code: ${json.code}, Message: ${errMsg}`);
|
|
|
|
let userTip = `数据获取失败 (Code: ${json.code})`
|
|
if (json.code === -101 || json.code === -400) {
|
|
userTip = "登录信息过期或缺失,请更新 BILI_COOKIE"; // 重点提示
|
|
}
|
|
|
|
$utils.toastError(userTip);
|
|
return jsonify({ list: [] });
|
|
}
|
|
|
|
const dataContainer = json.data || {};
|
|
|
|
const listData = dataContainer.item || dataContainer.items || dataContainer.list || dataContainer.archives || dataContainer.result || []
|
|
|
|
cards = listData.map(v => {
|
|
const vodId = v.bvid || v.aid || v.id || ""
|
|
if (!vodId) return null
|
|
|
|
return {
|
|
vod_id: vodId,
|
|
vod_name: (v.title || v.name || "未知标题").replace(/<[^>]+>/g, ""),
|
|
vod_pic: (v.pic || v.cover || "").replace(/@\d+w_\d+h.*$/, ""),
|
|
vod_remarks: formatDate(v.pubdate || v.ctime || v.created || v.publish_time),
|
|
vod_sub: formatVideoSubtitle(v),
|
|
ext: {
|
|
bvid: v.bvid || v.id || "",
|
|
cid: v.cid || 0,
|
|
season_id: v.season_id || 0,
|
|
aid: v.aid || 0
|
|
}
|
|
}
|
|
}).filter(c => c && c.vod_id)
|
|
}
|
|
|
|
if (cards.length === 0) {
|
|
$utils.toastInfo(`该页数据为空`)
|
|
}
|
|
|
|
if (wd) {
|
|
cards = cards.filter(c => c.vod_name.toLowerCase().includes(wd.toLowerCase()))
|
|
}
|
|
|
|
} catch (e) {
|
|
$print(`getCards error: ${e}`)
|
|
$utils.toastError(`加载失败: ${e.message}`)
|
|
}
|
|
|
|
return jsonify({ list: cards })
|
|
}
|
|
|
|
// 播放地址入口 (无变化)
|
|
async function getTracks(ext) {
|
|
ext = argsify(ext)
|
|
const { bvid, cid, vod_name, season_id, aid } = ext
|
|
|
|
if (season_id) {
|
|
return await handleSeasonVideo(season_id, bvid, cid, vod_name)
|
|
} else {
|
|
return await handleSingleVideo(bvid, cid, vod_name, aid)
|
|
}
|
|
}
|
|
|
|
// ... (以下函数均无逻辑变化,为保持代码完整性保留)
|
|
|
|
// 处理剧集视频
|
|
async function handleSeasonVideo(season_id, bvid, cid, vod_name) {
|
|
$utils.toastInfo(`加载剧集列表...`)
|
|
try {
|
|
const seasonApi = `https://api.bilibili.com/pgc/view/web/season?season_id=${season_id}`
|
|
const { data } = await $fetch.get(seasonApi, {
|
|
headers: { "User-Agent": UA, "Referer": "https://www.bilibili.com", "Cookie": BILI_COOKIE }
|
|
})
|
|
const json = argsify(data)
|
|
const episodes = json.result?.episodes || []
|
|
|
|
const trackGroups = []
|
|
for (const ep of episodes) {
|
|
const episodeTracks = await getEpisodeTracks(ep)
|
|
if (episodeTracks.length > 0) {
|
|
trackGroups.push({
|
|
title: ep.title || `第${ep.index}话`,
|
|
tracks: episodeTracks,
|
|
defaultQuality: "1080P"
|
|
})
|
|
}
|
|
}
|
|
if (trackGroups.length > 0) return jsonify({ list: trackGroups })
|
|
if (bvid && cid) return await handleSingleVideo(bvid, cid, vod_name)
|
|
} catch (e) {}
|
|
return createEmptyTrackGroup(vod_name)
|
|
}
|
|
|
|
// 获取剧集单集
|
|
async function getEpisodeTracks(episode) {
|
|
const { bvid, cid, id: ep_id } = episode
|
|
const tracks = []
|
|
const qualityLevels = getQualityLevels()
|
|
|
|
for (const { qn, n } of qualityLevels) {
|
|
const url = await tryPgcPlayUrl(ep_id, cid, qn, bvid)
|
|
if (url) {
|
|
tracks.push({ name: n, ext: { url } })
|
|
if (qn >= 80) break
|
|
}
|
|
}
|
|
return tracks
|
|
}
|
|
|
|
// 处理单视频
|
|
async function handleSingleVideo(bvid, cid, vod_name) {
|
|
if (!bvid) return createEmptyTrackGroup(vod_name)
|
|
|
|
let effectiveCid = cid
|
|
if (!effectiveCid) {
|
|
const info = await getVideoInfo(bvid)
|
|
if (info) {
|
|
effectiveCid = info.cid
|
|
if (!vod_name) vod_name = info.title
|
|
}
|
|
}
|
|
if (!effectiveCid) return createEmptyTrackGroup(vod_name)
|
|
|
|
const tracks = []
|
|
const qualityLevels = getQualityLevels()
|
|
|
|
for (const { qn, n } of qualityLevels) {
|
|
let url = await tryStandardPlayUrl(bvid, effectiveCid, qn)
|
|
if (!url) url = await tryPgcPlayUrl(0, effectiveCid, qn, bvid)
|
|
|
|
if (url) {
|
|
tracks.push({ name: n, ext: { url } })
|
|
if (qn >= 80) break
|
|
}
|
|
}
|
|
|
|
if (tracks.length === 0) {
|
|
$utils.toastError(`无可用源,可能需要大会员或地区限制`)
|
|
return createEmptyTrackGroup(vod_name)
|
|
}
|
|
|
|
return jsonify({
|
|
list: [{ title: vod_name || "视频", tracks, defaultQuality: "1080P" }]
|
|
})
|
|
}
|
|
|
|
// 接口:PGC (影视)
|
|
async function tryPgcPlayUrl(ep_id, cid, qn, bvid = "") {
|
|
try {
|
|
let apiUrl = `https://api.bilibili.com/pgc/player/web/playurl?cid=${cid}&qn=${qn}&fnval=1&fnver=0&otype=json&fourk=1`
|
|
if (ep_id) apiUrl += `&ep_id=${ep_id}`
|
|
if (bvid) apiUrl += `&bvid=${bvid}`
|
|
|
|
const { data } = await $fetch.get(apiUrl, {
|
|
headers: { "User-Agent": UA, "Referer": "https://www.bilibili.com", "Cookie": BILI_COOKIE }
|
|
})
|
|
const json = argsify(data)
|
|
if (json.code === 0 && json.result?.durl?.[0]?.url) return json.result.durl[0].url
|
|
} catch (e) {}
|
|
return null
|
|
}
|
|
|
|
// 接口:UGC (普通视频)
|
|
async function tryStandardPlayUrl(bvid, cid, qn) {
|
|
try {
|
|
const flvUrl = `https://api.bilibili.com/x/player/playurl?bvid=${bvid}&cid=${cid}&qn=${qn}&fnval=1&fnver=0&otype=json&fourk=1`
|
|
const { data } = await $fetch.get(flvUrl, {
|
|
headers: { "User-Agent": UA, "Referer": "https://www.bilibili.com", "Cookie": BILI_COOKIE }
|
|
})
|
|
const json = argsify(data)
|
|
if (json.code === 0 && json.data?.durl?.[0]?.url) return json.data.durl[0].url
|
|
} catch (e) {}
|
|
return null
|
|
}
|
|
|
|
// 辅助函数
|
|
function getQualityLevels() {
|
|
return BILI_COOKIE.includes("SESSDATA") ?
|
|
[{qn:112,n:"1080P+"},{qn:80,n:"1080P"},{qn:64,n:"720P"},{qn:32,n:"480P"}] :
|
|
[{qn:64,n:"480P"},{qn:16,n:"360P"}]
|
|
}
|
|
|
|
function createEmptyTrackGroup(title) {
|
|
return jsonify({
|
|
list: [{ title: title || "错误", tracks: [{ name: "无法播放", ext: { url: "" } }] }]
|
|
})
|
|
}
|
|
|
|
// 播放信息头
|
|
async function getPlayinfo(ext) {
|
|
ext = argsify(ext)
|
|
return jsonify({
|
|
urls: [ext.url],
|
|
headers: [{ "User-Agent": UA, "Referer": "https://www.bilibili.com", "Origin": "https://www.bilibili.com" }]
|
|
})
|
|
}
|
|
|
|
// 搜索 (无变化)
|
|
async function search(ext) {
|
|
ext = argsify(ext)
|
|
let cards = []
|
|
let text = encodeURIComponent(ext.text || ext.wd || "")
|
|
let page = ext.page || 1
|
|
|
|
if (!text) return jsonify({ list: [] })
|
|
|
|
const apiUrl = `https://api.bilibili.com/x/web-interface/search/type?search_type=video&keyword=${text}&page=${page}&page_size=20&order=totalrank`
|
|
|
|
try {
|
|
const { data } = await $fetch.get(apiUrl, {
|
|
headers: { "User-Agent": UA, "Cookie": BILI_COOKIE, "Referer": "https://search.bilibili.com/" }
|
|
})
|
|
const json = argsify(data)
|
|
if (json.code === 0 && json.data.result) {
|
|
cards = json.data.result.map(v => ({
|
|
vod_id: v.bvid,
|
|
vod_name: v.title.replace(/<[^>]+>/g, ""),
|
|
vod_pic: (v.pic.startsWith("http") ? v.pic : `https:${v.pic}`).replace(/@\d+w_\d+h.*$/, ""),
|
|
vod_remarks: formatDate(v.pubdate),
|
|
vod_sub: formatVideoSubtitle({author: v.author, play: v.play}),
|
|
ext: { bvid: v.bvid, cid: v.cid || 0, season_id: v.season_id || 0 }
|
|
}))
|
|
}
|
|
} catch (e) {
|
|
$print(`Search error: ${e}`)
|
|
}
|
|
return jsonify({ list: cards })
|
|
}
|
|
|
|
// 视频详情获取
|
|
async function getVideoInfo(bvid) {
|
|
try {
|
|
const { data } = await $fetch.get(`https://api.bilibili.com/x/web-interface/view?bvid=${bvid}`, {
|
|
headers: { "User-Agent": UA, "Cookie": BILI_COOKIE }
|
|
})
|
|
const json = argsify(data)
|
|
return (json.code === 0 && json.data) ? json.data : null
|
|
} catch (e) { return null }
|
|
}
|
|
|
|
function formatDate(timestamp) {
|
|
if (!timestamp) return ""
|
|
const date = new Date(timestamp * 1000)
|
|
return `${date.getFullYear()}-${String(date.getMonth() + 1).padStart(2, '0')}-${String(date.getDate()).padStart(2, '0')}`
|
|
}
|
|
|
|
function formatNumber(num) {
|
|
return num >= 10000 ? (num / 10000).toFixed(1) + '万' : num
|
|
}
|
|
|
|
function formatVideoSubtitle(video) {
|
|
const owner = video.owner?.name || video.author || video.up || '未知UP主'
|
|
const play = video.stat?.view || video.play || 0
|
|
return `${owner} | ${formatNumber(play)}`
|
|
}
|