search.html 64 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454
  1. <!DOCTYPE html>
  2. <html lang="zh-CN">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>知识检索 — 知识检索中心</title>
  7. <style>
  8. *, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
  9. :root {
  10. --primary: #2563EB;
  11. --primary-light: #EFF6FF;
  12. --primary-hover: #1D4ED8;
  13. --bg: #F1F5F9;
  14. --surface: #FFFFFF;
  15. --border: #E2E8F0;
  16. --border-focus: #A5B4FC;
  17. --text: #0F172A;
  18. --text-sub: #475569;
  19. --text-muted: #94A3B8;
  20. --radius: 8px;
  21. --substance-bg: #EFF6FF; --substance: #1D4ED8;
  22. --form-bg: #F5F3FF; --form: #6D28D9;
  23. --intent-bg: #F0FDF4; --intent: #15803D;
  24. --effect-bg: #FFF7ED; --effect: #C2410C;
  25. --feeling-bg: #FDF2F8; --feeling: #BE185D;
  26. --dim-bg: #F8FAFC; --dim: #64748B;
  27. }
  28. body {
  29. font-family: -apple-system, BlinkMacSystemFont, 'PingFang SC', 'Microsoft YaHei', sans-serif;
  30. background: var(--bg);
  31. color: var(--text);
  32. font-size: 14px;
  33. line-height: 1.5;
  34. }
  35. /* ── 搜索栏 ───────────────────────────────────────────────────── */
  36. .search-section {
  37. position: sticky; top: 0; z-index: 199;
  38. background: var(--surface);
  39. border-bottom: 1px solid var(--border);
  40. padding: 11px 16px;
  41. }
  42. .search-inner { max-width: 1400px; margin: 0 auto; display: flex; gap: 8px; }
  43. .search-input-wrap { flex: 0 1 680px; min-width: 240px; position: relative; }
  44. .search-icon {
  45. position: absolute; left: 10px; top: 50%;
  46. transform: translateY(-50%);
  47. color: var(--text-muted); pointer-events: none;
  48. display: flex; align-items: center;
  49. }
  50. .search-input {
  51. width: 100%; height: 38px;
  52. padding: 0 12px 0 34px;
  53. border: 1.5px solid var(--border); border-radius: 6px;
  54. font-size: 14px; outline: none;
  55. background: var(--bg); color: var(--text);
  56. transition: border-color .15s, background .15s;
  57. }
  58. .search-input:focus { border-color: var(--border-focus); background: #fff; }
  59. .btn-primary {
  60. height: 38px; padding: 0 18px;
  61. background: var(--primary); color: #fff;
  62. border: none; border-radius: 6px;
  63. font-size: 14px; font-weight: 500; cursor: pointer;
  64. transition: background .15s; white-space: nowrap;
  65. }
  66. .btn-primary:hover { background: var(--primary-hover); }
  67. /* ── 布局 ─────────────────────────────────────────────────────── */
  68. .layout {
  69. display: flex;
  70. max-width: 1400px; margin: 0 auto;
  71. padding: 16px; gap: 14px;
  72. align-items: flex-start;
  73. }
  74. /* ── 侧栏 ─────────────────────────────────────────────────────── */
  75. .sidebar {
  76. width: 232px; flex-shrink: 0;
  77. position: sticky; top: 126px;
  78. max-height: calc(100vh - 132px);
  79. overflow-y: auto;
  80. }
  81. .sidebar::-webkit-scrollbar { width: 3px; }
  82. .sidebar::-webkit-scrollbar-thumb { background: var(--border); border-radius: 2px; }
  83. .filter-card {
  84. background: var(--surface);
  85. border: 1px solid var(--border); border-radius: var(--radius);
  86. overflow: hidden;
  87. }
  88. .filter-section { border-bottom: 1px solid var(--border); }
  89. .filter-section:last-child { border-bottom: none; }
  90. .section-hd {
  91. display: flex; align-items: center; justify-content: space-between;
  92. padding: 10px 13px; cursor: pointer; user-select: none;
  93. font-size: 13px; font-weight: 600; color: var(--text);
  94. transition: background .1s;
  95. }
  96. .section-hd:hover { background: var(--bg); }
  97. .section-hd .chevron { color: var(--text-muted); font-size: 10px; transition: transform .2s; }
  98. .section-hd.open .chevron { transform: rotate(180deg); }
  99. .section-bd { display: none; padding: 2px 13px 10px; }
  100. .section-bd.open { display: block; }
  101. .sub-group { margin-bottom: 10px; }
  102. .sub-label {
  103. font-size: 10px; font-weight: 700; letter-spacing: .6px;
  104. text-transform: uppercase; color: var(--text-muted);
  105. padding: 5px 0 3px;
  106. }
  107. .semantic-input {
  108. width: 100%; height: 26px; padding: 0 8px;
  109. border: 1px solid var(--border); border-radius: 4px;
  110. font-size: 12px; outline: none; color: var(--text);
  111. background: var(--bg); margin-bottom: 5px;
  112. transition: border-color .15s;
  113. }
  114. .semantic-input:focus { border-color: var(--border-focus); background: #fff; }
  115. .semantic-input::placeholder { color: var(--text-muted); }
  116. .opt {
  117. display: flex; align-items: center; gap: 6px;
  118. padding: 3px 0; cursor: pointer;
  119. font-size: 13px; color: var(--text-sub);
  120. }
  121. .opt:hover { color: var(--primary); }
  122. .opt input[type="checkbox"] {
  123. accent-color: var(--primary);
  124. width: 13px; height: 13px; cursor: pointer; flex-shrink: 0;
  125. }
  126. .opt-text { flex: 1; }
  127. .opt-count {
  128. font-size: 11px; color: var(--text-muted);
  129. background: var(--bg); border-radius: 8px;
  130. padding: 0 5px; min-width: 20px; text-align: center;
  131. }
  132. .range-row { display: flex; align-items: center; gap: 5px; margin-top: 5px; }
  133. .range-inp {
  134. flex: 1; height: 28px; padding: 0 7px;
  135. border: 1px solid var(--border); border-radius: 4px;
  136. font-size: 12px; outline: none; color: var(--text); min-width: 0;
  137. }
  138. .range-inp:focus { border-color: var(--border-focus); }
  139. .range-sep { color: var(--text-muted); font-size: 12px; }
  140. /* ── 建议下拉(portal,fixed 定位到 body)──────────────────────── */
  141. .suggest-dropdown {
  142. background: var(--surface);
  143. border: 1px solid var(--border); border-radius: 4px;
  144. overflow: hidden;
  145. box-shadow: 0 6px 16px rgba(0,0,0,.12);
  146. max-height: 220px; overflow-y: auto;
  147. }
  148. .suggest-item {
  149. padding: 6px 10px; font-size: 12px;
  150. color: var(--text-sub); cursor: pointer;
  151. transition: background .1s;
  152. }
  153. .suggest-item:hover { background: var(--primary-light); color: var(--primary); }
  154. /* ── 结果区域 ─────────────────────────────────────────────────── */
  155. .results-area { flex: 1; min-width: 0; display: flex; flex-direction: column; gap: 10px; }
  156. .chips-bar {
  157. display: flex; align-items: center; flex-wrap: wrap; gap: 6px;
  158. padding: 9px 13px;
  159. background: var(--surface);
  160. border: 1px solid var(--border); border-radius: var(--radius);
  161. }
  162. .chips-label { font-size: 12px; color: var(--text-muted); white-space: nowrap; }
  163. .chip {
  164. display: inline-flex; align-items: center; gap: 3px;
  165. padding: 2px 8px; border-radius: 4px;
  166. background: var(--primary-light); color: var(--primary);
  167. font-size: 12px; font-weight: 500; cursor: pointer;
  168. }
  169. .chip-x { font-size: 13px; opacity: .55; margin-left: 1px; }
  170. .chip:hover .chip-x { opacity: 1; }
  171. .chip-clear { margin-left: auto; font-size: 12px; color: var(--text-muted); cursor: pointer; }
  172. .chip-clear:hover { color: #EF4444; }
  173. .results-bar {
  174. display: flex; align-items: center; justify-content: space-between;
  175. }
  176. .results-count { font-size: 13px; color: var(--text-sub); }
  177. .results-count strong { color: var(--text); font-weight: 600; }
  178. .sort-label { font-size: 12px; color: var(--text-muted); }
  179. /* ── 卡片 ─────────────────────────────────────────────────────── */
  180. .card {
  181. background: var(--surface);
  182. border: 1px solid var(--border); border-radius: var(--radius);
  183. padding: 15px 16px; cursor: pointer;
  184. transition: border-color .15s, box-shadow .15s;
  185. }
  186. .card:hover { border-color: #C7D2FE; box-shadow: 0 3px 12px rgba(79,70,229,.08); }
  187. .card-top {
  188. display: flex; align-items: flex-start;
  189. justify-content: space-between; gap: 10px; margin-bottom: 6px;
  190. }
  191. .card-title {
  192. font-size: 15px; font-weight: 600; color: var(--text);
  193. line-height: 1.4; flex: 1;
  194. }
  195. .card-title em {
  196. font-style: normal; color: var(--primary);
  197. background: var(--primary-light); padding: 0 2px; border-radius: 2px;
  198. }
  199. .score-badge {
  200. flex-shrink: 0; font-size: 11px; font-weight: 600;
  201. color: #15803D; background: #F0FDF4;
  202. padding: 2px 7px; border-radius: 10px; white-space: nowrap;
  203. }
  204. .card-meta {
  205. display: flex; align-items: center; flex-wrap: wrap;
  206. gap: 3px 10px; font-size: 12px; color: var(--text-muted); margin-bottom: 9px;
  207. }
  208. .source-badge {
  209. display: inline-flex; align-items: center;
  210. padding: 1px 6px;
  211. background: var(--bg); border: 1px solid var(--border);
  212. border-radius: 4px; font-size: 11px; color: var(--text-sub);
  213. }
  214. .meta-dot { color: var(--border); }
  215. .tag-row { display: flex; flex-wrap: wrap; gap: 4px; margin-bottom: 10px; }
  216. .tag {
  217. display: inline-flex; align-items: center; gap: 2px;
  218. padding: 2px 7px; border-radius: 4px;
  219. font-size: 11px; font-weight: 500; line-height: 1.6;
  220. }
  221. .tag-pre { opacity: .65; font-size: 10px; }
  222. .tag.s { background: var(--substance-bg); color: var(--substance); }
  223. .tag.fo { background: var(--form-bg); color: var(--form); }
  224. .tag.i { background: var(--intent-bg); color: var(--intent); }
  225. .tag.e { background: var(--effect-bg); color: var(--effect); }
  226. .tag.fe { background: var(--feeling-bg); color: var(--feeling); }
  227. .tag.d { background: var(--dim-bg); color: var(--dim); border: 1px solid var(--border); }
  228. .tag.hit { box-shadow: 0 0 0 1.5px currentColor; font-weight: 700; }
  229. .tag.d.hit { border-color: var(--dim); }
  230. .ext-item.hit .ext-key { color: var(--text-sub); }
  231. .ext-item.hit .ext-val { color: var(--primary); font-weight: 700; }
  232. .card-foot {
  233. display: flex; align-items: center; justify-content: space-between;
  234. padding-top: 9px; border-top: 1px solid var(--bg);
  235. }
  236. .ext-row { display: flex; flex-wrap: wrap; gap: 10px; }
  237. .ext-item { font-size: 11px; }
  238. .ext-key { color: var(--text-muted); }
  239. .ext-val { color: var(--text-sub); font-weight: 500; }
  240. .card-time { font-size: 11px; color: var(--text-muted); white-space: nowrap; }
  241. /* ── 分页 ─────────────────────────────────────────────────────── */
  242. .pagination {
  243. display: flex; justify-content: center; align-items: center;
  244. gap: 3px; padding: 6px 0 4px;
  245. }
  246. .pg-btn {
  247. min-width: 32px; height: 32px; padding: 0 8px;
  248. display: inline-flex; align-items: center; justify-content: center;
  249. border: 1px solid var(--border); border-radius: 4px;
  250. background: var(--surface); color: var(--text-sub);
  251. font-size: 13px; cursor: pointer; transition: all .15s; user-select: none;
  252. }
  253. .pg-btn:hover:not(.active):not([disabled]) { border-color: var(--primary); color: var(--primary); }
  254. .pg-btn.active { background: var(--primary); border-color: var(--primary); color: #fff; font-weight: 600; }
  255. .pg-btn[disabled] { color: var(--text-muted); cursor: default; }
  256. .pg-dots { font-size: 13px; color: var(--text-muted); padding: 0 2px; }
  257. /* ── 空/加载/错误状态 ─────────────────────────────────────────── */
  258. .state-box {
  259. text-align: center; padding: 60px 20px;
  260. color: var(--text-muted); font-size: 14px;
  261. background: var(--surface);
  262. border: 1px solid var(--border); border-radius: var(--radius);
  263. }
  264. .state-box.error { color: #EF4444; }
  265. .state-box .state-icon { font-size: 32px; margin-bottom: 10px; }
  266. .state-box p { margin-top: 6px; font-size: 13px; }
  267. /* ── 侧栏骨架 ─────────────────────────────────────────────────── */
  268. .sidebar-loading {
  269. padding: 20px 13px;
  270. display: flex; flex-direction: column; gap: 8px;
  271. }
  272. .skel {
  273. height: 12px; border-radius: 4px;
  274. background: linear-gradient(90deg, var(--border) 25%, var(--bg) 50%, var(--border) 75%);
  275. background-size: 200% 100%;
  276. animation: shimmer 1.4s infinite;
  277. }
  278. @keyframes shimmer { 0%{background-position:200% 0} 100%{background-position:-200% 0} }
  279. /* ── 语义输入框 loading 圈 ────────────────────────────────────────── */
  280. .sem-wrap { position: relative; }
  281. .sem-wrap .semantic-input { padding-right: 26px; }
  282. .sem-spinner {
  283. position: absolute; right: 7px; top: 50%;
  284. transform: translateY(-50%);
  285. width: 11px; height: 11px;
  286. border: 2px solid var(--border);
  287. border-top-color: var(--primary);
  288. border-radius: 50%;
  289. animation: sem-spin .65s linear infinite;
  290. display: none; pointer-events: none;
  291. }
  292. .sem-spinner.show { display: block; }
  293. @keyframes sem-spin { to { transform: translateY(-50%) rotate(360deg); } }
  294. /* ── 正文展开 ─────────────────────────────────────────────────── */
  295. .content-toggle {
  296. display: flex; align-items: center; gap: 5px;
  297. padding: 7px 0 2px;
  298. font-size: 12px; color: var(--text-muted);
  299. cursor: pointer; user-select: none;
  300. border-top: 1px solid var(--bg); margin-top: 8px;
  301. }
  302. .content-toggle:hover { color: var(--primary); }
  303. .content-caret {
  304. font-size: 9px; display: inline-block;
  305. transition: transform .2s; margin-left: auto;
  306. }
  307. .content-toggle.open .content-caret { transform: rotate(180deg); }
  308. .content-body { display: none; }
  309. .content-body.open { display: block; padding-top: 6px; }
  310. .content-pre {
  311. margin: 0; padding: 10px 12px;
  312. background: var(--bg); border: 1px solid var(--border); border-radius: 5px;
  313. font-family: 'Courier New', Consolas, monospace;
  314. font-size: 11px; line-height: 1.65; color: var(--text-sub);
  315. max-height: 300px; overflow-y: auto;
  316. white-space: pre-wrap; word-break: break-all;
  317. }
  318. /* ── 工序步骤表(复用自工序解构的 .steps 表格)─────────────────── */
  319. .steps-wrap { overflow-x: auto; margin-top: 6px; border: 1px solid var(--border); border-radius: 6px; }
  320. .steps { width: 100%; min-width: 1500px; table-layout: fixed; border-collapse: collapse; font-size: 12px; }
  321. .steps th { padding: 6px 8px; font-size: 11px; font-weight: 700; letter-spacing: 1px; color: #fff; text-align: left; }
  322. .steps thead tr:first-child th { text-align: center; font-size: 12px; letter-spacing: 4px; padding: 7px 4px; }
  323. .steps .h-req { background: #2b4a72; } .steps .h-req2 { background: #33547a; }
  324. .steps .h-in { background: #c2761f; } .steps .h-in2 { background: #cd7522; }
  325. .steps .h-im { background: #2f9c8a; } .steps .h-im2 { background: #2d8273; }
  326. .steps .h-out { background: #2563eb; } .steps .h-out2 { background: #4f7fe6; }
  327. .steps td { padding: 8px 9px; border: 1px solid var(--border); vertical-align: top; line-height: 1.6; color: var(--text); }
  328. .steps tbody tr:nth-child(odd) td { background: #fdfdf9; }
  329. .steps td.c-in { background: #FFF7ED !important; }
  330. .steps td.c-out { background: #eef3fe !important; }
  331. .steps .sid { font-family: 'Courier New', monospace; font-weight: 700; color: #2b4a72; white-space: nowrap; }
  332. .steps .vtxt { color: var(--text-sub); font-size: 11.5px; word-break: break-all; }
  333. .steps .anchor { font-family: 'Courier New', monospace; font-size: 10.5px; color: var(--text-muted); word-break: break-all; }
  334. .steps .pill { display: inline-block; padding: 1px 7px; border-radius: 4px; font-size: 11px; font-weight: 600; }
  335. .steps .pill.navy { background: #e7edf5; color: #2b4a72; }
  336. .steps .pill.amber { background: #fdebd3; color: #9a560f; }
  337. .steps .pill.teal { background: #d8f0ea; color: #1f6f62; }
  338. .steps .pill.green { background: #e3f3e8; color: #2e6b45; }
  339. .steps .pill.blue { background: #eef3fe; color: #2563eb; }
  340. /* 目的列 intent 胶囊: 底色 = 所引用列的分组色, 与表头/列 chip 一致
  341. (需求=navy / 输入=amber / 实现=teal / 输出=green; 口径同 procedure-dsl「token 色对应来源列」) */
  342. .intent-text { color: #1f2937; line-height: 1.6; }
  343. .intent-tok { display: inline-block; padding: 1px 6px; border-radius: 4px; margin: 0 1px; font-size: 11.5px; font-weight: 600; }
  344. .intent-tok.ik-effect { background: #e7edf5; color: #2b4a72; } /* 作用列 (需求组) */
  345. .intent-tok.ik-via { background: #d8f0ea; color: #1f6f62; font-family: ui-monospace, "SF Mono", monospace; } /* 外部工具列 (实现组) */
  346. .intent-tok.ik-act { background: #d8f0ea; color: #1f6f62; } /* 动作列 (实现组) */
  347. .intent-tok.ik-in-type { background: #fdebd3; color: #9a560f; border-radius: 99px; padding: 1px 8px; } /* 输入·类型 */
  348. .intent-tok.ik-out-type { background: #eef3fe; color: #2563eb; border: 1px solid #b6cdf7; border-radius: 99px; padding: 1px 8px; } /* 输出·类型 (蓝色,避免与实现组绿色混淆) */
  349. .intent-tok.ik-other { background: #fbeae5; color: #b3341d; text-decoration: line-through; } /* 非法类别(lint 警告) */
  350. .steps .inf { background: #fdf6e3 !important; position: relative; outline: 1px dashed #c9a227; outline-offset: -2px; }
  351. .steps .inf .ib { position: absolute; top: -1px; right: -1px; background: #c9a227; color: #fff; font-size: 9px; padding: 0 4px; border-radius: 0 0 0 4px; font-weight: 700; }
  352. .steps-empty { padding: 12px; color: var(--text-muted); font-size: 12px; }
  353. /* 输入/输出「值」单元格:超过 4 行加蒙版,点击展开/收起 */
  354. .clamp-val { position: relative; }
  355. .clamp-val.clampable { max-height: 6.6em; overflow: hidden; cursor: zoom-in; }
  356. .clamp-val.clampable::after {
  357. content: ''; position: absolute; left: 0; right: 0; bottom: 0; height: 2.4em; pointer-events: none;
  358. }
  359. .steps td.c-in .clamp-val.clampable::after { background: linear-gradient(180deg, rgba(255,247,237,0), rgba(255,247,237,1)); }
  360. .steps td.c-out .clamp-val.clampable::after { background: linear-gradient(180deg, rgba(238,243,254,0), rgba(238,243,254,1)); }
  361. .clamp-val.open { max-height: none; overflow: visible; cursor: zoom-out; }
  362. .clamp-val.open::after { display: none; }
  363. /* ── 工具解构表(移植自 index.html renderTools:案例逐行 rowspan + 限高展开)─ */
  364. .mw-ttwrap { overflow-x: auto; margin-top: 6px; border: 1px solid var(--border); border-radius: 8px; }
  365. .mw-tt { border-collapse: separate; border-spacing: 0; width: 100%; min-width: 1180px; background: #fff; font-size: 12.5px; }
  366. .mw-tt thead th {
  367. text-align: left; white-space: nowrap;
  368. background: linear-gradient(180deg, #2aa79b, #1c8076); color: #fff; font-weight: 700;
  369. padding: 9px 12px; letter-spacing: .3px; border-right: 1px solid rgba(255,255,255,.18);
  370. }
  371. .mw-tt thead th:last-child { border-right: none; }
  372. .mw-tt .th-group { text-align: center; }
  373. .mw-tt .th-sub { background: linear-gradient(180deg, #36bdb0, #23897f); font-weight: 600; }
  374. .mw-tt tbody td {
  375. padding: 9px 12px; vertical-align: top; line-height: 1.6;
  376. border-bottom: 1px solid #f0eee8; border-right: 1px solid #f5f3ee; color: #3a3a3a;
  377. }
  378. .mw-tt tbody td:last-child { border-right: none; }
  379. .mw-tt td.col-case { background: #fafdfc; }
  380. .mw-tt tbody td.col-tool {
  381. font-weight: 700; color: #176d64; white-space: nowrap;
  382. border-left: 3px solid #2aa79b; background: #f3faf8;
  383. }
  384. .mw-tt ul { margin: 0; padding-left: 17px; }
  385. .mw-tt ul li { margin: 3px 0; }
  386. .mw-tt ul li::marker { color: #2aa79b; }
  387. .mw-tt .layer-badge { display: inline-block; font-weight: 700; font-size: 11px; padding: 2px 10px; border-radius: 20px; white-space: nowrap; }
  388. .mw-tt .layer-badge.make { color: #0e7490; background: #d6f0ee; }
  389. .mw-tt .layer-badge.create { color: #b8731a; background: #fef0db; }
  390. .mw-tt .dash { color: #c9c2b6; }
  391. .mw-tt .tcell { position: relative; max-height: 7.8em; overflow: hidden; transition: max-height .15s; }
  392. .mw-tt .tcell.clamped { cursor: zoom-in; }
  393. .mw-tt .tcell.clamped::after {
  394. content: '▾ 展开'; position: absolute; left: 0; right: 0; bottom: 0; height: 2.6em;
  395. display: flex; align-items: flex-end; justify-content: center; padding-bottom: 2px;
  396. font-size: 11px; font-weight: 700; color: #176d64;
  397. background: linear-gradient(rgba(255,255,255,0), #fff 72%); pointer-events: none;
  398. }
  399. .mw-tt td.col-case .tcell.clamped::after { background: linear-gradient(rgba(250,253,252,0), #fafdfc 72%); }
  400. .mw-tt .tcell.open { max-height: none; cursor: zoom-out; }
  401. .mw-tt .tcell.open::after { content: ''; height: 0; }
  402. </style>
  403. </head>
  404. <body>
  405. <div class="search-section">
  406. <div class="search-inner">
  407. <div class="search-input-wrap">
  408. <span class="search-icon">
  409. <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.2">
  410. <circle cx="11" cy="11" r="7"/><line x1="21" y1="21" x2="16.65" y2="16.65"/>
  411. </svg>
  412. </span>
  413. <input id="search-input" class="search-input" type="text" placeholder="搜索知识标题、正文内容…">
  414. </div>
  415. <button id="search-btn" class="btn-primary">搜索</button>
  416. </div>
  417. </div>
  418. <div class="layout">
  419. <aside class="sidebar">
  420. <div class="filter-card" id="sidebar-card">
  421. <div class="sidebar-loading">
  422. <div class="skel" style="width:60%"></div>
  423. <div class="skel" style="width:90%"></div>
  424. <div class="skel" style="width:75%"></div>
  425. <div class="skel" style="width:80%"></div>
  426. <div class="skel" style="width:55%"></div>
  427. </div>
  428. </div>
  429. </aside>
  430. <div class="results-area">
  431. <div class="chips-bar" id="chips-bar" style="display:none"></div>
  432. <div class="results-bar">
  433. <div class="results-count" id="results-count">加载中…</div>
  434. <div class="sort-label" id="sort-label"></div>
  435. </div>
  436. <div id="results-list">
  437. <div class="state-box"><div class="state-icon">🔍</div><p>正在加载…</p></div>
  438. </div>
  439. <div class="pagination" id="pagination"></div>
  440. </div>
  441. </div>
  442. <script>
  443. const API = '/api/v1/knowledge';
  444. const SCOPE_TYPES = ['substance','form','intent','effect','feeling'];
  445. const SCOPE_LABELS = {substance:'实质', form:'形式', intent:'意图', effect:'作用', feeling:'感受'};
  446. const SCOPE_TAGS = {substance:'s', form:'fo', intent:'i', effect:'e', feeling:'fe'};
  447. // ── 转义工具 ──────────────────────────────────────────────────────────
  448. const esc = s => String(s ?? '').replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/"/g,'&quot;');
  449. // 目的列: 把 intent 里的 {类别:值} 标记渲染成彩色胶囊 (口径同 procedure-dsl renderer.py:render_intent)。
  450. // 合法类别 5 个: effect/via/act/in-type/out-type; 其余落 ik-other(红删除线)。标记外字面与值都转义。
  451. const INTENT_KIND = { effect:'ik-effect', via:'ik-via', act:'ik-act', 'in-type':'ik-in-type', 'out-type':'ik-out-type' };
  452. function renderIntent(text) {
  453. const s = String(text ?? ''), re = /\{([\w-]+):([^}]+)\}/g;
  454. let out = '', last = 0, m;
  455. while ((m = re.exec(s))) {
  456. out += esc(s.slice(last, m.index).replace(/`/g, ''));
  457. out += `<span class="intent-tok ${INTENT_KIND[m[1]] || 'ik-other'}">${esc(m[2])}</span>`;
  458. last = m.index + m[0].length;
  459. }
  460. return out + esc(s.slice(last).replace(/`/g, ''));
  461. }
  462. function debounce(fn, ms) {
  463. let t; return (...a) => { clearTimeout(t); t = setTimeout(() => fn(...a), ms); };
  464. }
  465. // ── 全局状态 ──────────────────────────────────────────────────────────
  466. let dims = null;
  467. let _portalDropdown = null;
  468. function clearPortalDropdown() {
  469. if (_portalDropdown) { _portalDropdown.remove(); _portalDropdown = null; }
  470. }
  471. document.addEventListener('mousedown', e => {
  472. if (_portalDropdown && !_portalDropdown.contains(e.target)) clearPortalDropdown();
  473. });
  474. const sel = {
  475. query: '',
  476. scopes: {substance: new Set(), form: new Set(), intent: new Set(), effect: new Set(), feeling: new Set()},
  477. scopeSemantic: {substance: '', form: '', intent: '', effect: '', feeling: ''},
  478. dimAttr: new Set(),
  479. dimCreate: new Set(),
  480. customValues: {}, // key → Set<string>
  481. customSemantics:{}, // key → string
  482. customRanges: {}, // key → {gte, lte}
  483. page: 1,
  484. };
  485. let _chips = [];
  486. // ── 初始化 ────────────────────────────────────────────────────────────
  487. async function init() {
  488. document.getElementById('search-btn').addEventListener('click', triggerSearch);
  489. document.getElementById('search-input').addEventListener('keydown', e => {
  490. if (e.key === 'Enter') triggerSearch();
  491. });
  492. try {
  493. const r = await fetch(API + '/dimensions');
  494. if (r.ok) {
  495. dims = await r.json();
  496. renderSidebar();
  497. }
  498. } catch(e) {
  499. console.warn('Failed to load dimensions', e);
  500. document.getElementById('sidebar-card').innerHTML =
  501. '<div class="sidebar-loading" style="color:var(--text-muted);font-size:12px;padding:16px 13px">筛选项加载失败</div>';
  502. }
  503. doSearch();
  504. }
  505. function triggerSearch() {
  506. sel.query = document.getElementById('search-input').value.trim();
  507. sel.page = 1;
  508. doSearch();
  509. }
  510. // ── 侧栏渲染 ──────────────────────────────────────────────────────────
  511. function renderSidebar() {
  512. const card = document.getElementById('sidebar-card');
  513. let html = '';
  514. // 作用域
  515. html += `<div class="filter-section">
  516. <div class="section-hd open" onclick="toggleSection(this)">作用域 <span class="chevron">▼</span></div>
  517. <div class="section-bd open" id="scope-bd">`;
  518. for (const sg of (dims?.scope_groups || [])) {
  519. html += renderScopeGroup(sg);
  520. }
  521. html += `</div></div>`;
  522. // 属性维度(从 facets 动态注入)
  523. html += `<div class="filter-section" id="dim-attr-section" style="display:none">
  524. <div class="section-hd open" onclick="toggleSection(this)">属性维度 <span class="chevron">▼</span></div>
  525. <div class="section-bd open" id="dim-attr-bd"></div>
  526. </div>`;
  527. // 创作维度(从 facets 动态注入)
  528. html += `<div class="filter-section" id="dim-create-section" style="display:none">
  529. <div class="section-hd open" onclick="toggleSection(this)">创作维度 <span class="chevron">▼</span></div>
  530. <div class="section-bd open" id="dim-create-bd"></div>
  531. </div>`;
  532. // 自定义字段(section 默认隐藏,有 facets 数据时才显示)
  533. if (dims?.custom_fields?.length) {
  534. html += `<div class="filter-section" id="custom-section" style="display:none">
  535. <div class="section-hd open" onclick="toggleSection(this)">自定义索引 <span class="chevron">▼</span></div>
  536. <div class="section-bd open" id="custom-bd">`;
  537. for (const cf of dims.custom_fields) {
  538. html += renderCustomField(cf);
  539. }
  540. html += `</div></div>`;
  541. }
  542. card.innerHTML = html;
  543. bindSidebarEvents();
  544. }
  545. function renderScopeGroup(sg) {
  546. const st = sg.scope_type;
  547. const label = SCOPE_LABELS[st] || st;
  548. return `<div class="sub-group" id="sg-${st}">
  549. <div class="sub-label">${label}</div>
  550. <div class="sem-wrap">
  551. <input class="semantic-input" type="text" id="sem-${st}" placeholder="语义搜索${label}标签…">
  552. <span class="sem-spinner" id="sem-${st}-sp"></span>
  553. </div>
  554. <div class="suggest-list" id="sug-${st}"></div>
  555. </div>`;
  556. }
  557. function renderCustomField(cf) {
  558. const label = esc(cf.description || cf.key);
  559. let html = `<div class="sub-group" id="cf-${esc(cf.key)}"><div class="sub-label">${label}</div>`;
  560. if (cf.value_type === 'str') {
  561. if (cf.semantic_enabled) {
  562. html += `<div class="sem-wrap">
  563. <input class="semantic-input" type="text" id="csem-${esc(cf.key)}" placeholder="语义搜索${label}…">
  564. <span class="sem-spinner" id="csem-${esc(cf.key)}-sp"></span>
  565. </div>`;
  566. }
  567. html += `<div class="suggest-list" id="csug-${esc(cf.key)}"></div>`;
  568. } else if (cf.value_type === 'num') {
  569. html += `<div class="range-row">
  570. <input class="range-inp" type="number" id="cgte-${esc(cf.key)}" placeholder="最小">
  571. <span class="range-sep">—</span>
  572. <input class="range-inp" type="number" id="clte-${esc(cf.key)}" placeholder="最大">
  573. </div>`;
  574. } else if (cf.value_type === 'date') {
  575. html += `<div class="range-row">
  576. <input class="range-inp" type="date" id="cgte-${esc(cf.key)}">
  577. <span class="range-sep">—</span>
  578. <input class="range-inp" type="date" id="clte-${esc(cf.key)}">
  579. </div>`;
  580. }
  581. html += `</div>`;
  582. return html;
  583. }
  584. function bindSidebarEvents() {
  585. // 作用域复选框
  586. document.querySelectorAll('input[data-scope]').forEach(cb => {
  587. cb.addEventListener('change', () => {
  588. const st = cb.dataset.scope, val = cb.dataset.val;
  589. if (cb.checked) sel.scopes[st].add(val); else sel.scopes[st].delete(val);
  590. sel.page = 1; doSearch();
  591. });
  592. });
  593. // 作用域语义输入
  594. const debouncedSuggestScope = debounce((st, q) => {
  595. if (q) suggestScope(st, q);
  596. else clearPortalDropdown();
  597. }, 300);
  598. for (const sg of (dims?.scope_groups || [])) {
  599. const st = sg.scope_type;
  600. const inp = document.getElementById('sem-' + st);
  601. if (!inp) continue;
  602. inp.addEventListener('input', () => {
  603. sel.scopeSemantic[st] = inp.value.trim();
  604. debouncedSuggestScope(st, inp.value.trim());
  605. });
  606. inp.addEventListener('keydown', e => { if (e.key === 'Enter') { sel.page = 1; doSearch(); } });
  607. inp.addEventListener('blur', clearPortalDropdown);
  608. }
  609. // 自定义字段复选框
  610. document.querySelectorAll('input[data-ckey]').forEach(cb => {
  611. cb.addEventListener('change', () => {
  612. const key = cb.dataset.ckey, val = cb.dataset.val;
  613. if (!sel.customValues[key]) sel.customValues[key] = new Set();
  614. if (cb.checked) sel.customValues[key].add(val); else sel.customValues[key].delete(val);
  615. sel.page = 1; doSearch();
  616. });
  617. });
  618. // 自定义语义输入
  619. const debouncedSuggestCustom = debounce((key, q) => {
  620. if (q) suggestCustom(key, q);
  621. else clearPortalDropdown();
  622. }, 300);
  623. for (const cf of (dims?.custom_fields || [])) {
  624. if (cf.value_type !== 'str' || !cf.semantic_enabled) continue;
  625. const inp = document.getElementById('csem-' + cf.key);
  626. if (!inp) continue;
  627. inp.addEventListener('input', () => {
  628. sel.customSemantics[cf.key] = inp.value.trim();
  629. debouncedSuggestCustom(cf.key, inp.value.trim());
  630. });
  631. inp.addEventListener('keydown', e => { if (e.key === 'Enter') { sel.page = 1; doSearch(); } });
  632. inp.addEventListener('blur', clearPortalDropdown);
  633. }
  634. // 自定义范围输入
  635. for (const cf of (dims?.custom_fields || [])) {
  636. if (cf.value_type !== 'num' && cf.value_type !== 'date') continue;
  637. const gteEl = document.getElementById('cgte-' + cf.key);
  638. const lteEl = document.getElementById('clte-' + cf.key);
  639. const onChange = debounce(() => {
  640. if (!sel.customRanges[cf.key]) sel.customRanges[cf.key] = {};
  641. if (cf.value_type === 'num') {
  642. sel.customRanges[cf.key].gte = gteEl?.value !== '' ? parseFloat(gteEl.value) : null;
  643. sel.customRanges[cf.key].lte = lteEl?.value !== '' ? parseFloat(lteEl.value) : null;
  644. } else {
  645. sel.customRanges[cf.key].gte = gteEl?.value ? new Date(gteEl.value).getTime() : null;
  646. sel.customRanges[cf.key].lte = lteEl?.value ? new Date(lteEl.value + 'T23:59:59').getTime() : null;
  647. }
  648. sel.page = 1; doSearch();
  649. }, 500);
  650. gteEl?.addEventListener('input', onChange);
  651. lteEl?.addEventListener('input', onChange);
  652. }
  653. }
  654. // ── 建议 API ──────────────────────────────────────────────────────────
  655. async function suggestScope(scopeType, q) {
  656. const inputEl = document.getElementById('sem-' + scopeType);
  657. const sp = document.getElementById('sem-' + scopeType + '-sp');
  658. if (!inputEl) return;
  659. if (sp) sp.classList.add('show');
  660. try {
  661. const r = await fetch(`${API}/scope-tags/suggest?scope_type=${encodeURIComponent(scopeType)}&q=${encodeURIComponent(q)}`);
  662. if (!r.ok) return;
  663. const data = await r.json();
  664. renderSuggestDropdown(inputEl, data.tags || [], val => {
  665. sel.scopes[scopeType].add(val);
  666. inputEl.value = '';
  667. sel.scopeSemantic[scopeType] = '';
  668. addInlineScopeTag(scopeType, val);
  669. sel.page = 1; doSearch();
  670. });
  671. } catch(e) {
  672. } finally {
  673. if (sp) sp.classList.remove('show');
  674. }
  675. }
  676. async function suggestCustom(key, q) {
  677. const inputEl = document.getElementById('csem-' + key);
  678. const sp = document.getElementById('csem-' + key + '-sp');
  679. if (!inputEl) return;
  680. if (sp) sp.classList.add('show');
  681. try {
  682. const r = await fetch(`${API}/custom-values/suggest?key=${encodeURIComponent(key)}&q=${encodeURIComponent(q)}`);
  683. if (!r.ok) return;
  684. const data = await r.json();
  685. renderSuggestDropdown(inputEl, data.values || [], val => {
  686. if (!sel.customValues[key]) sel.customValues[key] = new Set();
  687. sel.customValues[key].add(val);
  688. inputEl.value = '';
  689. sel.customSemantics[key] = '';
  690. addInlineCustomValue(key, val);
  691. sel.page = 1; doSearch();
  692. });
  693. } catch(e) {
  694. } finally {
  695. if (sp) sp.classList.remove('show');
  696. }
  697. }
  698. function renderSuggestDropdown(inputEl, items, onSelect) {
  699. clearPortalDropdown();
  700. if (!items.length) return;
  701. const rect = inputEl.getBoundingClientRect();
  702. const div = document.createElement('div');
  703. div.className = 'suggest-dropdown';
  704. div.style.cssText = `position:fixed;z-index:9999;top:${rect.bottom + 2}px;left:${rect.left}px;width:${rect.width}px;`;
  705. items.forEach(item => {
  706. const el = document.createElement('div');
  707. el.className = 'suggest-item';
  708. el.textContent = item;
  709. el.addEventListener('mousedown', e => e.preventDefault());
  710. el.addEventListener('click', () => { onSelect(item); clearPortalDropdown(); });
  711. div.appendChild(el);
  712. });
  713. document.body.appendChild(div);
  714. _portalDropdown = div;
  715. }
  716. function addInlineScopeTag(scopeType, val) {
  717. const container = document.getElementById('sg-' + scopeType);
  718. if (!container) return;
  719. const existing = container.querySelector(`input[data-val="${CSS.escape(val)}"]`);
  720. if (existing) { existing.checked = true; return; }
  721. const label = document.createElement('label');
  722. label.className = 'opt';
  723. label.innerHTML = `<input type="checkbox" checked data-scope="${esc(scopeType)}" data-val="${esc(val)}"><span class="opt-text">${esc(val)}</span>`;
  724. label.querySelector('input').addEventListener('change', function() {
  725. if (this.checked) sel.scopes[scopeType].add(val); else sel.scopes[scopeType].delete(val);
  726. sel.page = 1; doSearch();
  727. });
  728. const sugEl = container.querySelector('.suggest-list');
  729. if (sugEl) container.insertBefore(label, sugEl); else container.appendChild(label);
  730. }
  731. function addInlineCustomValue(key, val) {
  732. const container = document.getElementById('cf-' + key);
  733. if (!container) return;
  734. const existing = container.querySelector(`input[data-val="${CSS.escape(val)}"]`);
  735. if (existing) { existing.checked = true; return; }
  736. const label = document.createElement('label');
  737. label.className = 'opt';
  738. label.innerHTML = `<input type="checkbox" checked data-ckey="${esc(key)}" data-val="${esc(val)}"><span class="opt-text">${esc(val)}</span>`;
  739. label.querySelector('input').addEventListener('change', function() {
  740. if (!sel.customValues[key]) sel.customValues[key] = new Set();
  741. if (this.checked) sel.customValues[key].add(val); else sel.customValues[key].delete(val);
  742. sel.page = 1; doSearch();
  743. });
  744. const sugEl = container.querySelector('.suggest-list');
  745. if (sugEl) container.insertBefore(label, sugEl); else container.appendChild(label);
  746. }
  747. // ── 搜索请求 ──────────────────────────────────────────────────────────
  748. function buildRequest() {
  749. const req = {
  750. page: sel.page,
  751. page_size: 20,
  752. include_facets: true,
  753. highlight: !!sel.query,
  754. include_content: true,
  755. include_matched_conditions: true,
  756. };
  757. if (sel.query) req.query = sel.query;
  758. const scopes = [];
  759. for (const st of SCOPE_TYPES) {
  760. const values = [...sel.scopes[st]];
  761. const semantic = sel.scopeSemantic[st];
  762. if (values.length || semantic) {
  763. const sf = {scope_type: st};
  764. if (values.length) sf.values = values;
  765. if (semantic) sf.semantic_query = semantic;
  766. scopes.push(sf);
  767. }
  768. }
  769. if (scopes.length) req.scopes = scopes;
  770. if (sel.dimAttr.size) req.dim_attributes = [...sel.dimAttr];
  771. if (sel.dimCreate.size) req.dim_creations = [...sel.dimCreate];
  772. const customFilters = [];
  773. const allKeys = new Set([
  774. ...Object.keys(sel.customValues),
  775. ...Object.keys(sel.customSemantics),
  776. ...Object.keys(sel.customRanges),
  777. ]);
  778. for (const key of allKeys) {
  779. const cf = {key};
  780. const vals = [...(sel.customValues[key] || [])];
  781. const sem = sel.customSemantics[key];
  782. const rng = sel.customRanges[key];
  783. if (vals.length) cf.values = vals;
  784. if (sem) cf.semantic_query = sem;
  785. if (rng?.gte != null) cf.gte = rng.gte;
  786. if (rng?.lte != null) cf.lte = rng.lte;
  787. if (Object.keys(cf).length > 1) customFilters.push(cf);
  788. }
  789. if (customFilters.length) req.custom_filters = customFilters;
  790. return req;
  791. }
  792. async function doSearch() {
  793. renderChips();
  794. document.getElementById('results-list').innerHTML =
  795. '<div class="state-box"><div class="state-icon">⏳</div><p>搜索中…</p></div>';
  796. let resp;
  797. try {
  798. const r = await fetch(API + '/search', {
  799. method: 'POST',
  800. headers: {'Content-Type': 'application/json'},
  801. body: JSON.stringify(buildRequest()),
  802. });
  803. if (!r.ok) {
  804. const msg = (await r.json().catch(() => ({}))).detail || r.statusText;
  805. throw new Error(msg);
  806. }
  807. resp = await r.json();
  808. } catch(e) {
  809. document.getElementById('results-count').textContent = '';
  810. document.getElementById('results-list').innerHTML =
  811. `<div class="state-box error"><div class="state-icon">⚠️</div><p>搜索出错:${esc(e.message)}</p></div>`;
  812. return;
  813. }
  814. renderResults(resp);
  815. if (resp.facets) updateSidebarFacets(resp.facets);
  816. }
  817. // ── 渲染结果 ──────────────────────────────────────────────────────────
  818. function renderResults(data) {
  819. document.getElementById('results-count').innerHTML =
  820. `找到 <strong>${data.total.toLocaleString()}</strong> 条相关知识`;
  821. document.getElementById('sort-label').textContent =
  822. sel.query ? '按相关度排序' : '按最新更新排序';
  823. const list = document.getElementById('results-list');
  824. if (!data.hits.length) {
  825. list.innerHTML = '<div class="state-box"><div class="state-icon">📭</div><p>未找到匹配的知识条目,请调整搜索条件</p></div>';
  826. } else {
  827. list.innerHTML = data.hits.map(renderCard).join('');
  828. requestAnimationFrame(markStepClamps);
  829. requestAnimationFrame(markToolClamps);
  830. }
  831. renderPagination(data.total, data.page, data.page_size);
  832. }
  833. // 点击「值」单元格展开/收起(仅在内容溢出、已标记 clampable 时生效)
  834. function toggleClampVal(el) {
  835. if (!el.classList.contains('clampable') && !el.classList.contains('open')) return;
  836. el.classList.toggle('open');
  837. }
  838. // 渲染后标记真正溢出 4 行的「值」单元格,才显示蒙版 + 可点击
  839. // (先套上限高再测量:溢出则保留 clampable,放得下则移除)
  840. function markStepClamps() {
  841. document.querySelectorAll('.steps .clamp-val').forEach((el) => {
  842. if (el.classList.contains('open')) return;
  843. el.classList.add('clampable');
  844. if (el.scrollHeight <= el.clientHeight + 2) el.classList.remove('clampable');
  845. });
  846. }
  847. // ── 工序步骤表(自 index.html 工序解构复用:content 即一条 procedure)──────
  848. function fmtSF(v) {
  849. return v == null ? '' : Array.isArray(v) ? v.join('、') : v;
  850. }
  851. function ioCell(x, kind) {
  852. const cls = kind === 'in' ? 'c-in' : 'c-out';
  853. if (!x) return `<td class="${cls}"></td><td class="${cls}"></td><td class="${cls}"></td>`;
  854. const inf = x.inferred ? ` inf" title="推断理由:${esc(x.inferred_reason || '模型推断补全')}` : '';
  855. const badge = x.inferred ? '<span class="ib">推</span>' : '';
  856. return `<td class="${cls}"><span class="pill ${kind === 'in' ? 'amber' : 'blue'}">${esc(x.type || '')}</span></td>
  857. <td class="${cls}${inf}">${badge}<div class="clamp-val" onclick="toggleClampVal(this)"><span class="vtxt">${esc(x.value || '')}</span></div></td>
  858. <td class="${cls}"><span class="anchor">${esc(x.anchor || '')}</span></td>`;
  859. }
  860. function renderSteps(steps) {
  861. if (!steps || !steps.length) return '<div class="steps-empty">无步骤</div>';
  862. let rows = '';
  863. for (const s of steps) {
  864. const ins = s.inputs && s.inputs.length ? s.inputs : [null];
  865. const outs = s.outputs && s.outputs.length ? s.outputs : [null];
  866. const n = Math.max(ins.length, outs.length);
  867. for (let i = 0; i < n; i++) {
  868. rows += '<tr>';
  869. if (i === 0) {
  870. rows += `<td rowspan="${n}" class="sid">${esc(s.id || '')}</td>
  871. <td rowspan="${n}"><div class="intent-text">${renderIntent(s.intent || s.directive || '')}</div></td>
  872. <td rowspan="${n}">${s.effect ? `<span class="pill navy">${esc(s.effect)}</span>` : ''}</td>
  873. <td rowspan="${n}">${esc(fmtSF(s.substance))}</td>
  874. <td rowspan="${n}">${esc(fmtSF(s.form))}</td>`;
  875. }
  876. rows += ioCell(ins[i], 'in');
  877. if (i === 0) {
  878. rows += `<td rowspan="${n}">${s.via ? `<span class="pill teal">${esc(s.via)}</span>` : ''}</td>
  879. <td rowspan="${n}" class="vtxt">${esc(s.action || '')}</td>`;
  880. }
  881. rows += ioCell(outs[i], 'out');
  882. rows += '</tr>';
  883. }
  884. }
  885. return `<div class="steps-wrap"><table class="steps">
  886. <colgroup>
  887. <col style="width:44px"><col style="width:200px"><col style="width:92px">
  888. <col style="width:112px"><col style="width:100px">
  889. <col style="width:112px"><col style="width:330px"><col style="width:92px">
  890. <col style="width:118px"><col style="width:130px">
  891. <col style="width:112px"><col style="width:360px"><col style="width:110px">
  892. </colgroup>
  893. <thead>
  894. <tr><th class="h-req" colspan="5">需 求</th><th class="h-in" colspan="3">输 入</th><th class="h-im" colspan="2">实 现</th><th class="h-out" colspan="3">输 出</th></tr>
  895. <tr>
  896. <th class="h-req2">#</th><th class="h-req2">目的</th><th class="h-req2">作用</th><th class="h-req2">实质</th><th class="h-req2">形式</th>
  897. <th class="h-in2">类型</th><th class="h-in2">值</th><th class="h-in2">来源</th>
  898. <th class="h-im2">外部工具</th><th class="h-im2">动作</th>
  899. <th class="h-out2">类型</th><th class="h-out2">值</th><th class="h-out2">去处</th>
  900. </tr>
  901. </thead><tbody>${rows}</tbody></table></div>`;
  902. }
  903. // ── 工具解构表(自 index.html renderTools 移植:单条 content 即一个 tool 对象)──
  904. const TT_DASH = '<span class="dash">—</span>';
  905. function _ttToolCell(v) {
  906. if (v === null || v === undefined || v === '' || (Array.isArray(v) && !v.length)) return TT_DASH;
  907. if (Array.isArray(v)) return '<ul>' + v.map((x) => `<li>${esc(String(x))}</li>`).join('') + '</ul>';
  908. return esc(String(v));
  909. }
  910. function _ttScopeCell(v) {
  911. if (v === null || v === undefined || (Array.isArray(v) && !v.length) || v === '') return TT_DASH;
  912. return esc(Array.isArray(v) ? v.join('、') : String(v));
  913. }
  914. function _ttWrap(inner, clampable) {
  915. return clampable ? `<div class="tcell" onclick="this.classList.toggle('open')">${inner}</div>` : inner;
  916. }
  917. function _ttCellContent(c, t) {
  918. let inner, cls = '', clampable = true, style = '';
  919. if (c === '工具名称') {
  920. cls = 'col-tool'; clampable = false;
  921. inner = `🔧 ${esc(t[c] || '(未命名)')}`;
  922. } else if (c === '来源链接') {
  923. clampable = false;
  924. inner = t[c] ? `<a href="${esc(t[c])}" target="_blank" style="color:#176d64;font-weight:600;">🔗 打开</a>` : TT_DASH;
  925. } else if (c === '创作层级') {
  926. clampable = false;
  927. inner = t[c] ? `<span class="layer-badge ${t[c] === '制作层' ? 'make' : 'create'}">${esc(t[c])}</span>` : TT_DASH;
  928. } else if (c === '实质作用域' || c === '形式作用域') {
  929. inner = _ttScopeCell(t[c]);
  930. } else {
  931. inner = _ttToolCell(t[c]);
  932. }
  933. if (['输入', '输出', '用法', '缺点'].includes(c)) style = 'max-width:240px;';
  934. else if (c === '实质作用域' || c === '形式作用域') style = 'max-width:170px;';
  935. else if (!clampable) style = 'white-space:nowrap;';
  936. return { inner, cls, clampable, style };
  937. }
  938. function _ttTd(c, t, rowspan) {
  939. const { inner, cls, clampable, style } = _ttCellContent(c, t);
  940. const rs = rowspan > 1 ? ` rowspan="${rowspan}"` : '';
  941. return `<td class="${cls}" style="${style}"${rs}>${_ttWrap(inner, clampable)}</td>`;
  942. }
  943. function _ttCaseTd(cse, key) {
  944. const v = cse && cse[key] != null && cse[key] !== '' ? esc(String(cse[key])) : TT_DASH;
  945. return `<td class="col-case" style="max-width:210px;">${_ttWrap(v, true)}</td>`;
  946. }
  947. // content 是单个工具对象 → 渲染成一行(案例多条则逐行 rowspan)的工具解构表
  948. function renderToolTable(tool) {
  949. const before = ['工具名称', '创作层级', '实质作用域', '形式作用域', '输入', '输出', '用法'];
  950. const after = ['缺点', '来源链接', '最新更新时间'];
  951. const thead = `<thead>
  952. <tr>
  953. ${before.map((c) => `<th rowspan="2">${c}</th>`).join('')}
  954. <th colspan="3" class="th-group">案例</th>
  955. ${after.map((c) => `<th rowspan="2">${c}</th>`).join('')}
  956. </tr>
  957. <tr>${['输入', '输出', '效果'].map((c) => `<th class="th-sub">${c}</th>`).join('')}</tr>
  958. </thead>`;
  959. const cases = Array.isArray(tool['案例']) && tool['案例'].length ? tool['案例'] : [null];
  960. const K = cases.length;
  961. const rows = cases.map((cse, i) => {
  962. const caseTds = `${_ttCaseTd(cse, '输入')}${_ttCaseTd(cse, '输出')}${_ttCaseTd(cse, '效果')}`;
  963. if (i === 0) {
  964. return `<tr>${before.map((c) => _ttTd(c, tool, K)).join('')}${caseTds}${after.map((c) => _ttTd(c, tool, K)).join('')}</tr>`;
  965. }
  966. return `<tr>${caseTds}</tr>`;
  967. }).join('');
  968. return `<div class="mw-ttwrap"><table class="mw-tt">${thead}<tbody>${rows}</tbody></table></div>`;
  969. }
  970. // 渲染后标记真正溢出的工具单元格(才显示蒙版 + 可点击)
  971. function markToolClamps() {
  972. document.querySelectorAll('.mw-tt .tcell').forEach((el) => {
  973. if (el.classList.contains('open')) return;
  974. if (el.scrollHeight > el.clientHeight + 2) el.classList.add('clamped');
  975. else el.classList.remove('clamped');
  976. });
  977. }
  978. function renderCard(hit) {
  979. let titleHtml = esc(hit.title || '(无标题)');
  980. if (hit.highlight?.title?.[0]) titleHtml = hit.highlight.title[0];
  981. const score = hit.score != null
  982. ? `<span class="score-badge">相关度 ${hit.score.toFixed(2)}</span>` : '';
  983. let meta = '';
  984. if (hit.source_type) meta += `<span class="source-badge">${esc(hit.source_type)}</span>`;
  985. if (hit.author) meta += `<span>${esc(hit.author)}</span>`;
  986. if (hit.author && hit.source_title) meta += '<span class="meta-dot">·</span>';
  987. if (hit.source_title) meta += `<span>${esc(hit.source_title)}</span>`;
  988. // 从 matched_conditions 构建命中查找表
  989. const mc = hit.matched_conditions;
  990. const hitScopes = {}; // scope_type → Set<string>
  991. const hitCustom = {}; // key → Set<string>(str 类型)
  992. const hitRange = {}; // key → bool(num/date 类型)
  993. const hitDimAttr = new Set(mc?.dim_attributes || []);
  994. const hitDimCreate = new Set(mc?.dim_creations || []);
  995. for (const ms of (mc?.scopes || [])) hitScopes[ms.scope_type] = new Set(ms.matched_values);
  996. for (const cm of (mc?.custom || [])) {
  997. if (cm.matched_values != null) hitCustom[cm.key] = new Set(cm.matched_values);
  998. if (cm.in_range != null) hitRange[cm.key] = cm.in_range;
  999. }
  1000. let tags = '';
  1001. const scopeDefs = [
  1002. ['s','scope_substance','substance','实质'],
  1003. ['fo','scope_form','form','形式'],
  1004. ['i','scope_intent','intent','意图'],
  1005. ['e','scope_effect','effect','作用'],
  1006. ['fe','scope_feeling','feeling','感受'],
  1007. ];
  1008. for (const [cls, field, st, label] of scopeDefs) {
  1009. const hitSet = hitScopes[st] || new Set();
  1010. for (const t of (hit[field] || [])) {
  1011. const h = hitSet.has(t) ? ' hit' : '';
  1012. tags += `<span class="tag ${cls}${h}" title="${h ? '命中筛选条件' : ''}"><span class="tag-pre">${label}</span>${esc(t)}</span>`;
  1013. }
  1014. }
  1015. for (const d of (hit.dim_attributes || [])) {
  1016. const h = hitDimAttr.has(d) ? ' hit' : '';
  1017. tags += `<span class="tag d${h}" title="${h ? '命中筛选条件' : ''}">${esc(d)}</span>`;
  1018. }
  1019. for (const d of (hit.dim_creations || [])) {
  1020. const h = hitDimCreate.has(d) ? ' hit' : '';
  1021. tags += `<span class="tag d${h}" title="${h ? '命中筛选条件' : ''}">${esc(d)}</span>`;
  1022. }
  1023. let extHtml = '';
  1024. for (const item of (hit.custom_ext || [])) {
  1025. let val = item.value;
  1026. if (item.type === 'date' && typeof val === 'number') {
  1027. val = new Date(val).toLocaleDateString('zh-CN');
  1028. }
  1029. const isHit = item.type === 'str'
  1030. ? hitCustom[item.key]?.has(String(item.value ?? ''))
  1031. : hitRange[item.key] === true;
  1032. const h = isHit ? ' hit' : '';
  1033. extHtml += `<span class="ext-item${h}" title="${h ? '命中筛选条件' : ''}"><span class="ext-key">${esc(item.key)}:</span><span class="ext-val">${esc(String(val ?? ''))}</span></span>`;
  1034. }
  1035. const updatedAt = hit.updated_at
  1036. ? new Date(hit.updated_at).toLocaleDateString('zh-CN') : '';
  1037. // 正文展开区:content 可能是一条工序解构(procedure,含 steps)或一个工具解构(tool,含工具名称)。
  1038. // 工序 → 工序步骤表;工具 → 工具解构表;都解析不出时回退为格式化 JSON。
  1039. let contentSection = '';
  1040. if (hit.content) {
  1041. let inner = '';
  1042. try {
  1043. const obj = JSON.parse(hit.content);
  1044. if (obj && Array.isArray(obj.steps) && obj.steps.length) inner = renderSteps(obj.steps);
  1045. else if (obj && ('工具名称' in obj || '实质作用域' in obj || '创作层级' in obj)) inner = renderToolTable(obj);
  1046. } catch {}
  1047. if (!inner) {
  1048. let display = hit.content;
  1049. try { display = JSON.stringify(JSON.parse(hit.content), null, 2); } catch {}
  1050. inner = `<pre class="content-pre">${esc(display)}</pre>`;
  1051. }
  1052. // 去掉「正文」标题,默认直接展开
  1053. contentSection = `<div class="content-body open">${inner}</div>`;
  1054. }
  1055. return `<div class="card">
  1056. <div class="card-top"><div class="card-title">${titleHtml}</div>${score}</div>
  1057. ${meta ? `<div class="card-meta">${meta}</div>` : ''}
  1058. ${tags ? `<div class="tag-row">${tags}</div>` : ''}
  1059. <div class="card-foot">
  1060. <div class="ext-row">${extHtml}</div>
  1061. <span class="card-time">${updatedAt ? updatedAt + ' 更新' : ''}</span>
  1062. </div>
  1063. ${contentSection}
  1064. </div>`;
  1065. }
  1066. // ── 分页 ──────────────────────────────────────────────────────────────
  1067. function renderPagination(total, page, pageSize) {
  1068. const totalPages = Math.ceil(total / pageSize);
  1069. const pg = document.getElementById('pagination');
  1070. if (totalPages <= 1) { pg.innerHTML = ''; return; }
  1071. let html = `<button class="pg-btn" onclick="goPage(${page-1})" ${page<=1?'disabled':''}>‹</button>`;
  1072. const pages = buildPageList(page, totalPages);
  1073. let prev = null;
  1074. for (const p of pages) {
  1075. if (prev !== null && p - prev > 1) html += '<span class="pg-dots">…</span>';
  1076. html += `<button class="pg-btn ${p===page?'active':''}" onclick="goPage(${p})">${p}</button>`;
  1077. prev = p;
  1078. }
  1079. html += `<button class="pg-btn" onclick="goPage(${page+1})" ${page>=totalPages?'disabled':''}>›</button>`;
  1080. pg.innerHTML = html;
  1081. }
  1082. function buildPageList(cur, total) {
  1083. const s = new Set([1, total]);
  1084. for (let p = Math.max(1, cur-1); p <= Math.min(total, cur+1); p++) s.add(p);
  1085. return [...s].sort((a,b) => a-b);
  1086. }
  1087. function goPage(p) {
  1088. sel.page = p;
  1089. doSearch();
  1090. window.scrollTo({top: 0, behavior: 'smooth'});
  1091. }
  1092. // ── Facets → 侧栏 ─────────────────────────────────────────────────────
  1093. function updateSidebarFacets(facets) {
  1094. // 属性维度 / 创作维度:整体替换
  1095. updateFacetSection(
  1096. 'dim-attr-section', 'dim-attr-bd',
  1097. facets.dim_attributes || [],
  1098. sel.dimAttr, 'data-dim-attr',
  1099. cb => { const v = cb.dataset.dimAttr; cb.checked ? sel.dimAttr.add(v) : sel.dimAttr.delete(v); sel.page=1; doSearch(); }
  1100. );
  1101. updateFacetSection(
  1102. 'dim-create-section', 'dim-create-bd',
  1103. facets.dim_creations || [],
  1104. sel.dimCreate, 'data-dim-create',
  1105. cb => { const v = cb.dataset.dimCreate; cb.checked ? sel.dimCreate.add(v) : sel.dimCreate.delete(v); sel.page=1; doSearch(); }
  1106. );
  1107. // 作用域标签:完全由 facets 驱动,移除消失的项、补充新出现的项
  1108. const scopeMap = Object.fromEntries((facets.scopes || []).map(s => [s.scope_type, s.buckets || []]));
  1109. for (const st of SCOPE_TYPES) {
  1110. const buckets = scopeMap[st] || [];
  1111. const bucketMap = Object.fromEntries(buckets.map(b => [b.value, b.count]));
  1112. const sgContainer = document.getElementById('sg-' + st);
  1113. if (!sgContainer) continue;
  1114. // 移除不在当前 facets 且未被选中的项
  1115. sgContainer.querySelectorAll('label[data-facet-src]').forEach(label => {
  1116. const val = label.querySelector('input')?.dataset.val;
  1117. if (val && bucketMap[val] == null && !(sel.scopes[st]?.has(val))) label.remove();
  1118. });
  1119. // 更新已存在项的计数,补充 facets 中新出现的项
  1120. const sugEl = sgContainer.querySelector('.suggest-list');
  1121. for (const b of buckets) {
  1122. const existing = sgContainer.querySelector(`input[data-val="${CSS.escape(b.value)}"]`);
  1123. if (existing) {
  1124. syncCountBadge(existing.closest('.opt'), b.count);
  1125. continue;
  1126. }
  1127. const label = document.createElement('label');
  1128. label.className = 'opt';
  1129. label.dataset.facetSrc = 'true';
  1130. const chk = sel.scopes[st]?.has(b.value) ? 'checked' : '';
  1131. label.innerHTML = `<input type="checkbox" ${chk} data-scope="${esc(st)}" data-val="${esc(b.value)}"><span class="opt-text">${esc(b.value)}</span><span class="opt-count">${b.count}</span>`;
  1132. label.querySelector('input').addEventListener('change', function() {
  1133. if (this.checked) sel.scopes[st].add(b.value); else sel.scopes[st].delete(b.value);
  1134. sel.page = 1; doSearch();
  1135. });
  1136. if (sugEl) sgContainer.insertBefore(label, sugEl); else sgContainer.appendChild(label);
  1137. }
  1138. }
  1139. // 自定义字段
  1140. updateCustomExtFacets(facets.custom_ext || []);
  1141. }
  1142. function updateFacetSection(sectionId, bdId, buckets, selSet, dataAttr, onChange) {
  1143. const section = document.getElementById(sectionId);
  1144. const bd = document.getElementById(bdId);
  1145. if (!section || !bd) return;
  1146. if (!buckets.length) { section.style.display = 'none'; return; }
  1147. section.style.display = '';
  1148. let html = '';
  1149. for (const b of buckets) {
  1150. const checked = selSet.has(b.value) ? 'checked' : '';
  1151. html += `<label class="opt">
  1152. <input type="checkbox" ${checked} ${dataAttr}="${esc(b.value)}">
  1153. <span class="opt-text">${esc(b.value)}</span>
  1154. <span class="opt-count">${b.count}</span>
  1155. </label>`;
  1156. }
  1157. bd.innerHTML = html;
  1158. bd.querySelectorAll(`input[${dataAttr}]`).forEach(cb => {
  1159. cb.addEventListener('change', () => onChange(cb));
  1160. });
  1161. }
  1162. // 自定义字段 facets 处理:key 本身的显隐也由 facets 驱动
  1163. function updateCustomExtFacets(customExtFacets) {
  1164. if (!dims?.custom_fields?.length) return;
  1165. const facetMap = Object.fromEntries(customExtFacets.map(f => [f.key, f]));
  1166. let anyVisible = false;
  1167. for (const cf of dims.custom_fields) {
  1168. const container = document.getElementById('cf-' + cf.key);
  1169. if (!container) continue;
  1170. const facet = facetMap[cf.key];
  1171. // 有活跃筛选时即使 facets 无数据也保留显示
  1172. const hasActiveFilter =
  1173. (cf.value_type === 'str' &&
  1174. (sel.customValues[cf.key]?.size > 0 || !!sel.customSemantics[cf.key])) ||
  1175. ((cf.value_type === 'num' || cf.value_type === 'date') &&
  1176. (sel.customRanges[cf.key]?.gte != null || sel.customRanges[cf.key]?.lte != null));
  1177. const hasFacetData =
  1178. (cf.value_type === 'str' && facet?.buckets?.length > 0) ||
  1179. ((cf.value_type === 'num' || cf.value_type === 'date') && facet?.stats?.count > 0);
  1180. if (!hasFacetData && !hasActiveFilter) {
  1181. container.style.display = 'none';
  1182. continue;
  1183. }
  1184. container.style.display = '';
  1185. anyVisible = true;
  1186. if (cf.value_type === 'str') {
  1187. updateCustomStrCounts(cf.key, container, facet?.buckets || []);
  1188. } else if (cf.value_type === 'num' && facet?.stats?.count > 0) {
  1189. const s = facet.stats;
  1190. const gteEl = document.getElementById('cgte-' + cf.key);
  1191. const lteEl = document.getElementById('clte-' + cf.key);
  1192. if (gteEl && s.min != null) gteEl.placeholder = `最小 (${parseFloat(s.min.toFixed(2))})`;
  1193. if (lteEl && s.max != null) lteEl.placeholder = `最大 (${parseFloat(s.max.toFixed(2))})`;
  1194. }
  1195. // date 类型:range input 不显示 placeholder,仅控制可见性
  1196. }
  1197. // 整个"自定义索引"区块:有可见 key 才展示
  1198. const section = document.getElementById('custom-section');
  1199. if (section) section.style.display = anyVisible ? '' : 'none';
  1200. }
  1201. function updateCustomStrCounts(key, container, buckets) {
  1202. const bucketMap = Object.fromEntries(buckets.map(b => [b.value, b.count]));
  1203. // 移除不在当前 facets 且未被选中的项
  1204. container.querySelectorAll('label[data-facet-src]').forEach(label => {
  1205. const val = label.querySelector('input')?.dataset.val;
  1206. if (val && bucketMap[val] == null && !(sel.customValues[key]?.has(val))) label.remove();
  1207. });
  1208. // 更新已存在项的计数,补充 facets 中新出现的项
  1209. const sugEl = container.querySelector('.suggest-list');
  1210. for (const b of buckets) {
  1211. const existing = container.querySelector(`input[data-val="${CSS.escape(b.value)}"]`);
  1212. if (existing) {
  1213. syncCountBadge(existing.closest('.opt'), b.count);
  1214. continue;
  1215. }
  1216. const label = document.createElement('label');
  1217. label.className = 'opt';
  1218. label.dataset.facetSrc = 'true';
  1219. const chk = sel.customValues[key]?.has(b.value) ? 'checked' : '';
  1220. label.innerHTML = `<input type="checkbox" ${chk} data-ckey="${esc(key)}" data-val="${esc(b.value)}"><span class="opt-text">${esc(b.value)}</span><span class="opt-count">${b.count}</span>`;
  1221. label.querySelector('input').addEventListener('change', function() {
  1222. if (!sel.customValues[key]) sel.customValues[key] = new Set();
  1223. if (this.checked) sel.customValues[key].add(b.value); else sel.customValues[key].delete(b.value);
  1224. sel.page = 1; doSearch();
  1225. });
  1226. if (sugEl) container.insertBefore(label, sugEl); else container.appendChild(label);
  1227. }
  1228. }
  1229. // 在 .opt 元素上同步 .opt-count 徽章;count 为 undefined 时清除徽章
  1230. function syncCountBadge(optEl, count) {
  1231. if (!optEl) return;
  1232. let el = optEl.querySelector('.opt-count');
  1233. if (count != null) {
  1234. if (!el) { el = document.createElement('span'); el.className = 'opt-count'; optEl.appendChild(el); }
  1235. el.textContent = count;
  1236. } else if (el) {
  1237. el.remove();
  1238. }
  1239. }
  1240. // ── Chips 栏 ──────────────────────────────────────────────────────────
  1241. function renderChips() {
  1242. _chips = [];
  1243. for (const st of SCOPE_TYPES) {
  1244. for (const val of sel.scopes[st]) {
  1245. _chips.push({
  1246. label: `${SCOPE_LABELS[st]}:${val}`,
  1247. remove: () => { sel.scopes[st].delete(val); uncheckInput(`input[data-scope="${CSS.escape(st)}"][data-val="${CSS.escape(val)}"]`); sel.page=1; doSearch(); }
  1248. });
  1249. }
  1250. if (sel.scopeSemantic[st]) {
  1251. _chips.push({
  1252. label: `${SCOPE_LABELS[st]} 语义:${sel.scopeSemantic[st]}`,
  1253. remove: () => {
  1254. sel.scopeSemantic[st] = '';
  1255. const inp = document.getElementById('sem-' + st);
  1256. if (inp) inp.value = '';
  1257. sel.page = 1; doSearch();
  1258. }
  1259. });
  1260. }
  1261. }
  1262. for (const val of sel.dimAttr) _chips.push({label:`属性:${val}`, remove:()=>{sel.dimAttr.delete(val); uncheckInput(`input[data-dim-attr="${CSS.escape(val)}"]`); sel.page=1; doSearch();}});
  1263. for (const val of sel.dimCreate) _chips.push({label:`创作:${val}`, remove:()=>{sel.dimCreate.delete(val); uncheckInput(`input[data-dim-create="${CSS.escape(val)}"]`); sel.page=1; doSearch();}});
  1264. for (const [key, vals] of Object.entries(sel.customValues)) {
  1265. for (const val of vals) {
  1266. _chips.push({
  1267. label: `${key}:${val}`,
  1268. remove: () => { sel.customValues[key].delete(val); uncheckInput(`input[data-ckey="${CSS.escape(key)}"][data-val="${CSS.escape(val)}"]`); sel.page=1; doSearch(); }
  1269. });
  1270. }
  1271. }
  1272. for (const [key, sem] of Object.entries(sel.customSemantics)) {
  1273. if (!sem) continue;
  1274. _chips.push({
  1275. label: `${key} 语义:${sem}`,
  1276. remove: () => { sel.customSemantics[key]=''; const i=document.getElementById('csem-'+key); if(i)i.value=''; sel.page=1; doSearch(); }
  1277. });
  1278. }
  1279. for (const [key, rng] of Object.entries(sel.customRanges)) {
  1280. if (rng.gte == null && rng.lte == null) continue;
  1281. const parts = [];
  1282. if (rng.gte != null) parts.push(`≥ ${rng.gte}`);
  1283. if (rng.lte != null) parts.push(`≤ ${rng.lte}`);
  1284. _chips.push({
  1285. label: `${key}:${parts.join(' ')}`,
  1286. remove: () => {
  1287. sel.customRanges[key] = {};
  1288. const g = document.getElementById('cgte-'+key), l = document.getElementById('clte-'+key);
  1289. if(g) g.value=''; if(l) l.value='';
  1290. sel.page=1; doSearch();
  1291. }
  1292. });
  1293. }
  1294. const bar = document.getElementById('chips-bar');
  1295. if (!_chips.length) { bar.style.display = 'none'; return; }
  1296. bar.style.display = '';
  1297. let html = '<span class="chips-label">已选筛选:</span>';
  1298. _chips.forEach((c, i) => {
  1299. html += `<span class="chip" data-chip="${i}">${esc(c.label)}<span class="chip-x">×</span></span>`;
  1300. });
  1301. html += '<span class="chip-clear" id="chip-clear-all">清空全部</span>';
  1302. bar.innerHTML = html;
  1303. bar.querySelectorAll('[data-chip]').forEach(el => {
  1304. el.addEventListener('click', () => { _chips[+el.dataset.chip].remove(); });
  1305. });
  1306. document.getElementById('chip-clear-all')?.addEventListener('click', clearAllFilters);
  1307. }
  1308. function uncheckInput(selector) {
  1309. const el = document.querySelector(selector);
  1310. if (el) el.checked = false;
  1311. }
  1312. function clearAllFilters() {
  1313. SCOPE_TYPES.forEach(st => { sel.scopes[st].clear(); sel.scopeSemantic[st] = ''; const i=document.getElementById('sem-'+st); if(i)i.value=''; });
  1314. sel.dimAttr.clear();
  1315. sel.dimCreate.clear();
  1316. sel.customValues = {};
  1317. sel.customSemantics= {};
  1318. sel.customRanges = {};
  1319. document.querySelectorAll('input[data-scope],input[data-ckey],input[data-dim-attr],input[data-dim-create]').forEach(cb => cb.checked = false);
  1320. document.querySelectorAll('.range-inp').forEach(inp => inp.value = '');
  1321. sel.page = 1;
  1322. doSearch();
  1323. }
  1324. // ── 折叠展开 ──────────────────────────────────────────────────────────
  1325. function toggleSection(hd) {
  1326. hd.classList.toggle('open');
  1327. hd.nextElementSibling.classList.toggle('open');
  1328. }
  1329. document.addEventListener('DOMContentLoaded', init);
  1330. </script>
  1331. </body>
  1332. </html>