student-client.d.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  1. type Store = {
  2. /** 当前用户 */
  3. user: {
  4. id: number;
  5. /** 身份证号 */
  6. identityNumber: string;
  7. /** 学号 */
  8. studentCodeList: string[];
  9. /** 显示的姓名 */
  10. displayName: string;
  11. /** @deprecated 姓名,但可能为空 */
  12. name: string;
  13. /** 学生手机号 */
  14. phoneNumber: string;
  15. /** 属于那个学校 */
  16. schoolDomain: SchoolDomain;
  17. /** 顶级机构id */
  18. rootOrgId: number;
  19. /** 底照URL */
  20. photoPath: string;
  21. /** 机构名称 */
  22. orgName: string;
  23. /** 用户的另一个id */
  24. key: string;
  25. /** 登录认证信息 */
  26. token: string;
  27. /** 专业 */
  28. specialty: string;
  29. };
  30. /** 学生端配置。 Q代表qmth */
  31. QECSConfig: {
  32. /** 学号登录别名 */
  33. STUDENT_CODE_LOGIN_ALIAS: string;
  34. /** 身份证号登录别名 */
  35. IDENTITY_NUMBER_LOGIN_ALIAS: string;
  36. /** 防作弊配置 !!!需对json进行转换!!! */
  37. PREVENT_CHEATING_CONFIG: Partial<
  38. [
  39. "FULL_SCREEN_TOP",
  40. "DISABLE_REMOTE_ASSISTANCE",
  41. "DISABLE_VIRTUAL_CAMERA",
  42. "DISABLE_MULTISCREEN"
  43. ]
  44. >;
  45. /** 学校是否有定制logo !!!需对json进行转换!!! */
  46. IS_CUSTOM_MENU_LOGO: boolean;
  47. /** 定制logo的URL */
  48. CUS_MENU_LOGO_FILE_URL: string;
  49. /** 学生端控制台设置 */
  50. STUDENT_CLIENT_CONSOLE_CONFIG: string;
  51. /** 首页背景图 */
  52. LOGO_FILE_URL: string;
  53. /** 产品名称,登录页显示 */
  54. OE_STUDENT_SYS_NAME: string;
  55. /** 登录页可选的登录类型 !!!需对json进行转换!!! */
  56. LOGIN_TYPE: Partial<["STUDENT_CODE", "IDENTITY_NUMBER"]>;
  57. ROOT_ORG_ID: string;
  58. /** @deprecated 登录支持的客户端类型。新版只支持Electron包。 */
  59. LOGIN_SUPPORT: Partial<["NATIVE", "BROWSER"]>;
  60. };
  61. /** 电脑时间管理 */
  62. sysTime: {
  63. /** 与服务器差异 */
  64. difference: number;
  65. /** 网络延迟 通过网络请求来判断 */
  66. rtt: number;
  67. /** 上次的Date.now() 有的电脑上存在Date.now()不更新 */
  68. lastTime: number;
  69. /** 判断时钟是否不走动了 */
  70. isTimerFrozen: boolean;
  71. };
  72. /** 网络状况 */
  73. network: {
  74. /** 网络状况是即时的,如何获取? */
  75. /** computed */
  76. isOK: boolean;
  77. /** 百度不通,则网络不通 */
  78. pingBaidu: boolean;
  79. /** 云平台api是否通畅 */
  80. pingQmth: boolean;
  81. /** 阿里云文件存储 阿里云日志 分开? */
  82. pingAli: boolean;
  83. /** websocket */
  84. pingWebSocket: boolean;
  85. };
  86. /** 菜单项 */
  87. menus: Array<{
  88. /** 菜单名称 */
  89. name: string;
  90. /** 菜单指向链接,是path */
  91. link: string;
  92. /** 此处要细化为 type = 'STU_NOTICE' | 'STU_MODIFY_PWD',在router中也要用 */
  93. routeCode: string;
  94. }>;
  95. /** app下载 */
  96. appDownload: {
  97. /** 是否开放下载 */
  98. enabled: boolean;
  99. /** 下载链接 */
  100. url: string;
  101. };
  102. /** 站内消息 */
  103. siteMessage: {
  104. messages: SiteMessage[];
  105. /** 第一个未读消息 popup用 computed */
  106. firstUnreadMessage: SiteMessage;
  107. /** 忽略的站内消息,不弹popup */
  108. ignoreMessageIds: number[];
  109. /** 未读消息总数 */
  110. unreadCount: number;
  111. };
  112. /** 在线考试待考列表 */
  113. examList: OnlineExam[];
  114. /** 在线考试已结束的考试列表 */
  115. endedExamList: OnlineExam[];
  116. /** 在线作业 */
  117. homeworkList: OnlineExam[];
  118. /** 在线练习 */
  119. practiceList: PracticeExam[];
  120. /** 当前开考的exam,非断点续考才会需要发出这么网络请求,正常开考是从前页面带过来的 */
  121. exam: {
  122. /** 考生的考试记录id */
  123. examRecordDataId: number;
  124. /** 考试批次id */
  125. examId: number;
  126. /** 课程名称 */
  127. courseName: string;
  128. /** 是否开启微信作答 */
  129. weixinAnswerEnabled: boolean;
  130. /** 是否开启人脸比对 */
  131. faceCheckEnabled: boolean;
  132. /** 是否开启活体检测 */
  133. faceLivenessEnabled: boolean;
  134. /** 活体检测选项 */
  135. faceLivenessOption: {
  136. /** 进入考试后多久开启活体检测(分钟) */
  137. faceVerifyMinute: number;
  138. /** 活体检测类型 S1: FaceID S2: 自研活体 */
  139. identificationOfLivingBodyScheme: "S1" | "S2";
  140. };
  141. /** 抓拍间隔(秒) */
  142. SNAPSHOT_INTERVAL: number;
  143. /** 考试冻结(秒) */
  144. FREEZE_TIME: number;
  145. /** 练习显示答案的类型 IN_PRACTICE:在练习过程中显示答案 NO_ANSWER:练习过程中不显示答案 */
  146. PRACTICE_TYPE: "IN_PRACTICE" | "NO_ANSWER";
  147. /** 试卷结构 */
  148. paperStruct: PaperStruct;
  149. /** 试题结构 */
  150. examQuestionList: ExamQuestion[];
  151. };
  152. // /** 考试中的状态 */
  153. // examing: {};
  154. // /** 获取到的camera source */
  155. camera: {
  156. /** 获取摄像头,且在多页面共享,减少打开摄像头失败的次数 */
  157. stream: MediaStream | null;
  158. };
  159. };
  160. type SchoolDomain = `${string}.ecs.qmth.com.cn`;
  161. type SiteMessage = { id: number; hasRead: boolean; content: string };
  162. type BaseExam = {
  163. /** 考试批次id */
  164. examId: number;
  165. /** 考生id,用户的每场考试都有一个考生id */
  166. examStudentId: number;
  167. /** 考生的考试记录id */
  168. examRecordDataId: number;
  169. /** 考试类型 */
  170. examType: "ONLINE" | "ONLINE_HOMEWORK" | "PRACTICE";
  171. /** 课程id */
  172. courseId: number;
  173. /** 课程名称 */
  174. courseName: string;
  175. /** 课程层次 */
  176. courseLevel: string;
  177. /** 专业名称 */
  178. specialtyName: string;
  179. /** 考试开始时间。日期时间字符串,以前叫后台改为数字,但后台不改,遗留的设计错误。 */
  180. startTime: string;
  181. /** 考试结束时间 */
  182. endTime: string;
  183. };
  184. type ExamCycle = {
  185. /** 考试周期是否开启循环 */
  186. examCycleEnabled: boolean;
  187. /** 周循环中周几开启考试 */
  188. examCycleWeek: (1 | 2 | 3 | 4 | 5 | 6 | 7)[];
  189. /** 周期循环中开启的时间段 HHmm [['08:00', '10:30'], ['15:00', '17:00']] */
  190. examCycleTimeRange: [string, string][];
  191. };
  192. type OnlineExam = BaseExam &
  193. ExamCycle & {
  194. /** 是否显示考生承诺书 */
  195. showUndertaking: boolean;
  196. /** 是否启用人脸比对 */
  197. faceEnable: boolean;
  198. /** 剩余考试次数 */
  199. allowExamCount: number;
  200. /** 是否允许查看客观分 */
  201. isObjScoreView: boolean;
  202. };
  203. type PracticeExam = BaseExam &
  204. ExamCycle & {
  205. /** 考试批次名称 */
  206. examName: string;
  207. /** 练习次数 */
  208. practiceCount: number;
  209. /** 剩余练习次数 */
  210. allowExamCount: number;
  211. /** 最近正确率(后端已乘100) */
  212. recentObjectiveAccuracy: number;
  213. /** 平均正确率(后端已乘100) */
  214. aveObjectiveAccuracy: number;
  215. /** 最高正确率(后端已乘100) */
  216. maxObjectiveAccuracy: number;
  217. };
  218. type OfflineExam = BaseExam & {
  219. /** 机构名称 */
  220. orgName: string;
  221. /** 考试周期是否开启循环 */
  222. offlineFiles: Array<{ offlineFileUrl: string; originalFileName: string }>;
  223. /** 试卷id */
  224. paperId: string;
  225. };
  226. type OnlinePracticeRecord = {
  227. id: number;
  228. /** 考试开始时间。日期时间字符串,以前叫后台改为数字,但后台不改,遗留的设计错误。 */
  229. startTime: string;
  230. /** 考试结束时间 */
  231. endTime: string;
  232. /** 练习时长 类型待确认??? */
  233. usedExamTime: any;
  234. /** 总题量 */
  235. totalQuestionCount: number;
  236. /** 正确的数量 */
  237. succQuestionNum: number;
  238. /** 错误的数量 */
  239. failQuestionNum: number;
  240. /** 未答的数量 */
  241. notAnsweredCount: number;
  242. /** 客观题正确率(后端已乘100) */
  243. objectiveAccuracy: number;
  244. };
  245. type OnlinePracticeRecordResult = {
  246. courseName: string;
  247. /** 客观题正确率(后端已乘100) */
  248. objectiveAccuracy: string;
  249. paperStructInfos: Array<{
  250. index: number;
  251. /** 题目分类 */
  252. title: string;
  253. /** 题量 */
  254. questionCount: number;
  255. /** 正确的数量 */
  256. succQuestionNum: number;
  257. /** 错误的数量 */
  258. failQuestionNum: number;
  259. /** 未答的数量 */
  260. notAnsweredCount: number;
  261. }>;
  262. };
  263. type PaperStruct = {
  264. defaultPaper: {
  265. /** index = mainNumber - 1 ??待确定 */
  266. questionGroupList: Array<{
  267. groupName: string;
  268. groupScore: number;
  269. questionWrapperList: Array<{
  270. questionId: string;
  271. body: string;
  272. /** 小题的列表,学生端用不着,只用到它的length */
  273. questionUnitWrapperList: Array<{ id: string }>;
  274. questionUnitList: Array<{
  275. eqs: ExamQuestion[];
  276. body: string;
  277. }>;
  278. }>;
  279. }>;
  280. };
  281. };
  282. type ExamQuestion = {
  283. /** 试题id */
  284. questionId: string;
  285. /** 题目序号 */
  286. order: number;
  287. /** 题目在答题中的序号。 old groupOrder */
  288. inGroupOrder: number;
  289. /** 限制音频播放次数。从paperStruct[][]['limitedPlayTimes']得到。前端添加。 */
  290. limitedPlayTimes: number;
  291. questionScore: number;
  292. /** 学生填写的答案,和C端、App端结构一致 */
  293. studentAnswer: string;
  294. /** 试题类型 */
  295. questionType:
  296. | "SINGLE_CHOICE"
  297. | "MULTIPLE_CHOICE"
  298. | "TRUE_OR_FALSE"
  299. | "FILL_UP"
  300. | "ESSAY";
  301. /** 小题乱序。细化??? */
  302. optionPermutation: number[];
  303. /** 大题名称 */
  304. groupName: string;
  305. /** 什么的总分??? */
  306. groupTotal: number;
  307. /** 是否被标上星号 */
  308. isStarred: boolean;
  309. /** 大题号 */
  310. mainNumber: number;
  311. /** 小题号 */
  312. subNumber: number;
  313. /** 试题内容。重点要重构音频、填空题###的处理逻辑 */
  314. body: string;
  315. /** 是否为套题。此处要梳理数据结构,重新计算!!! */
  316. isNestedQuestion: boolean;
  317. /** 文本作答的题目,是否开启音频作答。和考试的weixinAnswerEnabled有关。 */
  318. answerType: "SINGLE_AUDIO" | null;
  319. /** 练习时有正确答案 */
  320. rightAnswer?: boolean | number | string;
  321. /** 后端给的类型前端好像用不着,待确认??? */
  322. questionOptionList: any;
  323. /** 题目中是否有音频 */
  324. hasAudio: boolean;
  325. /** 试题内容。通过网络获取。 */
  326. questionContent: string;
  327. /** 只有第一题有此数据,用来像服务器保存音频播放次数 */
  328. audioPlayTimes: { audioName: string; times: number }[];
  329. };
  330. type ExamInProgress = {
  331. /** "S-101000" 请重试 */
  332. code: "000000" | "S-101000";
  333. /** 是否超过断点次数限制 */
  334. isExceed: boolean;
  335. /** 允许的最大断点次数 */
  336. maxInterruptNum: number;
  337. /** 允许的最大切屏次数 */
  338. maxSwitchScreenCount: number;
  339. examId: number;
  340. examRecordDataId: number;
  341. };
  342. // export const REMOTE_APPS = [
  343. // ["qq", "QQ"],
  344. // ["teamviewer", "TeamViewer"],
  345. // ["lookmypc", "LookMyPC"],
  346. // ["xt", "协通"],
  347. // ["winaw32", "Symantec PCAnywhere"],
  348. // ["pcaquickconnect", "Symantec PCAnywhere"],
  349. // ["sessioncontroller", "Symantec PCAnywhere"],
  350. // [/sunloginclient/gi, "向日葵"],
  351. // [/sunloginremote/gi, "向日葵"],
  352. // [/选择免安装运行,截图识别/gi, "向日葵"],
  353. // ["wemeetapp", "腾讯会议"],
  354. // ["wechat", "微信"],
  355. // ] as const;
  356. // define process.env
  357. // clientVersion.ts 管理可用客户端版本。
  358. // api: getCurrentClientVersion isSupportedClientVersion
  359. /** 得到当前客户端版本 1.9.* */
  360. type GetCurrentClientVersion = () => string;
  361. type IsSupportedClientVersion = () => boolean;
  362. // updateManager.ts 应用(非客户端)是否应该更新了
  363. // api: isNewWebAppAvailable isNewBackendAvailable getNewBackendDesc
  364. // native.ts 管理原生及系统命令,详细说明path和相对path
  365. // api: isElectron execCmd readFile getRemoteApps getVirtualCams
  366. // runtimeMonitor.ts 检测当前运行的环境是否合法
  367. // 检测是否打开了控制台;检测是否使用了高版本的chrome(feature检测)
  368. // 检测onResize;截屏
  369. // cache design
  370. // service worker ; image cache; audio cache/prefetch
  371. // router design 严格受控的页面跳转,既考虑刷新,也考虑缓存
  372. // localStorage design
  373. // loginType accountValue
  374. // sessionStorage design
  375. // store 防破解?仅保存 user, QECSConfig 信息
  376. // 开考流程
  377. // 1. 确保每一次请求的结果都得到充分利用,即在页面中保存好信息,不做多余的重试,不在中途刷新页面,除敏感信息外要从上个页面带到下个页面
  378. // 2. 开考和考试中的信息获取要隔离,方便从不同的页面中进入考试
  379. // 3. 进入页面后,不轻易退出,要做足够的重试。退出的条件要设置充分,比如网络断了。