Bangumi快捷播放

首页追番卡片显示中文标题,浮动卡片增加资源搜索,可添加对应集数播放源,浮动卡片状态实时改变无需刷新

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name         Bangumi快捷播放
// @description  首页追番卡片显示中文标题,浮动卡片增加资源搜索,可添加对应集数播放源,浮动卡片状态实时改变无需刷新
// @namespace    https://github.com/RiverYale/Userscripts/
// @homepage     https://riveryale.github.io/Userscripts/
// @version      3.3
// @author       RiverYale
// @match        *://bangumi.tv
// @match        *://bgm.tv
// @match        *://chii.in
// @icon         https://riveryale.github.io/Userscripts/assets/pic/BangumiEasyPlay/icon.png
// @run-at       document-end
// @compatible   chrome
// @compatible   edge
// @license      MIT License
// ==/UserScript==

/*================= 更新脚本前注意保存自己修改的内容! =================*/

var autoMark = true;				// 默认点击链接后自动标记为看过
var authSrc = "";			// 若404则表明未更新资源,全部移除,若无需验证则删除引号中的内容
var src_dict = {					// 网址格式,番剧ID: [资源ID, 播放线路, 总体集数偏移, [集数, 增加偏移]...]
	"AGE动漫": {
		pattern: "https://www.agemys.org/play/${id}/${ch}/${ep}",
		search: "https://www.agemys.org/search?query=${keyword}",
		373247: [20220248, 1, 1],			// 无职转生:到了异世界就拿出真本事 ~ 第二季
		441233: [20230154, 1, 0],			// 总之就是非常可爱 女子高中篇
		376433: [20230138, 1, 0],			// 政宗君的复仇R
		403225: [20230131, 1, -12],			// 打工吧!!魔王大人 2nd Season
		376739: [20220244, 1, -87],			// 进击的巨人 最终季 Part.3
		414461: [20230128, 1, 0],			// 僵尸百分百~变成僵尸之前想做的100件事~
		386809: [20230077, 1, 0],			// 我推的孩子
	},
	"Bimi": {
		pattern: "http://m.dodoge.me/bangumi/${id}/play/${ch}/${ep}/",
		search: "http://m.dodoge.me/vod/search/wd/${keyword}",
	},
	"MX动漫": {
		pattern: "http://www.mxdm9.com/dongmanplay/${id}-${ch}-${ep}.html",
		search: "http://www.mxdm9.com/search/-------------.html?wd=${keyword}",
		373247: [8183, 1, 1],				// 无职转生:到了异世界就拿出真本事 ~ 第二季
		441233: [7112, 1, 12],				// 总之就是非常可爱 女子高中篇
		376433: [7622, 1, 0],				// 政宗君的复仇R
		403225: [8181, 1, -12],				// 打工吧!!魔王大人 2nd Season
		376739: [8049, 1, -87],				// 进击的巨人 最终季 Part.3
		414461: [8169, 1, 0],				// 僵尸百分百~变成僵尸之前想做的100件事~
		386809: [7963, 1, 0],				// 我推的孩子
	},
	"橘子动漫": {
		pattern: "https://www.mgnacg.com/bangumi/${id}-${ch}-${ep}/",
		search: "https://www.mgnacg.com/search/-------------/?wd=${keyword}",
	},
	"樱花动漫": {
		pattern: "https://www.vdm8.com/play/${id}-${ch}-${ep}.html",
		search: "https://www.vdm8.com/search/${keyword}-------------.html",
	},
	"宫下动漫": {
		pattern: "https://arlnigdm.com/vodplay/${id}-${ch}-${ep}.html",
		search: "https://arlnigdm.com/vodsearch/-------------.html?wd=${keyword}",
	},
	"新番组": {
		pattern: "https://bangumi.online",
		search: "https://bangumi.online",
	},
};
/*================= 更新脚本前注意保存自己修改的内容! =================*/

if($(".loginPanel").length == 1) return;

/* 标题中日文对调 */
var titles_A = Array.from($(".tinyHeader .textTip:not(.prgCheckIn)"))		// 平铺模式
	.concat(Array.from($(".l.textTip")))		// 列表模式右侧
if(titles_A.length == 0) return;
var handleTitle_A = function(title) {
	var text = $(title).text();
	var data_original_title = $(title).attr("data-original-title");
	if(!data_original_title){
		return true;
	}
	$(title).text(data_original_title);
	$(title).attr("data-original-title", text);
}

var titles_B = Array.from($(".subjectItem.title.textTip"))		// 列表模式左侧 - 1, 2, 3
var handleTitle_B = function(title) {
	var text = $(title).find('span').text();
	var data_original_title = $(title).attr("data-original-title");
	if(!data_original_title){
		return true;
	}
	$(title).find('span').text(data_original_title);
	$(title).attr('data-original-title', text);

	var preALink = $(title).prev().prev();
	var preALink_title = $(preALink).attr('data-original-title');
	$(preALink).attr('data-original-title', preALink_title.replace(text, data_original_title));
}

/* 生成配置模板 */
var _ul = $('<ul style="display:inline-block; float:right;"></ul>');
var _btn = $('<div style="padding: 3px 12px;margin: 8px 5px 0;;background: #F09199;color: white;border-radius: 50px;font-size: 11px;cursor: pointer;">生成模板</div>');
$(_ul).append(_btn);
$("#prgManagerHeader").append(_ul);
$(_btn).click((e) => {
	var resStr = "";
	document.querySelectorAll('.subjectItem[data-subject-id]').forEach((i) => {
		const id = i.getAttribute("data-subject-id");
		const name = i.getAttribute("data-subject-name-cn");
		resStr += `${id}: [0, 1, 0],			// ${name}\n`;
	})
	let allowCopy = confirm("将复制以下内容到剪切板:\n" + resStr);
	if (allowCopy) {
		navigator.clipboard.writeText(resStr);
		console.log(resStr);
		alert("请自行修改模板方括号内容。如果复制失败,请按F12打开控制台查看输出,手动复制。")
	}
})

/* 点击链接后是否自动标记为[看过] */
var _ul = $('<ul style="display:inline-block; float:right;"></ul>');
var _label = $('<label style="display:block; margin:10px;"></label>');
var _input = $('<input type="checkbox" style="vertical-align:middle;">');
var _span = $('<span style="vertical-align:middle;"> 自动标记</span>');
$(_ul).append(_label);
$(_label).append(_input);
$(_label).append(_span);
$(_label).attr("title", "点击播放链接后自动标记为[看过]");
$("#prgManagerHeader").append(_ul);
$(_input).prop("checked", autoMark);
$(_input).click(() => {
	autoMark = $(_input).prop("checked");
});


/* 为已出集数添加资源链接 */
var epLinkList = Array.from($(".load-epinfo"));
var getSrcHref = function(dict, subid, ep) {
	var value = dict[subid];
	ep = Number(ep);
	if(value.constructor == Array) {
		var pattern = dict["pattern"];
		var resId = value[0];
		var resCh = value[1];
		var resEp = ep + value[2];
		for(let i=3; i<value.length; i++) {
			if(ep >= value[i][0]) {
				resEp += value[i][1];
			}
		}
		return pattern.replace(/\$\{id\}/g, resId).replace(/\$\{ch\}/g, resCh).replace(/\$\{ep\}/g, resEp);
	} else if(value.constructor == String) {
		return value;
	}
	return '';
}
var handleEpLinkList = function(epLink) {
	// 标记该条目下集数的偏移量
	$(epLink.rel).attr("ep_offset", $(epLink).parent().index());

	// 未知集数跳过
	if(epLink.className.indexOf("epBtnNA") > -1) {
		return true;
	}
	var subid = $("#"+epLink.id).attr("subject_id");
	var ep = Number($("#"+epLink.id).text());
	var srcPanel = $("<div><hr class='board'></div>");
	for(let srcName in src_dict) {
		// 根据资源字典添加资源链接
		var dict = src_dict[srcName];
		if(dict[subid] == undefined) {
			var subjectName = $(`[data-subject-id=${subid}][class=textTip]`).text();
			if(dict['search'] == undefined) {
				$(srcPanel).append(`<a href="javascript:alert('未添加搜索地址格式!');" style="margin-right:10px; display:inline-block; color:#555; font-style:italic;">${srcName}</a>`);
			} else {
				var searchHref = dict['search'].replace(/\$\{keyword\}/g, subjectName)
				$(srcPanel).append(`<a href="${searchHref}" style="margin-right:10px; display:inline-block; color:#555; font-style:italic;" target="_blank">${srcName}</a>`);
			}
			continue;
		}
		var srcHref = getSrcHref(dict, subid, ep);
		var alink = $('<a style="margin-right:10px; display:inline-block; font-weight:bold;" target="_blank"></a>');
		$(alink).attr("href", srcHref);
		$(alink).text(srcName);
		$(srcPanel).append(alink);

		// 点击资源链接后标记为[看过]
		$(alink).click(() => {
			var watchedLink = "#Watched_" + epLink.id.slice(4);
			if(autoMark && $(watchedLink).length>0) {
				$(watchedLink).click();
			}
		});

		// 已看集数的不测试资源是否可达
		if(epLink.className.indexOf("epBtnWatched")==-1 && srcName==authSrc) {
			var authHref = srcHref;
		}
	}
	$(epLink.rel).append(srcPanel);

	// 测试资源是否可达
	if(authHref != undefined && authHref != "") {
		var isRunUrl = function(url){
			return new Promise(function (resolve, reject) {
				var dom = document.createElement('link');
				dom.href = url;
				dom.rel = 'stylesheet';
				dom.onload = function () {
					document.head.removeChild(dom);
					resolve();
				}
				dom.onerror = reject;
				document.head.appendChild(dom);
			});
		}
		isRunUrl(authHref).then(data => {}).catch(data => {
			// $(srcPanel).remove();
			$(epLink).css({"color":"red"});
			// $(epLink).css({"color":"#00F", "background-color":"#e0e0e0", "border":"1px solid #b6b6b6"});
		});
	}
}


// 根据状态更新单集进度情况面板
var updataEpStatusTool = function(prg, type) {
	var epid = prg.id.slice(8);
	var gh = $("a:first", prg).attr("href").split('=')[1];
	var epStatusTool = $(".epStatusTool", prg);

	var type_dict = { "看过": 1, "想看": 2, "抛弃": 3, "撤消": 0 };
	var status = type_dict[type];
	var statusLink = $(".epStatusTool a", prg)[0];
	statusLink = $(statusLink).clone(true);

	var epGrid = $(`[rel="#prginfo_${epid}"]`).parent().parent().parent();
	var progressText = $("#prgsPercentNum", epGrid).text();
	var progress = progressText.slice(1,-1).split('/');
	var old_type = $("p", epStatusTool).text();
	var old_status = type_dict[old_type] || 0;
	if((old_status == 1 || old_status == 3) && (status != 1 && status != 3)) {
		progress[0] = Number(progress[0]) - 1;
	} else if((old_status != 1 && old_status != 3) && (status == 1 || status == 3)) {
		progress[0] = Number(progress[0]) + 1;
	}
	$("#prgsPercentNum", epGrid).text(`[${progress[0]}/${progress[1]}]`);

	$(epStatusTool).empty();
	if(status == 0) {
		$(epStatusTool).append(`<p id="epBtnCu_${epid}"></p>`);
	} else {
		$(epStatusTool).append(`<p id="epBtnCu_${epid}" class="epBtnCu">${type}</p>`);
	}
	if(status != 1) {
		var watched = $(statusLink).clone(true);
		$(watched).attr({href: `/subject/ep/${epid}/status/watched?gh=${gh}`, id: `Watched_${epid}`});
		$(watched).text("看过")
		$(epStatusTool).append(watched);

		var watchedTill = $(statusLink).clone(true);
		$(watchedTill).attr({href: `/subject/ep/${epid}/status/watched?gh=${gh}`, id: `WatchedTill_${epid}`});
		$(watchedTill).text("看到")
		$(epStatusTool).append(watchedTill);
	}
	if(status != 2) {
		var queue = $(statusLink).clone(true);
		$(queue).attr({href: `/subject/ep/${epid}/status/queue?gh=${gh}`, id: `Queue_${epid}`});
		$(queue).text("想看")
		$(epStatusTool).append(queue);
	}
	if(status != 3) {
		var drop = $(statusLink).clone(true);
		$(drop).attr({href: `/subject/ep/${epid}/status/drop?gh=${gh}`, id: `Drop_${epid}`});
		$(drop).text("抛弃")
		$(epStatusTool).append(drop);
	}
	if(status != 0) {
		var remove = $(statusLink).clone(true);
		$(remove).attr({href: `/subject/ep/${epid}/status/remove?gh=${gh}`, id: `remove_${epid}`});
		$(remove).text("撤消")
		$(epStatusTool).append(remove);
	}
}


/* 修改进度后实时修改面板状态 */
var prgList = Array.from($("#subject_prg_content").children());
var handleprgList = function(prg) {
	$(".epStatusTool a", prg).click((event) => {
		var type = event.currentTarget.innerText;
		if(type == '看到') {
			var offset = Number($(prg).attr("ep_offset"));
			var curI = $(prg).index();
			while(curI >= 0 && offset >= 0) {
				if (offset == Number($(prgList[curI]).attr("ep_offset"))) {
					if ($(".epStatusTool p", prgList[curI]).text() != '抛弃') {
						updataEpStatusTool(prgList[curI], "看过");
					}
					offset -= 1;
				}
				curI -= 1;
			}
		} else {
			updataEpStatusTool(prg, type);
		}

		// 更新显示的面板
		var prg_new = $(prg).clone(true);
		$(prg_new).css("display", "block");
		$("#cluetip-inner").empty();
		$("#cluetip-inner").append(prg_new);
		return false;
	})
}


/* 功能运行进度指示文本 */
class Task {
	constructor(func, ...args) {
		this.func = func;
		this.args = args;
		this.delayMillsec = 0;
	}
	delay(millsec) {
		if (millsec) this.delayMillsec = millsec;
		return this.delayMillsec;
	}
	execute() {
		return this.func(...this.args);
	}
}

var taskList = [];
titles_A.forEach(title => {
	taskList.push(new Task(handleTitle_A, title));
});
titles_B.forEach(title => {
	taskList.push(new Task(handleTitle_B, title));
});
epLinkList.forEach(epLink => {
	taskList.push(new Task(handleEpLinkList, epLink));
});
prgList.forEach(prg => {
	taskList.push(new Task(handleprgList, prg));
});

var _progressUl = $('<ul style="display:inline-block; float:right; padding:10px"></ul>');
var _progressSpan = $('<span style="vertical-align:middle;">处理中...0%</span>');
if (taskList.length > 0) {
	$("#prgManagerHeader").append(_progressUl);
	$(_progressUl).append(_progressSpan);
	$(_progressSpan).attr("maxVal", taskList.length);
	$(_progressSpan).attr("curVal", 0);
}

var incProgress = function(val = 1) {
	let maxVal = Number($(_progressSpan).attr("maxVal"));
	let curVal = Number($(_progressSpan).attr("curVal"));
	curVal = Math.max(0, Math.min(curVal + val, maxVal));
	let progress = Math.round(curVal / maxVal * 100);
	$(_progressSpan).attr("curVal", curVal);
	$(_progressSpan).text(`处理中...${progress}%`);
	if (curVal == maxVal) {
		setTimeout(() => {
			$(_progressUl).css("transition", "0.5s ease");
			$(_progressUl).css("color", "rgba(0, 0, 0, 0)");
		}, 1000);
	}
}

var executor = function(taskList) {
	if (!taskList || taskList.length < 1) return;
	var task = taskList.shift();
	setTimeout(() => {
		task.execute();
		incProgress();
		executor(taskList);
	}, task.delay());
}
executor(taskList);