Explorar o código

add: 天津农学院教务适配

XingHeYuZhuan hai 14 horas
pai
achega
db1637fcbd
Modificáronse 3 ficheiros con 227 adicións e 1 borrados
  1. 6 1
      index/root_index.yaml
  2. 9 0
      resources/TJAU/adapters.yaml
  3. 212 0
      resources/TJAU/tjau.js

+ 6 - 1
index/root_index.yaml

@@ -445,4 +445,9 @@ schools:
   - id: "UPC"
     name: "中国石油大学(华东)"
     initial: "Z"
-    resource_folder: "UPC"   
+    resource_folder: "UPC"   
+    
+  - id: "TJAU"
+    name: "天津农学院"
+    initial: "T"
+    resource_folder: "TJAU"     

+ 9 - 0
resources/TJAU/adapters.yaml

@@ -0,0 +1,9 @@
+# resources/TJAU/adapters.yaml
+adapters:
+  - adapter_id: "TJAU"
+    adapter_name: "天津农学院树维教务"
+    category: "BACHELOR_AND_ASSOCIATE"
+    asset_js_path: "tjau.js"
+    import_url: "http://jwxt.tjau.edu.cn/eams/homeExt.action"
+    maintainer: "星河欲转"
+    description: "天津农学院树维教务适配,登录后直接点击导入,无视可能出现的显示异常,非本校开发者适配如果有误建议提交issues"

+ 212 - 0
resources/TJAU/tjau.js

@@ -0,0 +1,212 @@
+// 天津农学院(tjau.edu.cn) 拾光课程表适配脚本
+// 非该大学开发者适配,开发者无法及时发现问题
+// 出现问题请提issues或者提交pr更改,这更加快速
+
+
+function powerSplit(paramsRaw) {
+    const args = [];
+    let current = "";
+    let depth = 0; 
+    let inQuote = false;
+    let quoteChar = "";
+
+    for (let i = 0; i < paramsRaw.length; i++) {
+        let char = paramsRaw[i];
+        if ((char === '"' || char === "'") && (i === 0 || paramsRaw[i - 1] !== '\\')) {
+            if (!inQuote) { inQuote = true; quoteChar = char; }
+            else if (char === quoteChar) { inQuote = false; }
+        }
+        if (!inQuote) {
+            if (char === '(' || char === '[' || char === '{') depth++;
+            if (char === ')' || char === ']' || char === '}') depth--;
+        }
+        if (char === ',' && depth === 0 && !inQuote) {
+            args.push(cleanArg(current));
+            current = "";
+        } else {
+            current += char;
+        }
+    }
+    args.push(cleanArg(current)); 
+    return args;
+}
+
+function cleanArg(s) {
+    s = s.trim();
+    if (s === "null") return null;
+    return s.replace(/^["']|["']$/g, "");
+}
+
+/**
+ * 全局课程合并逻辑
+ */
+function mergeContinuousLessons(lessons) {
+    if (!lessons || lessons.length <= 1) return lessons;
+
+    lessons.sort((a, b) => {
+        if (a.name !== b.name) return a.name.localeCompare(b.name);
+        if (a.day !== b.day) return a.day - b.day;
+        if (JSON.stringify(a.weeks) !== JSON.stringify(b.weeks)) 
+            return JSON.stringify(a.weeks).localeCompare(JSON.stringify(b.weeks));
+        return a.startSection - b.startSection;
+    });
+
+    const merged = [];
+    let current = lessons[0];
+
+    for (let i = 1; i < lessons.length; i++) {
+        let next = lessons[i];
+
+        const isSameLesson = 
+            current.name === next.name &&
+            current.teacher === next.teacher &&
+            current.position === next.position &&
+            current.day === next.day &&
+            JSON.stringify(current.weeks) === JSON.stringify(next.weeks);
+        const isContinuous = current.endSection + 1 === next.startSection;
+
+        if (isSameLesson && isContinuous) {
+            current.endSection = next.endSection;
+        } else {
+            merged.push(current);
+            current = next;
+        }
+    }
+    merged.push(current);
+    return merged;
+}
+
+function parseTaskActivities(html) {
+    const rawResults = [];
+    const blocks = html.split(/var\s+teachers\s*=/);
+
+    for (let i = 1; i < blocks.length; i++) {
+        const block = blocks[i];
+        let teacherName = "未知教师";
+        const tMatch = block.match(/actTeachers\s*=\s*\[\s*\{[\s\S]*?name:\s*"(.*?)"/);
+        if (tMatch) teacherName = tMatch[1];
+
+        const activityMatch = block.match(/new\s+TaskActivity\(([\s\S]*?)\);/);
+        if (!activityMatch) continue;
+
+        const args = powerSplit(activityMatch[1]);
+        const courseName = (args[3] || "未知课程").split('(')[0];
+        const position = (args[5] || "未知地点").replace(/\(.*?\)/g, "");
+        const weeksBitmap = args[6] || "";
+        
+        const weeks = [];
+        for (let j = 0; j < weeksBitmap.length; j++) {
+            if (weeksBitmap[j] === '1') weeks.push(j);
+        }
+
+        const unitCountMatch = html.match(/unitCount\s*=\s*(\d+)/);
+        const unitCount = unitCountMatch ? parseInt(unitCountMatch[1]) : 14;
+
+        const idxRegex = /index\s*=\s*(\d+)\s*\*\s*unitCount\s*\+\s*(\d+);/g;
+        let m;
+        while ((m = idxRegex.exec(block)) !== null) {
+            const day = parseInt(m[1]) + 1; 
+            const section = parseInt(m[2]) + 1;
+
+            rawResults.push({
+                "name": courseName,
+                "teacher": teacherName,
+                "position": position,
+                "day": day,
+                "startSection": section,
+                "endSection": section,
+                "weeks": weeks
+            });
+        }
+    }
+
+    // 执行全局合并逻辑
+    const mergedResults = mergeContinuousLessons(rawResults);
+    return mergedResults;
+}
+
+
+async function request(url, options = {}) {
+    const res = await fetch(url, { credentials: "include", ...options });
+    if (!res.ok) throw new Error(`网络请求失败: ${res.status}`);
+    return await res.text();
+}
+
+async function detectParameters() {
+    const html = await request("http://jwxt.tjau.edu.cn/eams/courseTableForStd.action?sf_request_type=ajax");
+    const idsMatch = html.match(/bg\.form\.addInput\(form,"ids","(\d+)"\)/);
+    const tagIdMatch = html.match(/id="(semesterBar\d+Semester)"/);
+    if (!idsMatch || !tagIdMatch) return null;
+    return { ids: idsMatch[1], tagId: tagIdMatch[1] };
+}
+
+async function getSelectedSemester(tagId) {
+    const raw = await request(`http://jwxt.tjau.edu.cn/eams/dataQuery.action?sf_request_type=ajax`, {
+        method: "POST",
+        headers: { "Content-Type": "application/x-www-form-urlencoded" },
+        body: `tagId=${encodeURIComponent(tagId)}&dataType=semesterCalendar`
+    });
+    const data = Function(`return (${raw});`)();
+    const list = [];
+    for (let key in data.semesters) {
+        data.semesters[key].forEach(s => list.push({ id: s.id, name: `${s.schoolYear} ${s.name}学期` }));
+    }
+    const idx = await window.AndroidBridgePromise.showSingleSelection("选择学期", JSON.stringify(list.map(s => s.name)), 0);
+    return idx !== null ? list[idx] : null;
+}
+
+async function fetchAndParseCourses(semesterId, ids) {
+    const html = await request(`http://jwxt.tjau.edu.cn/eams/courseTableForStd!courseTable.action?sf_request_type=ajax`, {
+        method: "POST",
+        headers: { "Content-Type": "application/x-www-form-urlencoded" },
+        body: `ignoreHead=1&setting.kind=std&semester.id=${semesterId}&ids=${ids}`
+    });
+    return parseTaskActivities(html);
+}
+
+async function applyTimeSlots() {
+    const slots = [
+        { "number": 1, "startTime": "08:30", "endTime": "09:15" }, 
+        { "number": 2, "startTime": "09:20", "endTime": "10:05" },
+        { "number": 3, "startTime": "10:25", "endTime": "11:10" },
+        { "number": 4, "startTime": "11:15", "endTime": "12:00" },
+        { "number": 5, "startTime": "14:00", "endTime": "14:45" },
+        { "number": 6, "startTime": "14:50", "endTime": "15:35" },
+        { "number": 7, "startTime": "15:55", "endTime": "16:40" }, 
+        { "number": 8, "startTime": "16:45", "endTime": "17:30" },
+        { "number": 9, "startTime": "18:30", "endTime": "19:15" }, 
+        { "number": 10, "startTime": "19:20", "endTime": "20:05" },
+        { "number": 11, "startTime": "20:10", "endTime": "20:55" }
+    ];
+    return await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(slots));
+}
+
+
+async function runImportFlow() {
+    try {
+        AndroidBridge.showToast("开始探测教务参数...");
+        const params = await detectParameters();
+        if (!params) throw new Error("未能识别教务参数,请确认已登录");
+
+        const semester = await getSelectedSemester(params.tagId);
+        if (!semester) return; 
+
+        AndroidBridge.showToast("正在同步课表...");
+        const courses = await fetchAndParseCourses(semester.id, params.ids);
+        
+        if (!courses || courses.length === 0) throw new Error("未解析到课程数据");
+
+        await applyTimeSlots();
+        const saveResult = await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses));
+        
+        if (saveResult) {
+            AndroidBridge.showToast(`成功导入 ${courses.length} 个课程条目`);
+            AndroidBridge.notifyTaskCompletion();
+        }
+    } catch (e) {
+        console.error(`[异常] ${e.message}`);
+        AndroidBridge.showToast(e.message);
+    }
+}
+
+runImportFlow();