intelligent-exam-generation.blade.php 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581
  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. @php
  385. $questionType = $question['question_type'] ?? ($question['type'] ?? '解答题');
  386. $isChoice = ($questionType === 'choice' || $questionType === '选择题');
  387. $options = $question['options'] ?? [];
  388. // 如果选项不在单独的字段中,尝试从题干中解析选项
  389. if ($isChoice && empty($options)) {
  390. $stem = $question['stem'] ?? '';
  391. preg_match_all('/([A-D])[\.\、\:]\s*(.+?)(?=[A-D][\.\、\:]|$)/', $stem, $matches);
  392. if (!empty($matches[1]) && !empty($matches[2])) {
  393. $parsedOptions = [];
  394. for ($i = 0; $i < count($matches[1]); $i++) {
  395. $parsedOptions[$matches[1][$i]] = trim($matches[2][$i]);
  396. }
  397. $options = $parsedOptions;
  398. }
  399. }
  400. // 确保有4个选项(必须显示A、B、C、D)
  401. if ($isChoice) {
  402. $standardOptions = ['A', 'B', 'C', 'D'];
  403. $fullOptions = [];
  404. $optionIndex = 0;
  405. foreach ($standardOptions as $key) {
  406. // 检查键值形式 (A, B, C, D)
  407. if (isset($options[$key]) && !empty($options[$key])) {
  408. $fullOptions[$key] = $options[$key];
  409. $optionIndex++;
  410. }
  411. // 检查数组索引形式 (0, 1, 2, 3)
  412. elseif (is_array($options) && isset($options[$optionIndex]) && !empty($options[$optionIndex])) {
  413. $fullOptions[$key] = $options[$optionIndex];
  414. $optionIndex++;
  415. }
  416. // 如果没有值,补充占位符
  417. else {
  418. // 根据选项字母生成占位符文本
  419. $placeholders = [
  420. 'A' => '(待补充选项A)',
  421. 'B' => '(待补充选项B)',
  422. 'C' => '(待补充选项C)',
  423. 'D' => '(待补充选项D)'
  424. ];
  425. $fullOptions[$key] = $placeholders[$key];
  426. }
  427. }
  428. $options = $fullOptions;
  429. }
  430. @endphp
  431. <div class="border rounded-lg p-4">
  432. <div class="flex items-start gap-4">
  433. <div class="flex-shrink-0 w-10 h-10 bg-blue-100 rounded-full flex items-center justify-center text-blue-700 font-semibold">
  434. {{ $index + 1 }}
  435. </div>
  436. <div class="flex-1">
  437. <div class="flex items-center gap-2 mb-2">
  438. <span class="text-sm px-2 py-0.5 bg-gray-100 text-gray-700 rounded">
  439. {{ $question['kp_code'] }}
  440. </span>
  441. <span class="text-sm px-2 py-0.5 bg-blue-100 text-blue-700 rounded">
  442. {{ $question['question_type'] ?? '解答题' }}
  443. </span>
  444. @if(isset($question['difficulty']))
  445. <span class="text-sm px-2 py-0.5
  446. @if($question['difficulty'] <= 0.3) bg-green-100 text-green-700
  447. @elseif($question['difficulty'] <= 0.7) bg-yellow-100 text-yellow-700
  448. @else bg-red-100 text-red-700
  449. @endif
  450. rounded">
  451. {{ $question['difficulty'] <= 0.3 ? '基础' : ($question['difficulty'] <= 0.7 ? '中等' : '拔高') }}
  452. </span>
  453. @endif
  454. </div>
  455. <div class="prose prose-sm max-w-none text-gray-900">
  456. {!! $question['stem'] !!}
  457. </div>
  458. {{-- 选择题选项展示 --}}
  459. @if($isChoice && !empty($options))
  460. @php
  461. // 判断选项长度,决定布局
  462. $maxOptionLength = 0;
  463. foreach ($options as $key => $option) {
  464. $text = is_string($option) ? strip_tags($option) : (string)$option;
  465. $length = mb_strlen($text);
  466. $maxOptionLength = max($maxOptionLength, $length);
  467. }
  468. // 如果最长选项不超过20个字符,可以考虑一行显示
  469. $shouldDisplayInline = $maxOptionLength <= 20;
  470. @endphp
  471. <div class="mt-3 space-y-2">
  472. @if($shouldDisplayInline)
  473. {{-- 短选项:一行显示 --}}
  474. <div class="grid grid-cols-2 gap-3">
  475. @foreach($options as $key => $option)
  476. <div class="flex items-start gap-2 p-2 bg-gray-50 rounded
  477. {{ strpos($option, '待补充') !== false ? 'opacity-60 border border-dashed border-gray-300' : '' }}">
  478. <span class="flex-shrink-0 w-6 h-6 bg-blue-600 text-white text-sm rounded-full flex items-center justify-center font-semibold">
  479. {{ $key }}
  480. </span>
  481. <span class="text-sm text-gray-900 break-words">
  482. {!! $option !!}
  483. </span>
  484. </div>
  485. @endforeach
  486. </div>
  487. @else
  488. {{-- 长选项:单独显示 --}}
  489. <div class="space-y-2">
  490. @foreach($options as $key => $option)
  491. <div class="flex items-start gap-3 p-3 bg-gray-50 rounded-lg hover:bg-gray-100 transition-colors
  492. {{ strpos($option, '待补充') !== false ? 'opacity-60 border border-dashed border-gray-300' : '' }}">
  493. <span class="flex-shrink-0 w-7 h-7 bg-blue-600 text-white text-sm rounded-full flex items-center justify-center font-semibold">
  494. {{ $key }}
  495. </span>
  496. <span class="text-sm text-gray-900 leading-relaxed">
  497. {!! $option !!}
  498. </span>
  499. </div>
  500. @endforeach
  501. </div>
  502. @endif
  503. </div>
  504. @endif
  505. </div>
  506. </div>
  507. </div>
  508. @endforeach
  509. </div>
  510. {{-- 参考答案部分 --}}
  511. <div class="mt-8 bg-white p-6 rounded-lg border shadow-sm border-emerald-200">
  512. <x-slot name="header">
  513. <div class="flex items-center gap-3">
  514. <div class="w-10 h-10 bg-emerald-100 rounded-lg flex items-center justify-center">
  515. <x-heroicon-o-document-text class="w-6 h-6 text-emerald-600" />
  516. </div>
  517. <div>
  518. <h3 class="text-xl font-bold text-gray-900">参考答案</h3>
  519. <p class="text-sm text-gray-500">{{ $paperName ?: '智能生成试卷' }}</p>
  520. </div>
  521. </div>
  522. </x-slot>
  523. <div class="space-y-4">
  524. @foreach($generatedQuestions as $index => $question)
  525. @if(!empty($question['answer']))
  526. <div class="flex items-start gap-4 p-4 bg-emerald-50 rounded-lg">
  527. <div class="flex-shrink-0 w-10 h-10 bg-emerald-600 text-white rounded-full flex items-center justify-center font-semibold">
  528. {{ $index + 1 }}
  529. </div>
  530. <div class="flex-1">
  531. <div class="text-sm text-gray-600 mb-1">
  532. <strong>{{ $question['question_type'] ?? '题目' }} · {{ $question['kp_code'] ?? '' }}</strong>
  533. </div>
  534. <div class="text-sm text-emerald-700 font-medium">
  535. <strong>答案:</strong> {!! $question['answer'] !!}
  536. </div>
  537. @if(isset($question['score']))
  538. <div class="text-xs text-gray-500 mt-1">
  539. 分值:{{ $question['score'] }}分
  540. </div>
  541. @endif
  542. </div>
  543. </div>
  544. @endif
  545. @endforeach
  546. </div>
  547. </div>
  548. </div>
  549. @endif
  550. </div>
  551. </x-filament-pages::page>