Greasy Fork is available in English.

tame QTKJ

at the end of with it.

目前為 2019-12-26 提交的版本,檢視 最新版本

// ==UserScript==
// @name          tame QTKJ
// @namespace     Vionlentmonkey
// @version       2.7.0
// @description   at the end of with it.
// @author        someone

// @match         http://218-94-1-181.sft.ipv6.jiangsu.gov.cn:8087/sfxzwsxy/

// @match         http://218.94.1.181:8087/sfxzwsxy/*
// @match         http://218.94.1.179:8087/sfxzwsxy/*
// @match         http://218.94.1.175:8087/sfxzwsxy/*

// @match         http://218.94.1.181:5088/unzipapp/project/ware/attach/*

// @require       https://openuserjs.org/src/libs/sizzle/GM_config.js
// @require       https://cdn.jsdelivr.net/npm/vm.shortcut
// @require       https://greasyfork.org/scripts/381400-open-2345/code/open_2345.js
// @require       https://greasyfork.org/scripts/381401-addstyle/code/addStyle.js
// @require       https://greasyfork.org/scripts/381402-clearurls/code/clearURLs.js
// @require       https://greasyfork.org/scripts/381403-fakenavigators/code/fakeNavigators.js

// @grant         GM_getValue
// @grant         GM_setValue
// @grant         GM_registerMenuCommand
// @grant         GM_openInTab
// @grant         GM_notification

// @run-at        document-start

// ==/UserScript==

const body = document.body || document.documentElement;

const iframeCSS = `
height: 50% !important;
width: auto !important;
border: 1px solid;
position: fixed;
`;

const windowCSS = `
#tameCfg {background-color: lightblue;}
#tameCfg .reset_holder {float: left; position: relative; bottom: -1em;}
#tameCfg .saveclose_buttons {margin: 1em;}
`;

GM_registerMenuCommand('tame QTKJ Settings', opencfg);

function opencfg() {
  // 避免在包含框架的页面打开造成多个设置界面重叠
  if (window.top === window.self) {
    GM_config.open();
    keyCodeCfg.style = iframeCSS;
  }
}

GM_config.init({
  id: 'tameCfg',
  title: 'tame QTJK',
  fields: {
    open_unclose: {
      section: ['keyCode', '批量打开链接的快捷键'],
      label: '批量新标签页后台打开',
      labelPos: 'right',
      type: 'text',
      default: 'F8'
    },
    loginName: {
      section: ['网上学院', '完整填写后将尝试自动登录'],
      label: '账号',
      labelPos: 'right',
      type: 'text',
      default: ''
    },
    pwd: {
      label: '密码',
      labelPos: 'right',
      type: 'password',
      default: ''
    },
    batch: {
      label: '一次性批量打开',
      labelPos: 'right',
      type: 'int',
      default: 6
    },
    reload: {
      label: '是否无限次尝试登录',
      labelPos: 'right',
      type: 'checkbox',
      default: false
    }
  },
  css: windowCSS,
  events: {
    save: function() {
      GM_config.close();
    }
  }
});

// 从 QQ 等打开地址后会被加上奇葩后缀 http://218.94.1.179:8087/sfxzwsxy/#?tdsourcetag=s_pctim_aiomsg
clearURLs();

// 统一域名
if (location.host === '218-94-1-181.sft.ipv6.jiangsu.gov.cn:8087') {
  location.host = '218.94.1.181:8087';
}

// http://218.94.1.175:8087/sfxzwsxy/
if (location.href.match(/^http:\/\/218\.94\.1\.(181|179|175):8087\/sfxzwsxy\/?(serverSelect.jsp)?#?$/i)) {
  // 自动选择最空闲的服务器
  window.onload = () => {
    const servers = document.getElementsByClassName('num');
    if (servers.length > 0) {
      let servers_str = [];
      let notificationText = '自动选择最空闲的服务器!';
      for (let s of servers) {
        servers_str.push(s.textContent);
        notificationText += `\n服务器${servers_str.length}使用程度:${s.textContent}%`;
      }
      // 将字符串元素转为数字
      let servers_num = servers_str.map(Number);
      // 全满则自动刷新,否则选择最空服务器
      let mostFree = Math.min(...servers_num);
      if (mostFree === 100) {
        // https://developer.mozilla.org/docs/Web/API/Location/reload
        if (GM_config.get('reload')) {
          location.reload(true);
        }
      } else {
        document.getElementsByClassName('entrybtn')[servers_num.indexOf(mostFree)].click();
        GM_notification(notificationText);
      }
    }
  };
}

// 去除登陆验证码校验
// 曾使用 OCR 识别法,参考了 https://www.cnblogs.com/ziyunfei/archive/2012/10/05/2710349.html 但准确度有限。
if (
  location.href.startsWith('http://218.94.1.181:8087/sfxzwsxy/index.jsp') ||
  location.href.startsWith('http://218.94.1.179:8087/sfxzwsxy/index.jsp') ||
  location.href.startsWith('http://218.94.1.175:8087/sfxzwsxy/index.jsp')
) {
  window.onload = () => {
    // 原函数去掉验证码识别,去除 isBlank 函数依赖
    check = () => {
      if (document.getElementById('loginName').value === '') {
        alert('请输入用户名');
        document.getElementById('loginName').focus();
        return;
      }
      if (document.getElementById('pwd').value === '') {
        alert('请输入密码');
        document.getElementById('pwd').focus();
        return;
      }
      document.getElementById('form1').submit();
    };
    // 重新绑定点击事件
    document.getElementById('Submit').onclick = check;
    // 移除验证码并提示
    document.getElementById('verifyCode').remove();
    document.getElementById('imgCode').value = '已去除验证码可直接登录';
    // 以下尝试自动登录
    document.getElementById('loginName').value = GM_config.get('loginName'); // 写入预先设置的用户名
    document.getElementById('pwd').value = GM_config.get('pwd'); // 写入预先设置的密码
    // 自动获取用户名输入框焦点
    if (document.getElementById('loginName').value === '') {
      document.getElementById('loginName').focus();
    } else if (document.getElementById('pwd').value !== '') {
      // 用户名密码均已填写时才自动登录
      document.getElementById('Submit').click();
    }
  };
}

// 修复页面过长无法完整显示的 Bug; 隐藏 密码修改提示 和 每日一题
if (
  location.href === 'http://218.94.1.181:8087/sfxzwsxy/jypxks/index.html' ||
  location.href === 'http://218.94.1.181:8087/sfxzwsxy/jypxks/index.html#' ||
  location.href === 'http://218.94.1.179:8087/sfxzwsxy/jypxks/index.html' ||
  location.href === 'http://218.94.1.179:8087/sfxzwsxy/jypxks/index.html#' ||
  location.href === 'http://218.94.1.175:8087/sfxzwsxy/jypxks/index.html' ||
  location.href === 'http://218.94.1.175:8087/sfxzwsxy/jypxks/index.html#'
) {
  const css = `
  html[style="overflow: hidden;"] {
    overflow: visible !important;
  }
  #layui-layer1, #layui-layer-shade1, #layui-layer2, #layui-layer-shade2 {
    display: none !important;
  }
  `;
  addStyle(css);

  // 长时间不动会被弹出;暂时简单粗暴处理测试效果。
  setInterval(() => {
    location.reload(true);
  }, 600000); //十分钟刷新一次

  // 不保证成功率:自动聚焦
  let document_observer = new MutationObserver(() => {
    console.log('changing');
    if (document.getElementById('mainFrame')) {
      document.getElementById('mainFrame').focus();
      console.log('focus');
    }
  });
  document_observer.observe(body, {
    childList: true,
    subtree: true
  });
}

// 首页培训课程 iframe
if (
  location.href === 'http://218.94.1.181:8087/sfxzwsxy/jypxks/modules/homepage/homepage.jsp' ||
  location.href === 'http://218.94.1.179:8087/sfxzwsxy/jypxks/modules/homepage/homepage.jsp' ||
  location.href === 'http://218.94.1.175:8087/sfxzwsxy/jypxks/modules/homepage/homepage.jsp'
) {
  // F8 批量后台新标签页打开已报名课程播放页面
  VM.registerShortcut(GM_config.get('open_unclose'), () => {
    open_2345();
    let c = 0;
    const courses = document.getElementsByClassName('infomai');
    for (let course of courses) {
      let cID = course.getElementsByClassName('applyPk')[0].textContent;
      if (cID !== '') {
        let cURL =
          '//' +
          location.host +
          '/sfxzwsxy/jypxks/modules/train/ware/course_ware_view.jsp?applyPk=' +
          cID +
          '&courseType=1';
        GM_openInTab(cURL, true);
        c++;
        // 疑似超过 6 个课程同时学习后台不计数
        if (c >= GM_config.get('batch')) {
          console.log(`已批量打开${c}个课程`);
          return;
        }
      }
    }
  });
}

// 培训课程查询 iframe
if (
  location.href === 'http://218.94.1.181:8087/sfxzwsxy/jypxks/modules/train/query/course_query.jsp' ||
  location.href === 'http://218.94.1.179:8087/sfxzwsxy/jypxks/modules/train/query/course_query.jsp' ||
  location.href === 'http://218.94.1.175:8087/sfxzwsxy/jypxks/modules/train/query/course_query.jsp'
) {
  // 清理“参加培训”和“查看”链接
  const reallinks = () => {
    let courses = document.querySelectorAll('#trainCourseList a[onclick^=bindBeginTrainEvents]');
    for (let course of courses) {
      let cID = course.getAttribute('onclick').split('"')[1];
      //console.log(cID);
      let cURL =
        '//' +
        location.host +
        '/sfxzwsxy/jypxks/modules/train/ware/course_ware_view.jsp?applyPk=' +
        cID +
        '&courseType=1';
      //console.log(cURL);
      course.href = cURL;
      course.onclick = '';
      course.target = '_blank';
    }
    let answers = document.querySelectorAll('#trainCourseList a[onclick^=bindViewCourseInfoEvents]');
    for (let answer of answers) {
      let aID = answer.getAttribute('onclick').split('"')[1];
      //console.log(cID);
      let aURL = '//' + location.host + '/sfxzwsxy//jypxks/modules/train/course/course_view.jsp?coursePk=' + aID;
      //console.log(cURL);
      answer.href = aURL;
      answer.onclick = '';
      answer.target = '_blank';
    }
  };

  let document_observer = new MutationObserver(() => {
    reallinks();
  });
  document_observer.observe(body, {
    childList: true,
    subtree: true
  });

  // F8 批量后台新标签页打开课程播放页面
  VM.registerShortcut(GM_config.get('open_unclose'), () => {
    open_2345();
    let c = 0;
    const courses = document.querySelectorAll('#trainCourseList a[onclick^=bindBeginTrainEvents]');
    for (let course of courses) {
      let cID = course.getAttribute('onclick').split('"')[1];
      let cURL =
        '//' +
        location.host +
        '/sfxzwsxy/jypxks/modules/train/ware/course_ware_view.jsp?applyPk=' +
        cID +
        '&courseType=1';
      GM_openInTab(cURL, true);
      c++;
      // 疑似超过 6 个课程同时学习后台不计数
      if (c >= GM_config.get('batch')) {
        console.log(`已批量打开${c}个课程`);
        return;
      }
    }
  });
}

// 培训课程查询 - 查看 - 题干 iframe
if (
  location.href.startsWith(
    'http://218.94.1.181:8087/sfxzwsxy//jypxks/modules/train/course/subject_list.jsp?coursePk='
  ) ||
  location.href.startsWith(
    'http://218.94.1.179:8087/sfxzwsxy//jypxks/modules/train/course/subject_list.jsp?coursePk='
  ) ||
  location.href.startsWith('http://218.94.1.175:8087/sfxzwsxy//jypxks/modules/train/course/subject_list.jsp?coursePk=')
) {
  // 清理“题干”链接
  const views = document.querySelectorAll('a[onclick^=viewSubject]');
  for (let view of views) {
    const vID = view
      .getAttribute('onclick')
      .split('(')[1]
      .split(')')[0];
    //console.log(vID);
    const vURL = '//' + location.host + '/sfxzwsxy//jypxks/modules/train/course/subject_view.jsp?subjectPk=' + vID;
    //console.log(vURL);
    view.href = vURL;
    view.onclick = '';
    view.target = '_blank';
  }
}

// 课程视频播放 跨域 iframe
if (location.href.startsWith('http://218.94.1.181:5088/unzipapp/project/ware/attach/')) {
  // 旧播放器在 Windows 下要求 Flash,新播放器不兼容苹果系列。老版本播放器能否成功调用 HTML5 似乎是玄学问题。
  fakeUA('Linux');

  const learn = () => {
    // 自动从课程封面进入播放页面
    if (document.querySelector('img[src="courseware/iconImg/z3.png"]')) {
      document.querySelector('img[src="courseware/iconImg/z3.png"]').click();
    }
    // 老播放器用的 confirm 对话框手动点击即暂停,新版的 alert 无法模拟点击但手动点击不影响后台播放
    if (document.getElementById('cancel')) {
      //document.getElementById('confirm').click(); // 继续学习;可能需要多次重复才能完成一课时。
      document.getElementById('cancel').click(); // 大侠还请重新来过
    }
    // 不保证成功率:检测到载入了 Flash 播放器则刷新;HTML5 播放器的 ID 为 course_player5
    if (document.getElementById('course_player')) {
      if (document.getElementById('course_player').textContent.includes('flash')) {
        console.log('flash');
        location.reload(true);
      } else {
        console.log('html5');
      }
    }
  };

  const option = () => {
    // 旧播放器自动做题,找不到测试例进一步优化。
    if (document.querySelectorAll('div.option>label>input[name="que"]').length > 0) {
      document.querySelectorAll('div.option>label>input[name="que"]')[0].click();
      document.querySelectorAll('div.option>label>input[name="que"]')[1].click(); // 多选题
      document.getElementsByClassName('button')[0].click(); // 提交
      if (document.getElementsByClassName('button_xia').length === 1) {
        document.getElementsByClassName('button_xia')[0].click(); // 下一题
      }
      if (document.getElementsByClassName('button_wan').length === 1) {
        document.getElementsByClassName('button_wan')[0].click(); // 完成
      }
      document.getElementById('course_player5').click(); // 播放
    }
  };
  let document_observer = new MutationObserver(() => {
    learn();
    option();
  });
  document_observer.observe(body, {
    attributes: true,
    subtree: true
  });
}