// ==UserScript==
// @name 华医网助手2025
// @namespace https://greasyfork.org/zh-CN/scripts/553643-%E5%8D%8E%E5%8C%BB%E7%BD%91%E5%B0%8F%E5%8A%A9%E6%89%8B2025
// @version 202510
// @description 基于“🥇【华医网小助手】全网唯一真实免费|无人值守|自动静音|视频助手|考试助手|不疲劳”。[❌倍速播放✅视频助手✅屏蔽或者跳过课堂签到、提醒、疲劳✅考试助手(试错算法仅面向可多次提交的考试)✅双模选择:单刷视频or视频+考试。]
// @author 三创作者:citlalidsk 二创作者:境界程序员 原创作者:Dr.S
// @license AGPL License
// @match *://*.91huayi.com/course_ware/course_ware_polyv.aspx?*
// @match *://*.91huayi.com/course_ware/course_ware_cc.aspx*
// @match *://*.91huayi.com/pages/exam.aspx?*
// @match *://*.91huayi.com/pages/exam_result.aspx?*
// @match *://*.91huayi.com/*
// @grant none
// @downloadURL https://update.greasyfork.icu/scripts/553647/%E5%8D%8E%E5%8C%BB%E7%BD%91%E5%8A%A9%E6%89%8B2025.user.js
// @updateURL https://update.greasyfork.icu/scripts/553647/%E5%8D%8E%E5%8C%BB%E7%BD%91%E5%8A%A9%E6%89%8B2025.meta.js
// ==/UserScript==
var newupdate = "适配2025年";
//更新历史
//■2025.10适配2025年华医网
//原作https://greasyfork.org/zh-CN/scripts/502969-%E5%8D%8E%E5%8C%BB%E7%BD%91%E8%87%AA%E5%8A%A8%E5%AD%A6%E4%B9%A0-%E8%B7%B3%E8%BF%87%E7%AD%BE%E5%88%B0-%E8%87%AA%E5%8A%A8%E9%9D%99%E9%9F%B3-%E8%A7%86%E9%A2%91%E5%8A%A9%E6%89%8B-%E8%87%AA%E5%8A%A8%E8%80%83%E8%AF%95
//原作https://greasyfork.org/zh-CN/scripts/483418-%E5%8D%8E%E5%8C%BB%E7%BD%91%E5%B0%8F%E5%8A%A9%E6%89%8B-%E5%85%A8%E7%BD%91%E5%94%AF%E4%B8%80%E7%9C%9F%E5%AE%9E%E5%85%8D%E8%B4%B9-%E6%97%A0%E4%BA%BA%E5%80%BC%E5%AE%88-%E8%87%AA%E5%8A%A8%E9%9D%99%E9%9F%B3-%E8%A7%86%E9%A2%91%E5%8A%A9%E6%89%8B-%E8%80%83%E8%AF%95%E5%8A%A9%E6%89%8B-%E4%B8%8D%E7%96%B2%E5%8A%B3
//■2024.8.1网页布局和提示窗改版,调整检测逻辑;既然禁用倍速,不再显示变速按钮;得学分更快的双卫网小助手考试功能已开发完毕,正在优化缩短视频时间,完善后发布,欢迎天使投资人
//■2024.7.16因部分地区考试不用二维码,所以将进入考试的方式回滚到旧版本方便更多人使用,因此可能会导致部分全国通用版的用户依旧偶尔自动进入考试失败,以后再另行观察。感谢大家的意见
//■2024.7.14优化静音时间点;优化更新内容展示;优化播放逻辑,已完成的视频不再引起卡顿
//■2024.7.13优化进入考试的逻辑,不再依赖考试按钮
//■2024.7.11根据用户反馈,增加了登录界面关闭悬浮窗的按钮
//■2024.7.8增加了当前页面是否有对应代码的提示,增加了作者脚本的分享链接
//■2024.6.21智能检测剩余任务,以防有人直接看最后一节课导致脚本发呆
//■2024.6.19新增了从考试结果界面自动返回原课程的功能(官方网站改版,主动删除网页中的继续学习按钮)
//■2024.6.18针对华医网答题模块改版,已更新语法
//■2024.6.7根据赞赏和评论区反馈,修复了一种视频意外暂停的情况
//■2024.6.5增加视频过程中对温馨提示(疲劳)的检测
//■2024.6.3尝试修复CC播放器和保利威播放器加载事件bug
//■2024.4.28由于与用户无法取得联系,在页面上增加了反馈机制的说明
//■2024.4.15修复了不自动切换视频的问题(因网站版本限制,目前脚本倍速已失效)
//■2024.1.11在人脸识别页面增加温馨提醒,考试功能仅为答案遍历,而非自动搜索答案
//■2023.12.25添加了网页静音代码,润物细无声
//■2023.12.24优化了倍速调整的逻辑,无需刷新网页
//■2023.12.21将脚本控制台上移到显眼的位置,方便用户操作;增加生效的倍速按钮变色(删除了原先的文字提醒)
//■2023.12.15新增模式切换,可以选择先单刷视频(无人值守),刷完再打开考试开关,就可以连续考试了
//■2023.12.3优化了视频播放逻辑,能够自动切换下一个视频,而不是播完1个就卡在考试认证处(也导致了不修改代码就无法进入考试)
//■2023.12.1调整默认播放速度5倍(仅首次登录起效,后续以用户更改过的倍速保存),免得用户感觉不到脚本在运行
(function () {
'use strict';
var submitTime = 6100;//交卷时间控制
var reTryTime = 2100;//重考,视频进入考试延时控制
var examTime = 10000;//听课完成进入考试延时
var randomX = 5000;//随机延时上限
var vSpeed = 1; //首次使用脚本的默认播放速度
var autoSkip = false; //一个可能会封号的功能。
//记录字段
var keyPlayRate = "JJ_Playrate";
var keyTest = "JJ_Test";
var keyResult = "JJ_Result";
var keyThisTitle = "JJ_ThisTitle";
var keyTestAnswer = "JJ_TestAnswer";
var keyRightAnswer = "JJ_RightAnswer";
var keyAllAnswer = "JJ_AllAnswer";
//按钮样式
var btstyleA = "font-size: 16px;font-weight: 300;text-decoration: none;text-align: center;line-height: 40px;height: 40px;padding: 0 40px;display: inline-block;appearance: none;cursor: pointer;border: none;box-sizing: border-box;transition-property: all;transition-duration: .3s;background-color: #4cb0f9;border-color: #4cb0f9;border-radius: 4px;margin: 5px;color: #FFF;";
var btstyleB = "font-size: 12px;font-weight: 300;text-decoration: none;text-align: center;line-height: 20px;height: 20px;padding: 0 5px;display: inline-block;appearance: none;cursor: pointer;border: none;box-sizing: border-box;transition-property: all;transition-duration: .3s;background-color: #4cb0f9;border-color: #4cb0f9;border-radius: 4px;margin: 5px;color: #FFF;";
var btstyleC = "font-size: 12px;font-weight: 300;text-decoration: none;text-align: center;line-height: 20px;height: 20px;padding: 0 5px;display: inline-block;appearance: none;cursor: pointer;border: none;box-sizing: border-box;transition-property: all;transition-duration: .3s;background-color: #f15854;border-color: #f15854;border-radius: 4px;margin: 5px;color: #FFF;";
//页面判别
var urlInfos = window.location.href.split("/");
var urlTip = urlInfos[urlInfos.length - 1].split("?")[0];
var huayi = getHuayi();
var nspeed = 0;
//var mmcode = ``
var clock = null;
advis();
document.querySelector("span[id='tixing']").innerHTML = "当前网址已适配";
if (urlTip == "course_ware_polyv.aspx") { //保利威播放器视频页面
console.log("当前任务: 华医看视频");
document.querySelector("div[id='Div1']").style.top = "40px";
huayi.seeVideo(1);
} else if (urlTip == "course_ware_cc.aspx") { //CC播放器视频页面
console.log("当前任务: 华医看视频");
document.querySelector("div[id='Div1']").style.top = "40px";
huayi.seeVideo(2);
} else if (urlTip == "exam.aspx") { //考试页面
console.log("当前任务: 华医考试");
huayi.doTest();
} else if (urlTip == "course.aspx" || urlTip == "cme.aspx") { //课程列表页面
console.log("当前任务: 课程列表");
huayi.courseList();
} else if (urlTip == "exam_result.aspx") { //考试结果页面
console.log("当前任务: 华医考试结果审核");
huayi.doResult();
} else {
console.log("其它情况");
try {
document.querySelector("span[id='tixing']").innerHTML = "此页面非视频、考试或未适配";
//document.querySelector("img[id='Pic']").style.display = "block";
} catch (error) { };
};
//网页中存在子页面的检测,导致存在多个Div1
// ==================== 新增:全局试错函数 ====================
// 全局试错函数
// 全局试错函数 (动态 maxLetter)
function getChoiceCode(an) {
var charin = an || "A";
return charin.charCodeAt(0) - "A".charCodeAt(0);
};
function getNextChoice(an, maxLetter = 'D') {
var code = an.charCodeAt(0) + 1;
var maxCode = maxLetter.charCodeAt(0);
if (code > maxCode) code = "A".charCodeAt(0);
return String.fromCharCode(code);
};
// ==================== 新增结束 ====================
function getHuayi() {
return {
courseList: function () {
addAnwserCopybtn();
DelAllAnwser();
},
seeVideo: function (e) {
var tr = localStorage.getItem(keyPlayRate);
//console.log("存储读取" + tr);//读取倍速
//var playRateNow = tr ? tr : vSpeed;
var playRateNow = 1;
cleanKeyStorage();
asynckillsendQuestion();//屏蔽课堂问答的函数;
killsendQuestion2();//屏蔽课堂问答的函数2;
killsendQuestion3(); //循环检测问答对话框是否弹出。
// addrateinfo();//插入一些按钮
// addratebtn(1);
// addratebtn(1.5);
// addratebtn(2);
// addratebtn(3);
// addratebtn(5);
// addratebtn(10);
//addSkipbtn();//跳过按钮
addinfo();//脚本信息
changelayout();
//速度调节部分
window.onload = function () {
localStorage.setItem(keyThisTitle, JSON.stringify(window.document.title));//储存章节标题
// console.log("准备激活加速");
ratechg(playRateNow);
if (autoSkip == true) {//秒过功能,签完别尝试
setTimeout(function () {
skipVideo();
}, (submitTime + Math.ceil(Math.random() * randomX)));
console.log("秒过了!");
};
// ==================== 新增:单刷模式下隐藏考试按钮 ====================
var mode = localStorage.getItem("华医mode") || "1";
if (mode == "1") {
var examBtn = document.querySelector("#jrks"); // 考试按钮ID,根据实际页面调整
if (examBtn) {
examBtn.style.display = "none";
console.log("单刷模式:已隐藏考试按钮");
}
}
// ==================== 新增结束 ====================
// ==================== 新增:提前10秒切换视频的定时器(单刷模式) ====================
var preSwitchTimer = null;
var videoObj = document.querySelector("video");
if (videoObj && mode == "1") { // 只在单刷模式启用
preSwitchTimer = setInterval(function () {
try {
var currentTime = videoObj.currentTime;
var duration = videoObj.duration;
if (duration > 0 && (duration - currentTime) <= 10) { // 剩余 ≤10秒
console.log("视频剩余 ≤10秒(当前: " + Math.round(duration - currentTime) + "s),单刷模式下提前切换下一个视频");
clearInterval(preSwitchTimer); // 停止此定时器
playNextVideo(); // 立即切换
}
} catch (e) {
console.log("提前切换检查出错: ", e);
}
}, 1000); // 每秒检查一次
console.log("已启用提前10秒切换定时器(单刷模式)");
}
// ==================== 新增结束 ====================
// ==================== 优化:视频结束时立即检查 (视频+考试模式) ====================
// ==================== 优化:视频+考试模式 - 剩余60s 开始轮询 + onended 备用 ====================
var examPollingTimer = null;
if (videoObj && mode == "2") {
// 60s 轮询启动定时器
var examPollingStarter = setInterval(function () {
try {
var currentTime = videoObj.currentTime;
var duration = videoObj.duration;
if (duration > 0 && (duration - currentTime) <= 60) {
console.log("视频剩余 ≤60秒,开始考试轮询检查");
clearInterval(examPollingStarter); // 停止启动器
examPollingTimer = setInterval(examherftest, 500); // 启动专用轮询
}
} catch (e) {
console.log("60s 启动检查出错: ", e);
}
}, 1000); // 每秒检查剩余时间
// 结束时备用检查
videoObj.onended = function () {
console.log("视频 ended 事件触发,立即备用检查状态");
setTimeout(function () {
examherftest();
}, 500);
};
console.log("已启用60s轮询 + ended 备用检查 (视频+考试模式)");
}
// ==================== 优化结束 ====================
// 主定时器 (仅单刷用,视频+考试用以上专用的)
if (mode == "1") {
clock = setInterval(examherftest, 500);
}
// ==================== 修改结束 ====================
// 原有静音和播放逻辑保持不变...
switch (e) {
case 1:
window.s2j_onPlayerInitOver()
{
// console.log("polyv加载完毕,静音,稍后尝试触发一次播放");
player?.j2s_setVolume(0);
document.querySelector("video").defaultMuted = true;
setTimeout(function () {
try {
//document.querySelector("video").volume = 0;//实际测试,主要靠这一条静音
player.j2s_resumeVideo();
//document.querySelector("video").muted = true;
examherftest();
//document.querySelector("button[onclick='closeBangZhu()']").click();//关闭温馨提醒
} catch (error) {
console.log("上一段代码有误");
};
}, 2000); //延时点击播放,之前是5秒
};
break;
case 2:
window.on_CCH5player_ready()
{
//console.log("CCplayer加载完毕,静音,稍后尝试触发一次播放");
cc_js_Player?.setVolume(0);
document.querySelector("video").defaultMuted = true;
setTimeout(function () {
try {
//document.querySelector("video").volume = 0;//实际测试,主要靠这一条静音
cc_js_Player.play();
//document.querySelector("video").muted = true;
examherftest();
//document.querySelector("button[onclick='closeBangZhu()']").click();//关闭温馨提醒
} catch (error) {
console.log("上一段代码有误");
};
}, 2000); //延时点击播放,之前是5秒
};
break;
default:
console.log("其他播放器?");
};
};
},
doTest: function () {
var questions = JSON.parse(localStorage.getItem(keyTest)) || {};
var qRightAnswer = JSON.parse(localStorage.getItem(keyRightAnswer)) || {};
if (JSON.stringify(qRightAnswer) == "{}") {
qRightAnswer = LoadRightAnwser();
};
var qTestAnswer = {};
var index = 0;
console.log("=== 开始答题 ===");
console.log("keyTest:", questions);
while (true) {
var question = document.querySelectorAll("table[class='tablestyle']")[index];
if (question == null) break;
else {
var qRaw = question.querySelector(".q_name").innerText.substring(2);
var q = qRaw.replace(/\s*/g, ""); // 删空格
console.log("处理题 " + index + ": q = '" + q + "'");
if (qRightAnswer.hasOwnProperty(q)) {
console.log("有正确答案: " + qRightAnswer[q]);
var rightSelection = findAnwser("tbody", index, qRightAnswer[q]);
rightSelection.click();
} else {
console.log("无正确答案,当前: " + (questions[q] || '无'));
if (questions.hasOwnProperty(q)) {
// 保持上次
console.log("保持上次选项: " + questions[q]);
} else {
questions[q] = "A";
console.log("新题,默认A");
};
// 检测选项数
var labels = document.querySelectorAll("tbody")[index].getElementsByTagName("label");
var numOptions = labels.length;
var maxLetter = String.fromCharCode("A".charCodeAt(0) + numOptions - 1); // 4 → 'D', 5 → 'E'
console.log("题 " + index + " 选项数: " + numOptions + ", maxLetter: " + maxLetter);
var answer = getChoiceCode(questions[q]);
var element = labels[answer];
if (!element || answer >= numOptions) {
console.log("选项无效,重置A");
questions[q] = "A";
answer = 0;
element = labels[answer];
};
try {
var answerText = element.innerText.substring(3);
qTestAnswer[q] = answerText;
console.log("选 " + questions[q] + ": '" + answerText + "' (max: " + maxLetter + ")");
} catch (error) {
console.log("文本失败: " + error);
qTestAnswer[q] = "未知";
};
element.click();
};
index++;
};
};
localStorage.setItem(keyTest, JSON.stringify(questions));
localStorage.setItem(keyTestAnswer, JSON.stringify(qTestAnswer));
console.log("答题结束,qTestAnswer 键值对:", Object.entries(qTestAnswer));
console.log("更新 keyTest:", questions);
console.log("=== 答题结束 ===");
setTimeout(function () {
document.querySelector("#btn_submit").click();
}, (submitTime + Math.ceil(Math.random() * randomX)));
function findAnwser(qakey, index, rightAnwserText) {
var answerslist = document.querySelectorAll(qakey)[index];
var arr = answerslist.getElementsByTagName("label");
for (var i = 0; i < arr.length; i++) {
if (arr[i].innerText.substring(3) == rightAnwserText) {
return arr[i];
};
};
};
},
doResult: function () {
// 原有结果检测逻辑(增强 res 匹配,fallback 检查页面)
var tipsTextEl = $(".tips_text")[0] || document.querySelector(".tips_text");
var res = tipsTextEl ? tipsTextEl.innerText.trim() : ""; // trim() 去除空格
var dds = $(".state_cour_lis"); // 兼容旧版
var lis = document.querySelectorAll("ul.state_cour_ul li.state_cour_lis"); // 新结构
localStorage.removeItem(keyResult); // 移除错题表缓存
// ==================== 修复:优先图标 + 精确文本匹配,排除“未通过”误判 ====================
var tipsImgEl = document.querySelector(".tips_img");
var imgSrc = tipsImgEl ? tipsImgEl.src : "";
var isPassed = false;
// 优先:图标判断(更可靠,避免文本子串误判)
if (imgSrc.includes("tips_success.png")) {
isPassed = true;
console.log("通过图标匹配:tips_success.png");
saveRightAnwser(); // 记录最后一次答对的题目
SaveAllAnwser(); // 存储所有记录的答案
cleanKeyStorage(); // 如果通过清理答案
// ==================== 修改:只搜索并点击“立即学习”按钮 ====================
setTimeout(function () {
var nextBtn = null;
// 优先新结构
if (lis.length > 0) {
console.log("检测到新考试结果界面(ul.state_cour_ul),搜索'立即学习'按钮...");
for (var i = 0; i < lis.length; i++) {
var btn = lis[i].querySelector("input.state_lis_btn[value='立即学习']");
if (btn) {
nextBtn = btn;
var titleEl = lis[i].querySelector("p.state_lis_text");
console.log("找到下一个课程: " + (titleEl ? titleEl.title : "未知标题") + " (按钮: 立即学习)");
break; // 取第一个“立即学习”
}
}
} else if (dds.length > 0) {
// Fallback 旧结构
console.log("使用旧结构 fallback,搜索'立即学习'按钮...");
for (var i = 0; i < dds.length; i++) {
var btn = dds[i].querySelector("input.state_lis_btn[value='立即学习']");
if (btn) {
nextBtn = btn;
var titleEl = dds[i].querySelector("p");
console.log("找到下一个课程 (旧结构): " + (titleEl ? titleEl.title : "未知标题") + " (按钮: 立即学习)");
break;
}
}
}
if (nextBtn) {
// 有下一个:点击进入视频(学习后自动考试)
console.log("点击'立即学习'按钮,进入视频+考试循环");
nextBtn.click();
} else {
// 无下一个:所有完成,停止
console.log("未找到'立即学习'按钮,所有课程已完成或无需学习,无需进一步操作");
if (clock) clearInterval(clock); // 停止主定时器
}
}, (1000 + Math.ceil(Math.random() * 2000))); // 1-3秒随机延时
// ==================== 修改结束 ====================
} else { // 考试没过
console.log("考试未通过,标记错题并重考");
// 提示文本
try {
if (tipsTextEl) {
tipsTextEl.innerText = "本次未通过,正在尝试更换答案\r\n(此为正常现象,脚本几秒后刷新,请勿操作)";
}
} catch (e) {
console.log("设置提示文本失败: " + e);
}
var qWrong = {};
var elements = (lis.length > 0) ? Array.from(lis) : (dds.length > 0 ? Array.from(dds) : document.querySelectorAll('li[class*="lis"]'));
if (elements.length > 0) {
console.log("搜索错题... 元素数量: " + elements.length);
for (var i = 0; i < elements.length; i++) {
var imgEl = elements[i].querySelector("img");
var imgSrc = imgEl ? imgEl.src : "无";
var btn = elements[i].querySelector("input.state_lis_btn");
var btnValue = btn ? btn.value : "无";
var isWrong = false; // 补定义
console.log("检查元素 " + i + ": btnValue = '" + btnValue + "', imgSrc = '" + imgSrc + "'");
if (imgEl && imgEl.src && !imgEl.src.includes("bar_img")) {
isWrong = true;
console.log(" → img 错");
} else if (!imgEl) {
var textEl = elements[i].querySelector("p");
var text = textEl ? (textEl.title || textEl.innerText) : "无";
if (btnValue == "待考试" || text.includes("错") || btnValue.includes("未通过")) {
isWrong = true;
console.log(" → fallback 错 (btn: '" + btnValue + "')");
} else {
console.log(" → fallback 无错");
}
} else {
console.log(" → img 正确");
}
if (isWrong) {
var titleEl = elements[i].querySelector("p");
var qTitle = titleEl ? titleEl.title.replace(/\s*/g, "") : "未知";
qWrong[qTitle] = i;
console.log("标记错题: '" + qTitle + "'");
}
}
console.log("qWrong:", qWrong);
} else {
console.log("未找到元素");
}
localStorage.setItem(keyResult, JSON.stringify(qWrong));
saveRightAnwser();
var retryCount = parseInt(localStorage.getItem("retryCount") || "0") + 1;
localStorage.setItem("retryCount", retryCount.toString());
console.log("重考次数: " + retryCount);
setTimeout(function () {
if (retryCount > 10) {
console.log("超限10次,停止");
localStorage.removeItem("retryCount");
window.location.reload();
return;
}
var retryBtn = document.querySelector("input[value='重新考试']") ||
document.querySelector("input.state_foot_btn[value='重新考试']") ||
document.querySelector("input.state_edu[value='重新考试']");
if (retryBtn) {
retryBtn.click();
console.log("点击重新考试 (第 " + retryCount + " 次)");
} else {
var onclickBtn = document.querySelector("input[value='重新考试'][onclick*='exam.aspx']");
if (onclickBtn && onclickBtn.onclick) {
var onclickStr = onclickBtn.onclick.toString();
var cwidMatch = onclickStr.match(/cwid=([a-f0-9-]+)/i);
if (cwidMatch) {
var cwid = cwidMatch[1];
window.location.href = 'exam.aspx?cwid=' + cwid;
console.log("手动跳转重考: exam.aspx?cwid=" + cwid + " (第 " + retryCount + " 次)");
} else {
console.log("无 cwid,刷新");
window.location.reload();
}
} else {
console.log("无重考按钮,刷新");
window.location.reload();
}
}
}, (reTryTime + Math.ceil(Math.random() * randomX)) * 1);
}
},
};
};
//---------------------------------全局函数区------------------------------//
//答案记录函数区开始//
function SaveAllAnwser() {//保存历史题目答案
var qAllAnswer = JSON.parse(localStorage.getItem(keyAllAnswer)) || {};
var qRightAnswer = JSON.parse(localStorage.getItem(keyRightAnswer)) || {};
var qTitle = JSON.parse(localStorage.getItem(keyThisTitle)) || "没有记录到章节名称";
var qOldAnswer = qAllAnswer[qTitle] || {};
for (var q in qRightAnswer) {
qOldAnswer[q] = qRightAnswer[q];
};
qAllAnswer[qTitle] = qOldAnswer;
if (qAllAnswer != null) {//保存正确答案
localStorage.setItem(keyAllAnswer, JSON.stringify(qAllAnswer));
};
};
function LoadRightAnwser() {//加载历史题目答案
var qAllAnswer = JSON.parse(localStorage.getItem(keyAllAnswer)) || {};
//var qRightAnswer = JSON.parse(localStorage.getItem(keyRightAnswer)) ||{};
var qTitle = JSON.parse(localStorage.getItem(keyThisTitle)) || "没有记录到章节名称";
if (qTitle == "没有记录到章节名称") {
console.log("没找到章节名称");
return {};
};
var qOldAnswer = qAllAnswer[qTitle] || {};
return qOldAnswer
};
function saveRightAnwser() {
var qRightAnswer = JSON.parse(localStorage.getItem(keyRightAnswer)) || {};
var qTestAnswer = JSON.parse(localStorage.getItem(keyTestAnswer)) || {};
var qkeyTest = JSON.parse(localStorage.getItem(keyTest)) || {};
var qWrongs = JSON.parse(localStorage.getItem(keyResult)) || {};
// 简单键
var stdWrongs = {};
for (var wrongQ in qWrongs) {
stdWrongs[wrongQ.replace(/\s*/g, "")] = qWrongs[wrongQ];
}
qWrongs = stdWrongs;
console.log("=== saveRightAnwser ===");
console.log("qWrongs:", qWrongs);
console.log("qTestAnswer 键值对:", Object.entries(qTestAnswer));
for (var q in qTestAnswer) {
var stdQ = q.replace(/\s*/g, "");
var isWrong = qWrongs.hasOwnProperty(stdQ);
console.log("检查 q '" + q + "' (std: '" + stdQ + "'): 答 '" + qTestAnswer[q] + "', 错: " + isWrong);
if (!isWrong) {
console.log("正确: " + q + " = " + qTestAnswer[q]);
qRightAnswer[q] = qTestAnswer[q];
} else {
console.log("错题: " + q + ", 切下一个");
// 检测 maxLetter (从 qTestAnswer 推,假设上次答题有 labels)
var numOptions = 5; // 默认5,实际可从页面再检测,但 save 时无元素,固定或存
var maxLetter = String.fromCharCode("A".charCodeAt(0) + numOptions - 1);
if (qkeyTest.hasOwnProperty(q)) {
var oldOpt = qkeyTest[q];
qkeyTest[q] = getNextChoice(oldOpt, maxLetter);
console.log(" → 从 " + oldOpt + " 切到 " + qkeyTest[q] + " (max: " + maxLetter + ")");
} else {
qkeyTest[q] = "B";
console.log(" → 默认切B");
}
};
};
localStorage.removeItem(keyTestAnswer);
localStorage.setItem(keyRightAnswer, JSON.stringify(qRightAnswer));
localStorage.setItem(keyTest, JSON.stringify(qkeyTest));
console.log("更新 qRightAnswer:", qRightAnswer);
console.log("更新 keyTest:", qkeyTest);
console.log("=== 结束 ===");
};
//答案记录函数区结束//
//答案复制相关按钮
function addAnwserCopybtn() {//插入答案复制按钮
let alink = document.createElement("a");
alink.innerHTML = '显示已记录答案';
alink.style = btstyleB;
alink.onclick = function (event) {
var qAllAnswer = JSON.parse(localStorage.getItem(keyAllAnswer)) || {};
var Aout = JSON.stringify(qAllAnswer, null, "\t")
//Aout=encodeURIComponent(Aout);
//window.prompt("请复制",Aout);
if (document.getElementById("AnwserOut")) {
document.getElementById("AnwserOut").innerHTML = Aout;
} else {
let textout = document.createElement("textarea");
textout.id = "AnwserOut";
textout.innerHTML = Aout;
textout.rows = 20;
textout.cols = 30;
document.getElementById("main_div").parentNode.append(textout);
};
};
document.getElementById("main_div").parentNode.append(alink);
};
function DelAllAnwser() {//插入清除答案按钮
let alink = document.createElement("a");
alink.innerHTML = '清除已记录答案';
alink.style = btstyleB;
alink.onclick = function (event) {
var r = confirm("确定清除历史答案?!");
if (r) {
localStorage.removeItem(keyAllAnswer);
};
};
document.getElementById("main_div").parentNode.append(alink);
};
//答案复制相关按钮 end
function skipVideo() {//这是跳过视频的代码
var oVideo = document.getElementsByTagName('video')[0];
if (oVideo) {
oVideo.currentTime = oVideo.duration - 1
};
};
function clickexam() { //延时点击考试按钮。
console.log("已点击考试按钮");
setTimeout(function () {
document.querySelector("#jrks").click();
}, (Math.ceil(Math.random() * randomX)));
//}, (examTime + Math.ceil(Math.random() * randomX)));
};
//按钮插入函数相关
function addSkipbtn() {//插入按钮快进视频按钮
let alink = document.createElement("a");
alink.innerHTML = '快进视频';
alink.style = btstyleA;
alink.onclick = function (event) {
skipVideo();
};
document.querySelector("div[id='jj']").parentNode.append(alink);
};
function addratebtn(ra) {//倍率调整按钮
let alink = document.createElement("a");
alink.innerHTML = '' + ra + 'x';
alink.style = btstyleB;
alink.className = "speed";
alink.id = ra + "x";
alink.onclick = function (event) {
ratechg(ra);
try {
var arr = document.querySelectorAll("a[class='speed']");
arr.forEach(function (item, index, arr) {
arr[index].style = btstyleB;
});
} catch (error) {
};
alink.style = btstyleC;
};
document.querySelector("div[id='jj']").parentNode.append(alink);
}
function ratechg(ra) {//倍率调整
var videoObj = document.querySelector("video")
try {
clearInterval(nspeed);
nspeed = setInterval(() => {
videoObj.playbackRate = ra;
}, 1 * 1000);
localStorage.setItem(keyPlayRate, ra);
//document.querySelector("a[id=" + "'" + ra + "x']").style = btstyleC;
//document.getElementById("playrate").innerHTML = "当前播放速率" + ra + "x";
//console.log("倍率调整为" + ra);
} catch (error) { console.log("倍率调整错误" + error); };
};
function addrateinfo() {//插入说明
let adiv1 = document.createElement("div");
adiv1.innerHTML = '当前播放速率';
adiv1.id = 'playrate';
adiv1.style = "font-size: 15px;text-align: center;margin-top: 10px;";
document.querySelector("div[id='jj']").parentNode.append(adiv1);
};
function addinfo() {//插入说明
// 确定初始模式文本
var moderesult = localStorage.getItem("华医mode");
var modeText = (moderesult == '2') ? "当前模式:视频+考试" : "当前模式:单刷视频";
// 创建模式切换按钮
var checkbox = document.createElement('div');
checkbox.innerHTML = '' + modeText + '
[点击此处切换] ';
document.querySelector("div[id='jj']").parentNode.append(checkbox);
// 定义一个函数,用于根据当前模式更新UI(按钮文字和面板颜色)
function updateModeStyle() {
var currentMode = localStorage.getItem("华医mode");
var panel = document.getElementById('Div1');
var modeButton = document.getElementById('mode');
if (currentMode == '2') {
modeButton.innerText = "当前模式:视频+考试\n[点击此处切换]";
panel.style.backgroundColor = "rgba(255, 204, 203, 0.8)"; // 淡红色背景
} else {
modeButton.innerText = "当前模式:单刷视频\n[点击此处切换]";
panel.style.backgroundColor = "rgba(184, 247, 255, 0.7)"; // 原始淡蓝色背景
}
}
// 页面加载时立即应用一次样式
updateModeStyle();
// 为按钮绑定点击事件
document.getElementById('mode').onclick = function () {
var currentMode = localStorage.getItem("华医mode");
if (currentMode == '2') {
localStorage.setItem("华医mode", "1"); // 从模式2切换到模式1
} else {
localStorage.setItem("华医mode", "2"); // 从模式1切换到模式2
}
updateModeStyle(); // 每次点击后立即更新UI
};
// --- 原始脚本中的其他说明内容 ---
let adiv2 = document.createElement("div");
adiv2.innerHTML = '切换模式后请刷新一下网页';
adiv2.id = 'jsinfo';
adiv2.style = "position:relative;left:10px;top:5px;width:240px; font-size:13px;text-align: justify;border: 1px dashed #ff9595;padding:5px;";
document.querySelector("div[id='jj']").parentNode.append(adiv2);
$('div:contains("观看视频完成后,才能进入考试")').eq(-1).text('建议Chrome+tampermonkey');
};
function changelayout() {
document.querySelector("div[id='jj']").remove();
//document.querySelector("img[id='photo']").outerHTML = ``;
//document.querySelector("img[id='photo']").style.width = "120px";
//document.querySelector("img[id='photo']").style.height = "120px";
document.querySelectorAll("div[class='title']")[0].children[0].style = "color: #2600ffff;font-weight: bold";
document.querySelectorAll("div[class='title']")[0].children[0].innerText = "模式切换";
//document.querySelector("div[class='imgtext']").children[1].style.width = "125px";
//document.querySelector("div[class='imgtext']").children[1].style = "color: #ff0000;padding-top:10px";
//document.querySelector("div[class='imgtext']").children[1].innerText = "作 者\n创作优化不易\n投点小费吧\n❤谢啦❤\n❤"
//document.querySelector("div[class='imgtext']").children[1].style.left = "10px";
document.querySelector("div[class='top']").outerHTML = '