knowledge-points.blade.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. @php
  2. $stats = $this->stats;
  3. $points = $this->paginatedPoints['data'];
  4. $pagination = $this->paginatedPoints;
  5. $selectedPoint = $selectedPoint ?? $this->selectedPoint ?? null;
  6. $selectedSkills = isset($selectedPoint) ? collect($selectedPoint['skills'] ?? []) : collect();
  7. @endphp
  8. <x-filament-panels::page>
  9. <div class="space-y-8">
  10. <div class="grid gap-4 md:grid-cols-4">
  11. @foreach($stats as $stat)
  12. <div class="rounded-2xl border border-gray-200/80 bg-white/70 px-5 py-4 shadow-sm dark:border-gray-800 dark:bg-gray-900/40">
  13. <p class="text-sm text-gray-500">{{ $stat['label'] }}</p>
  14. <p class="mt-2 text-2xl font-semibold text-gray-900 dark:text-gray-100">{{ $stat['value'] }}</p>
  15. <p class="text-xs text-gray-400">{{ $stat['hint'] }}</p>
  16. </div>
  17. @endforeach
  18. </div>
  19. <div class="grid gap-6 lg:grid-cols-3">
  20. <div class="lg:col-span-1 space-y-6">
  21. <div class="rounded-2xl border border-gray-200 bg-white/80 px-4 py-3 shadow-sm dark:border-gray-800 dark:bg-gray-900/40">
  22. <div class="relative">
  23. <span class="pointer-events-none absolute inset-y-0 left-3 flex items-center text-gray-400">
  24. @svg('heroicon-m-magnifying-glass', 'h-4 w-4')
  25. </span>
  26. <input
  27. type="text"
  28. wire:model.debounce.500ms="search"
  29. placeholder="搜索知识点或编号..."
  30. class="w-full rounded-xl border border-gray-200 bg-white/80 py-2 pl-10 pr-3 text-sm placeholder:text-gray-400 focus:border-primary-300 focus:outline-none dark:border-gray-700 dark:bg-gray-900/60"
  31. />
  32. </div>
  33. <div class="mt-4 grid gap-3 md:grid-cols-2">
  34. <div class="relative">
  35. <span class="pointer-events-none absolute inset-y-0 left-3 flex items-center text-gray-400">
  36. @svg('heroicon-m-academic-cap', 'h-4 w-4')
  37. </span>
  38. <select
  39. wire:model="phaseFilter"
  40. class="w-full rounded-xl border border-gray-200 bg-white/80 py-2 pl-10 pr-3 text-sm focus:border-primary-300 focus:outline-none dark:border-gray-700 dark:bg-gray-900/60"
  41. >
  42. <option value="">全部学段</option>
  43. @foreach($this->phaseOptions as $value => $label)
  44. <option value="{{ $value }}">{{ $label }}</option>
  45. @endforeach
  46. </select>
  47. </div>
  48. <div class="relative">
  49. <span class="pointer-events-none absolute inset-y-0 left-3 flex items-center text-gray-400">
  50. @svg('heroicon-m-tag', 'h-4 w-4')
  51. </span>
  52. <select
  53. wire:model="categoryFilter"
  54. class="w-full rounded-xl border border-gray-200 bg-white/80 py-2 pl-10 pr-3 text-sm focus:border-primary-300 focus:outline-none dark:border-gray-700 dark:bg-gray-900/60"
  55. >
  56. <option value="">全部类别</option>
  57. @foreach($this->categoryOptions as $value => $label)
  58. <option value="{{ $value }}">{{ $label }}</option>
  59. @endforeach
  60. </select>
  61. </div>
  62. </div>
  63. </div>
  64. <div class="rounded-2xl border border-gray-200 bg-white/90 px-4 py-4 shadow-sm dark:border-gray-800 dark:bg-gray-900/40">
  65. <div class="flex items-center justify-between mb-4">
  66. <h3 class="text-sm font-semibold text-gray-900 dark:text-gray-100">知识点列表</h3>
  67. <p class="text-xs text-gray-500">{{ $pagination['total'] }} 个结果</p>
  68. </div>
  69. <div class="space-y-3">
  70. @forelse($points as $point)
  71. <button
  72. wire:click="selectPoint('{{ $point['kp_code'] }}')"
  73. class="w-full rounded-xl border px-4 py-3 text-left transition @if($selectedPoint && $selectedPoint['kp_code'] === $point['kp_code']) border-primary-500 bg-primary-50/70 dark:bg-primary-500/10 @else border-gray-200 hover:border-primary-200 dark:border-gray-800 @endif"
  74. >
  75. <p class="text-sm font-semibold text-gray-900 dark:text-gray-100">{{ $point['cn_name'] }}</p>
  76. <p class="text-xs text-gray-500">{{ $point['kp_code'] }} · {{ $point['phase'] ?? '未知学段' }}</p>
  77. <div class="mt-2 flex items-center gap-2 text-xs text-gray-500">
  78. <span class="inline-flex items-center rounded-full bg-gray-100 px-2 py-0.5 dark:bg-gray-800">重要度 {{ $point['importance'] ?? '-' }}</span>
  79. <span>{{ $point['category'] ?? '未分类' }}</span>
  80. </div>
  81. </button>
  82. @empty
  83. <p class="text-sm text-gray-500">暂无数据</p>
  84. @endforelse
  85. </div>
  86. <div class="mt-4 flex items-center justify-between text-xs text-gray-500">
  87. <button
  88. wire:click="changePage({{ $pagination['page'] - 1 }})"
  89. @disabled(!$pagination['has_prev'])"
  90. class="rounded-full border px-3 py-1 @if(!$pagination['has_prev']) opacity-40 cursor-not-allowed @endif"
  91. >
  92. 上一页
  93. </button>
  94. <span>第 {{ $pagination['page'] }} / {{ $pagination['total_pages'] }} 页</span>
  95. <button
  96. wire:click="changePage({{ $pagination['page'] + 1 }})"
  97. @disabled(!$pagination['has_next'])"
  98. class="rounded-full border px-3 py-1 @if(!$pagination['has_next']) opacity-40 cursor-not-allowed @endif"
  99. >
  100. 下一页
  101. </button>
  102. </div>
  103. </div>
  104. </div>
  105. <div class="lg:col-span-2 space-y-6">
  106. @if($selectedPoint)
  107. <div class="rounded-3xl bg-gradient-to-br from-primary-500 via-primary-600 to-indigo-600 px-6 py-6 text-white shadow-xl">
  108. <div class="flex flex-wrap items-start justify-between gap-4">
  109. <div class="space-y-1">
  110. <p class="text-xs uppercase tracking-[0.3em] text-white/80">Knowledge Node</p>
  111. <h1 class="text-3xl font-semibold">{{ $selectedPoint['cn_name'] }} <span class="text-sm font-normal text-white/70">({{ $selectedPoint['kp_code'] }})</span></h1>
  112. <div class="flex flex-wrap gap-2 text-sm text-white/90">
  113. <span class="inline-flex items-center gap-1 rounded-full bg-white/15 px-3 py-1">
  114. {{ $selectedPoint['category'] ?? '未分类' }}
  115. </span>
  116. <span class="inline-flex items-center gap-1 rounded-full bg-white/15 px-3 py-1">
  117. {{ $selectedPoint['phase'] ?? '未知学段' }} @if($selectedPoint['grade']) · {{ $selectedPoint['grade'] }} 年级 @endif
  118. </span>
  119. </div>
  120. </div>
  121. <div class="flex gap-4 text-right">
  122. <div>
  123. <p class="text-xs uppercase tracking-widest text-white/70">重要度</p>
  124. <p class="text-3xl font-semibold">{{ number_format((float) ($selectedPoint['importance'] ?? 0), 1) }}</p>
  125. <p class="text-xs text-white/80">教研权重</p>
  126. </div>
  127. <div>
  128. <p class="text-xs uppercase tracking-widest text-white/70">技能</p>
  129. <p class="text-3xl font-semibold">{{ $selectedSkills->count() }}</p>
  130. <p class="text-xs text-white/80">拆解单元</p>
  131. </div>
  132. </div>
  133. </div>
  134. </div>
  135. <div class="rounded-2xl border border-gray-200 bg-white/90 px-5 py-5 shadow-sm dark:border-gray-800 dark:bg-gray-900/50">
  136. <h3 class="text-sm font-semibold text-gray-600 dark:text-gray-300">知识路径</h3>
  137. <p class="mt-2 text-sm text-gray-900 dark:text-gray-100">{{ $selectedPoint['group_path'] ?? '尚未设置路径。' }}</p>
  138. <div class="mt-4 grid gap-4 lg:grid-cols-2">
  139. <div>
  140. <h4 class="text-sm font-semibold text-gray-600 dark:text-gray-300">描述 / 知识意图</h4>
  141. <p class="mt-2 rounded-xl bg-gray-50 px-4 py-3 text-sm text-gray-600 dark:bg-gray-800/40 dark:text-gray-300">
  142. {!! nl2br(e($selectedPoint['description'] ?? '暂无描述,建议补充课堂目标、典型错因等内容。')) !!}
  143. </p>
  144. </div>
  145. <div>
  146. <h4 class="text-sm font-semibold text-gray-600 dark:text-gray-300">先修关系</h4>
  147. <div class="mt-2 rounded-xl border border-gray-200 px-4 py-4 dark:border-gray-700">
  148. @if(!empty($selectedPoint['parents']))
  149. <ol class="relative border-l border-dashed border-primary-200 dark:border-primary-400/50 pl-4">
  150. @foreach($selectedPoint['parents'] as $parent)
  151. <li class="mb-4 ml-2">
  152. <span class="absolute -left-1.5 mt-1 h-3 w-3 rounded-full border-2 border-white bg-primary-400 dark:border-gray-900"></span>
  153. <p class="text-sm font-medium text-gray-900 dark:text-gray-100">{{ $parent }}</p>
  154. <p class="text-xs text-gray-500">触发该节点之前需掌握</p>
  155. </li>
  156. @endforeach
  157. </ol>
  158. @else
  159. <p class="text-sm text-gray-500">无父节点,可能是一级知识点。</p>
  160. @endif
  161. </div>
  162. </div>
  163. </div>
  164. </div>
  165. <div class="rounded-2xl border border-gray-200 bg-white/90 px-5 py-5 shadow-sm dark:border-gray-800 dark:bg-gray-900/50">
  166. <div class="flex items-center justify-between">
  167. <h3 class="text-base font-semibold text-gray-900 dark:text-gray-100">技能拆解</h3>
  168. <span class="text-xs text-gray-500">{{ $selectedSkills->count() }} 项</span>
  169. </div>
  170. @if($selectedSkills->isEmpty())
  171. <p class="mt-4 rounded-xl border border-dashed border-gray-300 px-4 py-6 text-center text-sm text-gray-500 dark:border-gray-700/60">
  172. 该知识点尚未配置技能,建议补充规则步骤、策略和应用场景,便于题库筛选。
  173. </p>
  174. @else
  175. <div class="mt-4 grid gap-4 md:grid-cols-2">
  176. @foreach($selectedSkills as $skill)
  177. <div class="rounded-2xl border border-gray-200 bg-white px-4 py-4 shadow-sm dark:border-gray-700/60 dark:bg-gray-900/40">
  178. <div class="flex items-start justify-between">
  179. <div>
  180. <p class="text-sm font-semibold text-gray-900 dark:text-gray-100">{{ $skill['skill_name'] ?? '未命名技能' }}</p>
  181. <p class="text-xs text-gray-500">{{ $skill['skill_type'] ?? '类型未知' }}</p>
  182. </div>
  183. <span class="inline-flex items-center gap-1 rounded-full bg-primary-50 px-2.5 py-0.5 text-xs font-medium text-primary-700 dark:bg-primary-500/10 dark:text-primary-300">
  184. 难度 {{ $skill['difficulty'] ?? '-' }}
  185. </span>
  186. </div>
  187. <p class="mt-3 text-sm leading-relaxed text-gray-600 dark:text-gray-300">
  188. {{ $skill['description'] ?? '暂无描述' }}
  189. </p>
  190. @if(!empty($skill['examples']))
  191. <div class="mt-3 rounded-lg bg-gray-50 px-3 py-2 text-xs text-gray-500 dark:bg-gray-800/50 dark:text-gray-300">
  192. <p class="font-medium text-gray-600 dark:text-gray-200">例题提示:</p>
  193. {{ $skill['examples'][0]['prompt'] ?? '' }}
  194. </div>
  195. @endif
  196. </div>
  197. @endforeach
  198. </div>
  199. @endif
  200. </div>
  201. @else
  202. <div class="rounded-2xl border border-dashed border-gray-300 px-6 py-12 text-center text-gray-500">
  203. 请选择左侧任意知识点查看详情。
  204. </div>
  205. @endif
  206. </div>
  207. </div>
  208. </div>
  209. </x-filament-panels::page>