| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543 |
- // 超星教务系统拾光课程表适配脚本
- // 理论上使用超星教务系统的学校通用
- // 课程数据处理部分来自sxgcxy_01.js,由GitHub Copilot生成
- /**
- * 从 HTML 字符串中提取纯文本内容。
- * 超星系统返回的部分字段包含 HTML 标签(如 <a> 标签)
- */
- function extractAnchorText(htmlStr) {
- if (!htmlStr) return '';
- // 移除 HTML 标签,返回剩余的文本内容
- const match = htmlStr.match(/>([^<]+)</);
- return match ? match[1].trim() : htmlStr.trim();
- }
- /**
- * 清理教师名称,去除括号及其内容。
- */
- function cleanTeacherName(name) {
- if (!name) return '';
- // 移除全角或半角的括号及其中的内容
- return name.replace(/([^)]*)/g, '').replace(/\([^)]*\)/g, '').trim();
- }
- /**
- * 解析周次字符串,超星系统直接提供逗号分隔的周次数字。
- * @param {string} weekStr - 周次字符串,如 "1,2,3,4,5"
- * @returns {number[]} - 排序后的周次数组
- */
- function parseWeeks(weekStr) {
- if (!weekStr) return [];
- return weekStr.split(',')
- .map(w => Number(w.trim()))
- .filter(w => !isNaN(w) && w > 0)
- .sort((a, b) => a - b);
- }
- /**
- * 从节次时间数据生成时间段列表。
- * @param {Array} jcsjszList - 节次时间数组,来自 getZclistByXnxq 接口
- * @returns {Array<Object>} - 时间段列表
- */
- 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<Object>} - 解析并合并后的课程列表
- */
- 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<string|null>} - 选中的学年学期参数(如 "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<Object|null>} - 包含 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<Array|null>} - 课程列表,或 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<boolean>} - 是否保存成功
- */
- 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<boolean>} - 是否导入成功
- */
- 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<boolean>} - 是否保存成功
- */
- 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<boolean>} - 用户是否确认
- */
- 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();
|