// ==UserScript==
// @name 成都文理学院刷课助手(自动填充验证码)
// @version 1.0.8
// @description 成都文理学院数字化实习实训平台刷课,在原基础上,添加了用户交互界面、自动识别填充验证码等功能。
// @author Fulling
// @match *://zxshixun.cdcas.com/user/node*
// @icon 
// @grant GM_xmlhttpRequest
// @license MIT
// @namespace https://github.com/iFulling/cdcasSK
// @downloadURL none
// ==/UserScript==
/**
/* @downloadURL https://update.greasyfork.org/scripts/512596/%E6%88%90%E9%83%BD%E6%96%87%E7%90%86%E5%AD%A6%E9%99%A2%E5%88%B7%E8%AF%BE%E5%8A%A9%E6%89%8B%EF%BC%88%E8%87%AA%E5%8A%A8%E5%A1%AB%E5%85%85%E9%AA%8C%E8%AF%81%E7%A0%81%EF%BC%89.user.js
/* @updateURL https://update.greasyfork.org/scripts/512596/%E6%88%90%E9%83%BD%E6%96%87%E7%90%86%E5%AD%A6%E9%99%A2%E5%88%B7%E8%AF%BE%E5%8A%A9%E6%89%8B%EF%BC%88%E8%87%AA%E5%8A%A8%E5%A1%AB%E5%85%85%E9%AA%8C%E8%AF%81%E7%A0%81%EF%BC%89.meta.js
*/
let videoElement = null;
let checkCaptchaTimer = null;
let containerTextElement = null;
let layuiLayerContent = null;
let timerCnt = 0;
let version = "1.0.8"
// 下一个视频
async function playNext() {
let links = $('a[target="_self"]');
let current = 0;
links.each((index, item) => {
if ($(item).hasClass("on")) {
return current = index
}
});
clearInterval(checkCaptchaTimer);
if (current === links.length - 1) {
addText("最后一个已看完!")
} else {
addText("准备播放下一个视频...")
await pause(3)
links[current + 1].click();
}
}
// 输入验证码
async function inputCaptcha() {
if (layuiLayerContent.length && layuiLayerContent.is(':visible')) {
addText("验证码弹窗出现,准备填写验证码...");
await pause(2, 4)
// 获取图片
let imgs = layuiLayerContent.find("img")
let img = imgs[0].style.opacity === '0' ? imgs[1] : imgs[0]
// 图片转base64
let canvas = document.createElement("canvas");
let ctx = canvas.getContext("2d");
canvas.width = img.width;
canvas.height = img.height;
ctx.drawImage(img, 0, 0, img.width, img.height);
let code = canvas.toDataURL("image/png").split("base64,")[1];
// 调用接口,识别验证码
let ans = await getCode(code)
// 获取input,填入验证码
let inputs = layuiLayerContent.find("input")
let input = inputs[0].style.display === 'none' ? inputs[1] : inputs[0]
$(input).mousedown()
input.value = ans
// 点击开始播放按钮
await pause(2, 4)
const playButton = $('.layui-layer-btn0');
if (playButton.length) {
playButton.click();
checkCaptchaTimer = setInterval(playVideo, 1000);
addText("已自动点击开始播放按钮!");
} else {
addText("未找到开始播放按钮,尝试刷新页面...");
location.reload(); // 刷新当前页面
}
}
}
// 使用sw1128的接口,油猴链接:https://greasyfork.org/zh-CN/scripts/459260
function getCode(code) {
return new Promise((resolve, reject) => {
const datas = {
"ImageBase64": String(code),
}
GM_xmlhttpRequest({
method: "POST",
url: "http://captcha.zwhyzzz.top:8092/identify_GeneralCAPTCHA",
data: JSON.stringify(datas),
headers: {
"Content-Type": "application/json",
},
responseType: "json",
onload: function (response) {
if (response.status == 200) {
if (response.responseText.indexOf("触发限流策略") != -1)
addText(response.response["msg"]);
try {
var result = response.response["result"];
addText("识别结果:" + result);
return resolve(result);
} catch (e) {
if (response.responseText.indexOf("接口请求频率过高") != -1)
addText(response.responseText);
}
} else {
addText("识别失败,请勿开启代理。");
}
}
});
});
}
// 播放视频,同时检测验证码
async function playVideo() {
timerCnt++;
if (timerCnt % 5 === 0) {
addText("等待加载,已加载:" + timerCnt + "秒")
}
if (timerCnt > 20) {
addText("刷新页面")
location.reload();
return
}
if (!videoElement) {
return getVideoElement();
}
// 验证码弹窗
layuiLayerContent = $('.layui-layer-content');
if (layuiLayerContent.length > 0) {
clearInterval(checkCaptchaTimer);
await inputCaptcha()
return;
}
// 检测视频是否加载成功且暂停
if (!videoElement) return;
if (videoElement.paused) {
videoElement.play();
if (videoElement.readyState === 4) {
const message = containerTextElement.text().includes("视频加载完成")
? "请将浏览器置于前台运行。" : "视频加载完成,准备播放";
addText(message);
}
} else {
timerCnt = 0;
}
}
// 获取视频元素
const getVideoElement = () => {
videoElement = document.querySelector("video");
videoElement.muted = true;
videoElement.playbackRate = 1.0;
videoElement.volume = 0;
videoElement.onended = async function () {
await playNext();
};
}
// 添加交互显示
const addContainer = () => {
const container = $('