|
|
@@ -0,0 +1,261 @@
|
|
|
+// 基于 HTML 页面抓取的拾光课表正方适配脚本
|
|
|
+
|
|
|
+/**
|
|
|
+ * 解析表格
|
|
|
+ */
|
|
|
+function parserTbale() {
|
|
|
+ const regexName = /[●★○]/g;
|
|
|
+ const courseInfoList = [];
|
|
|
+ const $ = window.jQuery;
|
|
|
+ if (!$) return courseInfoList;
|
|
|
+
|
|
|
+ $('#kbgrid_table_0 td').each((i, td) => {
|
|
|
+ if ($(td).hasClass('td_wrap') && $(td).text().trim() !== '') {
|
|
|
+ const day = parseInt($(td).attr('id').split('-')[0]);
|
|
|
+
|
|
|
+ $(td).find('.timetable_con.text-left').each((i, course) => {
|
|
|
+ const name = $(course).find('.title font').text().replace(regexName, '').trim();
|
|
|
+
|
|
|
+ const infoStr = $(course).find('p').eq(0).find('font').eq(1).text().trim();
|
|
|
+
|
|
|
+ const position = $(course).find('p').eq(1).find('font').text().trim();
|
|
|
+ const teacher = $(course).find('p').eq(2).find('font').text().trim();
|
|
|
+
|
|
|
+ if (infoStr && infoStr.match(/\((\d+-\d+节)\)/) && infoStr.split('节)')[1]) {
|
|
|
+ const [sections, weeks] = parserInfo(infoStr);
|
|
|
+
|
|
|
+ if (name && position && teacher && sections.length && weeks.length) {
|
|
|
+ const startSection = sections[0];
|
|
|
+ const endSection = sections[sections.length - 1];
|
|
|
+
|
|
|
+ const finalPosition = position.split(/\s+/).pop();
|
|
|
+
|
|
|
+ const data = { name, day, weeks, teacher, position: finalPosition, startSection, endSection };
|
|
|
+ courseInfoList.push(data);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ });
|
|
|
+ return courseInfoList;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * 解析列表
|
|
|
+ */
|
|
|
+function parserList() {
|
|
|
+ const regexName = /[●★○]/g;
|
|
|
+ const regexWeekNum = /周数:|周/g;
|
|
|
+ const regexPosition = /上课地点:/g;
|
|
|
+ const regexTeacher = /教师 :/g;
|
|
|
+
|
|
|
+ const $ = window.jQuery;
|
|
|
+ if (!$) return [];
|
|
|
+
|
|
|
+ let courseInfoList = [];
|
|
|
+ $('#kblist_table tbody').each((day, tbody) => {
|
|
|
+ if (day > 0 && day < 8) {
|
|
|
+ let sections;
|
|
|
+ $(tbody).find('tr:not(:first-child)').each((trIndex, tr) => {
|
|
|
+ let name, font;
|
|
|
+
|
|
|
+ if ($(tr).find('td').length > 1) {
|
|
|
+ sections = parserSections($(tr).find('td:first-child').text());
|
|
|
+ name = $(tr).find('td:nth-child(2)').find('.title').text().replace(regexName, '').trim();
|
|
|
+ font = $(tr).find('td:nth-child(2)').find('p font');
|
|
|
+ } else {
|
|
|
+ name = $(tr).find('td').find('.title').text().replace(regexName, '').trim();
|
|
|
+ font = $(tr).find('td').find('p font');
|
|
|
+ }
|
|
|
+
|
|
|
+ const weekStr = $(font[0]).text().replace(regexWeekNum, '').trim();
|
|
|
+ const weeks = parserWeeks(weekStr);
|
|
|
+
|
|
|
+ const positionRaw = $(font[1]).text().replace(regexPosition, '').trim();
|
|
|
+ const finalPosition = positionRaw.split(/\s+/).pop();
|
|
|
+
|
|
|
+ const teacher = $(font[2]).text().replace(regexTeacher, '').trim();
|
|
|
+
|
|
|
+ if (name && sections && weeks.length && teacher && finalPosition) {
|
|
|
+ const startSection = sections[0];
|
|
|
+ const endSection = sections[sections.length - 1];
|
|
|
+
|
|
|
+ const data = {
|
|
|
+ name,
|
|
|
+ day,
|
|
|
+ weeks,
|
|
|
+ teacher,
|
|
|
+ position: finalPosition,
|
|
|
+ startSection,
|
|
|
+ endSection
|
|
|
+ };
|
|
|
+ courseInfoList.push(data);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ });
|
|
|
+ return courseInfoList;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * 解析课程信息
|
|
|
+ */
|
|
|
+function parserInfo(str) {
|
|
|
+ const sections = parserSections(str.match(/\((\d+-\d+节)\)/)[1].replace(/节/g, ''));
|
|
|
+ const weekStrWithMarker = str.split('节)')[1];
|
|
|
+ const weeks = parserWeeks(weekStrWithMarker.replace(/周/g, '').trim());
|
|
|
+ return [sections, weeks];
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * 解析节次
|
|
|
+ */
|
|
|
+function parserSections(str) {
|
|
|
+ const [start, end] = str.split('-').map(Number);
|
|
|
+ if (isNaN(start) || isNaN(end) || start > end) return [];
|
|
|
+ return Array.from({ length: end - start + 1 }, (_, i) => start + i);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * 解析周次
|
|
|
+ */
|
|
|
+function parserWeeks(str) {
|
|
|
+ const segments = str.split(',');
|
|
|
+ let weeks = [];
|
|
|
+ const segmentRegex = /(\d+)(?:-(\d+))?\s*(\([单双]\))?/g;
|
|
|
+
|
|
|
+ for (const segment of segments) {
|
|
|
+ // 清理段落中的周字和多余空格
|
|
|
+ const cleanSegment = segment.replace(/周/g, '').trim();
|
|
|
+
|
|
|
+ // 重置正则的 lastIndex
|
|
|
+ segmentRegex.lastIndex = 0;
|
|
|
+
|
|
|
+ let match;
|
|
|
+ // 循环匹配一个段落内可能存在的所有周次定义(虽然通常只有一个)
|
|
|
+ while ((match = segmentRegex.exec(cleanSegment)) !== null) {
|
|
|
+ const start = parseInt(match[1]);
|
|
|
+ const end = match[2] ? parseInt(match[2]) : start;
|
|
|
+ const flagStr = match[3] || ''; // (单) 或 (双) 或 ""
|
|
|
+
|
|
|
+ let flag = 0;
|
|
|
+ if (flagStr.includes('单')) {
|
|
|
+ flag = 1;
|
|
|
+ } else if (flagStr.includes('双')) {
|
|
|
+ flag = 2;
|
|
|
+ }
|
|
|
+
|
|
|
+ for (let i = start; i <= end; i++) {
|
|
|
+ // 过滤单双周
|
|
|
+ if (flag === 1 && i % 2 !== 1) continue; // 仅保留单周
|
|
|
+ if (flag === 2 && i % 2 !== 0) continue; // 仅保留双周
|
|
|
+
|
|
|
+ // 防止重复添加
|
|
|
+ if (!weeks.includes(i)) {
|
|
|
+ weeks.push(i);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ // 排序并返回唯一的周次列表
|
|
|
+ return weeks.sort((a, b) => a - b);
|
|
|
+}
|
|
|
+
|
|
|
+async function scrapeAndParseCourses() {
|
|
|
+ AndroidBridge.showToast("正在检查页面并抓取课程数据...");
|
|
|
+ const ts = `1.登陆教务系统\n2.导航到学生课表查询页面\n3.等待课表信息加载,选择对应学年、学期,确认无误后点击【查询】\n4.确保页面上显示了课程表\n5.点击下方【一键导入】`
|
|
|
+ try {
|
|
|
+ const response = await fetch(window.location.href);
|
|
|
+ const text = await response.text();
|
|
|
+ if (!text.includes("课表查询")) {
|
|
|
+ console.log("页面内容检查失败!");
|
|
|
+ await window.AndroidBridgePromise.showAlert("导入失败", "当前页面似乎不是学生课表查询页面。请检查:\n" + ts, "确定");
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+ const typeElement = document.querySelector('#shcPDF');
|
|
|
+ if (!typeElement) {
|
|
|
+ console.log("未能找到视图类型元素 (#shcPDF)");
|
|
|
+ await window.AndroidBridgePromise.showAlert("导入失败", "未能识别课表视图类型,请确认您已点击查询且课表已加载完毕。", "确定");
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+ const type = typeElement.dataset['type'];
|
|
|
+ const tableElement = document.querySelector(type === 'list' ? '#kblist_table' : '#kbgrid_table_0');
|
|
|
+ if (!tableElement) {
|
|
|
+ console.log("未能找到课表主体 HTML");
|
|
|
+ await window.AndroidBridgePromise.showAlert("导入失败", `未能找到课表主体 (${type} 视图),请确认您已点击查询且课表已加载完毕。`, "确定");
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+ let result = [];
|
|
|
+ if (type === 'list') {
|
|
|
+ result = parserList();
|
|
|
+ } else {
|
|
|
+ result = parserTbale();
|
|
|
+ }
|
|
|
+ if (result.length === 0) {
|
|
|
+ AndroidBridge.showToast("未找到任何课程数据,请检查所选学年学期是否正确或本学期无课。");
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+ console.log(`JS: 课程数据解析成功,共找到 ${result.length} 门课程。`);
|
|
|
+ return { courses: result };
|
|
|
+ } catch (error) {
|
|
|
+ AndroidBridge.showToast(`抓取或解析失败: ${error.message}`);
|
|
|
+ console.error('JS: Scrape/Parse Error:', error);
|
|
|
+ await window.AndroidBridgePromise.showAlert("抓取或解析失败", `发生错误:${error.message}。请重试或联系开发者。`, "确定");
|
|
|
+ return null;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+async function saveCourses(parsedCourses) {
|
|
|
+ AndroidBridge.showToast(`正在保存 ${parsedCourses.length} 门课程...`);
|
|
|
+ console.log(`JS: 尝试保存 ${parsedCourses.length} 门课程...`);
|
|
|
+ try {
|
|
|
+ await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(parsedCourses, null, 2));
|
|
|
+ console.log("JS: 课程保存成功!");
|
|
|
+ return true;
|
|
|
+ } catch (error) {
|
|
|
+ AndroidBridge.showToast(`课程保存失败: ${error.message}`);
|
|
|
+ console.error('JS: Save Courses Error:', error);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+async function runImportFlow() {
|
|
|
+ const alertConfirmed = await window.AndroidBridgePromise.showAlert(
|
|
|
+ "教务系统课表导入",
|
|
|
+ "导入前请确保您已在浏览器中成功登录教务系统,并处于课表查询页面且已点击查询。",
|
|
|
+ "好的,开始导入"
|
|
|
+ );
|
|
|
+ if (!alertConfirmed) {
|
|
|
+ AndroidBridge.showToast("用户取消了导入。");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (typeof window.jQuery === 'undefined' && typeof $ === 'undefined') {
|
|
|
+ const errorMsg = "当前教务系统页面似乎没有加载 jQuery 库。本脚本依赖 jQuery 进行 DOM 解析。";
|
|
|
+ AndroidBridge.showToast(errorMsg);
|
|
|
+ await window.AndroidBridgePromise.showAlert("导入失败", errorMsg + "\n请尝试刷新页面或使用其他导入方式。", "确定");
|
|
|
+ console.error("JS: 缺少 jQuery 依赖,流程终止。");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ const result = await scrapeAndParseCourses();
|
|
|
+ if (result === null) {
|
|
|
+ console.log("JS: 课程获取或解析失败,流程终止。");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ const { courses } = result;
|
|
|
+
|
|
|
+ const saveResult = await saveCourses(courses);
|
|
|
+ if (!saveResult) {
|
|
|
+ console.log("JS: 课程保存失败,流程终止。");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ AndroidBridge.showToast(`课程导入成功,共导入 ${courses.length} 门课程!`);
|
|
|
+ console.log("JS: 整个导入流程执行完毕并成功。");
|
|
|
+ AndroidBridge.notifyTaskCompletion();
|
|
|
+}
|
|
|
+
|
|
|
+runImportFlow();
|