// 超星教务系统拾光课程表适配脚本 // 理论上使用超星教务系统的学校通用 // 课程数据处理部分来自sxgcxy_01.js,由GitHub Copilot生成 /** * 从 HTML 字符串中提取纯文本内容。 * 超星系统返回的部分字段包含 HTML 标签(如 标签) */ function extractAnchorText(htmlStr) { if (!htmlStr) return ''; // 移除 HTML 标签,返回剩余的文本内容 const match = htmlStr.match(/>([^<]+) Number(w.trim())) .filter(w => !isNaN(w) && w > 0) .sort((a, b) => a - b); } /** * 从节次时间数据生成时间段列表。 * @param {Array} jcsjszList - 节次时间数组,来自 getZclistByXnxq 接口 * @returns {Array} - 时间段列表 */ function generateTimeSlots(jcsjszList) { if (!jcsjszList || !Array.isArray(jcsjszList)) { console.warn("JS: 节次时间数据为空或格式错误。"); return []; } const timeSlots = jcsjszList.map(item => ({ number: Number(item.jc), startTime: item.kssj, endTime: item.jssj })).sort((a, b) => a.number - b.number); console.log(`JS: 生成了 ${timeSlots.length} 个时间段。`); return timeSlots; } /** * 从周次列表中获取开学日期(第1周的开始日期)。 * @param {Array} zclist - 周次列表,来自 getZclistByXnxq 接口 * @returns {string|null} - 开学日期,格式 YYYY-MM-DD */ function getSemesterStartDate(zclist) { if (!zclist || !Array.isArray(zclist) || zclist.length === 0) { console.warn("JS: 周次列表为空或格式错误。"); return null; } // 查找第1周的数据 const firstWeek = zclist.find(zc => Number(zc.zc) === 1); if (!firstWeek || !firstWeek.minrq) { console.warn("JS: 未找到第1周的开始日期。"); return null; } // 将 "2025-08-25 00:00:00" 格式转换为 "2025-08-25" const dateStr = firstWeek.minrq.split(' ')[0]; console.log(`JS: 获取到开学日期: ${dateStr}`); return dateStr; } /** * 解析课程数据,并合并连续节次的同一课程。 * @param {Object} jsonData - sdpkkbList 接口返回的 JSON 数据 * @returns {Array} - 解析并合并后的课程列表 */ function parseCourseData(jsonData) { console.log("JS: 开始解析超星课程数据..."); if (!jsonData || !Array.isArray(jsonData.data)) { console.warn("JS: 课程数据结构错误或缺少 data 字段。"); return []; } const rawCourseList = jsonData.data; // 1. 预处理课程数据,提取必要字段并标准化 const processedList = rawCourseList .map(rawCourse => { const name = extractAnchorText(rawCourse.kcmc); const teacher = cleanTeacherName(extractAnchorText(rawCourse.tmc)); const position = extractAnchorText(rawCourse.croommc) || '待定'; const day = Number(rawCourse.xingqi); const section = Number(rawCourse.djc); // 解析周次字符串并转换为标准 JSON 字符串(用于比较) const weeksArray = parseWeeks(rawCourse.zcstr); const standardizedWeeks = JSON.stringify(weeksArray); // 验证必填字段 if (!name || isNaN(day) || isNaN(section) || day < 1 || day > 7 || section < 1 || weeksArray.length === 0) { return null; } return { name, teacher, position, day, section, standardizedWeeks, weeksArray }; }) .filter(c => c !== null) // 排序:按星期 > 周次 > 课程名 > 教师 > 教室 > 节次 .sort((a, b) => a.day - b.day || a.standardizedWeeks.localeCompare(b.standardizedWeeks) || a.name.localeCompare(b.name) || a.teacher.localeCompare(b.teacher) || a.position.localeCompare(b.position) || a.section - b.section ); // 2. 合并连续节次的相同课程 const finalCourseList = []; let i = 0; while (i < processedList.length) { let current = processedList[i]; let startSection = current.section; let endSection = current.section; let j = i + 1; // 查找连续的节次 while (j < processedList.length) { let next = processedList[j]; // 检查是否可以合并:周次、星期、课程名、教师、教室必须相同,且节次连续 if ( next.day === current.day && next.name === current.name && next.teacher === current.teacher && next.position === current.position && next.standardizedWeeks === current.standardizedWeeks && next.section === endSection + 1 ) { endSection = next.section; j++; } else { break; } } // 添加合并后的课程 finalCourseList.push({ name: current.name, teacher: current.teacher, position: current.position, day: current.day, startSection: startSection, endSection: endSection, weeks: current.weeksArray }); i = j; } console.log(`JS: 课程数据解析完成,共 ${finalCourseList.length} 门课程(已合并连续节次)。`); return finalCourseList; } /** * 生成学年学期选项列表。 * @returns {Object} - 包含 labels(显示文本)、values(参数值)、defaultIndex(默认选项) */ function getSemesterOptions() { const currentYear = new Date().getFullYear(); const currentMonth = new Date().getMonth() + 1; // 根据当前月份判断默认学期(9月前为第二学期,9月后为第一学期) const defaultSemester = currentMonth < 9 ? 2 : 1; const defaultYear = currentMonth < 9 ? currentYear - 1 : currentYear; // 生成前后三年的学年学期选项 const years = [currentYear - 2, currentYear - 1, currentYear, currentYear + 1]; const semesterCodes = ["1", "2"]; let labels = []; let values = []; let defaultIndex = -1; let index = 0; for (let i = 0; i < years.length; i++) { const startYear = years[i]; const endYear = startYear + 1; const yearStr = `${startYear}-${endYear}`; for (let j = 0; j < semesterCodes.length; j++) { const code = semesterCodes[j]; const apiValue = `${yearStr}-${code}`; const semesterName = code === "1" ? "第一学期" : "第二学期"; labels.push(`${yearStr}学年 ${semesterName}`); values.push(apiValue); // 设置默认选项 if (startYear === defaultYear && Number(code) === defaultSemester) { defaultIndex = index; } index++; } } return { labels, values, defaultIndex }; } /** * 提示用户选择学年学期。 * @returns {Promise} - 选中的学年学期参数(如 "2025-2026-1"),或 null(取消) */ async function selectAcademicYearAndSemester() { console.log("JS: 提示用户选择学年学期。"); const { labels, values, defaultIndex } = getSemesterOptions(); const selectedIndex = await window.AndroidBridgePromise.showSingleSelection( "选择学年学期", JSON.stringify(labels), defaultIndex ); if (selectedIndex === null || selectedIndex === -1) { return null; } console.log(`JS: 用户选择了学年学期: ${values[selectedIndex]}`); return values[selectedIndex]; } /** * 从页面中提取必要的参数。 * @returns {Object|null} - 包含 xhid 和 xqdm 的对象,或 null(提取失败) */ async function extractPageParams() { console.log("JS: 尝试从页面中提取参数..."); // 方法1:从隐藏的 input 元素中获取 let xhid = document.querySelector('#xhid')?.value; let xqdm = document.querySelector('#xqdm')?.value; // 方法2:如果页面上找不到,尝试从 URL 参数中获取 if (!xhid || !xqdm) { const path = "/admin/pkgl/xskb/queryKbForXsd"; try { // 获取课表页的 HTML 文档 const htmlText = await fetch(path).then(res => res.text()); const contentDom = new DOMParser().parseFromString(htmlText, "text/html"); // 从转换后的 HTML 文档中获取 xhid 和 xqdm 的值 xhid = contentDom.querySelector("#xhid")?.value; xqdm = contentDom.querySelector("#xqdm")?.value; } catch (error) { console.warn("JS: 通过抓取课表页提取参数失败", error); } } console.log(`JS: 提取到参数 - xhid: ${xhid}, xqdm: ${xqdm}`); if (!xhid || !xqdm) { console.warn("JS: 无法从页面中提取必要参数。"); return null; } return { xhid, xqdm }; } /** * 获取节次时间和开学日期信息。 * @param {string} xnxq - 学年学期参数 * @param {string} xqdm - 校区代码 * @returns {Promise} - 包含 timeSlots 和 semesterStartDate,或 null(失败) */ async function fetchTimeAndWeekData(xnxq, xqdm) { console.log(`JS: 正在请求节次时间和周次数据...`); AndroidBridge.showToast("正在获取课表配置信息..."); const url = `/admin/api/getZclistByXnxq?xnxq=${xnxq}&xqid=${xqdm}`; const requestOptions = { "headers": { "Accept": "application/json, text/plain, */*", "X-Requested-With": "XMLHttpRequest" }, "method": "GET", "credentials": "include" }; try { const response = await fetch(url, requestOptions); if (!response.ok) { throw new Error(`网络请求失败。状态码: ${response.status}`); } const jsonData = await response.json(); if (jsonData.ret !== 0) { throw new Error(`API 返回错误: ${jsonData.msg || '未知错误'}`); } // 提取节次时间和开学日期 const timeSlots = generateTimeSlots(jsonData.data?.jcsjszList); const semesterStartDate = getSemesterStartDate(jsonData.data?.zclist); if (timeSlots.length === 0) { throw new Error("未能获取到有效的节次时间信息。"); } console.log(`JS: 成功获取节次时间(${timeSlots.length}个)和开学日期(${semesterStartDate})。`); return { timeSlots, semesterStartDate }; } catch (error) { AndroidBridge.showToast(`获取配置信息失败: ${error.message}`); console.error('JS: fetchTimeAndWeekData Error:', error); return null; } } /** * 获取课程数据。 * @param {string} xnxq - 学年学期参数 * @param {string} xhid - 学号ID * @param {string} xqdm - 校区代码 * @returns {Promise} - 课程列表,或 null(失败) */ async function fetchCourseData(xnxq, xhid, xqdm) { console.log(`JS: 正在请求课程数据...`); AndroidBridge.showToast(`正在获取 ${xnxq} 的课程数据...`); const url = `/admin/xsd/pkgl/xskb/sdpkkbList?xnxq=${xnxq}&xhid=${xhid}&xqdm=${xqdm}&xskbxslx=0`; const requestOptions = { "headers": { "Accept": "application/json, text/plain, */*", "X-Requested-With": "XMLHttpRequest" }, "method": "GET", "credentials": "include" }; try { const response = await fetch(url, requestOptions); if (!response.ok) { throw new Error(`网络请求失败。状态码: ${response.status}`); } const jsonData = await response.json(); if (jsonData.ret !== 0) { throw new Error(`API 返回错误: ${jsonData.msg || '未知错误'}`); } const courses = parseCourseData(jsonData); if (courses.length === 0) { AndroidBridge.showToast("未找到任何课程数据,本学期可能无课。"); return null; } console.log(`JS: 课程数据获取成功,共 ${courses.length} 门课程。`); return courses; } catch (error) { AndroidBridge.showToast(`获取课程数据失败: ${error.message}`); console.error('JS: fetchCourseData Error:', error); return null; } } /** * 保存课程数据到应用。 * @param {Array} courses - 课程列表 * @returns {Promise} - 是否保存成功 */ async function saveCourses(courses) { console.log(`JS: 正在保存 ${courses.length} 门课程...`); AndroidBridge.showToast(`正在保存 ${courses.length} 门课程...`); try { await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses, null, 2)); console.log("JS: 课程保存成功。"); return true; } catch (error) { AndroidBridge.showToast(`课程保存失败: ${error.message}`); console.error('JS: saveCourses Error:', error); return false; } } /** * 导入预设时间段到应用。 * @param {Array} timeSlots - 时间段列表 * @returns {Promise} - 是否导入成功 */ async function importPresetTimeSlots(timeSlots) { console.log(`JS: 正在导入 ${timeSlots.length} 个预设时间段...`); AndroidBridge.showToast(`正在导入作息时间...`); try { await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(timeSlots)); console.log("JS: 预设时间段导入成功。"); return true; } catch (error) { AndroidBridge.showToast("导入时间段失败: " + error.message); console.error('JS: importPresetTimeSlots Error:', error); return false; } } /** * 保存课表配置(开学日期等)。 * @param {string|null} semesterStartDate - 开学日期 * @returns {Promise} - 是否保存成功 */ async function saveCourseConfig(semesterStartDate) { if (!semesterStartDate) { console.log("JS: 开学日期为空,跳过课表配置保存。"); return true; } console.log(`JS: 正在保存课表配置(开学日期: ${semesterStartDate})...`); const config = { semesterStartDate: semesterStartDate }; try { await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify(config)); console.log("JS: 课表配置保存成功。"); return true; } catch (error) { AndroidBridge.showToast("保存课表配置失败: " + error.message); console.error('JS: saveCourseConfig Error:', error); return false; } } /** * 检查是否在登录页面。 * @returns {boolean} */ function isLoginPage() { const url = window.location.href; return url.includes('login') || url.includes('slogin'); } /** * 提示用户开始导入。 * @returns {Promise} - 用户是否确认 */ async function promptUserToStart() { return await window.AndroidBridgePromise.showAlert( "超星教务系统课表导入", "导入前请确保您已在成功登录教务系统,并打开课表页面。\n\n本脚本将自动获取作息时间、开学日期和课程数据。", "开始导入" ); } /** * 主导入流程。 */ async function runImportFlow() { console.log("JS: 开始执行超星教务系统课表导入流程..."); // 1. 检查是否在登录页面 if (isLoginPage()) { AndroidBridge.showToast("导入失败:请先登录教务系统!"); return; } // 2. 提示用户确认开始导入 const alertConfirmed = await promptUserToStart(); if (!alertConfirmed) { AndroidBridge.showToast("用户取消了导入。"); return; } // 3. 提取页面参数 const params = await extractPageParams(); if (!params) { AndroidBridge.showToast("无法从页面获取必要参数,请确保在正确的页面执行脚本。"); return; } const { xhid, xqdm } = params; // 4. 让用户选择学年学期 const xnxq = await selectAcademicYearAndSemester(); if (xnxq === null) { AndroidBridge.showToast("导入已取消,未选择学年学期。"); return; } // 5. 获取节次时间和开学日期 const timeData = await fetchTimeAndWeekData(xnxq, xqdm); if (!timeData) { return; } const { timeSlots, semesterStartDate } = timeData; // 6. 获取课程数据 const courses = await fetchCourseData(xnxq, xhid, xqdm); if (!courses) { return; } // 7. 保存课程数据 const saveResult = await saveCourses(courses); if (!saveResult) { return; } // 8. 导入预设时间段 await importPresetTimeSlots(timeSlots); // 9. 保存课表配置(开学日期) await saveCourseConfig(semesterStartDate); // 10. 完成 AndroidBridge.showToast(`导入成功!共导入 ${courses.length} 门课程。`); AndroidBridge.notifyTaskCompletion(); console.log("JS: 超星教务系统课表导入流程完成。"); } // 执行主流程 runImportFlow();