超星 - 考试与作业解除各种限制,允许拉伸输入框

合并类似脚本并大幅改进,解除选择和粘贴限制,允许拉伸输入框,支持在脚本中设置输入框默认高度。

  1. // ==UserScript==
  2. // @name 超星 - 考试与作业解除各种限制,允许拉伸输入框
  3. // @description 合并类似脚本并大幅改进,解除选择和粘贴限制,允许拉伸输入框,支持在脚本中设置输入框默认高度。
  4. // @namespace UnKnown
  5. // @author UnKnown
  6. // @license MIT
  7. // @version 1.1
  8. // @icon https://imgsrc.baidu.com/forum/pic/item/6a63f6246b600c33c3d714d61c4c510fd9f9a106.jpg
  9. // @match *://*.chaoxing.com/work/doHomeWorkNew
  10. // @match *://*.chaoxing.com/exam/test/reVersionTestStartNew
  11. // @match *://*.chaoxing.com/exam/test/reVersionPaperPreview
  12. // @match *://*.chaoxing.com/mooc2/work/dowork
  13. // @match *://*.chaoxing.com/mooc2/exam/preview
  14. // @grant none
  15. // @run-at document-start
  16. // ==/UserScript==
  17.  
  18. "use strict";
  19.  
  20. // 脚本 onload 函数执行超时
  21. // 单位为毫秒,一般不用修改
  22. const timeout = 500;
  23.  
  24. // 自定义输入框默认高度
  25. // 0 表示使用默认高度
  26. // 非零数字表示像素,负数无效
  27. // 字符串表示 CSS 长度,例如 "10rem",详搜 “CSS 长度单位”
  28. const customHeight = 0;
  29.  
  30. // 解除粘贴限制
  31. Object.defineProperties(
  32. unsafeWindow, {
  33. // 将 allowPaste 设死为 0 以允许粘贴
  34. // 注意,在神秘莫测的超星前端眼中,0 表示允许粘贴,1 表示禁止粘贴
  35. allowPaste: { configurable: false, enumerable: true, writable: false, value: 0 },
  36. // 将 umyEditor_paste 设死为 null,干掉阻止粘贴的函数
  37. // 会导致超星的代码在设置 umyEditor_paste 时报错,属于预期行为
  38. myEditor_paste: { configurable: false, enumerable: true, writable: false, value: null }
  39. }
  40. );
  41.  
  42. const ondom = () => {
  43.  
  44. // 允许选择(JS 事件)
  45. document.body.removeAttribute("onselectstart");
  46.  
  47. // 允许选择(CSS),允许拉伸输入框
  48. document.head.appendChild( document.createElement("style") ).textContent =
  49. "html { user-select: initial !important; }" +
  50. ".edui-default .edui-editor-iframeholder { resize: vertical; }";
  51.  
  52. // 设置页面标题
  53. const titleElement = document.querySelector('.CyTop ul.ul01 li.cur a');
  54. if (titleElement !== null) {
  55. document.title = titleElement.textContent;
  56. }
  57.  
  58. };
  59.  
  60. const onload = () => setTimeout(() => {
  61.  
  62. // 解除粘贴限制 备用方案
  63. if (
  64. unsafeWindow.allowPaste == 1 && // 超星在这里也用了 == 而非 ===
  65. unsafeWindow.myEditor_paste instanceof Function &&
  66. unsafeWindow.UE instanceof Object
  67. ) {
  68.  
  69. const UE = unsafeWindow.UE;
  70.  
  71. // 清除 beforepaste
  72. // UEditor 咋把 instances 叫成 instants…
  73. if ( Array.isArray(UE.instants) ) {
  74. UE.instants.forEach(
  75. instance => instance.removeListener("beforepaste", myEditor_paste)
  76. );
  77. } else if ( UE.getEditor instanceof Function ) {
  78. document.querySelectorAll('textarea[id]').forEach(
  79. textarea => UE.getEditor(textarea.id).removeListener("beforepaste", myEditor_paste)
  80. );
  81. }
  82.  
  83. }
  84.  
  85. // 设置输入框默认高度
  86. const height = (() => {
  87. switch (typeof customHeight) {
  88. case "number": return customHeight > 0 ? customHeight + "px" : false;
  89. case "string": return customHeight !== "" ? customHeight : false;
  90. default: return false;
  91. }
  92. })();
  93.  
  94. height &&
  95. document.querySelectorAll(".edui-editor-iframeholder").forEach(
  96. iframeholder => iframeholder.style.height = height
  97. );
  98.  
  99. }, timeout);
  100.  
  101. const once = { once: true };
  102.  
  103. switch (document.readyState) {
  104.  
  105. case "loading":
  106. document.addEventListener("DOMContentLoaded", ondom, once);
  107. window.addEventListener("load", onload, once);
  108. break;
  109.  
  110. case "interactive":
  111. ondom();
  112. window.addEventListener("load", onload, once);
  113. break;
  114.  
  115. case "complete":
  116. ondom();
  117. onload();
  118. break;
  119.  
  120. }

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址