chaoxing.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727
  1. // 超星教务系统拾光课程表适配脚本
  2. // 理论上使用超星教务系统的学校通用
  3. // 课程数据处理部分来自sxgcxy_01.js,由GitHub Copilot生成
  4. /**
  5. * 从 HTML 字符串中提取纯文本内容。
  6. * 超星系统返回的部分字段包含 HTML 标签(如 <a> 标签)
  7. */
  8. function extractAnchorText(htmlStr) {
  9. if (!htmlStr) return '';
  10. // 移除 HTML 标签,返回剩余的文本内容
  11. const match = htmlStr.match(/>([^<]+)</);
  12. return match ? match[1].trim() : htmlStr.trim();
  13. }
  14. /**
  15. * 清理教师名称,去除括号及其内容。
  16. */
  17. function cleanTeacherName(name) {
  18. if (!name) return '';
  19. // 移除全角或半角的括号及其中的内容
  20. return name.replace(/([^)]*)/g, '').replace(/\([^)]*\)/g, '').trim();
  21. }
  22. /**
  23. * 解析周次字符串,超星系统直接提供逗号分隔的周次数字。
  24. * @param {string} weekStr - 周次字符串,如 "1,2,3,4,5"
  25. * @returns {number[]} - 排序后的周次数组
  26. */
  27. function parseWeeks(weekStr) {
  28. if (!weekStr) return [];
  29. return weekStr.split(',')
  30. .map(w => Number(w.trim()))
  31. .filter(w => !isNaN(w) && w > 0)
  32. .sort((a, b) => a - b);
  33. }
  34. /**
  35. * 从周次列表中获取开学日期(第1周的开始日期)。
  36. * @param {Array} zclist - 周次列表,来自 getZclistByXnxq 接口
  37. * @returns {string|null} - 开学日期,格式 YYYY-MM-DD
  38. */
  39. function getSemesterStartDate(zclist) {
  40. if (!zclist || !Array.isArray(zclist) || zclist.length === 0) {
  41. console.warn("JS: 周次列表为空或格式错误。");
  42. return null;
  43. }
  44. // 查找第1周的数据
  45. const firstWeek = zclist.find(zc => Number(zc.zc) === 1);
  46. if (!firstWeek || !firstWeek.minrq) {
  47. console.warn("JS: 未找到第1周的开始日期。");
  48. return null;
  49. }
  50. // 将 "2025-08-25 00:00:00" 格式转换为 "2025-08-25"
  51. const dateStr = firstWeek.minrq.split(' ')[0];
  52. console.log(`JS: 获取到开学日期: ${dateStr}`);
  53. return dateStr;
  54. }
  55. /**
  56. * 解析课程数据,并合并连续节次的同一课程。
  57. * @param {Object} jsonData - sdpkkbList 接口返回的 JSON 数据
  58. * @returns {Array<Object>} - 解析并合并后的课程列表
  59. */
  60. function parseCourseData(jsonData) {
  61. console.log("JS: 开始解析超星课程数据...");
  62. if (!jsonData || !Array.isArray(jsonData.data)) {
  63. console.warn("JS: 课程数据结构错误或缺少 data 字段。");
  64. return [];
  65. }
  66. const rawCourseList = jsonData.data;
  67. // 1. 预处理课程数据,提取必要字段并标准化
  68. const processedList = rawCourseList
  69. .map(rawCourse => {
  70. const name = extractAnchorText(rawCourse.kcmc);
  71. const teacher = cleanTeacherName(extractAnchorText(rawCourse.tmc));
  72. const position = extractAnchorText(rawCourse.croommc) || '待定';
  73. const day = Number(rawCourse.xingqi);
  74. const section = Number(rawCourse.djc);
  75. // 解析周次字符串并转换为标准 JSON 字符串(用于比较)
  76. const weeksArray = parseWeeks(rawCourse.zcstr);
  77. const standardizedWeeks = JSON.stringify(weeksArray);
  78. // 验证必填字段
  79. if (!name || isNaN(day) || isNaN(section) || day < 1 || day > 7 || section < 1 || weeksArray.length === 0) {
  80. return null;
  81. }
  82. return { name, teacher, position, day, section, standardizedWeeks, weeksArray };
  83. })
  84. .filter(c => c !== null)
  85. // 排序:按星期 > 周次 > 课程名 > 教师 > 教室 > 节次
  86. .sort((a, b) =>
  87. a.day - b.day ||
  88. a.standardizedWeeks.localeCompare(b.standardizedWeeks) ||
  89. a.name.localeCompare(b.name) ||
  90. a.teacher.localeCompare(b.teacher) ||
  91. a.position.localeCompare(b.position) ||
  92. a.section - b.section
  93. );
  94. // 2. 合并连续节次的相同课程
  95. const finalCourseList = [];
  96. let i = 0;
  97. while (i < processedList.length) {
  98. let current = processedList[i];
  99. let startSection = current.section;
  100. let endSection = current.section;
  101. let j = i + 1;
  102. // 查找连续的节次
  103. while (j < processedList.length) {
  104. let next = processedList[j];
  105. // 检查是否可以合并:周次、星期、课程名、教师、教室必须相同,且节次连续
  106. if (
  107. next.day === current.day &&
  108. next.name === current.name &&
  109. next.teacher === current.teacher &&
  110. next.position === current.position &&
  111. next.standardizedWeeks === current.standardizedWeeks &&
  112. next.section === endSection + 1
  113. ) {
  114. endSection = next.section;
  115. j++;
  116. } else {
  117. break;
  118. }
  119. }
  120. // 添加合并后的课程
  121. finalCourseList.push({
  122. name: current.name,
  123. teacher: current.teacher,
  124. position: current.position,
  125. day: current.day,
  126. startSection: startSection,
  127. endSection: endSection,
  128. weeks: current.weeksArray
  129. });
  130. i = j;
  131. }
  132. console.log(`JS: 课程数据解析完成,共 ${finalCourseList.length} 门课程(已合并连续节次)。`);
  133. return finalCourseList;
  134. }
  135. /**
  136. * 生成学年学期选项列表。
  137. * @returns {Object} - 包含 labels(显示文本)、values(参数值)、defaultIndex(默认选项)
  138. */
  139. function getSemesterOptions() {
  140. const currentYear = new Date().getFullYear();
  141. const currentMonth = new Date().getMonth() + 1;
  142. // 根据当前月份判断默认学期(9月前为第二学期,9月后为第一学期)
  143. const defaultSemester = currentMonth < 9 ? 2 : 1;
  144. const defaultYear = currentMonth < 9 ? currentYear - 1 : currentYear;
  145. // 生成前后三年的学年学期选项
  146. const years = [currentYear - 2, currentYear - 1, currentYear, currentYear + 1];
  147. const semesterCodes = ["1", "2"];
  148. let labels = [];
  149. let values = [];
  150. let defaultIndex = -1;
  151. let index = 0;
  152. for (let i = 0; i < years.length; i++) {
  153. const startYear = years[i];
  154. const endYear = startYear + 1;
  155. const yearStr = `${startYear}-${endYear}`;
  156. for (let j = 0; j < semesterCodes.length; j++) {
  157. const code = semesterCodes[j];
  158. const apiValue = `${yearStr}-${code}`;
  159. const semesterName = code === "1" ? "第一学期" : "第二学期";
  160. labels.push(`${yearStr}学年 ${semesterName}`);
  161. values.push(apiValue);
  162. // 设置默认选项
  163. if (startYear === defaultYear && Number(code) === defaultSemester) {
  164. defaultIndex = index;
  165. }
  166. index++;
  167. }
  168. }
  169. return { labels, values, defaultIndex };
  170. }
  171. /**
  172. * 从学年学期下拉框 HTML 中解析选项。
  173. * @param {string} selectHtml - id=xnxq1 的 select 元素 HTML
  174. * @returns {Object|null} - 包含 labels、values、defaultIndex,失败返回 null
  175. */
  176. function parseSemesterOptionsFromHtml(selectHtml) {
  177. if (!selectHtml || typeof selectHtml !== 'string') {
  178. return null;
  179. }
  180. try {
  181. const doc = new DOMParser().parseFromString(selectHtml, "text/html");
  182. const selectElement = doc.querySelector('#xnxq1') || doc.querySelector('select[name="xnxq"]');
  183. if (!selectElement) {
  184. return null;
  185. }
  186. const optionElements = Array.from(selectElement.querySelectorAll('option'));
  187. const validOptions = optionElements.filter(option => (option.value || '').trim() !== '');
  188. if (validOptions.length === 0) {
  189. return null;
  190. }
  191. const labels = validOptions.map(option => (option.textContent || option.value || '').trim());
  192. const values = validOptions.map(option => (option.value || '').trim());
  193. let defaultIndex = validOptions.findIndex(option => option.selected);
  194. if (defaultIndex < 0) {
  195. defaultIndex = 0;
  196. }
  197. return { labels, values, defaultIndex };
  198. } catch (error) {
  199. console.warn("JS: 解析学年学期下拉 HTML 失败", error);
  200. return null;
  201. }
  202. }
  203. /**
  204. * 提示用户选择学年学期。
  205. * @param {string|null|undefined} xnxqSelectHtml - 页面中 id=xnxq1 的 select 元素 HTML
  206. * @returns {Promise<string|null>} - 选中的学年学期参数(如 "2025-2026-1"),或 null(取消)
  207. */
  208. async function selectAcademicYearAndSemester(xnxqSelectHtml) {
  209. console.log("JS: 提示用户选择学年学期。");
  210. let options = parseSemesterOptionsFromHtml(xnxqSelectHtml);
  211. if (!options) {
  212. console.warn("JS: 页面学年学期选项解析失败,回退为前后三年选项。");
  213. options = getSemesterOptions();
  214. }
  215. const { labels, values, defaultIndex } = options;
  216. const selectedIndex = await window.AndroidBridgePromise.showSingleSelection(
  217. "选择学年学期",
  218. JSON.stringify(labels),
  219. defaultIndex
  220. );
  221. if (selectedIndex === null || selectedIndex === -1) {
  222. return null;
  223. }
  224. console.log(`JS: 用户选择了学年学期: ${values[selectedIndex]}`);
  225. return values[selectedIndex];
  226. }
  227. /**
  228. * 获取校区列表。
  229. * @returns {Promise<Array<{xqdm: string, xqmc: string}>|null>} - 校区列表,或 null(失败)
  230. */
  231. async function fetchCampusList() {
  232. console.log("JS: 正在请求校区列表...");
  233. const url = "/admin/api/jcsj/xqsj/getXqList";
  234. const requestOptions = {
  235. "headers": {
  236. "Accept": "application/json, text/plain, */*",
  237. "X-Requested-With": "XMLHttpRequest"
  238. },
  239. "method": "GET",
  240. "credentials": "include"
  241. };
  242. try {
  243. const response = await fetch(url, requestOptions);
  244. if (!response.ok) {
  245. throw new Error(`网络请求失败。状态码: ${response.status}`);
  246. }
  247. const jsonData = await response.json();
  248. if (jsonData.ret !== 0) {
  249. throw new Error(`API 返回错误: ${jsonData.msg || '未知错误'}`);
  250. }
  251. if (!Array.isArray(jsonData.data)) {
  252. throw new Error("校区列表数据格式错误。");
  253. }
  254. const campusList = jsonData.data
  255. .map(item => ({
  256. xqdm: String(item.id || '').trim(),
  257. xqmc: String(item.xqmc || item.id || '').trim()
  258. }))
  259. .filter(item => item.xqdm && item.xqmc);
  260. if (campusList.length === 0) {
  261. throw new Error("未获取到有效校区信息。");
  262. }
  263. console.log(`JS: 校区列表获取成功,共 ${campusList.length} 个校区。`);
  264. return campusList;
  265. } catch (error) {
  266. AndroidBridge.showToast(`获取校区列表失败: ${error.message}`);
  267. console.error('JS: fetchCampusList Error:', error);
  268. return null;
  269. }
  270. }
  271. /**
  272. * 提示用户选择校区。
  273. * @param {string} defaultXqdm - 默认选中的校区代码
  274. * @returns {Promise<{xqdm: string, xqmc: string}|null>} - 选中的校区信息,或 null(取消/失败)
  275. */
  276. async function selectCampus(defaultXqdm) {
  277. const campusList = await fetchCampusList();
  278. if (!campusList) {
  279. console.warn(`JS: 获取校区列表失败,使用页面参数 xqdm=${defaultXqdm} 继续导入。`);
  280. AndroidBridge.showToast("获取校区列表失败,使用默认校区继续导入。");
  281. return {
  282. xqdm: defaultXqdm,
  283. xqmc: ""
  284. };
  285. }
  286. const labels = campusList.map(item => item.xqmc);
  287. const defaultIndex = Math.max(
  288. campusList.findIndex(item => item.xqdm === defaultXqdm),
  289. 0
  290. );
  291. const selectedIndex = await window.AndroidBridgePromise.showSingleSelection(
  292. "选择校区",
  293. JSON.stringify(labels),
  294. defaultIndex
  295. );
  296. if (selectedIndex === null || selectedIndex === -1) {
  297. return null;
  298. }
  299. const selectedCampus = campusList[selectedIndex];
  300. console.log(`JS: 用户选择了校区: ${selectedCampus.xqmc} (${selectedCampus.xqdm})`);
  301. return selectedCampus;
  302. }
  303. /**
  304. * 时间格式化
  305. * @returns {string|null} - 格式化后的时间字符串 "HH:MM",或 null(格式不合法)
  306. */
  307. function formatTime(timeStr) {
  308. if (typeof timeStr !== 'string') return null;
  309. const match = timeStr.match(/^(\d{1,2}):(\d{1,2})(?::\d{1,2})?$/);
  310. if (!match) return null;
  311. let [ , hour, minute ] = match.map(Number);
  312. return `${String(hour).padStart(2, '0')}:${String(minute).padStart(2, '0')}`;
  313. }
  314. /**
  315. * 从节次时间数据生成时间段列表。
  316. * @param {Array} jcsjszList - 节次时间数组,来自 getZclistByXnxq 接口
  317. * @returns {Array<Object>|null} - 时间段列表,格式不合法时返回 null
  318. */
  319. function generateTimeSlots(jcsjszList) {
  320. if (!jcsjszList || !Array.isArray(jcsjszList)) {
  321. console.warn("JS: 节次时间数据为空或格式错误。");
  322. return [];
  323. }
  324. const timeSlots = [];
  325. for (const item of jcsjszList) {
  326. const startTime = formatTime(item.kssj);
  327. const endTime = formatTime(item.jssj);
  328. if (!startTime || !endTime) {
  329. console.warn("JS: 节次时间格式不合法,跳过生成时间段列表。", item);
  330. return null;
  331. }
  332. timeSlots.push({
  333. number: Number(item.jc),
  334. startTime,
  335. endTime
  336. });
  337. }
  338. timeSlots.sort((a, b) => a.number - b.number);
  339. console.log(`JS: 生成了 ${timeSlots.length} 个时间段。`);
  340. return timeSlots;
  341. }
  342. /**
  343. * 从页面中提取必要的参数。
  344. * @returns {Object|null} - 包含 xhid、xqdm 和 xnxqSelectHtml 的对象,或 null(提取失败)
  345. */
  346. async function extractPageParams() {
  347. console.log("JS: 尝试从页面中提取参数...");
  348. // 方法1:从隐藏的 input 元素中获取
  349. let xhid = document.querySelector('#xhid')?.value;
  350. let xqdm = document.querySelector('#xqdm')?.value;
  351. let xnxqSelectHtml = document.querySelector('#xnxq1')?.outerHTML;
  352. // 方法2:如果页面上找不到,尝试抓取课表页并解析
  353. if (!xhid || !xqdm || !xnxqSelectHtml) {
  354. console.log("JS: 尝试抓取课表页并解析...");
  355. const path = "/admin/pkgl/xskb/queryKbForXsd";
  356. try {
  357. // 获取课表页的 HTML 文档
  358. const htmlText = await fetch(path).then(res => res.text());
  359. const contentDom = new DOMParser().parseFromString(htmlText, "text/html");
  360. // 从转换后的 HTML 文档中获取 xhid 和 xqdm 的值
  361. xhid = contentDom.querySelector("#xhid")?.value || xhid;
  362. xqdm = contentDom.querySelector("#xqdm")?.value || xqdm;
  363. xnxqSelectHtml = contentDom.querySelector("#xnxq1")?.outerHTML || xnxqSelectHtml;
  364. } catch (error) {
  365. console.warn("JS: 通过抓取课表页提取参数失败", error);
  366. }
  367. }
  368. console.log(`JS: 提取到参数 - xhid: ${xhid}, xqdm: ${xqdm}, xnxqSelectHtml: ${xnxqSelectHtml ? 'yes' : 'no'}`);
  369. if (!xhid || !xqdm) {
  370. console.warn("JS: 无法从页面中提取必要参数。");
  371. return null;
  372. }
  373. return { xhid, xqdm, xnxqSelectHtml };
  374. }
  375. /**
  376. * 获取节次时间和开学日期信息。
  377. * @param {string} xnxq - 学年学期参数
  378. * @param {string} xqdm - 校区代码
  379. * @returns {Promise<Object>} - 包含 timeSlots 和 semesterStartDate,请求失败时返回空配置继续流程
  380. */
  381. async function fetchTimeAndWeekData(xnxq, xqdm) {
  382. console.log(`JS: 正在请求节次时间和周次数据...`);
  383. AndroidBridge.showToast("正在获取课表配置信息...");
  384. const url = `/admin/api/getZclistByXnxq?xnxq=${xnxq}&xqid=${xqdm}`;
  385. const requestOptions = {
  386. "headers": {
  387. "Accept": "application/json, text/plain, */*",
  388. "X-Requested-With": "XMLHttpRequest"
  389. },
  390. "method": "GET",
  391. "credentials": "include"
  392. };
  393. try {
  394. const response = await fetch(url, requestOptions);
  395. if (!response.ok) {
  396. throw new Error(`网络请求失败。状态码: ${response.status}`);
  397. }
  398. const jsonData = await response.json();
  399. if (jsonData.ret !== 0) {
  400. throw new Error(`API 返回错误: ${jsonData.msg || '未知错误'}`);
  401. }
  402. // 提取节次时间和开学日期
  403. const timeSlots = generateTimeSlots(jsonData.data?.jcsjszList);
  404. const semesterStartDate = getSemesterStartDate(jsonData.data?.zclist);
  405. if (!timeSlots) {
  406. console.warn("JS: 未获取到节次时间");
  407. }
  408. if (!semesterStartDate) {
  409. console.warn("JS: 未获取到开学日期");
  410. }
  411. console.log(`JS: 成功获取节次时间(${Array.isArray(timeSlots) ? timeSlots.length : 0}个)和开学日期(${semesterStartDate})。`);
  412. return { timeSlots, semesterStartDate };
  413. } catch (error) {
  414. AndroidBridge.showToast(`获取配置信息失败,将继续导入课程: ${error.message}`);
  415. console.error('JS: fetchTimeAndWeekData Error:', error);
  416. return {
  417. timeSlots: null,
  418. semesterStartDate: null
  419. };
  420. }
  421. }
  422. /**
  423. * 获取课程数据。
  424. * @param {string} xnxq - 学年学期参数
  425. * @param {string} xhid - 学号ID
  426. * @param {string} xqdm - 校区代码
  427. * @returns {Promise<Array|null>} - 课程列表,或 null(失败)
  428. */
  429. async function fetchCourseData(xnxq, xhid, xqdm) {
  430. console.log(`JS: 正在请求课程数据...`);
  431. AndroidBridge.showToast(`正在获取 ${xnxq} 的课程数据...`);
  432. const url = `/admin/xsd/pkgl/xskb/sdpkkbList?xnxq=${xnxq}&xhid=${xhid}&xqdm=${xqdm}&xskbxslx=0`;
  433. const requestOptions = {
  434. "headers": {
  435. "Accept": "application/json, text/plain, */*",
  436. "X-Requested-With": "XMLHttpRequest"
  437. },
  438. "method": "GET",
  439. "credentials": "include"
  440. };
  441. try {
  442. const response = await fetch(url, requestOptions);
  443. if (!response.ok) {
  444. throw new Error(`网络请求失败。状态码: ${response.status}`);
  445. }
  446. const jsonData = await response.json();
  447. if (jsonData.ret !== 0) {
  448. throw new Error(`API 返回错误: ${jsonData.msg || '未知错误'}`);
  449. }
  450. const courses = parseCourseData(jsonData);
  451. if (courses.length === 0) {
  452. AndroidBridge.showToast("未找到任何课程数据,本学期可能无课。");
  453. return null;
  454. }
  455. console.log(`JS: 课程数据获取成功,共 ${courses.length} 门课程。`);
  456. return courses;
  457. } catch (error) {
  458. AndroidBridge.showToast(`获取课程数据失败: ${error.message}`);
  459. console.error('JS: fetchCourseData Error:', error);
  460. return null;
  461. }
  462. }
  463. /**
  464. * 保存课程数据到应用。
  465. * @param {Array} courses - 课程列表
  466. * @returns {Promise<boolean>} - 是否保存成功
  467. */
  468. async function saveCourses(courses) {
  469. console.log(`JS: 正在保存 ${courses.length} 门课程...`);
  470. AndroidBridge.showToast(`正在保存 ${courses.length} 门课程...`);
  471. try {
  472. await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses, null, 2));
  473. console.log("JS: 课程保存成功。");
  474. return true;
  475. } catch (error) {
  476. AndroidBridge.showToast(`课程保存失败: ${error.message}`);
  477. console.error('JS: saveCourses Error:', error);
  478. return false;
  479. }
  480. }
  481. /**
  482. * 导入预设时间段到应用。
  483. * @param {Array} timeSlots - 时间段列表
  484. * @returns {Promise<boolean>} - 是否导入成功
  485. */
  486. async function importPresetTimeSlots(timeSlots) {
  487. console.log(`JS: 正在导入 ${timeSlots.length} 个预设时间段...`);
  488. AndroidBridge.showToast(`正在导入作息时间...`);
  489. try {
  490. await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(timeSlots));
  491. console.log("JS: 预设时间段导入成功。");
  492. return true;
  493. } catch (error) {
  494. AndroidBridge.showToast("导入时间段失败: " + error.message);
  495. console.error('JS: importPresetTimeSlots Error:', error);
  496. return false;
  497. }
  498. }
  499. /**
  500. * 保存课表配置(开学日期等)。
  501. * @param {string|null} semesterStartDate - 开学日期
  502. * @returns {Promise<boolean>} - 是否保存成功
  503. */
  504. async function saveCourseConfig(semesterStartDate) {
  505. if (!semesterStartDate) {
  506. console.log("JS: 开学日期为空,跳过课表配置保存。");
  507. return true;
  508. }
  509. console.log(`JS: 正在保存课表配置(开学日期: ${semesterStartDate})...`);
  510. const config = {
  511. semesterStartDate: semesterStartDate
  512. };
  513. try {
  514. await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify(config));
  515. console.log("JS: 课表配置保存成功。");
  516. return true;
  517. } catch (error) {
  518. AndroidBridge.showToast("保存课表配置失败: " + error.message);
  519. console.error('JS: saveCourseConfig Error:', error);
  520. return false;
  521. }
  522. }
  523. /**
  524. * 检查是否在登录页面。
  525. * @returns {boolean}
  526. */
  527. function isLoginPage() {
  528. const url = window.location.href;
  529. return url.includes('login') || url.includes('slogin');
  530. }
  531. /**
  532. * 提示用户开始导入。
  533. * @returns {Promise<boolean>} - 用户是否确认
  534. */
  535. async function promptUserToStart() {
  536. return await window.AndroidBridgePromise.showAlert(
  537. "超星教务系统课表导入",
  538. "导入前请确保您已在成功登录教务系统,并打开课表页面。\n\n本脚本将自动获取作息时间、开学日期和课程数据。",
  539. "开始导入"
  540. );
  541. }
  542. /**
  543. * 主导入流程。
  544. */
  545. async function runImportFlow() {
  546. console.log("JS: 开始执行超星教务系统课表导入流程...");
  547. // 1. 检查是否在登录页面
  548. if (isLoginPage()) {
  549. AndroidBridge.showToast("导入失败:请先登录教务系统!");
  550. return;
  551. }
  552. // 2. 提示用户确认开始导入
  553. const alertConfirmed = await promptUserToStart();
  554. if (!alertConfirmed) {
  555. AndroidBridge.showToast("用户取消了导入。");
  556. return;
  557. }
  558. // 3. 提取页面参数
  559. const params = await extractPageParams();
  560. if (!params) {
  561. AndroidBridge.showToast("无法从页面获取必要参数,请确保在正确的页面执行脚本。");
  562. return;
  563. }
  564. const { xhid, xqdm: pageXqdm, xnxqSelectHtml } = params;
  565. // 4. 让用户选择学年学期
  566. const xnxq = await selectAcademicYearAndSemester(xnxqSelectHtml);
  567. if (xnxq === null) {
  568. AndroidBridge.showToast("导入已取消,未选择学年学期。");
  569. return;
  570. }
  571. // 5. 让用户选择校区(默认使用页面参数中的 xqdm,若校区接口失败则自动回退使用页面 xqdm)
  572. const selectedCampus = await selectCampus(pageXqdm);
  573. if (!selectedCampus) {
  574. AndroidBridge.showToast("导入已取消,未选择校区。");
  575. return;
  576. }
  577. const { xqdm } = selectedCampus;
  578. // 6. 获取节次时间和开学日期
  579. const timeData = await fetchTimeAndWeekData(xnxq, xqdm);
  580. const { timeSlots, semesterStartDate } = timeData;
  581. // 7. 获取课程数据
  582. const courses = await fetchCourseData(xnxq, xhid, xqdm);
  583. if (!courses) {
  584. return;
  585. }
  586. // 8. 保存课程数据
  587. const saveResult = await saveCourses(courses);
  588. if (!saveResult) {
  589. return;
  590. }
  591. // 9. 导入预设时间段
  592. if (Array.isArray(timeSlots) && timeSlots.length > 0) {
  593. await importPresetTimeSlots(timeSlots);
  594. } else {
  595. console.log("JS: 未生成有效时间段,跳过预设时间段导入。");
  596. }
  597. // 10. 保存课表配置(开学日期)
  598. if (semesterStartDate) {
  599. await saveCourseConfig(semesterStartDate);
  600. } else {
  601. console.log("JS: 未获取到开学日期,跳过课表配置保存。");
  602. }
  603. // 11. 完成
  604. AndroidBridge.showToast(`导入成功!共导入 ${courses.length} 门课程。`);
  605. AndroidBridge.notifyTaskCompletion();
  606. console.log("JS: 超星教务系统课表导入流程完成。");
  607. }
  608. // 执行主流程
  609. runImportFlow();