huel_01.js 7.0 KB

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