Files
IPTV3/bili.js
T
2025-12-03 17:26:58 +08:00

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)}`
}