Selaa lähdekoodia

add: 新疆工程学院教务适配

XingHeYuZhuan 2 viikkoa sitten
vanhempi
sitoutus
d137e9f184
3 muutettua tiedostoa jossa 238 lisäystä ja 1 poistoa
  1. 6 1
      index/root_index.yaml
  2. 9 0
      resources/XJIE/adapters.yaml
  3. 223 0
      resources/XJIE/xjie_01.js

+ 6 - 1
index/root_index.yaml

@@ -146,4 +146,9 @@ schools:
   - id: "HBLGXY"
     name: "淮北理工学院"
     initial: "H"
-    resource_folder: "HBLGXY"  
+    resource_folder: "HBLGXY"  
+
+  - id: "XJIE"
+    name: "新疆工程学院"
+    initial: "X"
+    resource_folder: "XJIE"    

+ 9 - 0
resources/XJIE/adapters.yaml

@@ -0,0 +1,9 @@
+# resources/XJIE/adapters.yaml
+adapters:
+  - adapter_id: "XJIE_01"
+    adapter_name: "新疆工程学院强智适配"
+    category: "BACHELOR_AND_ASSOCIATE"
+    asset_js_path: "xjie_01.js"
+    import_url: "https://authserver.xjie.edu.cn/authserver/login?service=https%3A%2F%2Fjwxt.xjie.edu.cn%2Fjsxsd%2F"
+    maintainer: "星河欲转"
+    description: "新疆工程学院强智适配,非本校开发者适配如果有误建议提交issues"

+ 223 - 0
resources/XJIE/xjie_01.js

@@ -0,0 +1,223 @@
+// 新疆工程学院(xjie.edu.cn) 拾光课程表适配脚本
+// 非该大学开发者适配,开发者无法及时发现问题
+// 出现问题请提联系开发者或者提交pr更改,这更加快速
+
+// 验证逻辑
+
+/**
+ * 年份输入验证函数
+ * @param {string} input 用户输入的年份
+ * @returns {boolean|string} 验证通过返回false,失败返回错误提示
+ */
+window.validateYearInput = function(input) {
+    return /^[0-9]{4}$/.test(input) ? false : "请输入四位数字的学年!";
+};
+
+// 数据解析函数
+
+/**
+ * 将周次字符串解析为数字数组
+ */
+function parseWeeks(weekStr) {
+    const weeks = [];
+    if (!weekStr) return weeks;
+    
+    // 适配 "1-9,11-17(周)[01-02节]" 或 "12-15(周)"
+    const pureWeekData = weekStr.split('(')[0]; 
+    
+    pureWeekData.split(',').forEach(seg => {
+        if (seg.includes('-')) {
+            const [s, e] = seg.split('-').map(Number);
+            if (!isNaN(s) && !isNaN(e)) {
+                for (let i = s; i <= e; i++) weeks.push(i);
+            }
+        } else {
+            const w = parseInt(seg);
+            if (!isNaN(w)) weeks.push(w);
+        }
+    });
+    return [...new Set(weeks)].sort((a, b) => a - b);
+}
+
+/**
+ * 转换课程 HTML 格式为应用模型
+ */
+function parseTimetableToModel(doc) {
+    const timetable = doc.getElementById('timetable');
+    if (!timetable) return [];
+
+    const results = [];
+    const rows = Array.from(timetable.querySelectorAll('tr')).slice(1);
+
+    rows.forEach(row => {
+        const cells = row.querySelectorAll('td');
+        if (cells.length < 7) return;
+
+        // 遍历周一到周日
+        cells.forEach((cell, dayIndex) => {
+            const day = dayIndex + 1;
+            const detailDiv = cell.querySelector('div.kbcontent[style*="none"]');
+            
+            if (detailDiv) {
+                const rawHtml = detailDiv.innerHTML.trim();
+                if (rawHtml === "" || rawHtml === "&nbsp;") return;
+
+                // 拆分可能存在的多个课程块
+                const courseBlocks = rawHtml.split(/---------------------|----------------------/);
+                
+                courseBlocks.forEach(block => {
+                    if (block.replace(/&nbsp;|<br\/?>/g, '').trim() === "") return;
+
+                    const tempDiv = document.createElement('div');
+                    tempDiv.innerHTML = block;
+
+                    // 提取课程名
+                    let name = "";
+                    for (let node of tempDiv.childNodes) {
+                        if (node.nodeType === 3 && node.textContent.trim() !== "") {
+                            name = node.textContent.trim();
+                            break;
+                        } else if (node.tagName === "BR" && name !== "") {
+                            break;
+                        } else if (node.nodeType === 3) {
+                            name += node.textContent.trim();
+                        }
+                    }
+
+                    // 提取教师、周次(节次)、地点
+                    const teacherRaw = tempDiv.querySelector('font[title="教师"]')?.innerText || "未知教师";
+                    const teacher = teacherRaw.replace("任课教师:", "");
+                    const weekStr = tempDiv.querySelector('font[title="周次(节次)"]')?.innerText || "";
+                    const position = tempDiv.querySelector('font[title="教室"]')?.innerText || "未知地点";
+
+                    let start = 0, end = 0;
+                    if (weekStr) {
+                        const secMatch = weekStr.match(/\[(\d+)(?:-(\d+))?节\]/);
+                        if (secMatch) {
+                            start = parseInt(secMatch[1]);
+                            end = secMatch[2] ? parseInt(secMatch[2]) : start;
+                        }
+                    }
+
+                    if (name && weekStr && start > 0) {
+                        results.push({
+                            "name": name,
+                            "teacher": teacher,
+                            "weeks": parseWeeks(weekStr),
+                            "position": position,
+                            "day": day,
+                            "startSection": start,
+                            "endSection": end
+                        });
+                    }
+                });
+            }
+        });
+    });
+
+    return results;
+}
+
+/**
+ * 保存课表全局配置
+ */
+async function saveAppConfig() {
+    const config = {
+        "semesterTotalWeeks": 20,
+        "firstDayOfWeek": 1
+    };
+    return await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify(config));
+}
+
+/**
+ * 保存时间段配置
+ */
+async function saveAppTimeSlots() {
+    const timeSlots = [
+        { "number": 1, "startTime": "10:00", "endTime": "10:45" },
+        { "number": 2, "startTime": "10:55", "endTime": "11:40" },
+        { "number": 3, "startTime": "11:55", "endTime": "12:40" },
+        { "number": 4, "startTime": "12:50", "endTime": "13:35" },
+        { "number": 5, "startTime": "13:45", "endTime": "14:30" },
+        { "number": 6, "startTime": "16:00", "endTime": "16:45" },
+        { "number": 7, "startTime": "16:55", "endTime": "17:40" },
+        { "number": 8, "startTime": "17:55", "endTime": "18:40" },
+        { "number": 9, "startTime": "18:50", "endTime": "19:35" },
+        { "number": 10, "startTime": "19:50", "endTime": "20:35" },
+        { "number": 11, "startTime": "20:45", "endTime": "21:30" }
+    ];
+    return await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(timeSlots));
+}
+
+/**
+ * 获取并让用户选择学期 ID
+ */
+async function getSelectedSemesterId() {
+    const currentYear = new Date().getFullYear();
+    // 绑定验证函数 validateYearInput
+    const year = await window.AndroidBridgePromise.showPrompt(
+        "选择学年", "请输入学年(如2025):", String(currentYear), "validateYearInput"
+    );
+    if (!year) return null;
+    
+    const semesterIndex = await window.AndroidBridgePromise.showSingleSelection(
+        "选择学期", JSON.stringify(["第一学期", "第二学期"]), 0
+    );
+    if (semesterIndex === null) return null;
+    
+    return `${year}-${parseInt(year) + 1}-${semesterIndex + 1}`;
+}
+
+// 流程控制
+
+async function runImportFlow() {
+    try {
+        const confirmed = await window.AndroidBridgePromise.showAlert(
+            "公告",
+            "请确保您已在当前页面成功登录教务系统,否则无法获取数据。是否继续?",
+            "确认已登录"
+        );
+        if (!confirmed) {
+            AndroidBridge.showToast("导入已取消");
+            return;
+        }
+
+        const semesterId = await getSelectedSemesterId();
+        if (!semesterId) {
+            AndroidBridge.showToast("导入已取消");
+            return;
+        }
+
+        AndroidBridge.showToast("正在获取教务数据...");
+        
+        const response = await fetch("https://jwxt.xjie.edu.cn/jsxsd/xskb/xskb_list.do", {
+            method: "POST",
+            headers: { "Content-Type": "application/x-www-form-urlencoded" },
+            body: `cj0701id=&zc=&demo=&xnxq01id=${semesterId}`,
+            credentials: "include"
+        });
+        
+        const html = await response.text();
+        const finalCourses = parseTimetableToModel(new DOMParser().parseFromString(html, "text/html"));
+
+        if (finalCourses.length === 0) {
+            AndroidBridge.showToast("未发现任何课程数据,检查是否登录或者学期选择是否正确");
+            return;
+        }
+
+        // 保存数据
+        AndroidBridge.showToast("正在保存配置...");
+        await saveAppConfig();
+        await saveAppTimeSlots();
+        await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(finalCourses));
+        
+        AndroidBridge.showToast(`成功导入 ${finalCourses.length} 门课程`);
+        AndroidBridge.notifyTaskCompletion();
+
+    } catch (error) {
+        AndroidBridge.showToast("异常: " + error.message);
+    }
+}
+
+// 启动导入流程
+runImportFlow();