| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 |
- // 基于 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();
|