"use strict"; // ==UserScript== // @name 51talk选择最好最合适的老师-经验-好评率-年龄-收藏数 // @version 2021.4.14001 // @namespace https://github.com/niubilityfrontend // @description 辅助选老师-排序显示,经验值计算|好评率|显示年龄|列表显示所有教师 // @author jimbo // @license OSL-3.0 // @supportURL https://github.com/niubilityfrontend/hunttingteacheron51talk // @match *://www.51talk.com/ReserveNew/index* // @match *://www.51talk.com/TeacherNew/* // @match *://www.51talk.com/user/* // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_registerMenuCommand // @require https://ajax.aspnetcdn.com/ajax/jQuery/jquery-3.5.0.min.js // @require https://ajax.aspnetcdn.com/ajax/jquery.ui/1.12.1/jquery-ui.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/pace/1.0.2/pace.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/free-jqgrid/4.15.5/i18n/grid.locale-cn.js // @require https://cdnjs.cloudflare.com/ajax/libs/free-jqgrid/4.15.5/jquery.jqgrid.min.js // @require https://greasyfork.org/scripts/388372-scrollfix/code/scrollfix.js?version=726657 // @require https://gitcdn.link/repo/kufii/My-UserScripts/fa4555701cf5a22eae44f06d9848df6966788fa8/libs/gm_config.js // // @downloadURL none // ==/UserScript== (function () { "use strict"; ///extend method parameters of window, get parameter's value with key case-insensitive (function ($) { var PropertiesCaseInsensitive = { has: function has(target, prop) { if (typeof prop === "symbol") { return prop in target; // pass through; or 'return;' if you want to block pass through } prop = prop.toLowerCase(); if (prop in target) return true; var keys = Object.keys(target), i = keys.length; while (i--) { if (keys[i] && keys[i].toLowerCase() == prop) return true; } return false; }, get: function get(target, prop, receiver) { if (typeof prop === "symbol") { return target[prop]; } prop = prop.toLowerCase(); if (prop in target) return target[prop]; var keys = Object.keys(target), i = keys.length; while (i--) { if (keys[i] && keys[i].toLowerCase() == prop) return target[keys[i]]; } return undefined; }, set: function set(target, prop, value) { if (typeof prop === "symbol") { target[prop] = value; } target[prop.toLowerCase()] = value; return true; } }; $.extend(window, { parameters: function parameters(url) { // get query string from url (optional) or window var queryString = url ? url.split("?")[1] : window.location.search.slice(1), cachedkey = "urlparameters" + queryString, obj = $(window).data(cachedkey); if (obj == undefined) { obj = new Proxy({}, PropertiesCaseInsensitive); $(window).data(cachedkey, obj); } else return obj; // we'll store the parameters here // if query string exists if (queryString) { // stuff after # is not part of query string, so get rid of it queryString = queryString.split("#")[0]; // split our query string into its component parts var arr = queryString.split("&"); for (var i = 0; i < arr.length; i++) { // separate the keys and the values var a = arr[i].split("="), paramName = a[0], paramValue = typeof a[1] === "undefined" ? true : a[1]; // set parameter name and value (use 'true' if empty) // if the paramName ends with square brackets, e.g. colors[] or colors[2] if (paramName.match(/\[(\d+)?\]$/)) { // create key if it doesn't exist var key = paramName.replace(/\[(\d+)?\]/, ""); if (!obj[key]) obj[key] = []; // if it's an indexed array e.g. colors[2] if (paramName.match(/\[\d+\]$/)) { // get the index value and add the entry at the appropriate position var index = /\[(\d+)\]/.exec(paramName)[1]; obj[key][index] = paramValue; } else { // otherwise add the value to the end of the array obj[key].push(paramValue); } } else { // we're dealing with a string if (!obj[paramName]) { // if it doesn't exist, create property obj[paramName] = paramValue; } else if (obj[paramName] && typeof obj[paramName] === "string") { // if property does exist and it's a string, convert it to an array obj[paramName] = [obj[paramName]]; obj[paramName].push(paramValue); } else { // otherwise add the property obj[paramName].push(paramValue); } } } } return obj; } }); })($); ///date to string with formater (function ($) { var getPaddedComp = function getPaddedComp(comp) { var len = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 2; if (len < 1) len = 1; comp = "" + comp; var paddedLen = len - ("" + comp).length; if (paddedLen > 0) { return [...Array(paddedLen).fill("0"), ...comp].join(""); } else return comp; }, o = { "[y|Y]{4}": function yY4(date) { return date.getFullYear(); }, // year "[y|Y]{2}": function yY2(date) { return date.getFullYear().toString().slice(2); }, // year MM: function MM(date) { return getPaddedComp(date.getMonth() + 1); }, //month M: function M(date) { return date.getMonth() + 1; }, //month "[d|D]{2}": function dD2(date) { return getPaddedComp(date.getDate()); }, //day "[d|D]{1}": function dD1(date) { return date.getDate(); }, //day "h{2}": function h2(date) { return getPaddedComp(date.getHours() > 12 ? date.getHours() % 12 : date.getHours()); }, //hour "h{1}": function h1(date) { return date.getHours() > 12 ? date.getHours() % 12 : date.getHours(); }, //hour "H{2}": function H2(date) { return getPaddedComp(date.getHours()); }, //hour "H{1}": function H1(date) { return date.getHours(); }, //hour "m{2}": function m2(date) { return getPaddedComp(date.getMinutes()); }, //minute "m{1}": function m1(date) { return date.getMinutes(); }, //minute "s+": function s(date) { return getPaddedComp(date.getSeconds()); }, //second "f+": function f(date) { return getPaddedComp(date.getMilliseconds(), 3); }, //millisecond, "f{1}": function f1(date) { return getPaddedComp(date.getMilliseconds(), 0); }, //millisecond, "b+": function b(date) { return date.getHours() >= 12 ? "PM" : "AM"; } }; $.extend(Date.prototype, { toString: function toString(format) { var formattedDate = format; for (var k in o) { if (new RegExp("(" + k + ")").test(format)) { formattedDate = formattedDate.replace(RegExp.$1, o[k](this)); } } return formattedDate; } }); })($); //扩展基本类型方法 array.clean(val), Number.toString(len),String.toFloat, String.toInt,String.startsWtih,String.endsWith, ** String.replaceAll区别育默认的string.replace (function ($) { $.extend(Array.prototype, { clean: function clean() { for (var deleteValue = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : "", i = 0; i < this.length; i++) { if (this[i] == deleteValue) { this.splice(i, 1); i--; } } return this; } }); $.extend(Number.prototype, { toString: function toString(num) { if (isNaN(num)) num = 2; return this.toFixed(num); } }); $.extend(String.prototype, { toFloat: function toFloat() { return parseFloat(this); }, toInt: function toInt() { return parseInt(this); }, startsWith: function startsWith(str) { return this.slice(0, str.length) == str; }, endsWith: function endsWith(str) { return this.slice(-str.length) == str; }, includes: function includes(str) { return this.indexOf(str) > -1; }, replaceAll: function replaceAll(search, replacement) { var target = this; return target.replace(new RegExp(search, "g"), replacement); } }); })($); var config = GM_config([{ key: "pagecount", label: "最大页数 (自动获取时)", "default": 20, type: "dropdown", values: [0, 5, 10, 20, 50, 1e3] }, { key: "newBatcherKeyHours", label: "批次更新间隔(小时),0为每次更新", "default": 24, type: "dropdown", values: [0, 1, 2, 3, 5, 10, 24, 168, 168e3] }, { key: "markRankRed", label: "突出前N名教师的名次", "default": 100, type: "dropdown", values: [5, 10, 30, 50, 120, 500, 3e3, 5e3, 10080] }, { key: "version", type: "hidden", "default": 1 }]), conf = config.load(); config.onsave = function (cfg) { conf = cfg; $("#autogetnextpage").text("自动获取" + getAutoNextPagesCount() + "页"); }; GM_registerMenuCommand("设置", config.setup); //*://www.51talk.com/ReserveNew/index* //https://www.51talk.com/TeacherNew/info/t26501111 //https://www.51talk.com/TeacherNew/teacherComment?tid=t26501111&type=all&has_msg=1 //https://www.51talk.com/TeacherNew/teacherComment?tid=t26501111&type=good&has_msg=1 //https://www.51talk.com/TeacherNew/teacherComment?tid=t26501111&type=bad&has_msg=1 //https://www.51talk.com/user/study_center_zx //https://www.51talk.com/user/study_center_zy //https://www.51talk.com/user/study_center_xx var url = window.location.href.toLocaleLowerCase(), settings = { url: url, tid: url.match(/(t\d+)/g), pagecount: conf.pagecount, isDetailPage: url.includes("teachernew"), isListPage: url.includes("reservenew"), isCoursePage: url.includes("study_center") }; function gettid() { return settings.tid; } /** * 提交运算函数到 document 的 fx 队列 */ var submit = function submit(fun) { var queue = $.queue(document, "fx", fun); if (queue[0] == "inprogress") { return; } $.dequeue(document); }; function getorAddSession(key, func) { if (!(key in sessionStorage)) { var data = typeof func == "function" ? func(key) : func; sessionStorage.setItem(key, data); return data; } return sessionStorage.getItem(key); } Pace.Options = { ajax: false, // disabled document: false, // disabled eventLag: false, // disabled elements: { selectors: ["#filterdialog"] } }; function sleep(delay) { var start = Date.now(); while (Date.now() - start < delay) { continue; } } var asc = function asc(a, b) { var av = $(a).attr("indicator"), bv = $(b).attr("indicator"); if (!av || !bv) return 0; return $(a).attr("indicator").toFloat() > $(b).attr("indicator").toFloat() ? 1 : -1; }, desc = function desc(a, b) { var av = $(a).attr("indicator"), bv = $(b).attr("indicator"); if (!av || !bv) return 0; return $(a).attr("indicator").toFloat() > $(b).attr("indicator").toFloat() ? -1 : 1; }, sortByIndicator = function sortByIndicator(sortBy) { var sortEle = $(".s-t-content.f-cb .item").sort(sortBy); $(".s-t-content.f-cb").empty().append(sortEle); }; function getBatchNumber() { if (conf.newBatcherKeyHours <= 0) return Date.now(); return parseInt(Date.now() / conf.newBatcherKeyHours / 36e5) * conf.newBatcherKeyHours * 36e5; } function getLeftPageCount() { var pages = Number($(".s-t-page>.next-page:first").prev().text()), curr = Number($(".s-t-page>.active:first").text()); if (pages) return pages - curr;else return 0; } function getAutoNextPagesCount() { var pages = getLeftPageCount(); if (settings.pagecount > pages) return pages;else return settings.pagecount; } $("head").append("\n \n \n "); $("head").append(""); var maxrate = 0, minrate = 99999, maxlabel = 0, minlabel = 9999999, maxfc = 0, minfc = 999999, maxage = 0, minage = 99999, configExprMilliseconds = 36e5 * GM_getValue("tinfoexprhours", 168), num = /[0-9]*/g; function updateTeacherinfoToUI(jqel, tinfo) { if (tinfo.label > maxlabel) maxlabel = tinfo.label; if (tinfo.label < minlabel) minlabel = tinfo.label; if (tinfo.favoritesCount > maxfc) maxfc = tinfo.favoritesCount; if (tinfo.favoritesCount < minfc) minfc = tinfo.favoritesCount; if (tinfo.thumbupRate > maxrate) maxrate = tinfo.thumbupRate; if (tinfo.thumbupRate < minrate) minrate = tinfo.thumbupRate; if (tinfo.age > maxage) maxage = tinfo.age ? tinfo.age : 100; if (tinfo.age < minage) minage = tinfo.age ? tinfo.age : 0; jqel.attr("teacherinfo", JSON.stringify(tinfo)); jqel.find(".teacher-name").html(jqel.find(".teacher-name").html() + "
|\n | ")); // jqel.find(".teacher-age").html(jqel.find(".teacher-age").html() + " | "); jqel.attr("indicator", tinfo.indicator); } function executeFilters(uifilters) { var tcount = 0, hidecount = 0; $.each($(".item"), function (i, item) { var node = $(item), tinfojson = node.attr("teacherinfo"); if (!tinfojson) { return true; } var tinfo = JSON.parse(tinfojson), ret = true; if (!isNaN(tinfo.thumbupRate)) ret = tinfo.thumbupRate >= uifilters.rate1 && tinfo.thumbupRate <= uifilters.rate2; if (!isNaN(tinfo.label)) ret = tinfo.label >= uifilters.l1 && tinfo.label <= uifilters.l2 && ret; if (!isNaN(tinfo.age)) tinfo.age >= uifilters.age1 && tinfo.age <= uifilters.age2 && ret; if (!isNaN(tinfo.favoritesCount)) tinfo.favoritesCount >= uifilters.fc1 && tinfo.favoritesCount <= uifilters.fc2 && ret; if (ret) { if (node.is(":hidden")) { //如果node是隐藏的则显示node元素,否则隐藏 node.show(); node.animate({ left: "+=50" }, 3500).animate({ left: "-=50" }, 3500); } else {//nothing todo } tcount++; } else { node.css("color", "white").hide(); hidecount++; } }); $("#tcount").text(tcount); $("#thidecount").text(hidecount); } function getUiFilters() { var l1 = $("#tlabelslider").slider("values", 0), l2 = $("#tlabelslider").slider("values", 1), rate1 = $("#thumbupRateslider").slider("values", 0), rate2 = $("#thumbupRateslider").slider("values", 1), age1 = $("#tAgeSlider").slider("values", 0), age2 = $("#tAgeSlider").slider("values", 1), fc1 = $("#fcSlider").slider("values", 0), fc2 = $("#fcSlider").slider("values", 1); return { l1: l1, l2: l2, rate1: rate1, rate2: rate2, age1: age1, age2: age2, fc1: fc1, fc2: fc2 }; } function getinfokey() { return "tinfo-" + gettid(); } if (settings.isListPage) { $(".item-top-cont").prop("innerHTML", function (i, val) { return val.replaceAll("", ""); }); // 自动获取时,显示停止按钮 submit(function (next) { var totalPages = Number($(".s-t-page>a:eq(-2)").text()), curPageId = window.parameters().pageID ? window.parameters().pageID : 1, remainPages = totalPages - curPageId, autonextpagecount = GM_getValue("autonextpagecount", 1); if (autonextpagecount > 0 && $(".s-t-page>.next-page").length > 0) { var dialog = $("
\n

\n\u6B63\u5728\u6839\u636E\u60A8\u7684\u9009\u62E9\u81EA\u52A8\u83B7\u53D6\u6559\u5E08\u4FE1\u606F

\n\u5269\u4F59".concat(sessionStorage.getItem("selectedTimeSlotsRemain"), "/").concat(sessionStorage.getItem("selectedTimeSlotsTotal"), "\u4E2A\u65F6\u6BB5\uFF0C

\n\u5F53\u524D\u65F6\u6BB5\u7EA6").concat(totalPages * 28, "\u4E2A\u6559\u5E08\uFF0C\u83B7\u53D6\u7B2C").concat(curPageId, "/").concat(totalPages, "\u9875\uFF0C\u8FDB\u5EA6").concat(Math.floor(curPageId / totalPages * 100), "%,
\n\n

\n
")); dialog.appendTo("body"); dialog.dialog({ resizable: false, height: "auto", width: 400, modal: false, buttons: { 立即停止: function _() { sessionStorage.setItem("selectedTimeSlots", ""); GM_setValue("autonextpagecount", 0); $(this).dialog("close"); } // [`取后${(remainPages*0.25).toFixed(0)}页`]: function() { // sessionStorage.setItem('selectedTimeSlots', ''); // GM_setValue('autonextpagecount', (remainPages * 0.25).toFixed(0)); // $(this).dialog("close"); // }, // [`取后${(remainPages*0.5).toFixed(0)}页`]: function() { // sessionStorage.setItem('selectedTimeSlots', ''); // GM_setValue('autonextpagecount', (remainPages * 0.5).toFixed(0)); // $(this).dialog("close"); // }, // [`取后${(remainPages*0.75).toFixed(0)}页`]: function() { // sessionStorage.setItem('selectedTimeSlots', ''); // GM_setValue('autonextpagecount', (remainPages * 0.75).toFixed(0)); // $(this).dialog("close"); // }, } }); } next(); }); function getTeacherInfoInList(jqel) { var age = 0, label = function () { var j_len = jqel.find(".label").text().match(num).clean("").length, l = 0; for (var j = 0; j < j_len; j++) { l += Number(jqel.find(".label").text().match(num).clean("")[j]); } return l; }(), name = jqel.find(".teacher-name").text(), type = $(".s-t-top-list .li-active").text(), effectivetime = getBatchNumber(); if (type == "收藏外教") { var isfavorite = true; return { age: age, label: label, name: name, effectivetime: effectivetime, isfavorite: isfavorite }; } else return { age: age, label: label, name: name, effectivetime: effectivetime, type: type }; } //获取列表中数据 $(".item").each(function (index, el) { submit(function (next) { Pace.track(function () { var jqel = $(el), tid = jqel.find(".teacher-details-link a").attr("href").replace("https://www.51talk.com/TeacherNew/info/", "").replace("http://www.51talk.com/TeacherNew/info/", ""), tinfokey = "tinfo-" + tid, teacherlistinfo = getTeacherInfoInList(jqel), tinfo = GM_getValue(tinfokey); if (tinfo) { var now = Date.now(); if (!tinfo.expire) { tinfo.expire = new Date(1970, 1, 1).getTime(); } tinfo = $.extend(tinfo, teacherlistinfo); GM_setValue(tinfokey, tinfo); if (now - tinfo.expire < configExprMilliseconds) { updateTeacherinfoToUI(jqel, tinfo); next(); return true; } } // ajax 请求一定要包含在一个函数中 var start = Date.now(); $.ajax({ url: window.location.protocol + "//www.51talk.com/TeacherNew/teacherComment?tid=" + tid + "&type=bad&has_msg=1", type: "GET", dateType: "html", success: function success(r) { var jqr = $(r); if (jqr.find(".teacher-name-tit").length > 0) { var tempitem = jqr.find(".teacher-name-tit")[0]; tempitem.innerHTML = tempitem.innerHTML.replace("", ""); } if (jqr.find(".evaluate-content-left span").length >= 3) { var thumbup = Number(jqr.find(".evaluate-content-left span:eq(1)").text().match(num).clean("")[0]), thumbdown = Number(jqr.find(".evaluate-content-left span:eq(2)").text().match(num).clean("")[0]), favoritesCount = Number(jqr.find(".clear-search").text().match(num).clean("")[0]), isfavorite = jqr.find(".go-search.cancel-collection").length > 0, agesstr = jqr.find(".teacher-name-tit > .age.age-line").text().match(num).clean(""), tage = Number(agesstr[1]), age = Number(agesstr[0]), slevel = jqr.find(".sui-students").text(); jqr.remove(); var tinfo = { slevel: slevel, tage: tage, age: age, thumbup: thumbup, thumbdown: thumbdown, thumbupRate: 100, favoritesCount: favoritesCount, isfavorite: isfavorite, expire: Date.now() }; tinfo = $.extend(tinfo, teacherlistinfo); tinfo.thumbupRate = calcThumbRate(tinfo); tinfo.indicator = calcIndicator(tinfo); GM_setValue(tinfokey, tinfo); updateTeacherinfoToUI(jqel, tinfo); } else { console.log("Teacher s detail info getting error:" + JSON.stringify(jqel) + ",error info:" + r); } }, error: function error(data) { console.log("xhr error when getting teacher " + JSON.stringify(jqel) + ",error msg:" + JSON.stringify(data)); } }).always(function () { while (Date.now() - start < 600) { continue; } next(); }); }); }); }); submit(function (next) { //翻页 var autonextpagecount = GM_getValue("autonextpagecount", 0); if (autonextpagecount > 0) { GM_setValue("autonextpagecount", autonextpagecount - 1); if ($(".s-t-page>.next-page").length == 0) { GM_setValue("autonextpagecount", 0); if (isStopShowboxAndAutoGetNextTimeTeachers()) return; } else { $(".s-t-page .next-page")[0].click(); return false; } } else { if (isStopShowboxAndAutoGetNextTimeTeachers()) return; } next(); }); } function calcIndicator(tinfo) { return Math.ceil(tinfo.label * tinfo.thumbupRate / 100) + tinfo.favoritesCount; } function calcThumbRate(tinfo) { var all = tinfo.thumbdown + tinfo.thumbup; if (all < 1) all = 1; return ((tinfo.thumbup + 1e-5) / all).toFixed(2) * 100; } function isStopShowboxAndAutoGetNextTimeTeachers() { var str = sessionStorage.getItem("selectedTimeSlots"); if (!str) return false; var selectedTimeSlots = JSON.parse(str), cur = selectedTimeSlots.shift(); if (cur) { GM_setValue("autonextpagecount", 500); sessionStorage.setItem("selectedTimeSlots", JSON.stringify(selectedTimeSlots)); sessionStorage.setItem("selectedTimeSlotsRemain", selectedTimeSlots.length); $('form[name="searchform"]>input[name="selectTime"]').val(cur); $('form[name="searchform"]>input[name="pageID"]').val(1); $(".go-search").trigger("click"); return true; } return false; } if (settings.isDetailPage) { function processTeacherDetailPage(jqr) { jqr.find(".teacher-name-tit").prop("innerHTML", function (i, val) { return val.replaceAll("", ""); }); var tinfo = GM_getValue(getinfokey(), {}); tinfo.label = function () { var l = 0; $.each(jqr.find(".t-d-label").text().match(num).clean(""), function (i, val) { l += Number(val); }); return l; }(); //if never set expire then if (!tinfo.expire) tinfo.expire = Date.now(); if (window.location.href.toLocaleLowerCase().includes("teachercomment")) { tinfo.thumbup = Number(jqr.find(".evaluate-content-left span:eq(1)").text().match(num).clean("")[0]); tinfo.thumbdown = Number(jqr.find(".evaluate-content-left span:eq(2)").text().match(num).clean("")[0]); tinfo.thumbupRate = calcThumbRate(tinfo); tinfo.slevel = jqr.find(".sui-students").text(); tinfo.expire = Date.now(); } tinfo.favoritesCount = Number(jqr.find(".clear-search").text().match(num).clean("")[0]); tinfo.isfavorite = jqr.find(".go-search.cancel-collection").length > 0; tinfo.name = jqr.find(".t-name").text().trim(); //无法获取type //tinfo.type = $('.s-t-top-list .li-active').text().trim(); var agesstr = jqr.find(".teacher-name-tit > .age.age-line").text().match(num).clean(""); tinfo.tage = Number(agesstr[1]); tinfo.age = Number(agesstr[0]); tinfo.effectivetime = getBatchNumber(); tinfo.indicator = calcIndicator(tinfo); GM_setValue(getinfokey(), tinfo); jqr.find(".teacher-name-tit").prop("innerHTML", function (i, val) { return "".concat(val, "\n \n \n \n \n \n \n "); }); } submit(function (next) { processTeacherDetailPage($(document)); next(); }); } function addCheckbox(val, lbl, group) { var container = $("#timesmutipulecheck"), inputs = container.find("input"), id = inputs.length + 1; $("", { type: "checkbox", id: "cb" + id, value: val, name: group }).appendTo(container); $("