imnc_01.js 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. /**
  2. * 呼和浩特民族学院 (IMNC) 课表解析脚本-通过 WebVPN 登录
  3. * 放置于测试目录用于真机测试
  4. */
  5. // 周次解析函数
  6. function parseWeeks(weekStr) {
  7. let weeks = [];
  8. if (!weekStr) return weeks;
  9. let isSingle = weekStr.includes('单');
  10. let isDouble = weekStr.includes('双');
  11. // 匹配 "1-16", "第1-9周"
  12. let match = weekStr.match(/(\d+)-(\d+)/);
  13. if (match) {
  14. let start = parseInt(match[1]);
  15. let end = parseInt(match[2]);
  16. for (let i = start; i <= end; i++) {
  17. if (isSingle && i % 2 === 0) continue;
  18. if (isDouble && i % 2 !== 0) continue;
  19. weeks.push(i);
  20. }
  21. } else {
  22. // 匹配 "第13周"
  23. let singleMatch = weekStr.match(/(\d+)/);
  24. if (singleMatch) {
  25. weeks.push(parseInt(singleMatch[1]));
  26. }
  27. }
  28. return weeks;
  29. }
  30. // 核心解析函数
  31. function fetchAndParseCourses() {
  32. let courses = [];
  33. let table = document.querySelector('#timetable');
  34. if (!table) return null;
  35. let rows = table.querySelectorAll('tr');
  36. // 第 0 行是表头,从第 1 行开始遍历节次
  37. for (let i = 1; i < rows.length; i++) {
  38. let row = rows[i];
  39. let cells = row.querySelectorAll('td');
  40. let section = i; // 1 到 13 节
  41. for (let j = 0; j < cells.length; j++) {
  42. let cell = cells[j];
  43. let day = j + 1; // 星期 1 到 7
  44. let html = cell.innerHTML.trim();
  45. if (!html || html === '&nbsp;') continue;
  46. // 按 <br> 分割
  47. let parts = html.split(/<br\s*\/?>/i).map(s => s.trim()).filter(s => s !== '');
  48. // 每 5 个元素代表一个完整的课程块
  49. for (let k = 0; k < parts.length; k += 5) {
  50. if (k + 3 >= parts.length) break;
  51. let namePart = parts[k];
  52. let position = parts[k+1];
  53. let teacher = parts[k+2];
  54. let weekStr = parts[k+3];
  55. // parts[k+4] 是 "讲授" 等类型,暂时不需要存入
  56. // 提取书名号内的课程名
  57. let nameMatch = namePart.match(/&lt;&lt;(.*?)&gt;&gt;/);
  58. let name = nameMatch ? nameMatch[1] : namePart;
  59. // 兼容某些浏览器可能将转义符还原的情况
  60. if (!nameMatch) {
  61. let nameMatch2 = namePart.match(/<<(.*?)>>/);
  62. if (nameMatch2) name = nameMatch2[1];
  63. }
  64. let weeks = parseWeeks(weekStr);
  65. // 查找同一天、同名、同老师、同地点、同周次,且正好是上一节的课程(合并连上的课)
  66. let existingCourse = courses.find(c =>
  67. c.name === name &&
  68. c.day === day &&
  69. c.teacher === teacher &&
  70. c.position === position &&
  71. JSON.stringify(c.weeks) === JSON.stringify(weeks) &&
  72. c.endSection === section - 1
  73. );
  74. if (existingCourse) {
  75. existingCourse.endSection = section;
  76. } else {
  77. courses.push({
  78. name: name,
  79. teacher: teacher,
  80. position: position,
  81. day: day,
  82. startSection: section,
  83. endSection: section,
  84. weeks: weeks
  85. });
  86. }
  87. }
  88. }
  89. }
  90. return courses;
  91. }
  92. function cleanCellText(cell) {
  93. return cell.textContent.replace(/\u00a0/g, ' ').replace(/\s+/g, ' ').trim();
  94. }
  95. function parseNoArrangementCourses() {
  96. let table = document.querySelector('#noArrangement');
  97. if (!table) return [];
  98. let rows = table.querySelectorAll('tr');
  99. let courses = [];
  100. let seen = {};
  101. for (let i = 1; i < rows.length; i++) {
  102. let cells = rows[i].querySelectorAll('td');
  103. if (cells.length < 8) continue;
  104. let course = {
  105. courseId: cleanCellText(cells[0]),
  106. name: cleanCellText(cells[1]),
  107. sequence: cleanCellText(cells[2]),
  108. teacher: cleanCellText(cells[3]) || '未填写教师',
  109. combinedClass: cleanCellText(cells[4]),
  110. weeks: cleanCellText(cells[5]),
  111. day: cleanCellText(cells[6]),
  112. position: cleanCellText(cells[7])
  113. };
  114. if (!course.courseId && !course.name) continue;
  115. let key = [course.courseId, course.name, course.sequence, course.teacher, course.combinedClass, course.weeks, course.day, course.position].join('|');
  116. if (seen[key]) continue;
  117. seen[key] = true;
  118. courses.push(course);
  119. }
  120. return courses;
  121. }
  122. function formatNoArrangementMessage(courses) {
  123. let lines = courses.map((course, index) => {
  124. let teacher = course.teacher || '未填写教师';
  125. let weeks = course.weeks || '未填写周次';
  126. return `${index + 1}. ${course.name} / ${teacher} / ${weeks}`;
  127. });
  128. return `检测到 ${courses.length} 门课程没有具体上课时间或地点,无法自动放入课表:\n\n${lines.join('\n')}\n\n请在确认课程时间后重新导入,点击【继续】将导入已知课程。`;
  129. }
  130. // 调度流程
  131. async function runImportFlow() {
  132. try {
  133. AndroidBridge.showToast("开始解析课表...");
  134. const alertConfirmed = await window.AndroidBridgePromise.showAlert(
  135. "导入确认",
  136. "请确保您目前处于教务系统的“学生课表”显示页面。\n是否立即提取并导入课表?",
  137. "开始提取"
  138. );
  139. if (!alertConfirmed) {
  140. AndroidBridge.showToast("导入已取消");
  141. return;
  142. }
  143. let courses = fetchAndParseCourses();
  144. if (!courses || courses.length === 0) {
  145. await window.AndroidBridgePromise.showAlert("错误", "未在当前页面找到课表数据,请确认是否处于课表页面,或联系适配开发者。", "好的");
  146. return;
  147. }
  148. let noArrangementCourses = parseNoArrangementCourses();
  149. if (noArrangementCourses.length > 0) {
  150. await window.AndroidBridgePromise.showAlert(
  151. "存在未安排课程",
  152. formatNoArrangementMessage(noArrangementCourses),
  153. "继续"
  154. );
  155. }
  156. await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses));
  157. AndroidBridge.showToast(`成功导入 ${courses.length} 门课程块!`);
  158. AndroidBridge.notifyTaskCompletion();
  159. } catch (error) {
  160. AndroidBridge.showToast("导入发生错误: " + error.message);
  161. }
  162. }
  163. // 启动执行
  164. runImportFlow();