knowledge-graph-explorer.blade.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. <div>
  2. <div class="space-y-4">
  3. {{-- 操作切换条(保留单一标题区) --}}
  4. <div class="flex items-center gap-2">
  5. <a
  6. href="{{ url('admin/knowledge-graph-integration') }}"
  7. class="inline-flex items-center gap-2 px-3 py-2 text-sm font-medium text-gray-700 bg-white border border-gray-300 rounded-lg hover:bg-gray-50 dark:text-gray-300 dark:bg-gray-800 dark:border-gray-700 dark:hover:bg-gray-700"
  8. >
  9. <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
  10. <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 10h16M4 14h16M4 18h16"></path>
  11. </svg>
  12. 综合视图
  13. </a>
  14. <a
  15. href="{{ url('admin/knowledge-mindmap') }}"
  16. class="inline-flex items-center gap-2 px-3 py-2 text-sm font-medium text-gray-700 bg-white border border-gray-300 rounded-lg hover:bg-gray-50 dark:text-gray-300 dark:bg-gray-800 dark:border-gray-700 dark:hover:bg-gray-700"
  17. >
  18. <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
  19. <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 10h16M4 14h16M4 18h16"></path>
  20. </svg>
  21. 原始脑图
  22. </a>
  23. </div>
  24. {{-- 知识图谱可视化区域 --}}
  25. <div
  26. class="rounded-2xl border border-slate-200 shadow-sm bg-white text-slate-900 knowledge-mindmap-card"
  27. x-data="{
  28. graphInstance: null,
  29. stats: { nodes: 0, extraEdges: 0 },
  30. livewireId: '{{ $this->getId() }}',
  31. selectedNode: null,
  32. initEventListener() {
  33. window.addEventListener('mindmap-node-selected', (evt) => {
  34. const model = evt.detail || {};
  35. const code = model?.meta?.code || model?.id;
  36. const mastery = model?.meta?.mastery_level ?? this.graphInstance?.masteryData?.[code]?.mastery_level ?? 0;
  37. const accuracy = model?.meta?.accuracy_rate ?? this.graphInstance?.masteryData?.[code]?.accuracy_rate ?? null;
  38. const attempts = model?.meta?.total_attempts ?? this.graphInstance?.masteryData?.[code]?.total_attempts ?? null;
  39. this.selectedNode = {
  40. id: model?.id,
  41. code,
  42. label: model?.label,
  43. mastery,
  44. accuracy,
  45. attempts,
  46. recommended: model?.meta?.recommended ?? false,
  47. skills: model?.meta?.skills || this.graphInstance?.masteryData?.[code]?.skills || [],
  48. };
  49. // 不再需要通知Livewire,直接在Alpine中处理
  50. // selectedNode 已在上面设置完成
  51. });
  52. },
  53. async initMindmap() {
  54. try {
  55. // 等待G6和自定义组件加载
  56. await this.waitForComponents();
  57. if (!window.G6 || !window.KnowledgeMindmapGraph) {
  58. console.error('G6组件未加载');
  59. return;
  60. }
  61. this.graphInstance = new window.KnowledgeMindmapGraph({
  62. containerId: 'knowledge-graph-viz',
  63. livewireMethod: null, // 不使用自动调用,由Alpine手动处理
  64. livewireId: this.livewireId,
  65. emitSelection: true,
  66. highlightLowMastery: false,
  67. });
  68. await this.graphInstance.init();
  69. this.stats = this.graphInstance.stats;
  70. } catch (error) {
  71. console.error('知识图谱初始化失败:', error);
  72. }
  73. },
  74. async waitForComponents() {
  75. let attempts = 0;
  76. const maxAttempts = 50;
  77. while ((!window.G6 || !window.KnowledgeMindmapGraph) && attempts < maxAttempts) {
  78. await new Promise(resolve => setTimeout(resolve, 100));
  79. attempts++;
  80. }
  81. if (attempts >= maxAttempts) {
  82. throw new Error('G6组件加载超时');
  83. }
  84. }
  85. }"
  86. x-init="initEventListener(); initMindmap()"
  87. data-knowledge-mindmap-root
  88. >
  89. <div class="relative overflow-hidden rounded-2xl border border-slate-200 shadow-sm bg-white text-slate-900">
  90. <div
  91. wire:ignore
  92. id="knowledge-graph-viz"
  93. class="knowledge-mindmap-canvas relative h-[82vh] min-h-[720px] w-full"
  94. >
  95. <div class="pointer-events-none absolute inset-0 bg-[radial-gradient(circle_at_20%_20%,rgba(255,255,255,0.08),transparent_40%),radial-gradient(circle_at_80%_60%,rgba(255,255,255,0.06),transparent_45%)]"></div>
  96. </div>
  97. </div>
  98. {{-- 选中节点信息 --}}
  99. <template x-if="selectedNode">
  100. <div class="mt-4 grid grid-cols-1 gap-4 lg:grid-cols-3">
  101. <div class="lg:col-span-2 rounded-xl border border-slate-200 bg-white p-4 shadow-sm">
  102. <div class="flex items-start justify-between">
  103. <div>
  104. <p class="text-sm text-slate-500">选中知识点</p>
  105. <h3 class="text-xl font-semibold text-slate-900" x-text="selectedNode.label || selectedNode.code"></h3>
  106. <p class="text-sm text-slate-500 mt-1" x-text="`ID: ${selectedNode.id || ''} · Code: ${selectedNode.code || ''}`"></p>
  107. </div>
  108. <div class="text-right">
  109. <p class="text-sm text-slate-500">掌握度</p>
  110. <p class="text-2xl font-bold" x-text="(selectedNode.mastery * 100).toFixed(1) + '%'"></p>
  111. <p class="text-xs text-slate-500" x-show="selectedNode.accuracy">准确率 <span x-text="(selectedNode.accuracy * 100).toFixed(1) + '%'"></span></p>
  112. <p class="text-xs text-slate-500" x-show="selectedNode.attempts">练习次数 <span x-text="selectedNode.attempts"></span></p>
  113. </div>
  114. </div>
  115. <div class="mt-4">
  116. <p class="text-sm font-semibold text-slate-700 mb-2">技能要点</p>
  117. <div class="flex flex-wrap gap-2">
  118. <template x-if="selectedNode.skills && selectedNode.skills.length">
  119. <template x-for="skill in selectedNode.skills" :key="skill">
  120. <span class="rounded-full bg-slate-100 px-3 py-1 text-xs font-medium text-slate-700" x-text="skill"></span>
  121. </template>
  122. </template>
  123. <span x-show="!selectedNode.skills || !selectedNode.skills.length" class="text-sm text-slate-500">暂无技能要点</span>
  124. </div>
  125. <div class="mt-4">
  126. <a
  127. class="inline-flex items-center rounded-md bg-indigo-600 px-3 py-2 text-sm font-medium text-white hover:bg-indigo-700"
  128. :href="selectedNode.code ? `/admin/knowledge-point-detail?kp_code=${encodeURIComponent(selectedNode.code)}` : '#'"
  129. target="_blank"
  130. >
  131. 查看知识点详情
  132. </a>
  133. </div>
  134. </div>
  135. </div>
  136. <div class="rounded-xl border border-slate-200 bg-white p-4 shadow-sm space-y-2">
  137. <p class="text-sm font-semibold text-slate-700">快速信息</p>
  138. <div class="text-sm text-slate-600 space-y-1">
  139. <p><span class="font-medium text-slate-900">ID:</span><span x-text="selectedNode.id"></span></p>
  140. <p><span class="font-medium text-slate-900">Code:</span><span x-text="selectedNode.code"></span></p>
  141. <p><span class="font-medium text-slate-900">推荐关注:</span><span x-text="selectedNode.recommended ? '是' : '否'"></span></p>
  142. </div>
  143. </div>
  144. </div>
  145. </template>
  146. </div>
  147. {{-- 知识点详情面板 --}}
  148. <template x-if="selectedNode && selectedNode.code !== 'root'">
  149. <div class="bg-white dark:bg-gray-800 rounded-lg border border-gray-200 dark:border-gray-700 overflow-hidden">
  150. {{-- 知识点标题 --}}
  151. <div class="p-6 border-b border-gray-200 dark:border-gray-700">
  152. <div class="flex items-center justify-between">
  153. <div>
  154. <h3 class="text-xl font-bold text-gray-900 dark:text-gray-100" x-text="selectedNode.label || selectedNode.code"></h3>
  155. <div class="mt-1 flex items-center gap-2 text-sm text-gray-500 dark:text-gray-400">
  156. <span x-text="selectedNode.code"></span>
  157. </div>
  158. </div>
  159. </div>
  160. </div>
  161. {{-- 统计概览 --}}
  162. <div class="p-6 border-b border-gray-200 dark:border-gray-700">
  163. <div class="grid grid-cols-2 md:grid-cols-4 gap-4">
  164. <div class="text-center">
  165. <div class="text-2xl font-bold text-blue-600 dark:text-blue-400">-</div>
  166. <div class="text-sm text-gray-500 dark:text-gray-400 mt-1">总题目数</div>
  167. </div>
  168. <div class="text-center">
  169. <div class="text-2xl font-bold text-green-600 dark:text-green-400">-</div>
  170. <div class="text-sm text-gray-500 dark:text-gray-400 mt-1">直接题目</div>
  171. </div>
  172. <div class="text-center">
  173. <div class="text-2xl font-bold text-yellow-600 dark:text-yellow-400">-</div>
  174. <div class="text-sm text-gray-500 dark:text-gray-400 mt-1">子知识点题目</div>
  175. </div>
  176. <div class="text-center">
  177. <div class="text-2xl font-bold text-purple-600 dark:text-purple-400">-</div>
  178. <div class="text-sm text-gray-500 dark:text-gray-400 mt-1">技能点题目</div>
  179. </div>
  180. </div>
  181. </div>
  182. {{-- 标签页导航 --}}
  183. <div class="px-6 border-b border-gray-200 dark:border-gray-700">
  184. <nav class="flex space-x-8" aria-label="Tabs">
  185. <span class="border-indigo-500 text-indigo-600 dark:text-indigo-400 py-4 px-1 border-b-2 font-medium text-sm">
  186. 知识点详情
  187. </span>
  188. </nav>
  189. </div>
  190. {{-- 标签页内容 --}}
  191. <div class="p-6">
  192. <div class="space-y-4">
  193. <div>
  194. <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">技能要点</h4>
  195. <div class="flex flex-wrap gap-2">
  196. <template x-if="selectedNode.skills && selectedNode.skills.length">
  197. <template x-for="skill in selectedNode.skills" :key="skill">
  198. <span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-indigo-100 text-indigo-800 dark:bg-indigo-900 dark:text-indigo-200" x-text="skill"></span>
  199. </template>
  200. </template>
  201. <span x-show="!selectedNode.skills || !selectedNode.skills.length" class="text-sm text-gray-500 dark:text-gray-400">暂无技能要点</span>
  202. </div>
  203. </div>
  204. </div>
  205. </div>
  206. </div>
  207. </template>
  208. {{-- 统计信息 --}}
  209. <div class="bg-white dark:bg-gray-800 rounded-lg border border-gray-200 dark:border-gray-700 p-4">
  210. <div class="flex items-center justify-between">
  211. <div class="flex items-center gap-2">
  212. <svg class="w-5 h-5 text-indigo-600 dark:text-indigo-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
  213. <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>
  214. </svg>
  215. <span class="text-sm font-medium text-gray-700 dark:text-gray-300">使用说明</span>
  216. </div>
  217. <div class="flex gap-4 text-sm text-gray-600 dark:text-gray-400">
  218. <div class="flex items-center gap-1">
  219. <div class="w-3 h-3 rounded-full bg-blue-500"></div>
  220. <span>知识点</span>
  221. </div>
  222. <div class="flex items-center gap-1">
  223. <div class="w-3 h-3 rounded-full bg-green-500"></div>
  224. <span>已掌握</span>
  225. </div>
  226. <div class="flex items-center gap-1">
  227. <div class="w-3 h-3 rounded-full bg-yellow-500"></div>
  228. <span>需加强</span>
  229. </div>
  230. </div>
  231. </div>
  232. </div>
  233. </div>
  234. </div>
  235. @push('styles')
  236. <style>
  237. .knowledge-mindmap-canvas {
  238. background: #ffffff;
  239. color: #0f172a;
  240. }
  241. .g6-grid {
  242. opacity: 0.08;
  243. }
  244. </style>
  245. <script src="https://unpkg.com/@antv/g6@4.8.24/dist/g6.min.js"></script>
  246. @endpush
  247. @push('scripts')
  248. {{-- 加载现有的知识图谱可视化类 --}}
  249. <script src="{{ asset('js/g6-custom-node.js') }}"></script>
  250. <script src="{{ asset('js/knowledge-mindmap-graph.js') }}"></script>
  251. <script>
  252. document.addEventListener('livewire:initialized', () => {
  253. // 监听错误事件
  254. Livewire.on('error', (event) => {
  255. console.error('Error:', event.message);
  256. });
  257. // 节点选择事件
  258. Livewire.on('nodeSelected', (event) => {
  259. console.log('Node selected:', event);
  260. });
  261. });
  262. </script>
  263. @endpush