student-analysis.blade.php 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <x-filament-panels::page>
  2. @push('styles')
  3. <style>
  4. .mastery-card {
  5. transition: all 0.3s ease;
  6. border-left: 4px solid transparent;
  7. }
  8. .mastery-card:hover {
  9. transform: translateX(4px);
  10. box-shadow: 0 4px 12px rgba(0,0,0,0.1);
  11. }
  12. .mastery-progress {
  13. height: 8px;
  14. border-radius: 4px;
  15. overflow: hidden;
  16. background: #e5e7eb;
  17. }
  18. .mastery-progress-bar {
  19. height: 100%;
  20. transition: width 0.5s ease;
  21. }
  22. .skill-badge {
  23. display: inline-block;
  24. padding: 4px 12px;
  25. margin: 4px;
  26. border-radius: 16px;
  27. font-size: 12px;
  28. font-weight: 500;
  29. }
  30. .weakness-tag {
  31. background: linear-gradient(135deg, #fee2e2 0%, #fecaca 100%);
  32. color: #991b1b;
  33. }
  34. .strength-tag {
  35. background: linear-gradient(135deg, #d1fae5 0%, #a7f3d0 100%);
  36. color: #065f46;
  37. }
  38. .chart-container {
  39. min-height: 300px;
  40. }
  41. </style>
  42. @endpush
  43. <div class="space-y-6">
  44. <!-- 页面标题 -->
  45. <div class="flex justify-between items-center">
  46. <div>
  47. <h2 class="text-2xl font-bold text-gray-900">学生掌握度分析</h2>
  48. <p class="mt-1 text-sm text-gray-500">
  49. 基于学生答题数据,分析知识点掌握情况并提供个性化学习建议
  50. </p>
  51. </div>
  52. <div class="flex gap-3">
  53. @if($selectedStudentId)
  54. button
  55. color="success"
  56. wire:click="generateStudyPlan"
  57. >
  58. <x-heroicon-m-academic-cap class="w-5 h-5 mr-2" />
  59. 生成学习计划
  60. /button>
  61. button
  62. color="gray"
  63. wire:click="exportAnalysis"
  64. >
  65. <x-heroicon-m-arrow-down-tray class="w-5 h-5 mr-2" />
  66. 导出报告
  67. /button>
  68. @endif
  69. </div>
  70. </div>
  71. <!-- 学生选择器 -->
  72. <x-filament::card>
  73. <div class="flex items-center gap-4">
  74. <div class="flex-1">
  75. <select
  76. wire:model.live="selectedStudentId"
  77. class="select select-bordered w-full"
  78. >
  79. <option value="">-- 选择学生 --</option>
  80. @foreach($this->students() as $student)
  81. <option value="{{ $student['student_id'] }}">
  82. {{ $student['name'] ?? $student['student_id'] }}
  83. </option>
  84. @endforeach
  85. </select>
  86. </div>
  87. @if($selectedStudentId)
  88. <div class="text-sm text-gray-600">
  89. 当前分析学生:<span class="font-semibold">{{ $selectedStudentId }}</span>
  90. </div>
  91. @endif
  92. </div>
  93. </x-filament::card>
  94. @if($selectedStudentId)
  95. <!-- 概览统计 -->
  96. <div class="grid grid-cols-1 md:grid-cols-4 gap-4">
  97. <x-filament::card class="text-center">
  98. <div class="text-3xl font-bold text-blue-600">
  99. {{ count($masteryData) }}
  100. </div>
  101. <div class="text-sm text-gray-600 mt-1">已学习知识点</div>
  102. </x-filament::card>
  103. <x-filament::card class="text-center">
  104. <div class="text-3xl font-bold text-green-600">
  105. {{ count(array_filter($masteryData, fn($m) => $m['mastery'] >= 0.8)) }}
  106. </div>
  107. <div class="text-sm text-gray-600 mt-1">掌握良好(≥80%)</div>
  108. </x-filament::card>
  109. <x-filament::card class="text-center">
  110. <div class="text-3xl font-bold text-red-600">
  111. {{ count($weaknesses) }}
  112. </div>
  113. <div class="text-sm text-gray-600 mt-1">薄弱知识点</div>
  114. </x-filament::card>
  115. <x-filament::card class="text-center">
  116. @php
  117. $avgMastery = count($masteryData) > 0
  118. ? round(array_sum(array_column($masteryData, 'mastery')) / count($masteryData) * 100, 1)
  119. : 0;
  120. @endphp
  121. <div class="text-3xl font-bold text-purple-600">{{ $avgMastery }}%</div>
  122. <div class="text-sm text-gray-600 mt-1">平均掌握度</div>
  123. </x-filament::card>
  124. </div>
  125. <div class="grid grid-cols-1 lg:grid-cols-2 gap-6">
  126. <!-- 掌握度详情 -->
  127. <x-filament::card>
  128. <x-slot name="header">
  129. <div class="flex items-center gap-3">
  130. <div class="w-10 h-10 bg-blue-100 rounded-lg flex items-center justify-center">
  131. <x-heroicon-o-chart-bar class="w-6 h-6 text-blue-600" />
  132. </div>
  133. <div>
  134. <h3 class="text-lg font-semibold text-gray-900">知识点掌握度详情</h3>
  135. <p class="text-sm text-gray-500">学生各知识点的掌握情况</p>
  136. </div>
  137. </div>
  138. </x-slot>
  139. @if(count($masteryData) > 0)
  140. <div class="space-y-4 max-h-96 overflow-y-auto">
  141. @foreach($masteryData as $item)
  142. <div class="mastery-card p-4 {{ getMasteryBgColor($item['mastery']) }}" style="border-left-color: {{ getMasteryColor($item['mastery']) }}">
  143. <div class="flex items-start justify-between mb-2">
  144. <div class="flex-1">
  145. <div class="font-medium text-gray-900">{{ $item['kp_name'] ?? $item['kp_code'] }}</div>
  146. <div class="text-xs text-gray-600 mt-1">{{ $item['kp_code'] }}</div>
  147. </div>
  148. <div class="text-right">
  149. <div class="text-lg font-bold" style="color: {{ getMasteryColor($item['mastery']) }}">
  150. {{ number_format($item['mastery'] * 100, 1) }}%
  151. </div>
  152. <div class="text-xs text-gray-600">{{ $item['mastery_level'] }}</div>
  153. </div>
  154. </div>
  155. <div class="mastery-progress">
  156. <div
  157. class="mastery-progress-bar"
  158. style="width: {{ $item['mastery'] * 100 }}%; background: {{ getMasteryColor($item['mastery']) }}"
  159. ></div>
  160. </div>
  161. @if($item['mastery'] < 0.7)
  162. <div class="mt-2 flex items-center gap-2">
  163. <span class="text-xs px-2 py-0.5 bg-red-200 text-red-700 rounded">
  164. 需重点关注
  165. </span>
  166. <span class="text-xs text-gray-600">
  167. 建议练习{{ ceil((0.8 - $item['mastery']) * 10) }}次
  168. </span>
  169. </div>
  170. @endif
  171. </div>
  172. @endforeach
  173. </div>
  174. @else
  175. <div class="text-center text-gray-500 py-8">
  176. 暂无掌握度数据
  177. </div>
  178. @endif
  179. </x-filament::card>
  180. <!-- 薄弱点分析 -->
  181. <x-filament::card>
  182. <x-slot name="header">
  183. <div class="flex items-center gap-3">
  184. <div class="w-10 h-10 bg-red-100 rounded-lg flex items-center justify-center">
  185. <x-heroicon-o-exclamation-triangle class="w-6 h-6 text-red-600" />
  186. </div>
  187. <div>
  188. <h3 class="text-lg font-semibold text-gray-900">薄弱点分析</h3>
  189. <p class="text-sm text-gray-500">需要加强练习的知识点</p>
  190. </div>
  191. </div>
  192. </x-slot>
  193. @if(count($weaknesses) > 0)
  194. <div class="space-y-3">
  195. @foreach($weaknesses as $weakness)
  196. <div class="border border-red-200 rounded-lg p-3 bg-red-50">
  197. <div class="flex items-start justify-between">
  198. <div class="flex-1">
  199. <div class="font-medium text-gray-900">
  200. {{ $weakness['kp_name'] ?? $weakness['kp_code'] }}
  201. </div>
  202. <div class="text-xs text-gray-600 mt-1">{{ $weakness['kp_code'] }}</div>
  203. </div>
  204. <div class="text-right">
  205. <div class="text-sm font-bold text-red-600">
  206. {{ number_format($weakness['mastery'] * 100, 1) }}%
  207. </div>
  208. </div>
  209. </div>
  210. <div class="mt-3 space-y-2">
  211. <div class="flex items-center gap-2">
  212. <span class="text-xs text-gray-600">掌握度:</span>
  213. <div class="flex-1 bg-gray-200 rounded-full h-2">
  214. <div
  215. class="bg-red-500 h-2 rounded-full"
  216. style="width: {{ $weakness['mastery'] * 100 }}%"
  217. ></div>
  218. </div>
  219. </div>
  220. <div class="text-xs text-gray-600">
  221. <strong>建议:</strong>
  222. @if($weakness['mastery'] < 0.3)
  223. 重新学习基础知识,从基础题开始练习
  224. @elseif($weakness['mastery'] < 0.5)
  225. 重点练习基础题型,复习相关概念
  226. @elseif($weakness['mastery'] < 0.7)
  227. 加强练习,增加题型熟悉度
  228. @endif
  229. </div>
  230. </div>
  231. </div>
  232. @endforeach
  233. </div>
  234. @else
  235. <div class="text-center text-green-600 py-8">
  236. <x-heroicon-o-check-circle class="w-12 h-12 mx-auto mb-2" />
  237. <div>学生表现优秀,暂无敌弱知识点!</div>
  238. </div>
  239. @endif
  240. </x-filament::card>
  241. </div>
  242. <!-- 优势与建议 -->
  243. <div class="grid grid-cols-1 lg:grid-cols-2 gap-6">
  244. <!-- 优势知识点 -->
  245. <x-filament::card>
  246. <x-slot name="header">
  247. <div class="flex items-center gap-3">
  248. <div class="w-10 h-10 bg-green-100 rounded-lg flex items-center justify-center">
  249. <x-heroicon-o-star class="w-6 h-6 text-green-600" />
  250. </div>
  251. <div>
  252. <h3 class="text-lg font-semibold text-gray-900">优势知识点</h3>
  253. <p class="text-sm text-gray-500">掌握良好的知识点</p>
  254. </div>
  255. </div>
  256. </x-slot>
  257. @php
  258. $strengths = array_filter($masteryData, fn($m) => $m['mastery'] >= 0.8);
  259. $strengths = array_slice($strengths, 0, 10);
  260. @endphp
  261. @if(count($strengths) > 0)
  262. <div class="flex flex-wrap gap-2">
  263. @foreach($strengths as $strength)
  264. <span class="skill-badge strength-tag">
  265. {{ $strength['kp_name'] ?? $strength['kp_code'] }}
  266. ({{ number_format($strength['mastery'] * 100, 0) }}%)
  267. </span>
  268. @endforeach
  269. </div>
  270. <div class="mt-4 p-3 bg-green-50 rounded-lg">
  271. <div class="text-sm text-green-800">
  272. <strong>分析:</strong>学生在这些知识点上表现优秀,可以作为学习其他内容的支撑点。
  273. </div>
  274. </div>
  275. @else
  276. <div class="text-center text-gray-500 py-4">
  277. 暂无优势知识点
  278. </div>
  279. @endif
  280. </x-filament::card>
  281. <!-- 学习建议 -->
  282. <x-filament::card>
  283. <x-slot name="header">
  284. <div class="flex items-center gap-3">
  285. <div class="w-10 h-10 bg-purple-100 rounded-lg flex items-center justify-center">
  286. <x-heroicon-o-light-bulb class="w-6 h-6 text-purple-600" />
  287. </div>
  288. <div>
  289. <h3 class="text-lg font-semibold text-gray-900">学习建议</h3>
  290. <p class="text-sm text-gray-500">个性化学习路径</p>
  291. </div>
  292. </div>
  293. </x-slot>
  294. @if(count($weaknesses) > 0)
  295. <div class="space-y-4">
  296. <div class="p-3 bg-amber-50 rounded-lg">
  297. <div class="text-sm font-medium text-amber-800 mb-2">📚 学习优先级</div>
  298. <ol class="text-sm text-amber-700 space-y-1 list-decimal list-inside">
  299. @foreach(array_slice($weaknesses, 0, 5) as $idx => $weakness)
  300. <li>
  301. {{ $weakness['kp_name'] ?? $weakness['kp_code'] }}
  302. - 建议练习{{ ceil((0.8 - $weakness['mastery']) * 15) }}道题
  303. </li>
  304. @endforeach
  305. </ol>
  306. </div>
  307. <div class="p-3 bg-blue-50 rounded-lg">
  308. <div class="text-sm font-medium text-blue-800 mb-2">🎯 学习策略</div>
  309. <ul class="text-sm text-blue-700 space-y-1 list-disc list-inside">
  310. <li>先巩固基础知识,再提升难度</li>
  311. <li>每天练习{{ min(10, ceil(50 / count($weaknesses))) }}道薄弱知识点题目</li>
  312. <li>定期复习已掌握的知识点</li>
  313. <li>结合优势知识点,尝试综合性题目</li>
  314. </ul>
  315. </div>
  316. </div>
  317. @else
  318. <div class="text-center text-gray-500 py-4">
  319. 学生掌握情况良好,建议继续提升挑战难度
  320. </div>
  321. @endif
  322. </x-filament::card>
  323. </div>
  324. @else
  325. <!-- 未选择学生的提示 -->
  326. <x-filament::card class="text-center py-12">
  327. <x-heroicon-o-users class="w-16 h-16 text-gray-400 mx-auto mb-4" />
  328. <div class="text-lg font-medium text-gray-900 mb-2">请选择学生</div>
  329. <div class="text-sm text-gray-500">
  330. 从上方下拉菜单中选择一个学生,查看详细的掌握度分析
  331. </div>
  332. </x-filament::card>
  333. @endif
  334. </div>
  335. </x-filament-pages::page>