|
|
@@ -0,0 +1,543 @@
|
|
|
+// 超星教务系统拾光课程表适配脚本
|
|
|
+// 理论上使用超星教务系统的学校通用
|
|
|
+// 课程数据处理部分来自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();
|