sdipct.js 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /**
  2. * 拾光课程表适配脚本 - 山东石油化工学院 (sdipct.edu.cn)
  3. * 非该大学开发者适配,开发者无法及时发现问题
  4. * 出现问题请提联系开发者或者提交pr更改,这更加快速
  5. */
  6. /**
  7. * 解析周次字符串,例如 "1,2,3,4,5" -> [1, 2, 3, 4, 5]
  8. */
  9. function parseWeeks(weekStr) {
  10. if (!weekStr) return [];
  11. const weeks = weekStr.split(',')
  12. .map(w => Number(w.trim()))
  13. .filter(w => !isNaN(w) && w > 0);
  14. return [...new Set(weeks)].sort((a, b) => a - b);
  15. }
  16. /**
  17. * 将教务系统返回的 JSON 转换为拾光标准的 CourseJsonModel 数组
  18. */
  19. function parseJsonData(jsonData) {
  20. console.log("JS: 开始解析课程 JSON...");
  21. if (!jsonData || jsonData.code !== 0 || !Array.isArray(jsonData.data)) {
  22. return [];
  23. }
  24. return jsonData.data.filter(raw => {
  25. return raw.kcmc && raw.xq && raw.ps && raw.pe && raw.zc;
  26. }).map(raw => {
  27. return {
  28. name: raw.kcmc.trim(),
  29. teacher: (raw.teaxms || "未知教师").trim(),
  30. position: (raw.jxcdmc || "未知地点").trim(),
  31. day: Number(raw.xq),
  32. startSection: Number(raw.ps),
  33. endSection: Number(raw.pe),
  34. weeks: parseWeeks(raw.zc)
  35. };
  36. }).filter(course => {
  37. return course.weeks.length > 0 && course.startSection <= course.endSection;
  38. });
  39. }
  40. /**
  41. * 校验函数:验证用户输入的学年格式
  42. */
  43. function validateYearInput(input) {
  44. return /^[0-9]{4}$/.test(input) ? false : "请输入四位数字的起始学年(如 2025)";
  45. }
  46. /**
  47. * 步骤 A: 显示引导公告
  48. */
  49. async function promptUserToStart() {
  50. return await window.AndroidBridgePromise.showAlert(
  51. "教务导入说明",
  52. "1. 请确保已在浏览器中成功登录教务系统\n2. 导入过程中请勿关闭页面",
  53. "好的,开始导入"
  54. );
  55. }
  56. /**
  57. * 步骤 B: 获取用户输入的学年
  58. */
  59. async function getAcademicYear() {
  60. const currentYear = new Date().getFullYear().toString();
  61. return await window.AndroidBridgePromise.showPrompt(
  62. "选择学年",
  63. "请输入要导入的起始学年(例如 2025-2026 应输入2025):",
  64. currentYear,
  65. "validateYearInput"
  66. );
  67. }
  68. /**
  69. * 步骤 C: 选择学期
  70. */
  71. async function selectSemester() {
  72. const semesters = ["第一学期 (秋季)", "第二学期 (春季)"];
  73. return await window.AndroidBridgePromise.showSingleSelection(
  74. "选择学期",
  75. JSON.stringify(semesters),
  76. 0
  77. );
  78. }
  79. /**
  80. * 发起网络请求并获取数据
  81. */
  82. async function fetchCourses(academicYear, semesterIndex) {
  83. AndroidBridge.showToast("正在请求教务数据...");
  84. const semesterCode = semesterIndex === 0 ? "01" : "02";
  85. const body = `xnxqdm=${academicYear}${semesterCode}`;
  86. const url = "https://webvpn.sdipct.edu.cn/https/77726476706e69737468656265737421fae0598869236c596e0b9de29d51367bd943/new/student/xsgrkb/getCalendarWeekDatas?vpn-12-o2-jwxt.sdipct.edu.cn";
  87. try {
  88. const response = await fetch(url, {
  89. method: "POST",
  90. headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8" },
  91. body: body,
  92. credentials: "include"
  93. });
  94. if (!response.ok) throw new Error(`HTTP Error: ${response.status}`);
  95. const jsonData = await response.json();
  96. return parseJsonData(jsonData);
  97. } catch (e) {
  98. console.error("Fetch Error:", e);
  99. AndroidBridge.showToast("网络请求失败,请检查登录状态");
  100. return null;
  101. }
  102. }
  103. /**
  104. * 预设时间段数据
  105. */
  106. const StandardTimeSlots = [
  107. { number: 1, startTime: "08:00", endTime: "08:50" },
  108. { number: 2, startTime: "09:00", endTime: "09:50" },
  109. { number: 3, startTime: "10:10", endTime: "11:00" },
  110. { number: 4, startTime: "11:10", endTime: "12:00" },
  111. { number: 5, startTime: "14:00", endTime: "14:50" },
  112. { number: 6, startTime: "15:00", endTime: "15:50" },
  113. { number: 7, startTime: "16:10", endTime: "17:00" },
  114. { number: 8, startTime: "17:10", endTime: "18:00" },
  115. { number: 9, startTime: "19:00", endTime: "19:50" },
  116. { number: 10, startTime: "20:00", endTime: "20:50" }
  117. ];
  118. // 主流程编排
  119. async function runImportFlow() {
  120. // 1. 前置检查
  121. const isReady = await promptUserToStart();
  122. if (!isReady) return; // 用户点击取消,静默退出
  123. // 2. 获取参数(学年、学期)
  124. const year = await getAcademicYear();
  125. if (!year) {
  126. AndroidBridge.showToast("导入已取消");
  127. return;
  128. }
  129. const semesterIdx = await selectSemester();
  130. if (semesterIdx === null) {
  131. AndroidBridge.showToast("导入已取消");
  132. return;
  133. }
  134. // 3. 执行获取与解析
  135. const courses = await fetchCourses(year, semesterIdx);
  136. if (!courses || courses.length === 0) {
  137. if (courses && courses.length === 0) AndroidBridge.showToast("该学期暂无课程数据");
  138. return;
  139. }
  140. // 4. 数据保存
  141. try {
  142. // 保存课程
  143. await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses));
  144. // 保存预设时间表
  145. await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(StandardTimeSlots));
  146. AndroidBridge.showToast(`成功导入 ${courses.length} 门课程!`);
  147. // 5. 流程收尾:通知原生端任务完成
  148. AndroidBridge.notifyTaskCompletion();
  149. console.log("JS: 流程成功完成");
  150. } catch (e) {
  151. AndroidBridge.showToast("保存失败: " + e.message);
  152. }
  153. }
  154. // 启动入口
  155. runImportFlow();