// ==UserScript==
// @name 超星网课助手/刷课/搜题(支持图片)/考试/all in one(fake题)
// @namespace lyj
// @version 3.2.8
// @description 考试版已经合并,自动答题,视频自动完成,章节测验自动答题提交,自动切换任务点等,开放自定义参数
// @author lyj
// @match *://*.chaoxing.com/*
// @match *://*.edu.cn/*
// @connect ti.fakev.cn
// @connect baidu.com
// @require https://cdn.bootcdn.net/ajax/libs/jquery/2.0.0/jquery.js
// @run-at document-end
// @grant unsafeWindow
// @grant GM_xmlhttpRequest
// @grant GM_setClipboard
// @grant GM_setValue
// @grant GM_getValue
// @license MIT
// @original-script https://greasyfork.org/scripts/369625
// @original-author wyn665817
// @original-license MIT
// @antifeature tracking
// @downloadURL none
// ==/UserScript==
var _hmt = _hmt || [];
(function() {
var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?c2daa8e62b938a0869a122a0d6da4e9a";
var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s);
})();
function getRandomInteger(min, max) {
return Math.floor(Math.random() * (max - min)) + min;
}
const api_array = [
"http://ti.fakev.cn/hashTopic?question=",
];
// 设置修改后,需要刷新或重新打开网课页面才会生效
var setting = {
api: getRandomInteger(0, api_array.length), // 答题接口编号,参考上方,默认随机
// 5E3 == 5000,科学记数法,表示毫秒数
time:4e3, // 默认响应速度为5秒,不建议小于3秒
review: 0, // 复习模式,完整挂机视频(音频)时长,支持挂机任务点已完成的视频和音频,默认关闭
queue: 1, // 队列模式,开启后任务点逐一完成,关闭则单页面所有任务点同时进行,默认开启
submit: 1, //答案收录,开启后可在作业完成界面自动收录题目,默认开启
// 1代表开启,0代表关闭
video: 1, // 视频支持后台、切换窗口不暂停,支持多视频,默认开启
work: 1, // 自动答题功能(章节测验),作业需要手动开启查询,高准确率,默认开启
audio: 1, // 音频自动播放,与视频功能共享vol和rate参数,默认开启
book: 1, // 图书阅读任务点,非课程阅读任务点,默认开启
docs: 1, // 文档阅读任务点,PPT类任务点自动完成阅读任务,默认开启
// 本区域参数,上方为任务点功能,下方为独立功能
jump: 1, // 自动切换任务点、章节、课程(需要配置course参数),默认开启
read: "65", // 挂机课程阅读时间,单位是分钟,'65'代表挂机65分钟,请手动打开阅读页面,默认'65'分钟
face: 0, // 解除面部识别(不支持二维码类面部采集),此功能仅为临时解除,默认关闭
total: 1, // 显示课程进度的统计数据,在学习进度页面的上方展示,默认开启
copy: 0, // 自动复制答案到剪贴板,也可以通过手动点击按钮或答案进行复制,默认关闭
hide: 0, // 不加载答案搜索提示框,键盘↑和↓可以临时移除和加载,默认关闭
// 仅开启video(audio)时,修改此处才会生效
line: "公网1", // 视频播放的默认资源线路,此功能适用于系统默认线路无资源,默认'公网1'
http: "标清", // 视频播放的默认清晰度,无效参数则使用系统默认清晰度,默认'标清'
// 本区域参数,上方为video功能独享,下方为audio功能共享
vol: "0", // 默认音量的百分数,设定范围:[0,100],'0'为静音,默认'0'
rate: "1", // 视频播放默认倍率,参数范围0∪[0.0625,16],'0'为秒过,默认'1'倍
// 仅开启work时,修改此处才会生效
auto: 1, // 答题完成后自动提交,默认打开
none: 0, // 无匹配答案时执行默认操作,关闭后若题目无匹配答案则会暂时保存已作答的题目,默认关闭
scale: 0, // 富文本编辑器高度自动拉伸,用于文本类题目,答题框根据内容自动调整大小,默认关闭
// 仅开启jump时,修改此处才会生效
course: 0, // 当前课程完成后自动切换课程,仅支持按照根目录课程顺序切换,默认关闭
lock: 1, // 跳过未开放(图标是锁)的章节,即闯关模式或定时发放的任务点,默认开启
//验证配置区
ischeck: 0, //是否检测完毕,可手动更改为1避免全部检测(如果改为1可能会不能使用)
},
_self = unsafeWindow,
url = location.pathname,
top = _self;
var currentURL = window.location.pathname; // 获取当前网页地址
if (currentURL.slice(0,32) == "/exam/test/reVersionTestStartNew"){
_self = unsafeWindow,
$ = _self.jQuery,
UE = _self.UE;
String.prototype.toCDB = function() {
return this.replace(/\s/g, '').replace(/[\uff01-\uff5e]/g, function(str) {
return String.fromCharCode(str.charCodeAt(0) - 65248);
}).replace(/[“”]/g, '"').replace(/[‘’]/g, "'").replace(/。/g, '.');
};
setting.TiMu = [
filterImg('.Cy_TItle .clearfix').replace(/\s*(\d+\.\d+分)$/, '').replace("题型说明:请输入题型说明",'').replace(/\s*/g,""),
$('[name^=type]:not([id])').val() || '-1',
$('.cur a').text().trim() || '无',
$('li .clearfix').map(function() {
return filterImg(this);
})
];
setting.div = $(
'
' +
'
' +
'
正在搜索答案...
' +
'
' +
'
' +
'
' +
'
' +
'
' +
'
' +
'' +
'' +
'' + ($('#randomOptions').val() == 'false' ? '' : '本次考试的选项为乱序 脚本会选择正确的选项') + ' | ' +
'
' +
'' +
'题目(点击可复制) | ' +
'答案(点击可复制) | ' +
'
' +
'' +
'' +
'' +
'已关闭 本次自动切换 | ' +
'
' +
'' +
'' +
'' +
' | ' +
'
' +
'' +
'
' +
'
' +
'
'
).appendTo('body').on('click', 'button, td', function() {
var num = setting.$btn.index(this);
if (num == -1) {
GM_setClipboard($(this).text());
} else if (num === 0) {
if (setting.loop) {
clearInterval(setting.loop);
delete setting.loop;
num = ['已暂停搜索', '继续答题'];
} else {
setting.loop = setInterval(findTiMu, 2E3);
num = ['正在搜索答案...', '暂停答题'];
}
setting.$div.html(function() {
return $(this).data('html') || num[0];
}).removeData('html');
$(this).html(num[1]);
} else if (num == 1) {
setting.jump = 0;
setting.$div.html(function() {
return arguments[1].replace('即将切换下一题', '未开启自动切换');
});
setting.div.find('tfoot').add(this).toggle();
} else if (num == 2) {
location.reload();
} else if (num == 3) {
GM_setClipboard(setting.div.find('td:last').text());
} else if (num == 4) {
($('.leftCard .saveYl')[0] || $()).click();
}
}).detach(setting.hide ? '*' : 'html');
setting.$btn = setting.div.children('button');
setting.$div = setting.div.children('div:eq(0)');
$(document).keydown(function(event) {
if (event.keyCode == 38) {
setting.div.detach();
} else if (event.keyCode == 40) {
setting.div.appendTo('body');
}
});
if (setting.scale) _self.UEDITOR_CONFIG.scaleEnabled = false;
$.each(UE.instants, function() {
var key = this.key;
this.ready(function() {
this.destroy();
UE.getEditor(key);
});
});
setting.loop = setInterval(findTiMu, 2E3);
function findTiMu() {
GM_xmlhttpRequest({
method: 'GET',
url:api_array[setting.api] + encodeURIComponent(setting.TiMu[0]) + '&type=' + setting.TiMu[1],
headers: {
'Authorization': setting.token,
},
timeout: setting.time,
onload: function(xhr) {
if (!setting.loop) {
} else if (xhr.status == 200) {
var obj = $.parseJSON(xhr.responseText) || {};
if (obj.code) {
var data = String(obj.data).replace(/&/g, '&').replace(/<(?!img)/g, '<'),
que = setting.TiMu[0].match('
' : obj.data;
setting.div.find('tbody').append(
'' +
'' + que + ' | ' +
'' + (/^http/.test(data) ? obj.data : '') + data + ' | ' +
'
'
);
setting.copy && GM_setClipboard(obj.data);
setting.$btn.eq(3).show();
fillAnswer(obj);
} else {
setting.$div.html(obj.data || '服务器繁忙,正在重试...');
}
setting.div.children('span').html(obj.msg || '');
} else if (xhr.status == 403) {
var html = xhr.responseText.indexOf('{') ? '请求过于频繁,建议稍后再试' : $.parseJSON(xhr.responseText).data;
setting.$div.data('html', html).siblings('button:eq(0)').click();
} else {
setting.$div.text('服务器异常,正在重试...');
}
},
ontimeout: function() {
setting.loop && setting.$div.text('服务器超时,正在重试...');
}
});
}
function fillAnswer(obj, tip) {
var $input = $(':radio, :checkbox', '.Cy_ulBottom'),
str = String(obj.data).toCDB() || new Date().toString(),
data = str.split(/#|\x01|\|/),
opt = obj.opt || str,
btn = $('.saveYl:contains(下一题)').offset();
// $input.filter(':radio:checked').prop('checked', false);
obj.code > 0 && $input.each(function(index) {
if (this.value == 'true') {
data.join().match(/(^|,)(正确|是|对|√|T|ri)(,|$)/) && this.click();
} else if (this.value == 'false') {
data.join().match(/(^|,)(错误|否|错|×|F|wr)(,|$)/) && this.click();
} else {
index = setting.TiMu[3][index].toCDB() || new Date().toString();
index = $.inArray(index, data) + 1 || (setting.TiMu[1] == '1' && str.indexOf(index) + 1);
Boolean(index) == this.checked || this.click();
}
}).each(function() {
if (!/^A?B?C?D?E?F?G?$/.test(opt)) return false;
Boolean(opt.match(this.value)) == this.checked || this.click();
});
if (setting.TiMu[1].match(/^[013]$/)) {
tip = $input.is(':checked') || setting.none && (($input[Math.floor(Math.random() * $input.length)] || $()).click(), ' ');
} else if (setting.TiMu[1].match(/^(2|[4-9]|1[08])$/)) {
data = String(obj.data).split(/#|\x01|\|/);
tip = $('.Cy_ulTk textarea').each(function(index) {
index = (obj.code > 0 && data[index]) || '';
UE.getEditor(this.name).setContent(index.trim());
}).length;
tip = (obj.code > 0 && data.length == tip) || setting.none && ' ';
setting.len = str.length * setting.time / 10;
}
if (tip == ' ') {
tip = '已执行默认操作';
} else if (tip) {
tip = '自动答题已完成';
} else if (tip === undefined) {
tip = '该题型不支持自动答题';
} else {
tip = '未找到有效答案';
}
if (btn) {
tip += setting.jump ? ',即将切换下一题' : ',未开启自动切换';
setInterval(function() {
if (!setting.jump) return;
var mouse = document.createEvent('MouseEvents'),
arr = [btn.left + Math.ceil(Math.random() * 80), btn.top + Math.ceil(Math.random() * 26)];
mouse.initMouseEvent('click', true, true, document.defaultView, 0, 0, 0, arr[0], arr[1], false, false, false, false, 0, null);
_self.event = $.extend(true, {}, mouse);
delete _self.event.isTrusted;
_self.getTheNextQuestion(1);
}, setting.len || Math.ceil(setting.time * Math.random()) * 2);
} else {
setting.$btn.eq(1).hide();
tip = '答题已完成,请自行查看答题详情';
}
setting.$div.data('html', tip).siblings('button:eq(0)').hide().click();
}
function filterImg(dom) {
return $(dom).clone().find('img[src]').replaceWith(function() {
return $('').text('
');
}).end().find('iframe[src]').replaceWith(function() {
return $('').text('