question-management.blade.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. <div>
  2. <div class="space-y-6">
  3. @php
  4. $questionsData = $this->questions;
  5. $metaData = $this->meta;
  6. $statisticsData = $this->statistics;
  7. @endphp
  8. {{-- 后台生成状态栏 - 仅在生成中显示 --}}
  9. @if($isGenerating && $currentTaskId)
  10. <div class="alert alert-info shadow-lg animate-pulse">
  11. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" class="stroke-current shrink-0 w-6 h-6"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"></path></svg>
  12. <div>
  13. <h3 class="font-bold">正在后台生成题目...</h3>
  14. <div class="text-xs">任务 ID: {{ $currentTaskId }} | AI生成完成后将自动刷新页面</div>
  15. </div>
  16. <button type="button" wire:click="$set('isGenerating', false)" class="btn btn-sm btn-ghost">关闭</button>
  17. </div>
  18. @endif
  19. <div class="flex justify-end">
  20. <button
  21. type="button"
  22. wire:click="$dispatch('ai-generate')"
  23. class="btn btn-primary"
  24. >
  25. <svg class="w-5 h-5 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
  26. <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4"></path>
  27. </svg>
  28. 生成题目
  29. </button>
  30. </div>
  31. {{-- 难度统计卡片 --}}
  32. @php
  33. // 显示统计数据的标签
  34. $statsLabel = $this->selectedKpCode ? "知识点 {$this->selectedKpCode}" : "全部题目";
  35. $displayStats = $statisticsData;
  36. @endphp
  37. <div class="grid grid-cols-1 md:grid-cols-4 gap-4">
  38. <div class="stats shadow bg-base-100 border">
  39. <div class="stat">
  40. <div class="stat-title">题目总数</div>
  41. <div class="stat-value text-primary">{{ $displayStats['total'] ?? 0 }}</div>
  42. <div class="stat-desc">当前题库总量</div>
  43. </div>
  44. </div>
  45. <div class="stats shadow bg-base-100 border">
  46. <div class="stat">
  47. <div class="stat-title">简单题 (≤0.4)</div>
  48. <div class="stat-value text-success">
  49. @php
  50. $basicCount = 0;
  51. foreach ($displayStats['by_difficulty'] ?? [] as $key => $value) {
  52. if ((float)$key <= 0.4) {
  53. $basicCount += $value;
  54. }
  55. }
  56. echo $basicCount;
  57. @endphp
  58. </div>
  59. </div>
  60. </div>
  61. <div class="stats shadow bg-base-100 border">
  62. <div class="stat">
  63. <div class="stat-title">中等题 (0.4-0.7)</div>
  64. <div class="stat-value text-warning">
  65. @php
  66. $mediumCount = 0;
  67. foreach ($displayStats['by_difficulty'] ?? [] as $key => $value) {
  68. if ((float)$key > 0.4 && (float)$key <= 0.7) {
  69. $mediumCount += $value;
  70. }
  71. }
  72. echo $mediumCount;
  73. @endphp
  74. </div>
  75. </div>
  76. </div>
  77. <div class="stats shadow bg-base-100 border">
  78. <div class="stat">
  79. <div class="stat-title">拔高题 (>0.7)</div>
  80. <div class="stat-value text-error">
  81. @php
  82. $advancedCount = 0;
  83. foreach ($displayStats['by_difficulty'] ?? [] as $key => $value) {
  84. if ((float)$key > 0.7) {
  85. $advancedCount += $value;
  86. }
  87. }
  88. echo $advancedCount;
  89. @endphp
  90. </div>
  91. </div>
  92. </div>
  93. </div>
  94. {{-- 筛选区域 --}}
  95. <div class="card bg-base-100 shadow-sm border">
  96. <div class="card-body p-4">
  97. <div class="grid grid-cols-1 md:grid-cols-5 gap-4">
  98. <div class="form-control">
  99. <label class="label"><span class="label-text">搜索题目</span></label>
  100. <input type="text" wire:model.live.debounce.300ms="search" placeholder="输入关键词" class="input input-bordered w-full input-sm">
  101. </div>
  102. <div class="form-control">
  103. <label class="label"><span class="label-text">知识点筛选</span></label>
  104. <input type="text" wire:model.live="selectedKpCode" placeholder="KP1001" class="input input-bordered w-full input-sm">
  105. </div>
  106. <div class="form-control">
  107. <label class="label"><span class="label-text">难度筛选</span></label>
  108. <select wire:model.live="selectedDifficulty" class="select select-bordered w-full select-sm">
  109. <option value="">全部难度</option>
  110. <option value="0.3">简单 (0.3)</option>
  111. <option value="0.6">中等 (0.6)</option>
  112. <option value="0.85">拔高 (0.85)</option>
  113. </select>
  114. </div>
  115. <div class="form-control">
  116. <label class="label"><span class="label-text">题目类型</span></label>
  117. <select wire:model.live="selectedType" class="select select-bordered w-full select-sm">
  118. <option value="">全部类型</option>
  119. @foreach($this->questionTypeOptions as $value => $label)
  120. <option value="{{ $value }}">{{ $label }}</option>
  121. @endforeach
  122. </select>
  123. </div>
  124. <div class="form-control">
  125. <label class="label"><span class="label-text">每页显示</span></label>
  126. <select wire:model.live="perPage" class="select select-bordered w-full select-sm">
  127. <option value="10">10 条</option>
  128. <option value="25">25 条</option>
  129. <option value="50">50 条</option>
  130. <option value="100">100 条</option>
  131. </select>
  132. </div>
  133. </div>
  134. </div>
  135. </div>
  136. {{-- 题目列表 --}}
  137. <div class="overflow-x-auto bg-base-100 rounded-lg shadow border">
  138. <table class="table table-zebra w-full">
  139. <thead>
  140. <tr>
  141. <th>题目编号</th>
  142. <th>知识点</th>
  143. <th>类型</th>
  144. <th>题干</th>
  145. <th>难度</th>
  146. <th>操作</th>
  147. </tr>
  148. </thead>
  149. <tbody>
  150. @forelse($questionsData as $question)
  151. <tr class="hover">
  152. <td class="font-mono text-xs">{{ $question['question_code'] ?? 'N/A' }}</td>
  153. <td>
  154. <div class="badge badge-ghost">{{ $question['kp_code'] ?? 'N/A' }}</div>
  155. </td>
  156. <td>
  157. @php
  158. $type = $question['type'] ?? 'CHOICE';
  159. $typeLabel = $this->questionTypeOptions[$type] ?? $type;
  160. $typeClass = match($type) {
  161. 'CHOICE', 'MULTIPLE_CHOICE' => 'badge-info',
  162. 'FILL_IN_THE_BLANK' => 'badge-warning',
  163. 'CALCULATION', 'WORD_PROBLEM', 'PROOF' => 'badge-error',
  164. default => 'badge-ghost'
  165. };
  166. @endphp
  167. <div class="badge {{ $typeClass }} badge-outline text-xs">{{ $typeLabel }}</div>
  168. </td>
  169. <td class="max-w-md">
  170. <div class="prose prose-sm max-w-none">
  171. <x-math-render :content="\Illuminate\Support\Str::limit($question['stem'] ?? 'N/A', 150)" class="text-sm" />
  172. </div>
  173. </td>
  174. <td>
  175. @php
  176. $difficulty = $question['difficulty'] ?? null;
  177. $label = match (true) {
  178. !$difficulty => 'N/A',
  179. (float)$difficulty <= 0.4 => '基础',
  180. (float)$difficulty <= 0.7 => '中等',
  181. default => '拔高',
  182. };
  183. $colorClass = match (true) {
  184. !$difficulty => 'badge-ghost',
  185. (float)$difficulty <= 0.4 => 'badge-success',
  186. (float)$difficulty <= 0.7 => 'badge-warning',
  187. default => 'badge-error',
  188. };
  189. @endphp
  190. <div class="badge {{ $colorClass }} gap-1">
  191. {{ $label }}
  192. @if(app()->environment('local'))
  193. <span class="text-[10px] opacity-70">({{ $difficulty }})</span>
  194. @endif
  195. </div>
  196. </td>
  197. <td class="space-x-2">
  198. <a href="{{ url('/admin/question-detail') }}?question_id={{ urlencode($question['id'] ?? $question['question_code'] ?? '') }}" class="btn btn-ghost btn-xs text-primary">
  199. 查看
  200. </a>
  201. <button
  202. wire:click="deleteQuestion('{{ $question['question_code'] }}')"
  203. wire:confirm="确定要删除这道题目吗?此操作不可恢复。"
  204. class="btn btn-outline btn-xs text-error border-error"
  205. >
  206. 删除
  207. </button>
  208. </td>
  209. </tr>
  210. @empty
  211. <tr><td colspan="6" class="text-center py-10 text-gray-500">暂无数据</td></tr>
  212. @endforelse
  213. </tbody>
  214. </table>
  215. </div>
  216. {{-- 分页 --}}
  217. @if(!empty($metaData) && ($metaData['total'] ?? 0) > 0)
  218. <div class="flex justify-between items-center bg-base-100 p-4 rounded-lg border shadow-sm">
  219. <div class="text-sm text-gray-500">共 {{ $metaData['total'] ?? 0 }} 条记录</div>
  220. <div class="join">
  221. <button class="join-item btn btn-sm" wire:click="previousPage" @disabled($currentPage <= 1)>«</button>
  222. @foreach($this->getPages() as $page)
  223. <button
  224. class="join-item btn btn-sm {{ $page === $currentPage ? 'btn-active btn-primary' : '' }}"
  225. wire:click="gotoPage({{ $page }})"
  226. >
  227. {{ $page }}
  228. </button>
  229. @endforeach
  230. <button class="join-item btn btn-sm" wire:click="nextPage" @disabled($currentPage >= ($metaData['total_pages'] ?? 1))>»</button>
  231. </div>
  232. </div>
  233. @endif
  234. {{-- 生成模态框 --}}
  235. @if($showGenerateModal)
  236. <div class="modal modal-open">
  237. <div class="modal-box w-11/12 max-w-2xl">
  238. <h3 class="font-bold text-lg mb-6">智能题目生成</h3>
  239. <div class="grid grid-cols-1 md:grid-cols-2 gap-6">
  240. <div class="form-control w-full">
  241. <label class="label"><span class="label-text font-semibold">知识点 <span class="text-error">*</span></span></label>
  242. <select wire:model.live="generateKpCode" class="select select-bordered w-full">
  243. <option value="">请选择知识点</option>
  244. @foreach($this->knowledgePointOptions as $code => $name)
  245. <option value="{{ $code }}">{{ $code }} - {{ $name }}</option>
  246. @endforeach
  247. </select>
  248. </div>
  249. <div class="form-control w-full">
  250. <label class="label"><span class="label-text font-semibold">题目数量</span></label>
  251. <input type="number" wire:model="questionCount" min="1" max="500" class="input input-bordered w-full">
  252. </div>
  253. <div class="form-control w-full">
  254. <label class="label"><span class="label-text font-semibold">难度偏好</span></label>
  255. <select wire:model="generateDifficulty" class="select select-bordered w-full">
  256. <option value="">随机难度</option>
  257. <option value="0.3">简单 (0.3)</option>
  258. <option value="0.6">中等 (0.6)</option>
  259. <option value="0.85">拔高 (0.85)</option>
  260. </select>
  261. </div>
  262. <div class="form-control w-full">
  263. <label class="label"><span class="label-text font-semibold">题目类型</span></label>
  264. <select wire:model="generateType" class="select select-bordered w-full">
  265. <option value="">随机类型</option>
  266. @foreach($this->questionTypeOptions as $value => $label)
  267. <option value="{{ $value }}">{{ $label }}</option>
  268. @endforeach
  269. </select>
  270. </div>
  271. </div>
  272. @if(!empty($this->skillsOptions))
  273. <div class="divider">关联技能</div>
  274. <div class="form-control">
  275. <div class="flex justify-between items-center mb-2">
  276. <label class="label-text font-semibold">选择技能 <span class="text-error">*</span></label>
  277. <button type="button" class="btn btn-xs btn-ghost text-primary" wire:click="toggleAllSkills">
  278. {{ count($selectedSkills) === count($this->skillsOptions) ? '取消全选' : '全选' }}
  279. </button>
  280. </div>
  281. <div class="grid grid-cols-2 gap-2 max-h-48 overflow-y-auto p-2 border rounded-lg bg-base-50">
  282. @foreach($this->skillsOptions as $skill)
  283. <label class="label cursor-pointer justify-start gap-2 hover:bg-base-200 rounded p-1">
  284. <input type="checkbox" value="{{ $skill['code'] }}" wire:model="selectedSkills" class="checkbox checkbox-primary checkbox-sm">
  285. <span class="label-text text-xs">
  286. <span class="font-bold">{{ $skill['code'] }}</span>
  287. {{ $skill['name'] }}
  288. </span>
  289. </label>
  290. @endforeach
  291. </div>
  292. </div>
  293. @elseif($generateKpCode)
  294. <div class="alert alert-warning mt-4">
  295. <svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z" /></svg>
  296. <span>该知识点下暂无关联技能,无法生成题目。</span>
  297. </div>
  298. @endif
  299. <div class="modal-action">
  300. <button type="button" wire:click="closeGenerateModal" class="btn" @disabled($isGenerating)>取消</button>
  301. <button
  302. type="button"
  303. wire:click="executeGenerate"
  304. class="btn btn-primary"
  305. @disabled($isGenerating || empty($selectedSkills))
  306. >
  307. @if($isGenerating)
  308. <span class="loading loading-spinner"></span>
  309. 生成中...
  310. @else
  311. 开始生成
  312. @endif
  313. </button>
  314. </div>
  315. </div>
  316. </div>
  317. @endif
  318. <script>
  319. document.addEventListener('livewire:init', () => {
  320. Livewire.on('ai-generate', () => {
  321. @this.call('openGenerateModal');
  322. });
  323. Livewire.on('refresh-page', () => {
  324. document.dispatchEvent(new Event('math:render'));
  325. });
  326. // 监听页面刷新事件
  327. Livewire.on('refresh-page', () => {
  328. console.log('[QuestionGen] 收到刷新页面事件');
  329. setTimeout(() => {
  330. window.location.reload();
  331. }, 1000);
  332. });
  333. // 任务监控逻辑
  334. Livewire.on('start-async-task-monitoring', () => {
  335. console.log('[QuestionGen] 开始监控任务状态');
  336. const taskId = @this.currentTaskId;
  337. if (!taskId) return;
  338. window.currentTaskId = taskId;
  339. let checkCount = 0;
  340. const maxChecks = 10; // 增加检查次数
  341. function checkCallbackStatus() {
  342. checkCount++;
  343. console.log(`[QuestionGen] 检查回调 #${checkCount}/${maxChecks}`);
  344. fetch(`/api/questions/callback/${taskId}`, {
  345. method: 'GET',
  346. headers: {
  347. 'X-Requested-With': 'XMLHttpRequest',
  348. 'Accept': 'application/json',
  349. }
  350. })
  351. .then(response => response.json())
  352. .then(data => {
  353. if (data.status) {
  354. if (data.status === 'completed') {
  355. @this.set('isGenerating', false);
  356. @this.set('currentTaskId', null);
  357. setTimeout(() => window.location.reload(), 1000);
  358. } else if (data.status === 'failed') {
  359. @this.set('isGenerating', false);
  360. @this.set('currentTaskId', null);
  361. }
  362. } else if (checkCount < maxChecks) {
  363. setTimeout(checkCallbackStatus, 3000);
  364. } else {
  365. @this.set('isGenerating', false);
  366. @this.set('currentTaskId', null);
  367. }
  368. })
  369. .catch(error => {
  370. if (checkCount < maxChecks) {
  371. setTimeout(checkCallbackStatus, 3000);
  372. }
  373. });
  374. }
  375. checkCallbackStatus();
  376. });
  377. });
  378. </script>
  379. </div>
  380. </div>