intelligent-exam-generation.blade.php 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448
  1. <x-filament-panels::page>
  2. @push('styles')
  3. <style>
  4. .exam-card {
  5. transition: all 0.3s ease;
  6. }
  7. .exam-card:hover {
  8. transform: translateY(-2px);
  9. box-shadow: 0 10px 25px rgba(0,0,0,0.1);
  10. }
  11. .skill-tag {
  12. display: inline-block;
  13. padding: 4px 12px;
  14. margin: 4px;
  15. background: #e0f2fe;
  16. color: #0369a1;
  17. border-radius: 12px;
  18. font-size: 12px;
  19. }
  20. .difficulty-indicator {
  21. width: 100%;
  22. height: 8px;
  23. border-radius: 4px;
  24. overflow: hidden;
  25. }
  26. .difficulty-easy { background: #86efac; }
  27. .difficulty-medium { background: #fde047; }
  28. .difficulty-hard { background: #fca5a5; }
  29. </style>
  30. @endpush
  31. <div class="space-y-6">
  32. <!-- 页面标题和操作 -->
  33. <div class="flex justify-between items-center">
  34. <div>
  35. <h2 class="text-2xl font-bold text-gray-900">智能出卷系统</h2>
  36. <p class="mt-1 text-sm text-gray-500">
  37. 基于知识点掌握度和技能依赖关系,智能生成个性化试卷
  38. </p>
  39. </div>
  40. <div class="flex gap-3">
  41. button
  42. color="gray"
  43. wire:click="resetForm"
  44. >
  45. 重置
  46. /button>
  47. </div>
  48. </div>
  49. <!-- 主要内容区 -->
  50. <div class="grid grid-cols-1 lg:grid-cols-3 gap-6">
  51. <!-- 左侧:配置表单 -->
  52. <div class="lg:col-span-2 space-y-6">
  53. <!-- 基本信息 -->
  54. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  55. <x-slot name="header">
  56. <div class="flex items-center gap-3">
  57. <div class="w-10 h-10 bg-blue-100 rounded-lg flex items-center justify-center">
  58. <x-heroicon-o-document-text class="w-6 h-6 text-blue-600" />
  59. </div>
  60. <div>
  61. <h3 class="text-lg font-semibold text-gray-900">基本信息</h3>
  62. <p class="text-sm text-gray-500">设置试卷名称、难度和题目数量</p>
  63. </div>
  64. </div>
  65. </x-slot>
  66. <div class="space-y-4">
  67. <input
  68. wire:model="paperName"
  69. label="试卷名称"
  70. placeholder="例如:因式分解专项练习(基础版)"
  71. required
  72. />
  73. <textarea
  74. wire:model="paperDescription"
  75. label="试卷描述"
  76. placeholder="描述本试卷的特点、适用对象等(可选)"
  77. rows="3"
  78. />
  79. <div class="grid grid-cols-3 gap-4">
  80. <select
  81. wire:model="difficultyCategory"
  82. label="难度分类"
  83. >
  84. <option value="基础">基础</option>
  85. <option value="进阶">进阶</option>
  86. <option value="竞赛">竞赛</option>
  87. /select>
  88. <input
  89. wire:model="totalQuestions"
  90. type="number"
  91. label="题目数量"
  92. min="5"
  93. max="100"
  94. required
  95. />
  96. <input
  97. wire:model="totalScore"
  98. type="number"
  99. label="总分"
  100. min="0"
  101. max="200"
  102. />
  103. </div>
  104. </div>
  105. </div>
  106. <!-- 知识点选择 -->
  107. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  108. <x-slot name="header">
  109. <div class="flex items-center justify-between">
  110. <div class="flex items-center gap-3">
  111. <div class="w-10 h-10 bg-green-100 rounded-lg flex items-center justify-center">
  112. <x-heroicon-o-academic-cap class="w-6 h-6 text-green-600" />
  113. </div>
  114. <div>
  115. <h3 class="text-lg font-semibold text-gray-900">知识点选择</h3>
  116. <p class="text-sm text-gray-500">选择要考查的知识点(可多选)</p>
  117. </div>
  118. </div>
  119. <div class="text-sm text-gray-500">
  120. 已选择: {{ count($selectedKpCodes) }} 个
  121. </div>
  122. </div>
  123. </x-slot>
  124. <div class="space-y-3">
  125. <div class="grid grid-cols-1 md:grid-cols-2 gap-3 max-h-64 overflow-y-auto">
  126. @foreach($this->knowledgePoints as $kp)
  127. <label class="flex items-start gap-3 p-3 border rounded-lg hover:bg-gray-50 cursor-pointer">
  128. <input type="checkbox"
  129. wire:model="selectedKpCodes"
  130. value="{{ $kp['kp_code'] }}"
  131. class="mt-1"
  132. />
  133. <div class="flex-1">
  134. <div class="font-medium text-gray-900">{{ $kp['cn_name'] ?? $kp['kp_code'] }}</div>
  135. @if(!empty($kp['description']))
  136. <div class="text-sm text-gray-500 mt-1">{{ Str::limit($kp['description'], 80) }}</div>
  137. @endif
  138. <div class="flex items-center gap-2 mt-2">
  139. <span class="text-xs px-2 py-0.5 bg-blue-100 text-blue-700 rounded">
  140. {{ $kp['kp_code'] }}
  141. </span>
  142. @if(!empty($kp['level']))
  143. <span class="text-xs px-2 py-0.5 bg-gray-100 text-gray-700 rounded">
  144. Level {{ $kp['level'] }}
  145. </span>
  146. @endif
  147. </div>
  148. </div>
  149. </label>
  150. @endforeach
  151. </div>
  152. </div>
  153. </div>
  154. <!-- 技能点选择 -->
  155. @if(count($this->skills) > 0)
  156. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  157. <x-slot name="header">
  158. <div class="flex items-center justify-between">
  159. <div class="flex items-center gap-3">
  160. <div class="w-10 h-10 bg-purple-100 rounded-lg flex items-center justify-center">
  161. <x-heroicon-o-adjustments-horizontal class="w-6 h-6 text-purple-600" />
  162. </div>
  163. <div>
  164. <h3 class="text-lg font-semibold text-gray-900">技能点选择</h3>
  165. <p class="text-sm text-gray-500">根据知识点自动获取相关技能点</p>
  166. </div>
  167. </div>
  168. </div>
  169. </x-slot>
  170. <div class="space-y-3">
  171. <div class="flex flex-wrap gap-2">
  172. @foreach($this->skills as $skill)
  173. <label class="skill-tag cursor-pointer">
  174. <input type="checkbox"
  175. wire:model="selectedSkills"
  176. value="{{ $skill['skill_name'] }}"
  177. class="sr-only"
  178. />
  179. {{ $skill['skill_name'] }}
  180. </label>
  181. @endforeach
  182. </div>
  183. </div>
  184. </div>
  185. @endif
  186. <!-- 题型配比 -->
  187. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  188. <x-slot name="header">
  189. <div class="flex items-center gap-3">
  190. <div class="w-10 h-10 bg-yellow-100 rounded-lg flex items-center justify-center">
  191. <x-heroicon-o-chart-pie class="w-6 h-6 text-yellow-600" />
  192. </div>
  193. <div>
  194. <h3 class="text-lg font-semibold text-gray-900">题型配比</h3>
  195. <p class="text-sm text-gray-500">设置各类题型的比例(总和为100%)</p>
  196. </div>
  197. </div>
  198. </x-slot>
  199. <div class="space-y-3">
  200. @foreach($questionTypeRatio as $type => $percentage)
  201. <div class="flex items-center gap-4">
  202. <div class="w-24 text-sm font-medium text-gray-700">{{ $type }}</div>
  203. <div class="flex-1">
  204. <input
  205. type="range"
  206. min="0"
  207. max="100"
  208. wire:model="questionTypeRatio.{{ $type }}"
  209. class="w-full"
  210. />
  211. </div>
  212. <div class="w-16 text-sm text-gray-600">{{ $percentage }}%</div>
  213. </div>
  214. @endforeach
  215. <div class="text-xs text-gray-500">
  216. 总计: {{ array_sum($questionTypeRatio) }}%
  217. @if(array_sum($questionTypeRatio) !== 100)
  218. <span class="text-red-500 ml-2">(应为100%)</span>
  219. @endif
  220. </div>
  221. </div>
  222. </div>
  223. <!-- 难度配比 -->
  224. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  225. <x-slot name="header">
  226. <div class="flex items-center gap-3">
  227. <div class="w-10 h-10 bg-indigo-100 rounded-lg flex items-center justify-center">
  228. <x-heroicon-o-signal class="w-6 h-6 text-indigo-600" />
  229. </div>
  230. <div>
  231. <h3 class="text-lg font-semibold text-gray-900">难度配比</h3>
  232. <p class="text-sm text-gray-500">设置各难度题目的比例</p>
  233. </div>
  234. </div>
  235. </x-slot>
  236. <div class="space-y-3">
  237. @foreach($difficultyRatio as $level => $percentage)
  238. <div class="flex items-center gap-4">
  239. <div class="w-24 text-sm font-medium text-gray-700">{{ $level }}</div>
  240. <div class="flex-1">
  241. <input
  242. type="range"
  243. min="0"
  244. max="100"
  245. wire:model="difficultyRatio.{{ $level }}"
  246. class="w-full"
  247. />
  248. </div>
  249. <div class="w-16 text-sm text-gray-600">{{ $percentage }}%</div>
  250. </div>
  251. @endforeach
  252. <div class="text-xs text-gray-500">
  253. 总计: {{ array_sum($difficultyRatio) }}%
  254. @if(array_sum($difficultyRatio) !== 100)
  255. <span class="text-red-500 ml-2">(应为100%)</span>
  256. @endif
  257. </div>
  258. </div>
  259. </div>
  260. </div>
  261. <!-- 右侧:操作面板 -->
  262. <div class="space-y-6">
  263. <!-- 学生选择(可选) -->
  264. <div class="bg-white p-6 rounded-lg border shadow-sm" class="exam-card">
  265. <x-slot name="header">
  266. <div class="flex items-center gap-3">
  267. <div class="w-10 h-10 bg-pink-100 rounded-lg flex items-center justify-center">
  268. <x-heroicon-o-user class="w-6 h-6 text-pink-600" />
  269. </div>
  270. <div>
  271. <h3 class="text-lg font-semibold text-gray-900">个性化设置</h3>
  272. <p class="text-sm text-gray-500">根据学生情况定制</p>
  273. </div>
  274. </div>
  275. </x-slot>
  276. <div class="space-y-4">
  277. <select
  278. wire:model="selectedStudentId"
  279. label="选择学生(可选)"
  280. placeholder="不指定则生成通用试卷"
  281. >
  282. <option value="">-- 不指定 --</option>
  283. @foreach($this->students as $student)
  284. <option value="{{ $student['student_id'] }}">
  285. {{ $student['name'] ?? $student['student_id'] }}
  286. </option>
  287. @endforeach
  288. /select>
  289. <label class="flex items-start gap-3">
  290. <input type="checkbox"
  291. wire:model="filterByStudentWeakness"
  292. wire:click="$refresh"
  293. />
  294. <div>
  295. <div class="text-sm font-medium text-gray-900">基于学生薄弱点</div>
  296. <div class="text-xs text-gray-500">
  297. 根据学生历史答题数据,自动筛选其薄弱知识点
  298. </div>
  299. </div>
  300. </label>
  301. @if(count($this->studentWeaknesses) > 0)
  302. <div class="mt-4 p-3 bg-amber-50 rounded-lg">
  303. <div class="text-sm font-medium text-amber-800 mb-2">检测到学生的薄弱点:</div>
  304. <div class="space-y-1">
  305. @foreach($this->studentWeaknesses as $weakness)
  306. <div class="text-xs text-amber-700 flex items-center gap-2">
  307. <span>{{ $weakness['kp_name'] ?? $weakness['kp_code'] }}</span>
  308. <span class="text-amber-600">
  309. (掌握度: {{ number_format($weakness['mastery'] * 100, 1) }}%)
  310. </span>
  311. </div>
  312. @endforeach
  313. </div>
  314. </div>
  315. @endif
  316. </div>
  317. </div>
  318. <!-- 生成按钮 -->
  319. <div class="bg-white p-6 rounded-lg border shadow-sm">
  320. <div class="space-y-4">
  321. button
  322. wire:click="generateExam"
  323. color="primary"
  324. class="w-full"
  325. size="lg"
  326. :disabled="$isGenerating"
  327. >
  328. @if($isGenerating)
  329. <svg class="animate-spin -ml-1 mr-3 h-5 w-5 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
  330. <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
  331. <path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
  332. </svg>
  333. 生成中...
  334. @else
  335. <x-heroicon-m-sparkles class="w-5 h-5 mr-2" />
  336. 智能生成试卷
  337. @endif
  338. /button>
  339. @if($generatedPaperId)
  340. button
  341. wire:click="exportToPdf"
  342. color="success"
  343. class="w-full"
  344. size="lg"
  345. >
  346. <x-heroicon-m-arrow-down-tray class="w-5 h-5 mr-2" />
  347. 导出PDF
  348. /button>
  349. @endif
  350. @if(!empty($generatedQuestions))
  351. <div class="mt-4 p-4 bg-green-50 rounded-lg">
  352. <div class="flex items-center gap-2 text-green-800">
  353. <x-heroicon-o-check-circle class="w-5 h-5" />
  354. <div class="font-medium">生成成功</div>
  355. </div>
  356. <div class="mt-2 text-sm text-green-700">
  357. 已生成试卷ID: <span class="font-mono">{{ $generatedPaperId }}</span>
  358. </div>
  359. <div class="mt-1 text-sm text-green-700">
  360. 题目数量: {{ count($generatedQuestions) }} 题
  361. </div>
  362. </div>
  363. @endif
  364. </div>
  365. </div>
  366. </div>
  367. </div>
  368. <!-- 生成的试卷预览 -->
  369. @if(!empty($generatedQuestions))
  370. <div class="bg-white p-6 rounded-lg border shadow-sm" class="mt-6">
  371. <x-slot name="header">
  372. <div class="flex items-center gap-3">
  373. <div class="w-10 h-10 bg-emerald-100 rounded-lg flex items-center justify-center">
  374. <x-heroicon-o-document-magnifying-glass class="w-6 h-6 text-emerald-600" />
  375. </div>
  376. <div>
  377. <h3 class="text-lg font-semibold text-gray-900">试卷预览</h3>
  378. <p class="text-sm text-gray-500">生成的题目列表</p>
  379. </div>
  380. </div>
  381. </x-slot>
  382. <div class="space-y-4">
  383. @foreach($generatedQuestions as $index => $question)
  384. <div class="border rounded-lg p-4">
  385. <div class="flex items-start gap-4">
  386. <div class="flex-shrink-0 w-10 h-10 bg-blue-100 rounded-full flex items-center justify-center text-blue-700 font-semibold">
  387. {{ $index + 1 }}
  388. </div>
  389. <div class="flex-1">
  390. <div class="flex items-center gap-2 mb-2">
  391. <span class="text-sm px-2 py-0.5 bg-gray-100 text-gray-700 rounded">
  392. {{ $question['kp_code'] }}
  393. </span>
  394. <span class="text-sm px-2 py-0.5 bg-blue-100 text-blue-700 rounded">
  395. {{ $question['question_type'] ?? '解答题' }}
  396. </span>
  397. @if(isset($question['difficulty']))
  398. <span class="text-sm px-2 py-0.5
  399. @if($question['difficulty'] <= 0.3) bg-green-100 text-green-700
  400. @elseif($question['difficulty'] <= 0.7) bg-yellow-100 text-yellow-700
  401. @else bg-red-100 text-red-700
  402. @endif
  403. rounded">
  404. {{ $question['difficulty'] <= 0.3 ? '基础' : ($question['difficulty'] <= 0.7 ? '中等' : '拔高') }}
  405. </span>
  406. @endif
  407. </div>
  408. <div class="prose prose-sm max-w-none text-gray-900">
  409. {!! $question['stem'] !!}
  410. </div>
  411. @if(!empty($question['answer']))
  412. <div class="mt-2 text-sm text-gray-600">
  413. <strong>参考答案:</strong> {!! $question['answer'] !!}
  414. </div>
  415. @endif
  416. </div>
  417. </div>
  418. </div>
  419. @endforeach
  420. </div>
  421. </div>
  422. @endif
  423. </div>
  424. </x-filament-pages::page>