zjut_01.js 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. // 文件: ZJUT_01.js
  2. // 功能:从浙江工业大学正方教务系统获取课程表,解析后导入到拾光课程表
  3. // 适配:浙江工业大学正方教务系统
  4. // 维护者:glxgo
  5. const BASE = `${window.location.origin}/jwglxt`;
  6. const GNMKDM = 'N253508';
  7. const INDEX_PATH = `/kbcx/xskbcx_cxXskbcxIndex.html?gnmkdm=${GNMKDM}&layout=default`;
  8. const COURSE_API_PATH = `/kbcx/xskbcx_cxXsgrkb.html?gnmkdm=${GNMKDM}`;
  9. const TIME_API_PATH = `/kbcx/xskbcx_cxRjc.html?gnmkdm=${GNMKDM}`;
  10. async function req(url, method = 'GET', body) {
  11. const res = await fetch(url, {
  12. method,
  13. credentials: 'include',
  14. headers: {
  15. 'content-type': 'application/x-www-form-urlencoded;charset=UTF-8',
  16. 'x-requested-with': 'XMLHttpRequest',
  17. 'accept': '*/*'
  18. },
  19. body
  20. });
  21. if (!res.ok) throw new Error(`请求失败: ${res.status}`);
  22. return await res.text();
  23. }
  24. function isOnTimetablePage() {
  25. return window.location.pathname === '/jwglxt/kbcx/xskbcx_cxXskbcxIndex.html';
  26. }
  27. function readCurrentPageTerm() {
  28. const xnmEl = document.querySelector('#xnm');
  29. const xqmEl = document.querySelector('#xqm');
  30. const xnm = xnmEl ? String(xnmEl.value || '').trim() : '';
  31. const xqm = xqmEl ? String(xqmEl.value || '').trim() : '';
  32. if (!xnm || !xqm) throw new Error('当前课表页未读到学年学期,请先选择后再导入');
  33. return { xnm, xqm };
  34. }
  35. function parseSelectOptions(selectEl) {
  36. if (!selectEl) return { options: [], defaultIndex: 0 };
  37. const options = [];
  38. let defaultIndex = 0;
  39. Array.from(selectEl.querySelectorAll('option')).forEach((opt) => {
  40. const value = String(opt.value || '').trim();
  41. if (!value) return;
  42. const text = String(opt.textContent || '').trim() || value;
  43. if (opt.selected) defaultIndex = options.length;
  44. options.push({ value, text });
  45. });
  46. return { options, defaultIndex };
  47. }
  48. function parseTermOptionsFromDoc(doc) {
  49. const yearData = parseSelectOptions(doc.querySelector('#xnm'));
  50. const semesterData = parseSelectOptions(doc.querySelector('#xqm'));
  51. if (!yearData.options.length || !semesterData.options.length) {
  52. throw new Error('课表页学年学期选项解析失败');
  53. }
  54. return { yearData, semesterData };
  55. }
  56. async function fetchIndexDoc() {
  57. const html = await fetch(`${BASE}${INDEX_PATH}`, { credentials: 'include' }).then(res => {
  58. if (!res.ok) throw new Error(`课表页请求失败: ${res.status}`);
  59. return res.text();
  60. });
  61. return new DOMParser().parseFromString(html, 'text/html');
  62. }
  63. async function selectTermByUserFromDoc(doc) {
  64. const { yearData, semesterData } = parseTermOptionsFromDoc(doc);
  65. const yearIndex = await window.AndroidBridgePromise.showSingleSelection(
  66. '选择学年',
  67. JSON.stringify(yearData.options.map(item => item.text)),
  68. yearData.defaultIndex
  69. );
  70. if (yearIndex === null || yearIndex === -1) throw new Error('已取消学年选择');
  71. const semesterIndex = await window.AndroidBridgePromise.showSingleSelection(
  72. '选择学期',
  73. JSON.stringify(semesterData.options.map(item => item.text)),
  74. semesterData.defaultIndex
  75. );
  76. if (semesterIndex === null || semesterIndex === -1) throw new Error('已取消学期选择');
  77. return {
  78. xnm: yearData.options[yearIndex].value,
  79. xqm: semesterData.options[semesterIndex].value
  80. };
  81. }
  82. async function resolveTerm() {
  83. if (isOnTimetablePage()) {
  84. return readCurrentPageTerm();
  85. }
  86. const doc = await fetchIndexDoc();
  87. return await selectTermByUserFromDoc(doc);
  88. }
  89. function parseWeeks(zcd) {
  90. if (!zcd) return [];
  91. const result = new Set();
  92. String(zcd).replace(/\s+/g, '').split(/[,,]/).forEach((seg) => {
  93. const odd = seg.includes('单');
  94. const even = seg.includes('双');
  95. const normalized = seg.replace(/周|\(|\)|单|双/g, '');
  96. const match = normalized.match(/(\d+)(?:-(\d+))?/);
  97. if (!match) return;
  98. const start = Number(match[1]);
  99. const end = Number(match[2] || match[1]);
  100. for (let week = start; week <= end; week++) {
  101. if (odd && week % 2 === 0) continue;
  102. if (even && week % 2 !== 0) continue;
  103. result.add(week);
  104. }
  105. });
  106. return [...result].sort((a, b) => a - b);
  107. }
  108. function parseCourses(data) {
  109. if (!data || !Array.isArray(data.kbList)) {
  110. return { courses: [], xqhId: '01' };
  111. }
  112. const courses = [];
  113. let xqhId = '01';
  114. data.kbList.forEach((course) => {
  115. if (course.xqh_id) xqhId = String(course.xqh_id).trim() || xqhId;
  116. const day = Number(course.xqj);
  117. const secRaw = String(course.jcs || course.jc || '').replace(/节/g, '').trim();
  118. const sectionNums = (secRaw.match(/\d+/g) || []).map(Number).filter(n => !Number.isNaN(n));
  119. const weeks = parseWeeks(course.zcd);
  120. if (!course.kcmc || !sectionNums.length || !weeks.length || !(day >= 1 && day <= 7)) return;
  121. courses.push({
  122. name: String(course.kcmc).trim(),
  123. teacher: String(course.xm || '未知').trim(),
  124. position: String(course.cdmc || course.cdbh || '未排地点').trim(),
  125. day,
  126. startSection: sectionNums[0],
  127. endSection: sectionNums[sectionNums.length - 1],
  128. weeks
  129. });
  130. });
  131. const deduped = new Map();
  132. courses.forEach((course) => {
  133. const key = `${course.name}|${course.teacher}|${course.position}|${course.day}|${course.startSection}|${course.endSection}|${course.weeks.join(',')}`;
  134. if (!deduped.has(key)) deduped.set(key, course);
  135. });
  136. return { courses: [...deduped.values()], xqhId };
  137. }
  138. function parseTimeSlots(data) {
  139. if (!Array.isArray(data) || !data.length) throw new Error('未获取到节次时间数据');
  140. return data.map((item) => ({
  141. number: Number(item.jcmc),
  142. startTime: String(item.qssj || '').trim(),
  143. endTime: String(item.jssj || '').trim()
  144. })).filter(item => item.number > 0 && item.startTime && item.endTime);
  145. }
  146. async function fetchCourses(xnm, xqm) {
  147. const body = `xnm=${encodeURIComponent(xnm)}&xqm=${encodeURIComponent(xqm)}&kzlx=ck&xsdm=&kclbdm=&kclxdm=`;
  148. const text = await req(`${BASE}${COURSE_API_PATH}`, 'POST', body);
  149. return JSON.parse(text);
  150. }
  151. async function fetchTimeSlots(xnm, xqm) {
  152. const body = `xnm=${encodeURIComponent(xnm)}&xqm=${encodeURIComponent(xqm)}`;
  153. const text = await req(`${BASE}${TIME_API_PATH}`, 'POST', body);
  154. return parseTimeSlots(JSON.parse(text));
  155. }
  156. async function run() {
  157. try {
  158. const { xnm, xqm } = await resolveTerm();
  159. AndroidBridge.showToast('正在解析课表数据...');
  160. const rawData = await fetchCourses(xnm, xqm);
  161. const { courses, xqhId } = parseCourses(rawData);
  162. if (!courses.length) throw new Error('未获取到课表数据');
  163. const timeSlots = await fetchTimeSlots(xnm, xqm).catch(() => null);
  164. const allWeeks = courses.flatMap(course => course.weeks);
  165. const semesterTotalWeeks = allWeeks.length ? Math.max(...allWeeks) : 20;
  166. await window.AndroidBridgePromise.saveCourseConfig(JSON.stringify({
  167. semesterTotalWeeks,
  168. semesterStartDate: null,
  169. firstDayOfWeek: 1
  170. }));
  171. if (timeSlots && timeSlots.length) {
  172. await window.AndroidBridgePromise.savePresetTimeSlots(JSON.stringify(timeSlots));
  173. }
  174. await window.AndroidBridgePromise.saveImportedCourses(JSON.stringify(courses));
  175. AndroidBridge.showToast(`导入成功:${courses.length} 门`);
  176. AndroidBridge.notifyTaskCompletion();
  177. } catch (error) {
  178. console.error(error);
  179. AndroidBridge.showToast(`导入失败: ${error.message}`);
  180. }
  181. }
  182. run();