myschool.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. // 西安工业大学(http://jwgl2018.xatu.edu.cn) 拾光课程表适配脚本,基于天津农学院适配脚本
  2. // 本校开发者适配,出现问题请提issues或者提交pr更改,这更加快速
  3. //感谢XingHeYuZhuan、aryunm、jursin...等的帮助,感谢trae的辅助
  4. function powerSplit(paramsRaw) {
  5. const args = [];
  6. let current = "";
  7. let depth = 0;
  8. let inQuote = false;
  9. let quoteChar = "";
  10. for (let i = 0; i < paramsRaw.length; i++) {
  11. let char = paramsRaw[i];
  12. if ((char === '"' || char === "'") && (i === 0 || paramsRaw[i - 1] !== '\\')) {
  13. if (!inQuote) { inQuote = true; quoteChar = char; }
  14. else if (char === quoteChar) { inQuote = false; }
  15. }
  16. if (!inQuote) {
  17. if (char === '(' || char === '[' || char === '{') depth++;
  18. if (char === ')' || char === ']' || char === '}') depth--;
  19. }
  20. if (char === ',' && depth === 0 && !inQuote) {
  21. args.push(cleanArg(current));
  22. current = "";
  23. } else {
  24. current += char;
  25. }
  26. }
  27. args.push(cleanArg(current));
  28. return args;
  29. }
  30. function cleanArg(s) {
  31. s = s.trim();
  32. if (s === "null") return null;
  33. return s.replace(/^["']|["']$/g, "");
  34. }
  35. /**
  36. * 全局课程合并逻辑
  37. */
  38. function mergeContinuousLessons(lessons) {
  39. if (!lessons || lessons.length === 0) return [];
  40. // 1. 建立基于 (课程名|教师|地点|星期几) 的分组
  41. const groups = {};
  42. lessons.forEach(l => {
  43. const key = `${l.name}|${l.teacher}|${l.position}|${l.day}`;
  44. if (!groups[key]) {
  45. groups[key] = {
  46. name: l.name,
  47. teacher: l.teacher,
  48. position: l.position,
  49. day: l.day,
  50. isTeachingBuilding3: l.isTeachingBuilding3,
  51. // 假设大学最多 50 周,构建一个:第 N 周对应哪些节次的矩阵
  52. weeksMatrix: Array.from({ length: 50 }, () => new Set())
  53. };
  54. }
  55. // 将系统传来的凌乱数据彻底打散,按“周”填入对应的“节”中,Set自动去重
  56. if (l.weeks && Array.isArray(l.weeks)) {
  57. l.weeks.forEach(w => {
  58. if (w >= 0 && w < 50) {
  59. for (let s = l.startSection; s <= l.endSection; s++) {
  60. groups[key].weeksMatrix[w].add(s);
  61. }
  62. }
  63. });
  64. }
  65. });
  66. const merged = [];
  67. // 2. 根据矩阵重新组装绝对精确的课程块
  68. for (const key in groups) {
  69. const group = groups[key];
  70. const matrix = group.weeksMatrix;
  71. // 用于记录相同的“连续节次块”分布在哪些周次
  72. // 例如 blockMap["1-2"] = [1, 2, 3, 4, 5, 6, 7, 8, 9]
  73. // 例如 blockMap["2-2"] = [10]
  74. const blockMap = {};
  75. for (let w = 0; w < matrix.length; w++) {
  76. const sections = Array.from(matrix[w]).sort((a, b) => a - b);
  77. if (sections.length === 0) continue;
  78. // 寻找当前周的连续节次块
  79. let start = sections[0];
  80. let prev = sections[0];
  81. for (let i = 1; i < sections.length; i++) {
  82. const curr = sections[i];
  83. if (curr === prev + 1) {
  84. prev = curr; // 节次连续,继续延伸
  85. } else {
  86. // 节次断开,结算上一个块
  87. const blockKey = `${start}-${prev}`;
  88. if (!blockMap[blockKey]) blockMap[blockKey] = [];
  89. blockMap[blockKey].push(w);
  90. // 开启新块
  91. start = curr;
  92. prev = curr;
  93. }
  94. }
  95. // 结算每周最后一个块
  96. const blockKey = `${start}-${prev}`;
  97. if (!blockMap[blockKey]) blockMap[blockKey] = [];
  98. blockMap[blockKey].push(w);
  99. }
  100. // 3. 将聚合好的 blockMap 转换为最终的 JSON 对象
  101. for (const blockKey in blockMap) {
  102. const [startSec, endSec] = blockKey.split('-').map(Number);
  103. merged.push({
  104. name: group.name,
  105. teacher: group.teacher,
  106. position: group.position,
  107. day: group.day,
  108. startSection: startSec,
  109. endSection: endSec,
  110. weeks: blockMap[blockKey],
  111. isTeachingBuilding3: group.isTeachingBuilding3
  112. });
  113. }
  114. }
  115. // 4. 排序以便输出整洁美观
  116. merged.sort((a, b) => {
  117. if (a.day !== b.day) return a.day - b.day;
  118. if (a.startSection !== b.startSection) return a.startSection - b.startSection;
  119. return a.name.localeCompare(b.name);
  120. });
  121. return merged;
  122. }
  123. function isTeachingBuilding3(position) {
  124. return /教3/.test(position);
  125. }
  126. function parseTaskActivities(html) {
  127. const rawResults = [];
  128. const blocks = html.split(/var\s+teachers\s*=/);
  129. for (let i = 1; i < blocks.length; i++) {
  130. const block = blocks[i];
  131. let teacherName = "未知教师";
  132. const tMatch = block.match(/actTeachers\s*=\s*\[\s*\{[\s\S]*?name:\s*"(.*?)"/);
  133. if (tMatch) teacherName = tMatch[1];
  134. const activityMatch = block.match(/new\s+TaskActivity\(([\s\S]*?)\);/);
  135. if (!activityMatch) continue;
  136. const args = powerSplit(activityMatch[1]);
  137. const courseName = (args[3] || "未知课程").split('(')[0];
  138. const position = (args[5] || "未知地点").replace(/\(.*?\)/g, "");
  139. const weeksBitmap = args[6] || "";
  140. const weeks = [];
  141. for (let j = 0; j < weeksBitmap.length; j++) {
  142. if (weeksBitmap[j] === '1') weeks.push(j);
  143. }
  144. const unitCountMatch = html.match(/unitCount\s*=\s*(\d+)/);
  145. const unitCount = unitCountMatch ? parseInt(unitCountMatch[1]) : 14;
  146. const idxRegex = /index\s*=\s*(\d+)\s*\*\s*unitCount\s*\+\s*(\d+);/g;
  147. let m;
  148. while ((m = idxRegex.exec(block)) !== null) {
  149. const day = parseInt(m[1]) + 1;
  150. const section = parseInt(m[2]) + 1;
  151. rawResults.push({
  152. "name": courseName,
  153. "teacher": teacherName,
  154. "position": position,
  155. "day": day,
  156. "startSection": section,
  157. "endSection": section,
  158. "weeks": weeks,
  159. "isTeachingBuilding3": isTeachingBuilding3(position)
  160. });
  161. }
  162. }
  163. // 执行全局合并逻辑
  164. return mergeContinuousLessons(rawResults);
  165. }
  166. async function request(url, options = {}) {
  167. const res = await fetch(url, { credentials: "include", ...options });
  168. if (!res.ok) throw new Error(`网络请求失败: ${res.status}`);
  169. return await res.text();
  170. }
  171. async function detectParameters() {
  172. const html = await request("http://jwgl2018.xatu.edu.cn/eams/courseTableForStd.action?sf_request_type=ajax");
  173. const idsMatch = html.match(/bg\.form\.addInput\(form,"ids","(\d+)"\)/);
  174. const tagIdMatch = html.match(/id="(semesterBar\d+Semester)"/);
  175. if (!idsMatch || !tagIdMatch) return null;
  176. return { ids: idsMatch[1], tagId: tagIdMatch[1] };
  177. }
  178. async function getSelectedSemester(tagId) {
  179. const raw = await request(`http://jwgl2018.xatu.edu.cn/eams/dataQuery.action?sf_request_type=ajax`, {
  180. method: "POST",
  181. headers: { "Content-Type": "application/x-www-form-urlencoded" },
  182. body: `tagId=${encodeURIComponent(tagId)}&dataType=semesterCalendar`
  183. });
  184. const data = Function(`return (${raw});`)();
  185. const list = [];
  186. for (let key in data.semesters) {
  187. data.semesters[key].forEach(s => list.push({ id: s.id, name: `${s.schoolYear} ${s.name}学期` }));
  188. }
  189. const idx = await window.AndroidBridgePromise.showSingleSelection("选择学期", JSON.stringify(list.map(s => s.name)), -1);
  190. return idx !== null ? list[idx] : null;
  191. }
  192. async function fetchAndParseCourses(semesterId, ids) {
  193. const html = await request(`http://jwgl2018.xatu.edu.cn/eams/courseTableForStd!courseTable.action?sf_request_type=ajax`, {
  194. method: "POST",
  195. headers: { "Content-Type": "application/x-www-form-urlencoded" },
  196. body: `ignoreHead=1&setting.kind=std&semester.id=${semesterId}&ids=${ids}`
  197. });
  198. return parseTaskActivities(html);
  199. }
  200. async function applyTimeSlots() {
  201. const slots = [
  202. { "number": 1, "startTime": "08:20", "endTime": "09:05" },
  203. { "number": 2, "startTime": "09:15", "endTime": "10:00" },
  204. { "number": 3, "startTime": "10:20", "endTime": "11:05" },
  205. { "number": 4, "startTime": "11:15", "endTime": "12:00" },
  206. { "number": 5, "startTime": "14:00", "endTime": "14:45" },
  207. { "number": 6, "startTime": "14:55", "endTime": "15:40" },
  208. { "number": 7, "startTime": "16:00", "endTime": "16:45" },
  209. { "number": 8, "startTime": "16:55", "endTime": "17:40" },
  210. { "number": 9, "startTime": "18:10", "endTime": "18:55" },
  211. { "number": 10, "startTime": "19:05", "endTime": "19:50" },
  212. { "number": 11, "startTime": "20:00", "endTime": "20:45" },
  213. { "number": 11, "startTime": "20:55", "endTime": "21:40" },
  214. ];
  215. return await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(slots));
  216. }
  217. function adjustTeachingBuilding3Courses(courses) {
  218. return courses.map(course => {
  219. // 检查是否是教3楼且正好是第3-4节两节课
  220. if (course.isTeachingBuilding3 &&
  221. course.startSection === 3 &&
  222. course.endSection === 4) {
  223. // 设置为自定义时间模式
  224. course.isCustomTime = true;
  225. course.customStartTime = "10:10";
  226. course.customEndTime = "11:40";
  227. }
  228. return course;
  229. });
  230. }
  231. async function runImportFlow() {
  232. try {
  233. AndroidBridge.showToast("开始探测教务参数...");
  234. const params = await detectParameters();
  235. if (!params) throw new Error("未能识别教务参数,请确认已登录");
  236. const semester = await getSelectedSemester(params.tagId);
  237. if (!semester) return;
  238. AndroidBridge.showToast("正在同步课表...");
  239. let courses = await fetchAndParseCourses(semester.id, params.ids);
  240. if (!courses || courses.length === 0) throw new Error("未解析到课程数据");
  241. // 调整教3楼课程时间
  242. courses = adjustTeachingBuilding3Courses(courses);
  243. await applyTimeSlots();
  244. const saveResult = await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses));
  245. if (saveResult) {
  246. AndroidBridge.showToast(`成功导入 ${courses.length} 个课程条目`);
  247. AndroidBridge.notifyTaskCompletion();
  248. }
  249. } catch (e) {
  250. console.error(`[异常] ${e.message}`);
  251. AndroidBridge.showToast(e.message);
  252. }
  253. }
  254. runImportFlow();