CFEC.js 7.0 KB

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