xzit_01.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. // 徐州工程学院(xzit.edu.cn) 拾光课程表适配脚本
  2. // 基于正方教务 HTML 页面抓取,出现问题请去学校校园墙寻找管理员 "盐泡"
  3. const TIME_SLOTS = [
  4. { number: 1, startTime: "08:00", endTime: "08:45" },
  5. { number: 2, startTime: "08:55", endTime: "09:40" },
  6. { number: 3, startTime: "10:05", endTime: "10:50" },
  7. { number: 4, startTime: "11:00", endTime: "11:45" },
  8. { number: 5, startTime: "12:00", endTime: "12:45" },
  9. { number: 6, startTime: "12:55", endTime: "13:40" },
  10. { number: 7, startTime: "14:00", endTime: "14:45" },
  11. { number: 8, startTime: "14:55", endTime: "15:40" },
  12. { number: 9, startTime: "16:05", endTime: "16:50" },
  13. { number: 10, startTime: "17:00", endTime: "17:45" },
  14. { number: 11, startTime: "17:55", endTime: "18:40" },
  15. { number: 12, startTime: "18:45", endTime: "19:30" },
  16. { number: 13, startTime: "19:40", endTime: "20:25" },
  17. { number: 14, startTime: "20:35", endTime: "21:20" }
  18. ];
  19. function parseTable() {
  20. const regexName = /[●★○]/g;
  21. const courseInfoList = [];
  22. const $ = window.jQuery;
  23. if (!$) return courseInfoList;
  24. $("#kbgrid_table_0 td").each((i, td) => {
  25. if ($(td).hasClass("td_wrap") && $(td).text().trim() !== "") {
  26. const day = parseInt($(td).attr("id").split("-")[0], 10);
  27. $(td).find(".timetable_con.text-left").each((index, course) => {
  28. const name = $(course).find(".title font").text().replace(regexName, "").trim();
  29. const infoStr = $(course).find("p").eq(0).find("font").eq(1).text().trim();
  30. const position = $(course).find("p").eq(1).find("font").text().trim();
  31. const teacher = $(course).find("p").eq(2).find("font").text().trim();
  32. if (infoStr && infoStr.match(/\((\d+-\d+节)\)/) && infoStr.split("节)")[1]) {
  33. const [sections, weeks] = parseCourseInfo(infoStr);
  34. if (name && position && teacher && sections.length && weeks.length) {
  35. courseInfoList.push({
  36. name: name,
  37. day: day,
  38. weeks: weeks,
  39. teacher: teacher,
  40. position: position.split(/\s+/).pop(),
  41. startSection: sections[0],
  42. endSection: sections[sections.length - 1]
  43. });
  44. }
  45. }
  46. });
  47. }
  48. });
  49. return courseInfoList;
  50. }
  51. function parseList() {
  52. const regexName = /[●★○]/g;
  53. const regexWeekNum = /周数:|周/g;
  54. const regexPosition = /上课地点:/g;
  55. const regexTeacher = /教师 :/g;
  56. const $ = window.jQuery;
  57. if (!$) return [];
  58. const courseInfoList = [];
  59. $("#kblist_table tbody").each((day, tbody) => {
  60. if (day > 0 && day < 8) {
  61. let sections;
  62. $(tbody).find("tr:not(:first-child)").each((trIndex, tr) => {
  63. let name;
  64. let font;
  65. if ($(tr).find("td").length > 1) {
  66. sections = parseSections($(tr).find("td:first-child").text());
  67. name = $(tr).find("td:nth-child(2)").find(".title").text().replace(regexName, "").trim();
  68. font = $(tr).find("td:nth-child(2)").find("p font");
  69. } else {
  70. name = $(tr).find("td").find(".title").text().replace(regexName, "").trim();
  71. font = $(tr).find("td").find("p font");
  72. }
  73. const weekStr = $(font[0]).text().replace(regexWeekNum, "").trim();
  74. const weeks = parseWeeks(weekStr);
  75. const positionRaw = $(font[1]).text().replace(regexPosition, "").trim();
  76. const teacher = $(font[2]).text().replace(regexTeacher, "").trim();
  77. if (name && sections && weeks.length && teacher && positionRaw) {
  78. courseInfoList.push({
  79. name: name,
  80. day: day,
  81. weeks: weeks,
  82. teacher: teacher,
  83. position: positionRaw.split(/\s+/).pop(),
  84. startSection: sections[0],
  85. endSection: sections[sections.length - 1]
  86. });
  87. }
  88. });
  89. }
  90. });
  91. return courseInfoList;
  92. }
  93. function parseCourseInfo(str) {
  94. const sections = parseSections(str.match(/\((\d+-\d+节)\)/)[1].replace(/节/g, ""));
  95. const weekStrWithMarker = str.split("节)")[1];
  96. const weeks = parseWeeks(weekStrWithMarker.replace(/周/g, "").trim());
  97. return [sections, weeks];
  98. }
  99. function parseSections(str) {
  100. const [start, end] = str.split("-").map(Number);
  101. if (isNaN(start) || isNaN(end) || start > end) return [];
  102. return Array.from({ length: end - start + 1 }, (_, i) => start + i);
  103. }
  104. function parseWeeks(str) {
  105. const segments = str.split(",");
  106. const weeks = [];
  107. const segmentRegex = /(\d+)(?:-(\d+))?\s*(\([单双]\))?/g;
  108. for (const segment of segments) {
  109. const cleanSegment = segment.replace(/周/g, "").trim();
  110. segmentRegex.lastIndex = 0;
  111. let match;
  112. while ((match = segmentRegex.exec(cleanSegment)) !== null) {
  113. const start = parseInt(match[1], 10);
  114. const end = match[2] ? parseInt(match[2], 10) : start;
  115. const flagStr = match[3] || "";
  116. let flag = 0;
  117. if (flagStr.includes("单")) {
  118. flag = 1;
  119. } else if (flagStr.includes("双")) {
  120. flag = 2;
  121. }
  122. for (let i = start; i <= end; i += 1) {
  123. if (flag === 1 && i % 2 !== 1) continue;
  124. if (flag === 2 && i % 2 !== 0) continue;
  125. if (!weeks.includes(i)) {
  126. weeks.push(i);
  127. }
  128. }
  129. }
  130. }
  131. return weeks.sort((a, b) => a - b);
  132. }
  133. function buildCourseConfig(courses) {
  134. let maxWeek = 0;
  135. for (const course of courses) {
  136. for (const week of course.weeks) {
  137. if (week > maxWeek) {
  138. maxWeek = week;
  139. }
  140. }
  141. }
  142. return {
  143. semesterTotalWeeks: maxWeek || 20,
  144. firstDayOfWeek: 1
  145. };
  146. }
  147. async function scrapeAndParseCourses() {
  148. AndroidBridge.showToast("正在检查页面并抓取课程数据...");
  149. const tips = "1. 登录徐州工程学院教务系统\n2. 进入学生个人课表页面\n3. 选择正确学年、学期并点击【查询】\n4. 确认页面已显示课表\n5. 点击下方【一键导入】";
  150. try {
  151. const response = await fetch(window.location.href);
  152. const text = await response.text();
  153. if (!text.includes("课表查询")) {
  154. await window.AndroidBridgePromise.showAlert("导入失败", `当前页面似乎不是学生课表查询页面。请检查:\n${tips}`, "确定");
  155. return null;
  156. }
  157. const typeElement = document.querySelector("#shcPDF");
  158. if (!typeElement) {
  159. await window.AndroidBridgePromise.showAlert("导入失败", "未能识别课表视图类型,请确认您已点击查询且课表已加载完毕。", "确定");
  160. return null;
  161. }
  162. const type = typeElement.dataset.type;
  163. const tableElement = document.querySelector(type === "list" ? "#kblist_table" : "#kbgrid_table_0");
  164. if (!tableElement) {
  165. await window.AndroidBridgePromise.showAlert("导入失败", `未能找到课表主体(${type} 视图),请确认您已点击查询且课表已加载完毕。`, "确定");
  166. return null;
  167. }
  168. const courses = type === "list" ? parseList() : parseTable();
  169. if (!courses.length) {
  170. AndroidBridge.showToast("未找到任何课程数据,请检查学年学期是否正确或本学期无课。");
  171. return null;
  172. }
  173. return {
  174. courses: courses,
  175. config: buildCourseConfig(courses)
  176. };
  177. } catch (error) {
  178. AndroidBridge.showToast(`抓取或解析失败: ${error.message}`);
  179. console.error("JS: Scrape/Parse Error:", error);
  180. await window.AndroidBridgePromise.showAlert("抓取或解析失败", `发生错误:${error.message}`, "确定");
  181. return null;
  182. }
  183. }
  184. async function saveCourses(parsedCourses) {
  185. AndroidBridge.showToast(`正在保存 ${parsedCourses.length} 门课程...`);
  186. try {
  187. await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(parsedCourses, null, 2));
  188. return true;
  189. } catch (error) {
  190. AndroidBridge.showToast(`课程保存失败: ${error.message}`);
  191. console.error("JS: Save Courses Error:", error);
  192. return false;
  193. }
  194. }
  195. async function saveCourseConfig(config) {
  196. try {
  197. await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify(config));
  198. } catch (error) {
  199. AndroidBridge.showToast(`课表配置保存失败: ${error.message}`);
  200. console.error("JS: Save Config Error:", error);
  201. }
  202. }
  203. async function importPresetTimeSlots() {
  204. try {
  205. await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(TIME_SLOTS));
  206. AndroidBridge.showToast("预设时间段导入成功!");
  207. } catch (error) {
  208. AndroidBridge.showToast(`导入时间段失败: ${error.message}`);
  209. console.error("JS: Save Time Slots Error:", error);
  210. }
  211. }
  212. async function runImportFlow() {
  213. const alertConfirmed = await window.AndroidBridgePromise.showAlert(
  214. "徐州工程学院课表导入",
  215. "导入前请确保您已在浏览器中成功登录教务系统,并处于课表查询页面且已点击查询。",
  216. "好的,开始导入"
  217. );
  218. if (!alertConfirmed) {
  219. AndroidBridge.showToast("用户取消了导入。");
  220. return;
  221. }
  222. if (typeof window.jQuery === "undefined" && typeof $ === "undefined") {
  223. const errorMsg = "当前教务系统页面似乎没有加载 jQuery 库。本脚本依赖 jQuery 进行 DOM 解析。";
  224. AndroidBridge.showToast(errorMsg);
  225. await window.AndroidBridgePromise.showAlert("导入失败", `${errorMsg}\n请尝试刷新页面后重试。`, "确定");
  226. return;
  227. }
  228. const result = await scrapeAndParseCourses();
  229. if (result === null) {
  230. return;
  231. }
  232. const { courses, config } = result;
  233. const saveResult = await saveCourses(courses);
  234. if (!saveResult) {
  235. return;
  236. }
  237. await saveCourseConfig(config);
  238. await importPresetTimeSlots();
  239. AndroidBridge.showToast(`课程导入成功,共导入 ${courses.length} 门课程!`);
  240. AndroidBridge.notifyTaskCompletion();
  241. }
  242. runImportFlow();