| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336 |
- // 三峡大学(ctgu.edu.cn)拾光课程表适配脚本
- // 非该大学开发者适配,开发者无法及时发现问题
- // 出现问题请提联系开发者或者提交pr更改,这更加快速
- // 核心工具函数:数据验证
- function validateYearInput(input) {
- if (/^[0-9]{4}$/.test(input) && parseInt(input) > 2000) {
- return false;
- } else {
- return "请输入有效的四位数字学年(例如:2025)!";
- }
- }
- /**
- * 辅助函数:解析周次字符串 "111000..." 为数字数组 [1, 2, 3]
- */
- function parseWeeksFromSkzc(skzc) {
- const weeks = [];
- const rawSkzc = skzc || '';
- for (let i = 0; i < rawSkzc.length; i++) {
- if (rawSkzc[i] === '1') {
- weeks.push(Number(i + 1));
- }
- }
- return weeks;
- }
- /**
- * 将教务系统的课程数据转换成 CourseJsonModel 结构
- */
- function parseSingleCourse(rawCourse) {
- const courseName = rawCourse.KCM;
- const teacherName = rawCourse.SKJS ? rawCourse.SKJS.split('/')[0] : '';
- const position = rawCourse.JASMC;
- const day = rawCourse.SKXQ;
- const startSection = rawCourse.KSJC;
- const endSection = rawCourse.JSJC;
- const weeks = parseWeeksFromSkzc(rawCourse.SKZC);
- if (!courseName || !day || !startSection || !endSection || weeks.length === 0) {
- return null;
- }
- const course = {
- "name": courseName,
- "teacher": teacherName,
- "position": position || '待定',
- "day": parseInt(day),
- "startSection": parseInt(startSection),
- "endSection": parseInt(endSection),
- "weeks": weeks
- };
-
- course._kbId = rawCourse.KBID;
- course._day = course.day;
- course._startSection = course.startSection;
- course._endSection = course.endSection;
-
- return course;
- }
- /**
- * 将调课数据应用到已解析的课程列表上
- */
- function applyCourseChanges(parsedCourses, rawChanges) {
- let successCount = 0;
-
- for (const change of rawChanges) {
- const kbID = change.KBID;
- const originalTeacher = change.YSKJS ? change.YSKJS.split('/')[0] : '';
-
- const weeksToRemove = parseWeeksFromSkzc(change.SKZC);
-
- let changeApplied = false;
- const affectedOriginalCourses = parsedCourses.filter(c =>
- c._kbId === kbID &&
- c._day === parseInt(change.SKXQ) &&
- c._startSection === parseInt(change.KSJC) &&
- c._endSection === parseInt(change.JSJC)
- );
-
- if (affectedOriginalCourses.length === 0) {
- continue;
- }
- if (weeksToRemove.length > 0) {
- affectedOriginalCourses.forEach(originalCourse => {
- const beforeLength = originalCourse.weeks.length;
- originalCourse.weeks = originalCourse.weeks.filter(w => !weeksToRemove.includes(w));
- if (originalCourse.weeks.length < beforeLength) {
- changeApplied = true;
- }
- });
- }
-
- const isTimeLocationChange = (change.TKLXDM === '01' || change.TKLXDM === '03');
- if (isTimeLocationChange && change.XSKZC && change.XSKXQ && change.XKSJC && change.XJSJC) {
- const newWeeks = parseWeeksFromSkzc(change.XSKZC);
-
- if (newWeeks.length > 0) {
- const newCourse = {
- "name": change.KCM,
- "teacher": change.XSKJS ? change.XSKJS.split('/')[0] : originalTeacher,
- "position": change.XJASMC || change.JASMC || '待定',
- "day": parseInt(change.XSKXQ),
- "startSection": parseInt(change.XKSJC),
- "endSection": parseInt(change.XJSJC),
- "weeks": newWeeks,
- "_kbId": kbID,
- "_day": parseInt(change.XSKXQ),
- "_startSection": parseInt(change.XKSJC),
- "_endSection": parseInt(change.XJSJC)
- };
- parsedCourses.push(newCourse);
- changeApplied = true;
- }
- }
-
- if (changeApplied) {
- successCount++;
- }
- }
-
- if (successCount > 0) {
- AndroidBridge.showToast(`已应用 ${successCount} 条调课/停课变更,获得实际课表。`);
- }
-
- return parsedCourses.map(c => {
- delete c._kbId;
- delete c._day;
- delete c._startSection;
- delete c._endSection;
- return c;
- }).filter(c => c.weeks.length > 0);
- }
- async function promptUserToStart() {
- const confirmed = await window.AndroidBridgePromise.showAlert(
- "重要通知:三峡大学课表导入",
- "本流程将通过教务系统接口获取您的个人课表。\n重要提示:\n导入前请确保您已在浏览器中成功登录教务系统,且未关闭登录窗口,确认当前页面有显示课表不然获取不了数据",
- "好的,开始导入"
- );
- if (!confirmed) {
- AndroidBridge.showToast("用户取消了导入。");
- return null;
- }
- return true;
- }
- async function getAcademicYear() {
- const currentYear = new Date().getFullYear();
- const yearSelection = await window.AndroidBridgePromise.showPrompt(
- "选择学年",
- "请输入要导入课程的学年(例如 2025):",
- String(currentYear),
- "validateYearInput"
- );
- return yearSelection;
- }
- async function selectSemester() {
- const semesters = ["1 (秋季学期/上学期)", "2 (春季学期/下学期)"];
- const semesterIndex = await window.AndroidBridgePromise.showSingleSelection(
- "选择学期",
- JSON.stringify(semesters),
- 0
- );
-
- if (semesterIndex === null) return null;
- return String(semesterIndex + 1);
- }
- // 数据获取和解析部分
- async function fetchAndParseCourses(academicYear, semesterCode) {
- const XNXQDM = `${academicYear}-${parseInt(academicYear) + 1}-${semesterCode}`;
- const headers = {
- "content-type": "application/x-www-form-urlencoded; charset=UTF-8",
- "x-requested-with": "XMLHttpRequest",
- };
-
- // 获取个人课表数据
- const courseUrl = "http://jwxt.ctgu.edu.cn/jwapp/sys/wdkb/modules/xskcb/cxxszhxqkb.do";
- const courseBody = `XNXQDM=${XNXQDM}`;
- let rawCourseData;
- try {
- const response = await fetch(courseUrl, { "headers": headers, "body": courseBody, "method": "POST", "credentials": "include" });
- rawCourseData = JSON.parse(await response.text());
- } catch (e) {
- AndroidBridge.showToast("请求课表 API 失败,请检查网络和登录状态,以及是否跳转到课表页面");
- console.error("Fetch Course Error:", e);
- return null;
- }
- const rawCourses = rawCourseData?.datas?.cxxszhxqkb?.rows || [];
- if (rawCourses.length === 0) {
- AndroidBridge.showToast("该学期未查询到您的课程数据。");
- return null;
- }
- let parsedCourses = rawCourses.map(c => parseSingleCourse(c)).filter(c => c !== null);
-
- const changeUrl = "http://jwxt.ctgu.edu.cn/jwapp/sys/wdkb/modules/xskcb/xsdkkc.do";
- const changeBody = `XNXQDM=${XNXQDM}&*order=-SQSJ`;
- let rawChangeData;
- try {
- const response = await fetch(changeUrl, { "headers": headers, "body": changeBody, "method": "POST", "credentials": "include" });
- rawChangeData = JSON.parse(await response.text());
- } catch (e) {
- AndroidBridge.showToast("请求调课 API 失败,将使用未调整的课表数据。");
- console.error("Fetch Change Error:", e);
- }
-
- const rawChanges = rawChangeData?.datas?.xsdkkc?.rows || [];
-
- // 应用调课变更
- if (rawChanges.length > 0) {
- parsedCourses = applyCourseChanges(parsedCourses, rawChanges);
- }
-
- // 课表配置数据
- const courseConfig = {
- semesterTotalWeeks: 20
- };
- return {
- courses: parsedCourses,
- config: courseConfig
- };
- }
- async function saveCourses(parsedCourses) {
- if (parsedCourses.length === 0) {
- AndroidBridge.showToast("没有有效的课程数据可供保存。");
- return true;
- }
- try {
- await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(parsedCourses));
- AndroidBridge.showToast(`成功导入 ${parsedCourses.length} 门课程!`);
- return true;
- } catch (error) {
- AndroidBridge.showToast(`保存课程数据失败: ${error.message}`);
- return false;
- }
- }
- /**
- * 导入预设时间段数据
- */
- async function importPresetTimeSlots() {
- AndroidBridge.showToast("正在导入预设节次时间...");
-
- const presetTimeSlots = [
- { "number": 1, "startTime": "08:00", "endTime": "08:45" },
- { "number": 2, "startTime": "08:50", "endTime": "09:35" },
- { "number": 3, "startTime": "09:55", "endTime": "10:40" },
- { "number": 4, "startTime": "10:45", "endTime": "11:30" },
- { "number": 5, "startTime": "11:35", "endTime": "12:20" },
- { "number": 6, "startTime": "14:20", "endTime": "15:05" },
- { "number": 7, "startTime": "15:10", "endTime": "15:55" },
- { "number": 8, "startTime": "16:15", "endTime": "17:00" },
- { "number": 9, "startTime": "17:05", "endTime": "17:50" },
- { "number": 10, "startTime": "19:00", "endTime": "19:45" },
- { "number": 11, "startTime": "19:50", "endTime": "20:35" },
- { "number": 12, "startTime": "20:40", "endTime": "21:25" }
- ];
- try {
- await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(presetTimeSlots));
- AndroidBridge.showToast("预设时间段导入成功!");
- return true;
- } catch (error) {
- AndroidBridge.showToast("导入时间段失败: " + error.message);
- return false;
- }
- }
- async function saveConfig(configData) {
- try {
- await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify(configData));
- AndroidBridge.showToast("课表配置更新成功!");
- return true;
- } catch (error) {
- AndroidBridge.showToast("保存配置失败: " + error.message);
- return false;
- }
- }
- // 主流程入口
- async function runImportFlow() {
- AndroidBridge.showToast("三峡大学课程导入流程启动...");
- // 1. 公告和前置检查。
- const alertConfirmed = await promptUserToStart();
- if (!alertConfirmed) return;
-
- // 2. 获取用户输入参数 (学年和学期)。
- const academicYear = await getAcademicYear();
- if (academicYear === null) {
- AndroidBridge.showToast("导入已取消。");
- return;
- }
-
- const semesterCode = await selectSemester();
- if (semesterCode === null) {
- AndroidBridge.showToast("导入已取消。");
- return;
- }
- // 3. 导入预设时间段
- await importPresetTimeSlots();
- // 4. 网络请求和数据解析。
- const courseData = await fetchAndParseCourses(academicYear, semesterCode);
- if (courseData === null) return;
- // 5. 保存配置数据
- const configSaveResult = await saveConfig(courseData.config);
- if (!configSaveResult) return;
- // 6. 课程数据保存。
- const saveResult = await saveCourses(courseData.courses);
- if (!saveResult) return;
- // 7. 流程完全成功,发送结束信号。
- AndroidBridge.showToast("所有任务已完成!课表导入成功。");
- AndroidBridge.notifyTaskCompletion();
- }
- // 启动导入流程
- runImportFlow();
|