@robot-admin/naive-ui-components 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (352) hide show
  1. package/README.md +257 -0
  2. package/dist/C_ActionBar-DWN-woTc.css.map +1 -0
  3. package/dist/C_ActionBar.cjs +5 -0
  4. package/dist/C_ActionBar.d.cts +2 -0
  5. package/dist/C_ActionBar.d.ts +2 -0
  6. package/dist/C_ActionBar.js +4 -0
  7. package/dist/C_ActionBar2.js +196 -0
  8. package/dist/C_ActionBar2.js.map +1 -0
  9. package/dist/C_AntV-AFKyK6hH.css.map +1 -0
  10. package/dist/C_AntV.cjs +8 -0
  11. package/dist/C_AntV.d.cts +2 -0
  12. package/dist/C_AntV.d.ts +2 -0
  13. package/dist/C_AntV.js +4 -0
  14. package/dist/C_AntV2.js +3150 -0
  15. package/dist/C_AntV2.js.map +1 -0
  16. package/dist/C_Barcode-P_EFj8dC.css.map +1 -0
  17. package/dist/C_Barcode.cjs +4 -0
  18. package/dist/C_Barcode.d.cts +2 -0
  19. package/dist/C_Barcode.d.ts +2 -0
  20. package/dist/C_Barcode.js +3 -0
  21. package/dist/C_Barcode2.js +68 -0
  22. package/dist/C_Barcode2.js.map +1 -0
  23. package/dist/C_Captcha-C-ef41xw.css.map +1 -0
  24. package/dist/C_Captcha.cjs +4 -0
  25. package/dist/C_Captcha.d.cts +2 -0
  26. package/dist/C_Captcha.d.ts +2 -0
  27. package/dist/C_Captcha.js +3 -0
  28. package/dist/C_Captcha2.js +155 -0
  29. package/dist/C_Captcha2.js.map +1 -0
  30. package/dist/C_Cascade-D9kNsjsV.css.map +1 -0
  31. package/dist/C_Cascade.cjs +4 -0
  32. package/dist/C_Cascade.d.cts +2 -0
  33. package/dist/C_Cascade.d.ts +2 -0
  34. package/dist/C_Cascade.js +3 -0
  35. package/dist/C_Cascade2.js +103 -0
  36. package/dist/C_Cascade2.js.map +1 -0
  37. package/dist/C_City-BCQ4ipiK.css.map +1 -0
  38. package/dist/C_City.cjs +4 -0
  39. package/dist/C_City.d.cts +2 -0
  40. package/dist/C_City.d.ts +2 -0
  41. package/dist/C_City.js +3 -0
  42. package/dist/C_City2.js +841 -0
  43. package/dist/C_City2.js.map +1 -0
  44. package/dist/C_Code-C9kvvEmO.css.map +1 -0
  45. package/dist/C_Code.cjs +5 -0
  46. package/dist/C_Code.d.cts +2 -0
  47. package/dist/C_Code.d.ts +2 -0
  48. package/dist/C_Code.js +4 -0
  49. package/dist/C_Code2.js +346 -0
  50. package/dist/C_Code2.js.map +1 -0
  51. package/dist/C_CollapsePanel-BUJHuYcU.css.map +1 -0
  52. package/dist/C_CollapsePanel.cjs +6 -0
  53. package/dist/C_CollapsePanel.d.cts +2 -0
  54. package/dist/C_CollapsePanel.d.ts +2 -0
  55. package/dist/C_CollapsePanel.js +4 -0
  56. package/dist/C_CollapsePanel2.js +319 -0
  57. package/dist/C_CollapsePanel2.js.map +1 -0
  58. package/dist/C_Cron-yx2Ob4Jl.css.map +1 -0
  59. package/dist/C_Cron.cjs +15 -0
  60. package/dist/C_Cron.d.cts +2 -0
  61. package/dist/C_Cron.d.ts +2 -0
  62. package/dist/C_Cron.js +4 -0
  63. package/dist/C_Cron2.js +1209 -0
  64. package/dist/C_Cron2.js.map +1 -0
  65. package/dist/C_Date.cjs +4 -0
  66. package/dist/C_Date.d.cts +2 -0
  67. package/dist/C_Date.d.ts +2 -0
  68. package/dist/C_Date.js +3 -0
  69. package/dist/C_Date2.js +219 -0
  70. package/dist/C_Date2.js.map +1 -0
  71. package/dist/C_Draggable-C483syRC.css.map +1 -0
  72. package/dist/C_Draggable.cjs +5 -0
  73. package/dist/C_Draggable.d.cts +2 -0
  74. package/dist/C_Draggable.d.ts +2 -0
  75. package/dist/C_Draggable.js +3 -0
  76. package/dist/C_Draggable2.js +295 -0
  77. package/dist/C_Draggable2.js.map +1 -0
  78. package/dist/C_Editor-Bp0SyIEw.css.map +1 -0
  79. package/dist/C_Editor.cjs +4 -0
  80. package/dist/C_Editor.d.cts +2 -0
  81. package/dist/C_Editor.d.ts +2 -0
  82. package/dist/C_Editor.js +3 -0
  83. package/dist/C_Editor2.js +160 -0
  84. package/dist/C_Editor2.js.map +1 -0
  85. package/dist/C_FilePreview-CPqvhoCy.css.map +1 -0
  86. package/dist/C_FilePreview.cjs +6 -0
  87. package/dist/C_FilePreview.d.cts +2 -0
  88. package/dist/C_FilePreview.d.ts +2 -0
  89. package/dist/C_FilePreview.js +3 -0
  90. package/dist/C_FilePreview2.js +1031 -0
  91. package/dist/C_FilePreview2.js.map +1 -0
  92. package/dist/C_Form-Jx7PY3sT.css.map +1 -0
  93. package/dist/C_Form.cjs +15 -0
  94. package/dist/C_Form.d.cts +2 -0
  95. package/dist/C_Form.d.ts +2 -0
  96. package/dist/C_Form.js +4 -0
  97. package/dist/C_Form2.js +2510 -0
  98. package/dist/C_Form2.js.map +1 -0
  99. package/dist/C_FormSearch-DvRgxlRn.css.map +1 -0
  100. package/dist/C_FormSearch.cjs +6 -0
  101. package/dist/C_FormSearch.d.cts +2 -0
  102. package/dist/C_FormSearch.d.ts +2 -0
  103. package/dist/C_FormSearch.js +3 -0
  104. package/dist/C_FormSearch2.js +356 -0
  105. package/dist/C_FormSearch2.js.map +1 -0
  106. package/dist/C_FormulaEditor-DtGkt4T_.css.map +1 -0
  107. package/dist/C_FormulaEditor.cjs +13 -0
  108. package/dist/C_FormulaEditor.d.cts +2 -0
  109. package/dist/C_FormulaEditor.d.ts +2 -0
  110. package/dist/C_FormulaEditor.js +4 -0
  111. package/dist/C_FormulaEditor2.js +1433 -0
  112. package/dist/C_FormulaEditor2.js.map +1 -0
  113. package/dist/C_FullCalendar-BF7H0YIx.css.map +1 -0
  114. package/dist/C_FullCalendar.cjs +9 -0
  115. package/dist/C_FullCalendar.d.cts +2 -0
  116. package/dist/C_FullCalendar.d.ts +2 -0
  117. package/dist/C_FullCalendar.js +3 -0
  118. package/dist/C_FullCalendar2.js +377 -0
  119. package/dist/C_FullCalendar2.js.map +1 -0
  120. package/dist/C_Guide.cjs +4 -0
  121. package/dist/C_Guide.d.cts +2 -0
  122. package/dist/C_Guide.d.ts +2 -0
  123. package/dist/C_Guide.js +3 -0
  124. package/dist/C_Guide2.js +58 -0
  125. package/dist/C_Guide2.js.map +1 -0
  126. package/dist/C_Icon.cjs +4 -0
  127. package/dist/C_Icon.d.cts +2 -0
  128. package/dist/C_Icon.d.ts +2 -0
  129. package/dist/C_Icon.js +3 -0
  130. package/dist/C_Icon2.js +286 -0
  131. package/dist/C_Icon2.js.map +1 -0
  132. package/dist/C_ImageCropper-BVJfUufl.css.map +1 -0
  133. package/dist/C_ImageCropper.cjs +6 -0
  134. package/dist/C_ImageCropper.d.cts +2 -0
  135. package/dist/C_ImageCropper.d.ts +2 -0
  136. package/dist/C_ImageCropper.js +4 -0
  137. package/dist/C_ImageCropper2.js +723 -0
  138. package/dist/C_ImageCropper2.js.map +1 -0
  139. package/dist/C_Language.cjs +4 -0
  140. package/dist/C_Language.d.cts +2 -0
  141. package/dist/C_Language.d.ts +2 -0
  142. package/dist/C_Language.js +3 -0
  143. package/dist/C_Language2.js +72 -0
  144. package/dist/C_Language2.js.map +1 -0
  145. package/dist/C_Map-DpzeuWdX.css.map +1 -0
  146. package/dist/C_Map.cjs +7 -0
  147. package/dist/C_Map.d.cts +2 -0
  148. package/dist/C_Map.d.ts +2 -0
  149. package/dist/C_Map.js +3 -0
  150. package/dist/C_Map2.js +199 -0
  151. package/dist/C_Map2.js.map +1 -0
  152. package/dist/C_Markdown-BEjxknqd.css.map +1 -0
  153. package/dist/C_Markdown.cjs +4 -0
  154. package/dist/C_Markdown.d.cts +2 -0
  155. package/dist/C_Markdown.d.ts +2 -0
  156. package/dist/C_Markdown.js +3 -0
  157. package/dist/C_Markdown2.js +186 -0
  158. package/dist/C_Markdown2.js.map +1 -0
  159. package/dist/C_NotificationCenter-0l3TY2Gn.css.map +1 -0
  160. package/dist/C_NotificationCenter.cjs +20 -0
  161. package/dist/C_NotificationCenter.d.cts +2 -0
  162. package/dist/C_NotificationCenter.d.ts +2 -0
  163. package/dist/C_NotificationCenter.js +4 -0
  164. package/dist/C_NotificationCenter2.js +1383 -0
  165. package/dist/C_NotificationCenter2.js.map +1 -0
  166. package/dist/C_Progress.cjs +4 -0
  167. package/dist/C_Progress.d.cts +2 -0
  168. package/dist/C_Progress.d.ts +2 -0
  169. package/dist/C_Progress.js +3 -0
  170. package/dist/C_Progress2.js +103 -0
  171. package/dist/C_Progress2.js.map +1 -0
  172. package/dist/C_QRCode-DbdiAIPg.css.map +1 -0
  173. package/dist/C_QRCode.cjs +5 -0
  174. package/dist/C_QRCode.d.cts +2 -0
  175. package/dist/C_QRCode.d.ts +2 -0
  176. package/dist/C_QRCode.js +3 -0
  177. package/dist/C_QRCode2.js +218 -0
  178. package/dist/C_QRCode2.js.map +1 -0
  179. package/dist/C_Signature-zhHCbra9.css.map +1 -0
  180. package/dist/C_Signature.cjs +8 -0
  181. package/dist/C_Signature.d.cts +2 -0
  182. package/dist/C_Signature.d.ts +2 -0
  183. package/dist/C_Signature.js +4 -0
  184. package/dist/C_Signature2.js +618 -0
  185. package/dist/C_Signature2.js.map +1 -0
  186. package/dist/C_SplitPane-C6sBsfKY.css.map +1 -0
  187. package/dist/C_SplitPane.cjs +6 -0
  188. package/dist/C_SplitPane.d.cts +2 -0
  189. package/dist/C_SplitPane.d.ts +2 -0
  190. package/dist/C_SplitPane.js +4 -0
  191. package/dist/C_SplitPane2.js +356 -0
  192. package/dist/C_SplitPane2.js.map +1 -0
  193. package/dist/C_Steps-CODHN5Hs.css.map +1 -0
  194. package/dist/C_Steps.cjs +4 -0
  195. package/dist/C_Steps.d.cts +2 -0
  196. package/dist/C_Steps.d.ts +2 -0
  197. package/dist/C_Steps.js +3 -0
  198. package/dist/C_Steps2.js +82 -0
  199. package/dist/C_Steps2.js.map +1 -0
  200. package/dist/C_Table-DSNsntmT.css.map +1 -0
  201. package/dist/C_Table.cjs +19 -0
  202. package/dist/C_Table.d.cts +2 -0
  203. package/dist/C_Table.d.ts +2 -0
  204. package/dist/C_Table.js +5 -0
  205. package/dist/C_Table2.js +3009 -0
  206. package/dist/C_Table2.js.map +1 -0
  207. package/dist/C_Theme.cjs +4 -0
  208. package/dist/C_Theme.d.cts +2 -0
  209. package/dist/C_Theme.d.ts +2 -0
  210. package/dist/C_Theme.js +3 -0
  211. package/dist/C_Theme2.js +60 -0
  212. package/dist/C_Theme2.js.map +1 -0
  213. package/dist/C_Time-BvZLYraL.css.map +1 -0
  214. package/dist/C_Time.cjs +5 -0
  215. package/dist/C_Time.d.cts +2 -0
  216. package/dist/C_Time.d.ts +2 -0
  217. package/dist/C_Time.js +3 -0
  218. package/dist/C_Time2.js +199 -0
  219. package/dist/C_Time2.js.map +1 -0
  220. package/dist/C_Tree-0GDv--jX.css.map +1 -0
  221. package/dist/C_Tree.cjs +7 -0
  222. package/dist/C_Tree.d.cts +2 -0
  223. package/dist/C_Tree.d.ts +2 -0
  224. package/dist/C_Tree.js +4 -0
  225. package/dist/C_Tree2.js +441 -0
  226. package/dist/C_Tree2.js.map +1 -0
  227. package/dist/C_Upload-BXd3YYLx.css.map +1 -0
  228. package/dist/C_Upload.cjs +12 -0
  229. package/dist/C_Upload.d.cts +2 -0
  230. package/dist/C_Upload.d.ts +2 -0
  231. package/dist/C_Upload.js +4 -0
  232. package/dist/C_Upload2.js +1388 -0
  233. package/dist/C_Upload2.js.map +1 -0
  234. package/dist/C_VideoPlayer-DYG3RL0Q.css.map +1 -0
  235. package/dist/C_VideoPlayer.cjs +23 -0
  236. package/dist/C_VideoPlayer.d.cts +2 -0
  237. package/dist/C_VideoPlayer.d.ts +2 -0
  238. package/dist/C_VideoPlayer.js +3 -0
  239. package/dist/C_VideoPlayer2.js +1932 -0
  240. package/dist/C_VideoPlayer2.js.map +1 -0
  241. package/dist/C_VtableGantt-fhItIiHE.css.map +1 -0
  242. package/dist/C_VtableGantt.cjs +6 -0
  243. package/dist/C_VtableGantt.d.cts +2 -0
  244. package/dist/C_VtableGantt.d.ts +2 -0
  245. package/dist/C_VtableGantt.js +4 -0
  246. package/dist/C_VtableGantt2.js +873 -0
  247. package/dist/C_VtableGantt2.js.map +1 -0
  248. package/dist/C_WaterFall-8sQDFXKg.css.map +1 -0
  249. package/dist/C_WaterFall.cjs +13 -0
  250. package/dist/C_WaterFall.d.cts +2 -0
  251. package/dist/C_WaterFall.d.ts +2 -0
  252. package/dist/C_WaterFall.js +3 -0
  253. package/dist/C_WaterFall2.js +365 -0
  254. package/dist/C_WaterFall2.js.map +1 -0
  255. package/dist/C_WorkFlow-J-dyIuh9.css.map +1 -0
  256. package/dist/C_WorkFlow.cjs +8 -0
  257. package/dist/C_WorkFlow.d.cts +2 -0
  258. package/dist/C_WorkFlow.d.ts +2 -0
  259. package/dist/C_WorkFlow.js +4 -0
  260. package/dist/C_WorkFlow2.js +1984 -0
  261. package/dist/C_WorkFlow2.js.map +1 -0
  262. package/dist/chunk.js +22 -0
  263. package/dist/city.js +4817 -0
  264. package/dist/city.js.map +1 -0
  265. package/dist/constants.d.ts +273 -0
  266. package/dist/constants.d.ts.map +1 -0
  267. package/dist/constants2.d.ts +178 -0
  268. package/dist/constants2.d.ts.map +1 -0
  269. package/dist/constants3.d.ts +475 -0
  270. package/dist/constants3.d.ts.map +1 -0
  271. package/dist/constants4.d.ts +430 -0
  272. package/dist/constants4.d.ts.map +1 -0
  273. package/dist/constants5.d.ts +4283 -0
  274. package/dist/constants5.d.ts.map +1 -0
  275. package/dist/data.d.ts +67 -0
  276. package/dist/data.d.ts.map +1 -0
  277. package/dist/export-helper.js +9 -0
  278. package/dist/index.cjs +409 -0
  279. package/dist/index.d.cts +96 -0
  280. package/dist/index.d.cts.map +1 -0
  281. package/dist/index.d.ts +103 -0
  282. package/dist/index.d.ts.map +1 -0
  283. package/dist/index.js +230 -0
  284. package/dist/index.js.map +1 -0
  285. package/dist/index.vue.d.ts +80 -0
  286. package/dist/index.vue.d.ts.map +1 -0
  287. package/dist/index10.vue.d.ts +72 -0
  288. package/dist/index10.vue.d.ts.map +1 -0
  289. package/dist/index11.vue.d.ts +26 -0
  290. package/dist/index11.vue.d.ts.map +1 -0
  291. package/dist/index12.vue.d.ts +81 -0
  292. package/dist/index12.vue.d.ts.map +1 -0
  293. package/dist/index13.vue.d.ts +55 -0
  294. package/dist/index13.vue.d.ts.map +1 -0
  295. package/dist/index14.vue.d.ts +33 -0
  296. package/dist/index14.vue.d.ts.map +1 -0
  297. package/dist/index15.vue.d.ts +18 -0
  298. package/dist/index15.vue.d.ts.map +1 -0
  299. package/dist/index16.vue.d.ts +662 -0
  300. package/dist/index16.vue.d.ts.map +1 -0
  301. package/dist/index2.vue.d.ts +38 -0
  302. package/dist/index2.vue.d.ts.map +1 -0
  303. package/dist/index3.vue.d.ts +45 -0
  304. package/dist/index3.vue.d.ts.map +1 -0
  305. package/dist/index4.vue.d.ts +31 -0
  306. package/dist/index4.vue.d.ts.map +1 -0
  307. package/dist/index5.vue.d.ts +35 -0
  308. package/dist/index5.vue.d.ts.map +1 -0
  309. package/dist/index6.vue.d.ts +48 -0
  310. package/dist/index6.vue.d.ts.map +1 -0
  311. package/dist/index7.vue.d.ts +56 -0
  312. package/dist/index7.vue.d.ts.map +1 -0
  313. package/dist/index8.vue.d.ts +41 -0
  314. package/dist/index8.vue.d.ts.map +1 -0
  315. package/dist/index9.vue.d.ts +30 -0
  316. package/dist/index9.vue.d.ts.map +1 -0
  317. package/dist/storage.js +31 -0
  318. package/dist/storage.js.map +1 -0
  319. package/dist/style.css +7725 -0
  320. package/dist/useCalendarEvents.d.ts +148 -0
  321. package/dist/useCalendarEvents.d.ts.map +1 -0
  322. package/dist/useCollapsePanel.d.ts +132 -0
  323. package/dist/useCollapsePanel.d.ts.map +1 -0
  324. package/dist/useCropperCore.d.ts +102 -0
  325. package/dist/useCropperCore.d.ts.map +1 -0
  326. package/dist/useDraggableLayout.d.ts +194 -0
  327. package/dist/useDraggableLayout.d.ts.map +1 -0
  328. package/dist/useDynamicFormState.d.ts +4248 -0
  329. package/dist/useDynamicFormState.d.ts.map +1 -0
  330. package/dist/useEdgeInteraction.d.ts +7614 -0
  331. package/dist/useEdgeInteraction.d.ts.map +1 -0
  332. package/dist/useFullscreen.d.ts +166 -0
  333. package/dist/useFullscreen.d.ts.map +1 -0
  334. package/dist/useInfiniteScroll.d.ts +169 -0
  335. package/dist/useInfiniteScroll.d.ts.map +1 -0
  336. package/dist/useModalEdit.d.ts +960 -0
  337. package/dist/useModalEdit.d.ts.map +1 -0
  338. package/dist/useQRCode.d.ts +87 -0
  339. package/dist/useQRCode.d.ts.map +1 -0
  340. package/dist/useSearchState.d.ts +180 -0
  341. package/dist/useSearchState.d.ts.map +1 -0
  342. package/dist/useSignatureHistory.d.ts +189 -0
  343. package/dist/useSignatureHistory.d.ts.map +1 -0
  344. package/dist/useSplitResize.d.ts +158 -0
  345. package/dist/useSplitResize.d.ts.map +1 -0
  346. package/dist/useTimeSelection.d.ts +105 -0
  347. package/dist/useTimeSelection.d.ts.map +1 -0
  348. package/dist/useTreeOperations.d.ts +183 -0
  349. package/dist/useTreeOperations.d.ts.map +1 -0
  350. package/dist/useWorkflowValidation.d.ts +1052 -0
  351. package/dist/useWorkflowValidation.d.ts.map +1 -0
  352. package/package.json +342 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_VtableGantt2.js","names":["showToolbar","title","showFullscreenButton"],"sources":["../src/components/C_VtableGantt/data.ts","../src/components/C_VtableGantt/index.vue","../src/components/C_VtableGantt/index.vue","../src/components/C_VtableGantt/index.vue"],"sourcesContent":["// 甘特图相关类型定义和预设配置\r\n\r\n// ==================== 类型定义 ====================\r\nexport interface GanttTask {\r\n id: string | number\r\n title: string\r\n start: string | Date\r\n end?: string | Date\r\n progress?: number\r\n priority?: string\r\n type?: 'milestone' | 'task'\r\n children?: GanttTask[]\r\n [key: string]: any\r\n}\r\n\r\nexport interface GanttOptions {\r\n [key: string]: any\r\n}\r\n\r\nexport type GanttPreset =\r\n | 'basic'\r\n | 'project'\r\n | 'timeline'\r\n | 'milestone'\r\n | 'official'\r\n\r\n// ==================== 通用主题配置 ====================\r\n// 移除硬编码颜色,让背景透明以跟随系统主题\r\nexport const commonTheme = {\r\n headerStyle: {\r\n borderColor: 'rgba(128, 128, 128, 0.2)',\r\n borderLineWidth: 1,\r\n fontSize: 14,\r\n fontWeight: 'bold',\r\n // bgColor 留空或透明,自动跟随主题\r\n },\r\n bodyStyle: {\r\n borderColor: 'rgba(128, 128, 128, 0.2)',\r\n borderLineWidth: [1, 0, 1, 0],\r\n fontSize: 13,\r\n // bgColor 留空或透明,自动跟随主题\r\n },\r\n}\r\n\r\n// ==================== 预设配置 ====================\r\nexport const presetConfigs = {\r\n basic: {\r\n overscrollBehavior: 'none',\r\n headerRowHeight: 40,\r\n rowHeight: 40,\r\n taskListTable: {\r\n columns: [\r\n { field: 'title', title: '任务名称', width: 220, tree: true },\r\n { field: 'start', title: '开始时间', width: 120 },\r\n { field: 'end', title: '结束时间', width: 120 },\r\n { field: 'progress', title: '进度', width: 80 },\r\n ],\r\n tableWidth: 420,\r\n minTableWidth: 300,\r\n theme: commonTheme,\r\n },\r\n frame: {\r\n outerFrameStyle: {\r\n borderLineWidth: 1,\r\n borderColor: '#e1e4e8',\r\n cornerRadius: 6,\r\n },\r\n verticalSplitLineMoveable: true,\r\n verticalSplitLine: { lineColor: '#e1e4e8', lineWidth: 2 },\r\n },\r\n grid: {\r\n // 移除 weekendBackgroundColor,让它透明跟随主题\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n taskBar: {\r\n startDateField: 'start',\r\n endDateField: 'end',\r\n progressField: 'progress',\r\n moveable: true,\r\n resizable: true,\r\n labelText: '{title}',\r\n labelTextStyle: { fontSize: 12, textAlign: 'left' },\r\n barStyle: {\r\n width: 18,\r\n barColor: '#1890ff',\r\n completedBarColor: '#52c41a',\r\n cornerRadius: 4,\r\n borderLineWidth: 0,\r\n },\r\n milestoneStyle: {\r\n borderColor: '#fa8c16',\r\n borderLineWidth: 1,\r\n fillColor: '#ffd666',\r\n width: 12,\r\n },\r\n },\r\n timelineHeader: {\r\n // 移除 backgroundColor,让它透明跟随主题\r\n colWidth: 60,\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n scales: [\r\n {\r\n unit: 'week',\r\n step: 1,\r\n format: (date: any) => `第${date.dateIndex}周`,\r\n style: { fontSize: 12, fontWeight: 'bold' },\r\n },\r\n {\r\n unit: 'day',\r\n step: 1,\r\n format: (date: any) => `${date.dateIndex}`,\r\n style: { fontSize: 11 },\r\n },\r\n ],\r\n },\r\n scrollStyle: {\r\n scrollRailColor: 'rgba(246,246,246,0.5)',\r\n visible: 'scrolling',\r\n width: 6,\r\n scrollSliderCornerRadius: 2,\r\n scrollSliderColor: '#1890ff',\r\n },\r\n },\r\n\r\n project: {\r\n overscrollBehavior: 'none',\r\n headerRowHeight: 40,\r\n rowHeight: 40,\r\n taskListTable: {\r\n columns: [\r\n {\r\n field: 'title',\r\n title: '任务名称',\r\n width: 200,\r\n tree: true,\r\n editor: 'input',\r\n },\r\n {\r\n field: 'start',\r\n title: '开始时间',\r\n width: 120,\r\n editor: 'date-input',\r\n },\r\n { field: 'end', title: '结束时间', width: 120, editor: 'date-input' },\r\n { field: 'priority', title: '优先级', width: 80, editor: 'input' },\r\n { field: 'progress', title: '进度%', width: 80, editor: 'input' },\r\n ],\r\n tableWidth: 480,\r\n minTableWidth: 350,\r\n maxTableWidth: 800,\r\n theme: commonTheme,\r\n hierarchyExpandLevel: 2,\r\n },\r\n frame: {\r\n outerFrameStyle: {\r\n borderLineWidth: 2,\r\n borderColor: '#e1e4e8',\r\n cornerRadius: 8,\r\n },\r\n verticalSplitLineMoveable: true,\r\n verticalSplitLine: { lineColor: '#e1e4e8', lineWidth: 3 },\r\n },\r\n grid: {\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n taskBar: {\r\n startDateField: 'start',\r\n endDateField: 'end',\r\n progressField: 'progress',\r\n moveable: true,\r\n resizable: true,\r\n hoverBarStyle: { barOverlayColor: 'rgba(24, 144, 255, 0.3)' },\r\n labelText: '{title} {progress}%',\r\n labelTextStyle: {\r\n fontSize: 12,\r\n textAlign: 'left',\r\n textOverflow: 'ellipsis',\r\n },\r\n barStyle: {\r\n width: 20,\r\n barColor: '#1890ff',\r\n completedBarColor: '#52c41a',\r\n cornerRadius: 6,\r\n borderLineWidth: 0,\r\n },\r\n },\r\n timelineHeader: {\r\n colWidth: 60,\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n scales: [\r\n {\r\n unit: 'week',\r\n step: 1,\r\n startOfWeek: 'monday',\r\n format: (date: any) => `第${date.dateIndex}周`,\r\n style: { fontSize: 13, fontWeight: 'bold', textAlign: 'center' },\r\n },\r\n {\r\n unit: 'day',\r\n step: 1,\r\n format: (date: any) => `${date.dateIndex}`,\r\n style: { fontSize: 11, textAlign: 'center' },\r\n },\r\n ],\r\n },\r\n rowSeriesNumber: {\r\n title: '序号',\r\n dragOrder: true,\r\n headerStyle: { borderColor: 'rgba(128, 128, 128, 0.2)' },\r\n style: { borderColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n scrollStyle: {\r\n scrollRailColor: 'rgba(246,246,246,0.5)',\r\n visible: 'scrolling',\r\n width: 6,\r\n scrollSliderCornerRadius: 2,\r\n scrollSliderColor: '#1890ff',\r\n },\r\n },\r\n\r\n timeline: {\r\n overscrollBehavior: 'none',\r\n headerRowHeight: 45,\r\n rowHeight: 40,\r\n taskListTable: {\r\n columns: [\r\n { field: 'title', title: '事件名称', width: 250, tree: true },\r\n { field: 'start', title: '时间', width: 150 },\r\n ],\r\n tableWidth: 400,\r\n minTableWidth: 300,\r\n theme: {\r\n headerStyle: { ...commonTheme.headerStyle },\r\n bodyStyle: commonTheme.bodyStyle,\r\n },\r\n },\r\n frame: {\r\n outerFrameStyle: {\r\n borderLineWidth: 1,\r\n borderColor: 'rgba(128, 128, 128, 0.3)',\r\n cornerRadius: 4,\r\n },\r\n verticalSplitLineMoveable: true,\r\n },\r\n grid: {\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n taskBar: {\r\n startDateField: 'start',\r\n endDateField: 'end',\r\n progressField: 'progress',\r\n moveable: true,\r\n resizable: true,\r\n labelText: '{title}',\r\n labelTextStyle: { fontSize: 12, textAlign: 'left' },\r\n barStyle: {\r\n width: 22,\r\n barColor: '#722ed1',\r\n completedBarColor: '#b37feb',\r\n cornerRadius: 8,\r\n borderLineWidth: 0,\r\n },\r\n },\r\n timelineHeader: {\r\n colWidth: 60,\r\n scales: [\r\n {\r\n unit: 'month',\r\n step: 1,\r\n format: (date: any) => `${date.dateIndex}月`,\r\n style: { fontSize: 13, fontWeight: 'bold' },\r\n },\r\n {\r\n unit: 'day',\r\n step: 1,\r\n format: (date: any) => `${date.dateIndex}`,\r\n style: { fontSize: 11 },\r\n },\r\n ],\r\n },\r\n scrollStyle: {\r\n scrollRailColor: 'rgba(246,246,246,0.5)',\r\n visible: 'scrolling',\r\n width: 6,\r\n scrollSliderCornerRadius: 2,\r\n scrollSliderColor: '#1890ff',\r\n },\r\n },\r\n\r\n milestone: {\r\n overscrollBehavior: 'none',\r\n headerRowHeight: 40,\r\n rowHeight: 38,\r\n taskListTable: {\r\n columns: [\r\n { field: 'title', title: '里程碑', width: 200, tree: true },\r\n { field: 'start', title: '目标日期', width: 120 },\r\n { field: 'priority', title: '重要性', width: 100 },\r\n ],\r\n tableWidth: 400,\r\n minTableWidth: 300,\r\n theme: {\r\n headerStyle: {\r\n ...commonTheme.headerStyle,\r\n borderColor: '#ffa940',\r\n },\r\n bodyStyle: { ...commonTheme.bodyStyle, borderColor: '#ffa940' },\r\n },\r\n },\r\n frame: {\r\n outerFrameStyle: {\r\n borderLineWidth: 2,\r\n borderColor: '#ffa940',\r\n cornerRadius: 6,\r\n },\r\n verticalSplitLineMoveable: true,\r\n },\r\n grid: {\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(255, 168, 64, 0.3)' },\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(255, 168, 64, 0.3)' },\r\n },\r\n taskBar: {\r\n startDateField: 'start',\r\n endDateField: 'start',\r\n labelText: '{title}',\r\n labelTextStyle: { fontSize: 12, fontWeight: 'bold' },\r\n barStyle: { width: 0, barColor: 'transparent' },\r\n milestoneStyle: {\r\n borderColor: '#fa8c16',\r\n borderLineWidth: 2,\r\n fillColor: '#ffd666',\r\n width: 16,\r\n },\r\n },\r\n timelineHeader: {\r\n colWidth: 80,\r\n scales: [\r\n {\r\n unit: 'month',\r\n step: 1,\r\n format: (date: any) => `${date.dateIndex}月`,\r\n style: { fontSize: 13, fontWeight: 'bold' },\r\n },\r\n {\r\n unit: 'week',\r\n step: 1,\r\n format: (date: any) => `第${date.dateIndex}周`,\r\n style: { fontSize: 11 },\r\n },\r\n ],\r\n },\r\n scrollStyle: {\r\n scrollRailColor: 'rgba(246,246,246,0.5)',\r\n visible: 'scrolling',\r\n width: 6,\r\n scrollSliderCornerRadius: 2,\r\n scrollSliderColor: '#1890ff',\r\n },\r\n },\r\n\r\n official: {\r\n overscrollBehavior: 'none',\r\n headerRowHeight: 40,\r\n rowHeight: 40,\r\n taskListTable: {\r\n columns: [\r\n {\r\n field: 'title',\r\n title: 'title',\r\n width: 180,\r\n sort: true,\r\n tree: true,\r\n editor: 'input',\r\n },\r\n {\r\n field: 'start',\r\n title: 'start',\r\n width: 120,\r\n sort: true,\r\n editor: 'date-input',\r\n },\r\n {\r\n field: 'end',\r\n title: 'end',\r\n width: 120,\r\n sort: true,\r\n editor: 'date-input',\r\n },\r\n {\r\n field: 'priority',\r\n title: 'priority',\r\n width: 80,\r\n sort: true,\r\n editor: 'input',\r\n },\r\n {\r\n field: 'progress',\r\n title: 'progress',\r\n width: 80,\r\n sort: true,\r\n editor: 'input',\r\n },\r\n ],\r\n tableWidth: 460,\r\n minTableWidth: 350,\r\n maxTableWidth: 800,\r\n theme: {\r\n headerStyle: {\r\n ...commonTheme.headerStyle,\r\n fontSize: 18,\r\n color: 'red',\r\n },\r\n bodyStyle: {\r\n ...commonTheme.bodyStyle,\r\n fontSize: 16,\r\n color: '#4D4D4D',\r\n },\r\n },\r\n hierarchyExpandLevel: 2,\r\n },\r\n frame: {\r\n outerFrameStyle: {\r\n borderLineWidth: 2,\r\n borderColor: '#e1e4e8',\r\n cornerRadius: 8,\r\n },\r\n verticalSplitLineMoveable: true,\r\n verticalSplitLine: { lineColor: '#e1e4e8', lineWidth: 3 },\r\n horizontalSplitLine: { lineColor: '#e1e4e8', lineWidth: 3 },\r\n },\r\n grid: {\r\n verticalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n horizontalLine: { lineWidth: 1, lineColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n taskBar: {\r\n startDateField: 'start',\r\n endDateField: 'end',\r\n progressField: 'progress',\r\n moveable: true,\r\n resizable: true,\r\n hoverBarStyle: { barOverlayColor: 'rgba(99, 144, 0, 0.4)' },\r\n labelText: '{title} {progress}%',\r\n labelTextStyle: {\r\n fontSize: 16,\r\n textAlign: 'left',\r\n textOverflow: 'ellipsis',\r\n },\r\n barStyle: {\r\n width: 20,\r\n barColor: '#ee8800',\r\n completedBarColor: '#91e8e0',\r\n cornerRadius: 8,\r\n borderLineWidth: 1,\r\n borderColor: 'black',\r\n },\r\n milestoneStyle: {\r\n borderColor: 'red',\r\n borderLineWidth: 1,\r\n fillColor: 'green',\r\n width: 15,\r\n },\r\n },\r\n timelineHeader: {\r\n backgroundColor: '#EEF1F5',\r\n colWidth: 60,\r\n horizontalLine: { lineWidth: 1, lineColor: '#e1e4e8' },\r\n verticalLine: { lineWidth: 1, lineColor: '#e1e4e8' },\r\n scales: [\r\n {\r\n unit: 'week',\r\n step: 1,\r\n startOfWeek: 'sunday',\r\n format: (date: any) => `Week ${date.dateIndex}`,\r\n style: {\r\n fontSize: 20,\r\n fontWeight: 'bold',\r\n color: 'white',\r\n strokeColor: 'black',\r\n textAlign: 'right',\r\n textBaseline: 'bottom',\r\n textStick: true,\r\n },\r\n },\r\n {\r\n unit: 'day',\r\n step: 1,\r\n format: (date: any) => date.dateIndex.toString(),\r\n style: {\r\n fontSize: 20,\r\n fontWeight: 'bold',\r\n color: 'white',\r\n strokeColor: 'black',\r\n textAlign: 'right',\r\n textBaseline: 'bottom',\r\n },\r\n },\r\n ],\r\n },\r\n markLine: [\r\n {\r\n date: '2024-07-28',\r\n style: { lineWidth: 1, lineColor: 'blue', lineDash: [8, 4] },\r\n },\r\n {\r\n date: '2024-08-17',\r\n style: { lineWidth: 2, lineColor: 'red', lineDash: [8, 4] },\r\n },\r\n ],\r\n rowSeriesNumber: {\r\n title: '行号',\r\n dragOrder: true,\r\n headerStyle: { borderColor: 'rgba(128, 128, 128, 0.2)' },\r\n style: { borderColor: 'rgba(128, 128, 128, 0.2)' },\r\n },\r\n scrollStyle: {\r\n scrollRailColor: 'rgba(246,246,246,0.5)',\r\n visible: 'scrolling',\r\n width: 6,\r\n scrollSliderCornerRadius: 2,\r\n scrollSliderColor: '#1890ff',\r\n },\r\n },\r\n}\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-08-01\r\n * @Description: VTable 甘特图组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div class=\"c-vtable-gantt-wrapper\">\r\n <div v-if=\"showToolbar\" class=\"gantt-toolbar\">\r\n <div class=\"toolbar-left\">\r\n <span class=\"gantt-title\">{{ title || \"甘特图\" }}</span>\r\n </div>\r\n <div class=\"toolbar-right\">\r\n <NButton\r\n v-if=\"showFullscreenButton\"\r\n size=\"small\"\r\n @click=\"toggleFullscreen\"\r\n >\r\n <template #icon>\r\n <C_Icon\r\n :name=\"isFullscreen ? 'carbon:minimize' : 'carbon:maximize'\"\r\n :size=\"16\"\r\n color=\"currentColor\"\r\n />\r\n </template>\r\n {{ isFullscreen ? \"退出全屏\" : \"全屏\" }}\r\n </NButton>\r\n </div>\r\n </div>\r\n <div\r\n ref=\"ganttContainerRef\"\r\n class=\"gantt-container\"\r\n :style=\"{ height: containerHeight }\"\r\n />\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport {\r\n presetConfigs,\r\n type GanttTask,\r\n type GanttOptions,\r\n type GanttPreset,\r\n} from \"./data\";\r\n\r\ndefineOptions({ name: \"C_VtableGantt\" });\r\n\r\ninterface Props {\r\n data?: GanttTask[];\r\n options?: GanttOptions;\r\n preset?: GanttPreset;\r\n height?: string | number;\r\n title?: string;\r\n showToolbar?: boolean;\r\n showFullscreenButton?: boolean;\r\n theme?: \"light\" | \"dark\";\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n data: () => [],\r\n options: () => ({}),\r\n preset: \"basic\",\r\n height: \"600px\",\r\n title: \"\",\r\n showToolbar: true,\r\n showFullscreenButton: true,\r\n theme: \"light\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n ganttCreated: [gantt: any];\r\n taskClick: [task: GanttTask, event: Event];\r\n taskChange: [task: GanttTask, changes: any];\r\n}>();\r\n\r\nconst ganttContainerRef = ref<HTMLDivElement>();\r\nconst ganttInstance = ref<any>();\r\nconst isFullscreen = ref(false);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst deepMerge = (target: any, source: any, seen = new WeakMap()): any => {\r\n if (!isObject(target)) return source;\r\n if (!isObject(source)) return target;\r\n if (seen.has(source)) return seen.get(source);\r\n return createMergeResult(target, source, seen);\r\n};\r\n\r\nconst isObject = (value: any): boolean => {\r\n return value !== null && typeof value === \"object\";\r\n};\r\n\r\nconst isSpecialObject = (value: any): boolean => {\r\n return value instanceof Date || value instanceof RegExp;\r\n};\r\n\r\nconst createMergeResult = (\r\n target: any,\r\n source: any,\r\n seen: WeakMap<any, any>,\r\n): any => {\r\n const result = Array.isArray(target) ? [...target] : { ...target };\r\n seen.set(source, result);\r\n for (const key in source) {\r\n if (!source.hasOwnProperty(key)) continue;\r\n const sourceValue = source[key];\r\n const shouldDeepMerge =\r\n isObject(sourceValue) &&\r\n !Array.isArray(sourceValue) &&\r\n !isSpecialObject(sourceValue);\r\n result[key] = shouldDeepMerge\r\n ? deepMerge(target[key] || {}, sourceValue, seen)\r\n : sourceValue;\r\n }\r\n return result;\r\n};\r\n\r\nconst processData = (data: GanttTask[]): GanttTask[] => {\r\n return data.map((item) => ({\r\n ...item,\r\n title: item.title || `任务${item.id}`,\r\n children: item.children ? processData(item.children) : undefined,\r\n }));\r\n};\r\n\r\nconst getThemeConfig = (isDark: boolean) => ({\r\n underlayBackgroundColor: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n timelineHeaderBg: isDark ? \"#2d2d2d\" : \"#EEF1F5\",\r\n gridBg: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n lineColor: isDark ? \"rgba(255, 255, 255, 0.1)\" : \"rgba(128, 128, 128, 0.2)\",\r\n textColor: isDark ? \"#ffffff\" : \"#000000\",\r\n});\r\n\r\nconst applyThemeToTimelineHeader = (timelineHeader: any, themeColors: any) => ({\r\n ...timelineHeader,\r\n backgroundColor: themeColors.timelineHeaderBg,\r\n horizontalLine: {\r\n ...timelineHeader?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...timelineHeader?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n scales: timelineHeader?.scales?.map((scale: any) => ({\r\n ...scale,\r\n style: {\r\n ...scale.style,\r\n color: themeColors.textColor,\r\n },\r\n })),\r\n});\r\n\r\nconst applyThemeToGrid = (grid: any, themeColors: any) => ({\r\n ...grid,\r\n backgroundColor: themeColors.gridBg,\r\n horizontalLine: {\r\n ...grid?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...grid?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n});\r\n\r\nconst buildGanttOptions = (\r\n finalConfig: any,\r\n processedData: GanttTask[],\r\n tableTheme: any,\r\n themeColors: any,\r\n) => ({\r\n ...finalConfig,\r\n records: processedData,\r\n underlayBackgroundColor: themeColors.underlayBackgroundColor,\r\n taskListTable: {\r\n ...finalConfig.taskListTable,\r\n theme: tableTheme,\r\n },\r\n timelineHeader: applyThemeToTimelineHeader(\r\n finalConfig.timelineHeader,\r\n themeColors,\r\n ),\r\n grid: applyThemeToGrid(finalConfig.grid, themeColors),\r\n});\r\n\r\nconst bindGanttEvents = (instance: any) => {\r\n instance.on(\"click_cell\", (args: any) => {\r\n const { record, event } = args || {};\r\n if (record) emit(\"taskClick\", record, event);\r\n });\r\n instance.on(\"change_data\", (args: any) => {\r\n const { record, changes } = args || {};\r\n if (record && changes) emit(\"taskChange\", record, changes);\r\n });\r\n};\r\n\r\nconst initGantt = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n const { Gantt } = await import(\"@visactor/vtable-gantt\");\r\n const { themes } = await import(\"@visactor/vtable\");\r\n const isDark = props.theme === \"dark\";\r\n const presetConfig = presetConfigs[props.preset] || presetConfigs.basic;\r\n const finalConfig = deepMerge(presetConfig, props.options);\r\n const processedData = processData(props.data || []);\r\n if (ganttInstance.value) ganttInstance.value.release();\r\n const tableTheme = isDark ? themes.DARK : themes.DEFAULT;\r\n const themeColors = getThemeConfig(isDark);\r\n const ganttOptions = buildGanttOptions(\r\n finalConfig,\r\n processedData,\r\n tableTheme,\r\n themeColors,\r\n );\r\n ganttInstance.value = new Gantt(ganttContainerRef.value, ganttOptions);\r\n bindGanttEvents(ganttInstance.value);\r\n emit(\"ganttCreated\", ganttInstance.value);\r\n } catch (error) {\r\n console.error(\"甘特图初始化失败:\", error);\r\n }\r\n};\r\n\r\nconst toggleFullscreen = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n if (!document.fullscreenElement) {\r\n await ganttContainerRef.value.requestFullscreen();\r\n isFullscreen.value = true;\r\n } else {\r\n await document.exitFullscreen();\r\n isFullscreen.value = false;\r\n }\r\n setTimeout(() => ganttInstance.value?.resize?.(), 100);\r\n } catch (error) {\r\n console.warn(\"全屏切换失败:\", error);\r\n isFullscreen.value = !isFullscreen.value;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n }\r\n};\r\n\r\nconst handleFullscreenChange = () => {\r\n isFullscreen.value = !!document.fullscreenElement;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n};\r\n\r\nconst updateData = (newData: GanttTask[]) => {\r\n if (ganttInstance.value && newData) {\r\n try {\r\n ganttInstance.value.setRecords(processData(newData));\r\n } catch (error) {\r\n console.warn(\"更新数据失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst updateOptions = (newOptions: GanttOptions) => {\r\n if (ganttInstance.value && newOptions) {\r\n try {\r\n ganttInstance.value.updateOption(newOptions);\r\n } catch (error) {\r\n console.warn(\"更新配置失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst destroyGantt = () => {\r\n if (ganttInstance.value) {\r\n try {\r\n ganttInstance.value.release();\r\n } catch (error) {\r\n console.warn(\"销毁甘特图失败:\", error);\r\n } finally {\r\n ganttInstance.value = undefined;\r\n }\r\n }\r\n};\r\n\r\nwatch(\r\n () => props.data,\r\n (newData) => {\r\n if (newData && ganttInstance.value) updateData(newData);\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => [props.options, props.preset],\r\n () => nextTick(() => initGantt()),\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.theme,\r\n () => nextTick(() => initGantt()),\r\n);\r\n\r\nonMounted(() => {\r\n document.addEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n nextTick(() => setTimeout(() => initGantt(), 100));\r\n});\r\n\r\nonUnmounted(() => {\r\n document.removeEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n destroyGantt();\r\n});\r\n\r\ndefineExpose({\r\n ganttInstance,\r\n updateData,\r\n updateOptions,\r\n toggleFullscreen,\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n@use \"./index.scss\";\r\n</style>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-08-01\r\n * @Description: VTable 甘特图组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div class=\"c-vtable-gantt-wrapper\">\r\n <div v-if=\"showToolbar\" class=\"gantt-toolbar\">\r\n <div class=\"toolbar-left\">\r\n <span class=\"gantt-title\">{{ title || \"甘特图\" }}</span>\r\n </div>\r\n <div class=\"toolbar-right\">\r\n <NButton\r\n v-if=\"showFullscreenButton\"\r\n size=\"small\"\r\n @click=\"toggleFullscreen\"\r\n >\r\n <template #icon>\r\n <C_Icon\r\n :name=\"isFullscreen ? 'carbon:minimize' : 'carbon:maximize'\"\r\n :size=\"16\"\r\n color=\"currentColor\"\r\n />\r\n </template>\r\n {{ isFullscreen ? \"退出全屏\" : \"全屏\" }}\r\n </NButton>\r\n </div>\r\n </div>\r\n <div\r\n ref=\"ganttContainerRef\"\r\n class=\"gantt-container\"\r\n :style=\"{ height: containerHeight }\"\r\n />\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport {\r\n presetConfigs,\r\n type GanttTask,\r\n type GanttOptions,\r\n type GanttPreset,\r\n} from \"./data\";\r\n\r\ndefineOptions({ name: \"C_VtableGantt\" });\r\n\r\ninterface Props {\r\n data?: GanttTask[];\r\n options?: GanttOptions;\r\n preset?: GanttPreset;\r\n height?: string | number;\r\n title?: string;\r\n showToolbar?: boolean;\r\n showFullscreenButton?: boolean;\r\n theme?: \"light\" | \"dark\";\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n data: () => [],\r\n options: () => ({}),\r\n preset: \"basic\",\r\n height: \"600px\",\r\n title: \"\",\r\n showToolbar: true,\r\n showFullscreenButton: true,\r\n theme: \"light\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n ganttCreated: [gantt: any];\r\n taskClick: [task: GanttTask, event: Event];\r\n taskChange: [task: GanttTask, changes: any];\r\n}>();\r\n\r\nconst ganttContainerRef = ref<HTMLDivElement>();\r\nconst ganttInstance = ref<any>();\r\nconst isFullscreen = ref(false);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst deepMerge = (target: any, source: any, seen = new WeakMap()): any => {\r\n if (!isObject(target)) return source;\r\n if (!isObject(source)) return target;\r\n if (seen.has(source)) return seen.get(source);\r\n return createMergeResult(target, source, seen);\r\n};\r\n\r\nconst isObject = (value: any): boolean => {\r\n return value !== null && typeof value === \"object\";\r\n};\r\n\r\nconst isSpecialObject = (value: any): boolean => {\r\n return value instanceof Date || value instanceof RegExp;\r\n};\r\n\r\nconst createMergeResult = (\r\n target: any,\r\n source: any,\r\n seen: WeakMap<any, any>,\r\n): any => {\r\n const result = Array.isArray(target) ? [...target] : { ...target };\r\n seen.set(source, result);\r\n for (const key in source) {\r\n if (!source.hasOwnProperty(key)) continue;\r\n const sourceValue = source[key];\r\n const shouldDeepMerge =\r\n isObject(sourceValue) &&\r\n !Array.isArray(sourceValue) &&\r\n !isSpecialObject(sourceValue);\r\n result[key] = shouldDeepMerge\r\n ? deepMerge(target[key] || {}, sourceValue, seen)\r\n : sourceValue;\r\n }\r\n return result;\r\n};\r\n\r\nconst processData = (data: GanttTask[]): GanttTask[] => {\r\n return data.map((item) => ({\r\n ...item,\r\n title: item.title || `任务${item.id}`,\r\n children: item.children ? processData(item.children) : undefined,\r\n }));\r\n};\r\n\r\nconst getThemeConfig = (isDark: boolean) => ({\r\n underlayBackgroundColor: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n timelineHeaderBg: isDark ? \"#2d2d2d\" : \"#EEF1F5\",\r\n gridBg: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n lineColor: isDark ? \"rgba(255, 255, 255, 0.1)\" : \"rgba(128, 128, 128, 0.2)\",\r\n textColor: isDark ? \"#ffffff\" : \"#000000\",\r\n});\r\n\r\nconst applyThemeToTimelineHeader = (timelineHeader: any, themeColors: any) => ({\r\n ...timelineHeader,\r\n backgroundColor: themeColors.timelineHeaderBg,\r\n horizontalLine: {\r\n ...timelineHeader?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...timelineHeader?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n scales: timelineHeader?.scales?.map((scale: any) => ({\r\n ...scale,\r\n style: {\r\n ...scale.style,\r\n color: themeColors.textColor,\r\n },\r\n })),\r\n});\r\n\r\nconst applyThemeToGrid = (grid: any, themeColors: any) => ({\r\n ...grid,\r\n backgroundColor: themeColors.gridBg,\r\n horizontalLine: {\r\n ...grid?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...grid?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n});\r\n\r\nconst buildGanttOptions = (\r\n finalConfig: any,\r\n processedData: GanttTask[],\r\n tableTheme: any,\r\n themeColors: any,\r\n) => ({\r\n ...finalConfig,\r\n records: processedData,\r\n underlayBackgroundColor: themeColors.underlayBackgroundColor,\r\n taskListTable: {\r\n ...finalConfig.taskListTable,\r\n theme: tableTheme,\r\n },\r\n timelineHeader: applyThemeToTimelineHeader(\r\n finalConfig.timelineHeader,\r\n themeColors,\r\n ),\r\n grid: applyThemeToGrid(finalConfig.grid, themeColors),\r\n});\r\n\r\nconst bindGanttEvents = (instance: any) => {\r\n instance.on(\"click_cell\", (args: any) => {\r\n const { record, event } = args || {};\r\n if (record) emit(\"taskClick\", record, event);\r\n });\r\n instance.on(\"change_data\", (args: any) => {\r\n const { record, changes } = args || {};\r\n if (record && changes) emit(\"taskChange\", record, changes);\r\n });\r\n};\r\n\r\nconst initGantt = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n const { Gantt } = await import(\"@visactor/vtable-gantt\");\r\n const { themes } = await import(\"@visactor/vtable\");\r\n const isDark = props.theme === \"dark\";\r\n const presetConfig = presetConfigs[props.preset] || presetConfigs.basic;\r\n const finalConfig = deepMerge(presetConfig, props.options);\r\n const processedData = processData(props.data || []);\r\n if (ganttInstance.value) ganttInstance.value.release();\r\n const tableTheme = isDark ? themes.DARK : themes.DEFAULT;\r\n const themeColors = getThemeConfig(isDark);\r\n const ganttOptions = buildGanttOptions(\r\n finalConfig,\r\n processedData,\r\n tableTheme,\r\n themeColors,\r\n );\r\n ganttInstance.value = new Gantt(ganttContainerRef.value, ganttOptions);\r\n bindGanttEvents(ganttInstance.value);\r\n emit(\"ganttCreated\", ganttInstance.value);\r\n } catch (error) {\r\n console.error(\"甘特图初始化失败:\", error);\r\n }\r\n};\r\n\r\nconst toggleFullscreen = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n if (!document.fullscreenElement) {\r\n await ganttContainerRef.value.requestFullscreen();\r\n isFullscreen.value = true;\r\n } else {\r\n await document.exitFullscreen();\r\n isFullscreen.value = false;\r\n }\r\n setTimeout(() => ganttInstance.value?.resize?.(), 100);\r\n } catch (error) {\r\n console.warn(\"全屏切换失败:\", error);\r\n isFullscreen.value = !isFullscreen.value;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n }\r\n};\r\n\r\nconst handleFullscreenChange = () => {\r\n isFullscreen.value = !!document.fullscreenElement;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n};\r\n\r\nconst updateData = (newData: GanttTask[]) => {\r\n if (ganttInstance.value && newData) {\r\n try {\r\n ganttInstance.value.setRecords(processData(newData));\r\n } catch (error) {\r\n console.warn(\"更新数据失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst updateOptions = (newOptions: GanttOptions) => {\r\n if (ganttInstance.value && newOptions) {\r\n try {\r\n ganttInstance.value.updateOption(newOptions);\r\n } catch (error) {\r\n console.warn(\"更新配置失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst destroyGantt = () => {\r\n if (ganttInstance.value) {\r\n try {\r\n ganttInstance.value.release();\r\n } catch (error) {\r\n console.warn(\"销毁甘特图失败:\", error);\r\n } finally {\r\n ganttInstance.value = undefined;\r\n }\r\n }\r\n};\r\n\r\nwatch(\r\n () => props.data,\r\n (newData) => {\r\n if (newData && ganttInstance.value) updateData(newData);\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => [props.options, props.preset],\r\n () => nextTick(() => initGantt()),\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.theme,\r\n () => nextTick(() => initGantt()),\r\n);\r\n\r\nonMounted(() => {\r\n document.addEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n nextTick(() => setTimeout(() => initGantt(), 100));\r\n});\r\n\r\nonUnmounted(() => {\r\n document.removeEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n destroyGantt();\r\n});\r\n\r\ndefineExpose({\r\n ganttInstance,\r\n updateData,\r\n updateOptions,\r\n toggleFullscreen,\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n@use \"./index.scss\";\r\n</style>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-08-01\r\n * @Description: VTable 甘特图组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div class=\"c-vtable-gantt-wrapper\">\r\n <div v-if=\"showToolbar\" class=\"gantt-toolbar\">\r\n <div class=\"toolbar-left\">\r\n <span class=\"gantt-title\">{{ title || \"甘特图\" }}</span>\r\n </div>\r\n <div class=\"toolbar-right\">\r\n <NButton\r\n v-if=\"showFullscreenButton\"\r\n size=\"small\"\r\n @click=\"toggleFullscreen\"\r\n >\r\n <template #icon>\r\n <C_Icon\r\n :name=\"isFullscreen ? 'carbon:minimize' : 'carbon:maximize'\"\r\n :size=\"16\"\r\n color=\"currentColor\"\r\n />\r\n </template>\r\n {{ isFullscreen ? \"退出全屏\" : \"全屏\" }}\r\n </NButton>\r\n </div>\r\n </div>\r\n <div\r\n ref=\"ganttContainerRef\"\r\n class=\"gantt-container\"\r\n :style=\"{ height: containerHeight }\"\r\n />\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport {\r\n presetConfigs,\r\n type GanttTask,\r\n type GanttOptions,\r\n type GanttPreset,\r\n} from \"./data\";\r\n\r\ndefineOptions({ name: \"C_VtableGantt\" });\r\n\r\ninterface Props {\r\n data?: GanttTask[];\r\n options?: GanttOptions;\r\n preset?: GanttPreset;\r\n height?: string | number;\r\n title?: string;\r\n showToolbar?: boolean;\r\n showFullscreenButton?: boolean;\r\n theme?: \"light\" | \"dark\";\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n data: () => [],\r\n options: () => ({}),\r\n preset: \"basic\",\r\n height: \"600px\",\r\n title: \"\",\r\n showToolbar: true,\r\n showFullscreenButton: true,\r\n theme: \"light\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n ganttCreated: [gantt: any];\r\n taskClick: [task: GanttTask, event: Event];\r\n taskChange: [task: GanttTask, changes: any];\r\n}>();\r\n\r\nconst ganttContainerRef = ref<HTMLDivElement>();\r\nconst ganttInstance = ref<any>();\r\nconst isFullscreen = ref(false);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst deepMerge = (target: any, source: any, seen = new WeakMap()): any => {\r\n if (!isObject(target)) return source;\r\n if (!isObject(source)) return target;\r\n if (seen.has(source)) return seen.get(source);\r\n return createMergeResult(target, source, seen);\r\n};\r\n\r\nconst isObject = (value: any): boolean => {\r\n return value !== null && typeof value === \"object\";\r\n};\r\n\r\nconst isSpecialObject = (value: any): boolean => {\r\n return value instanceof Date || value instanceof RegExp;\r\n};\r\n\r\nconst createMergeResult = (\r\n target: any,\r\n source: any,\r\n seen: WeakMap<any, any>,\r\n): any => {\r\n const result = Array.isArray(target) ? [...target] : { ...target };\r\n seen.set(source, result);\r\n for (const key in source) {\r\n if (!source.hasOwnProperty(key)) continue;\r\n const sourceValue = source[key];\r\n const shouldDeepMerge =\r\n isObject(sourceValue) &&\r\n !Array.isArray(sourceValue) &&\r\n !isSpecialObject(sourceValue);\r\n result[key] = shouldDeepMerge\r\n ? deepMerge(target[key] || {}, sourceValue, seen)\r\n : sourceValue;\r\n }\r\n return result;\r\n};\r\n\r\nconst processData = (data: GanttTask[]): GanttTask[] => {\r\n return data.map((item) => ({\r\n ...item,\r\n title: item.title || `任务${item.id}`,\r\n children: item.children ? processData(item.children) : undefined,\r\n }));\r\n};\r\n\r\nconst getThemeConfig = (isDark: boolean) => ({\r\n underlayBackgroundColor: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n timelineHeaderBg: isDark ? \"#2d2d2d\" : \"#EEF1F5\",\r\n gridBg: isDark ? \"#1e1e1e\" : \"#ffffff\",\r\n lineColor: isDark ? \"rgba(255, 255, 255, 0.1)\" : \"rgba(128, 128, 128, 0.2)\",\r\n textColor: isDark ? \"#ffffff\" : \"#000000\",\r\n});\r\n\r\nconst applyThemeToTimelineHeader = (timelineHeader: any, themeColors: any) => ({\r\n ...timelineHeader,\r\n backgroundColor: themeColors.timelineHeaderBg,\r\n horizontalLine: {\r\n ...timelineHeader?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...timelineHeader?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n scales: timelineHeader?.scales?.map((scale: any) => ({\r\n ...scale,\r\n style: {\r\n ...scale.style,\r\n color: themeColors.textColor,\r\n },\r\n })),\r\n});\r\n\r\nconst applyThemeToGrid = (grid: any, themeColors: any) => ({\r\n ...grid,\r\n backgroundColor: themeColors.gridBg,\r\n horizontalLine: {\r\n ...grid?.horizontalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n verticalLine: {\r\n ...grid?.verticalLine,\r\n lineColor: themeColors.lineColor,\r\n },\r\n});\r\n\r\nconst buildGanttOptions = (\r\n finalConfig: any,\r\n processedData: GanttTask[],\r\n tableTheme: any,\r\n themeColors: any,\r\n) => ({\r\n ...finalConfig,\r\n records: processedData,\r\n underlayBackgroundColor: themeColors.underlayBackgroundColor,\r\n taskListTable: {\r\n ...finalConfig.taskListTable,\r\n theme: tableTheme,\r\n },\r\n timelineHeader: applyThemeToTimelineHeader(\r\n finalConfig.timelineHeader,\r\n themeColors,\r\n ),\r\n grid: applyThemeToGrid(finalConfig.grid, themeColors),\r\n});\r\n\r\nconst bindGanttEvents = (instance: any) => {\r\n instance.on(\"click_cell\", (args: any) => {\r\n const { record, event } = args || {};\r\n if (record) emit(\"taskClick\", record, event);\r\n });\r\n instance.on(\"change_data\", (args: any) => {\r\n const { record, changes } = args || {};\r\n if (record && changes) emit(\"taskChange\", record, changes);\r\n });\r\n};\r\n\r\nconst initGantt = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n const { Gantt } = await import(\"@visactor/vtable-gantt\");\r\n const { themes } = await import(\"@visactor/vtable\");\r\n const isDark = props.theme === \"dark\";\r\n const presetConfig = presetConfigs[props.preset] || presetConfigs.basic;\r\n const finalConfig = deepMerge(presetConfig, props.options);\r\n const processedData = processData(props.data || []);\r\n if (ganttInstance.value) ganttInstance.value.release();\r\n const tableTheme = isDark ? themes.DARK : themes.DEFAULT;\r\n const themeColors = getThemeConfig(isDark);\r\n const ganttOptions = buildGanttOptions(\r\n finalConfig,\r\n processedData,\r\n tableTheme,\r\n themeColors,\r\n );\r\n ganttInstance.value = new Gantt(ganttContainerRef.value, ganttOptions);\r\n bindGanttEvents(ganttInstance.value);\r\n emit(\"ganttCreated\", ganttInstance.value);\r\n } catch (error) {\r\n console.error(\"甘特图初始化失败:\", error);\r\n }\r\n};\r\n\r\nconst toggleFullscreen = async () => {\r\n if (!ganttContainerRef.value) return;\r\n try {\r\n if (!document.fullscreenElement) {\r\n await ganttContainerRef.value.requestFullscreen();\r\n isFullscreen.value = true;\r\n } else {\r\n await document.exitFullscreen();\r\n isFullscreen.value = false;\r\n }\r\n setTimeout(() => ganttInstance.value?.resize?.(), 100);\r\n } catch (error) {\r\n console.warn(\"全屏切换失败:\", error);\r\n isFullscreen.value = !isFullscreen.value;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n }\r\n};\r\n\r\nconst handleFullscreenChange = () => {\r\n isFullscreen.value = !!document.fullscreenElement;\r\n nextTick(() => ganttInstance.value?.resize?.());\r\n};\r\n\r\nconst updateData = (newData: GanttTask[]) => {\r\n if (ganttInstance.value && newData) {\r\n try {\r\n ganttInstance.value.setRecords(processData(newData));\r\n } catch (error) {\r\n console.warn(\"更新数据失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst updateOptions = (newOptions: GanttOptions) => {\r\n if (ganttInstance.value && newOptions) {\r\n try {\r\n ganttInstance.value.updateOption(newOptions);\r\n } catch (error) {\r\n console.warn(\"更新配置失败:\", error);\r\n }\r\n }\r\n};\r\n\r\nconst destroyGantt = () => {\r\n if (ganttInstance.value) {\r\n try {\r\n ganttInstance.value.release();\r\n } catch (error) {\r\n console.warn(\"销毁甘特图失败:\", error);\r\n } finally {\r\n ganttInstance.value = undefined;\r\n }\r\n }\r\n};\r\n\r\nwatch(\r\n () => props.data,\r\n (newData) => {\r\n if (newData && ganttInstance.value) updateData(newData);\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => [props.options, props.preset],\r\n () => nextTick(() => initGantt()),\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.theme,\r\n () => nextTick(() => initGantt()),\r\n);\r\n\r\nonMounted(() => {\r\n document.addEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n nextTick(() => setTimeout(() => initGantt(), 100));\r\n});\r\n\r\nonUnmounted(() => {\r\n document.removeEventListener(\"fullscreenchange\", handleFullscreenChange);\r\n destroyGantt();\r\n});\r\n\r\ndefineExpose({\r\n ganttInstance,\r\n updateData,\r\n updateOptions,\r\n toggleFullscreen,\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n@use \"./index.scss\";\r\n</style>\r\n"],"mappings":"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"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_WaterFall-8sQDFXKg.css","names":[],"sources":["../src/components/C_WaterFall/index.vue?vue&type=style&index=0&scoped=d436c326&lang.scss"],"sourcesContent":[".c-waterfall[data-v-d436c326] {\n width: 100%;\n}\n.waterfall__body[data-v-d436c326] {\n overflow: hidden;\n}\n.waterfall__item[data-v-d436c326] {\n cursor: pointer;\n}\n.waterfall__card[data-v-d436c326] {\n overflow: hidden;\n border-radius: 8px;\n border: 1px solid var(--border-color);\n background: var(--card-color);\n box-shadow: 0 1px 4px rgba(0, 0, 0, 0.04);\n transition: box-shadow 0.2s ease, transform 0.2s ease;\n}\n.waterfall__card[data-v-d436c326]:hover {\n box-shadow: 0 4px 16px rgba(0, 0, 0, 0.1);\n transform: translateY(-2px);\n}\n.waterfall__image[data-v-d436c326] {\n display: block;\n width: 100%;\n object-fit: cover;\n background: var(--body-color);\n}\n.waterfall__title[data-v-d436c326] {\n padding: 10px 12px;\n font-size: 13px;\n line-height: 1.5;\n color: var(--text-color-1);\n word-break: break-all;\n}\n.waterfall__skeleton[data-v-d436c326] {\n overflow: hidden;\n border-radius: 8px;\n}\n.waterfall__skeleton-inner[data-v-d436c326] {\n width: 100%;\n height: 100%;\n border-radius: 8px;\n background: linear-gradient(90deg, var(--body-color) 25%, color-mix(in srgb, var(--body-color) 80%, var(--border-color)) 37%, var(--body-color) 63%);\n background-size: 400% 100%;\n animation: shimmer-d436c326 1.4s ease infinite;\n}\n@keyframes shimmer-d436c326 {\n0% {\n background-position: 100% 50%;\n}\n100% {\n background-position: 0 50%;\n}\n}\n.waterfall__footer[data-v-d436c326] {\n padding: 24px 0;\n text-align: center;\n}\n.waterfall__sentinel[data-v-d436c326] {\n height: 1px;\n}\n.waterfall__status[data-v-d436c326] {\n display: inline-flex;\n gap: 8px;\n align-items: center;\n font-size: 13px;\n color: var(--text-color-3);\n}\n.waterfall__status--done[data-v-d436c326] {\n color: var(--text-color-4);\n}"],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA"}
@@ -0,0 +1,13 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_C_WaterFall = require('./C_WaterFall.js');
3
+
4
+ exports.C_WaterFall = require_C_WaterFall.C_WaterFall_default;
5
+ exports.DEFAULT_ANIMATION_DURATION = require_C_WaterFall.DEFAULT_ANIMATION_DURATION;
6
+ exports.DEFAULT_BREAKPOINTS = require_C_WaterFall.DEFAULT_BREAKPOINTS;
7
+ exports.DEFAULT_GAP = require_C_WaterFall.DEFAULT_GAP;
8
+ exports.DEFAULT_SKELETON_COUNT = require_C_WaterFall.DEFAULT_SKELETON_COUNT;
9
+ exports.INFINITE_SCROLL_THRESHOLD = require_C_WaterFall.INFINITE_SCROLL_THRESHOLD;
10
+ exports.SKELETON_HEIGHT_RANGE = require_C_WaterFall.SKELETON_HEIGHT_RANGE;
11
+ exports.useInfiniteScroll = require_C_WaterFall.useInfiniteScroll;
12
+ exports.useResponsiveColumns = require_C_WaterFall.useResponsiveColumns;
13
+ exports.useWaterFallLayout = require_C_WaterFall.useWaterFallLayout;
@@ -0,0 +1,2 @@
1
+ import { _ as WaterFallLayoutItem, a as DEFAULT_BREAKPOINTS, c as INFINITE_SCROLL_THRESHOLD, d as InfiniteScrollStatus, f as WaterFallBreakpoint, g as WaterFallItem, h as WaterFallExpose, i as DEFAULT_ANIMATION_DURATION, l as SKELETON_HEIGHT_RANGE, m as WaterFallEmits, n as useWaterFallLayout, o as DEFAULT_GAP, p as WaterFallColumn, r as useResponsiveColumns, s as DEFAULT_SKELETON_COUNT, t as useInfiniteScroll, u as _default, v as WaterFallProps } from "./useInfiniteScroll.js";
2
+ export { _default as C_WaterFall, DEFAULT_ANIMATION_DURATION, DEFAULT_BREAKPOINTS, DEFAULT_GAP, DEFAULT_SKELETON_COUNT, INFINITE_SCROLL_THRESHOLD, type InfiniteScrollStatus, SKELETON_HEIGHT_RANGE, type WaterFallBreakpoint, type WaterFallColumn, type WaterFallEmits, type WaterFallExpose, type WaterFallItem, type WaterFallLayoutItem, type WaterFallProps, useInfiniteScroll, useResponsiveColumns, useWaterFallLayout };
@@ -0,0 +1,2 @@
1
+ import { _ as WaterFallLayoutItem, a as DEFAULT_BREAKPOINTS, c as INFINITE_SCROLL_THRESHOLD, d as InfiniteScrollStatus, f as WaterFallBreakpoint, g as WaterFallItem, h as WaterFallExpose, i as DEFAULT_ANIMATION_DURATION, l as SKELETON_HEIGHT_RANGE, m as WaterFallEmits, n as useWaterFallLayout, o as DEFAULT_GAP, p as WaterFallColumn, r as useResponsiveColumns, s as DEFAULT_SKELETON_COUNT, t as useInfiniteScroll, u as _default, v as WaterFallProps } from "./useInfiniteScroll.js";
2
+ export { _default as C_WaterFall, DEFAULT_ANIMATION_DURATION, DEFAULT_BREAKPOINTS, DEFAULT_GAP, DEFAULT_SKELETON_COUNT, INFINITE_SCROLL_THRESHOLD, type InfiniteScrollStatus, SKELETON_HEIGHT_RANGE, type WaterFallBreakpoint, type WaterFallColumn, type WaterFallEmits, type WaterFallExpose, type WaterFallItem, type WaterFallLayoutItem, type WaterFallProps, useInfiniteScroll, useResponsiveColumns, useWaterFallLayout };
@@ -0,0 +1,3 @@
1
+ import { a as DEFAULT_ANIMATION_DURATION, c as DEFAULT_SKELETON_COUNT, i as useResponsiveColumns, l as INFINITE_SCROLL_THRESHOLD, n as useInfiniteScroll, o as DEFAULT_BREAKPOINTS, r as useWaterFallLayout, s as DEFAULT_GAP, t as C_WaterFall_default, u as SKELETON_HEIGHT_RANGE } from "./C_WaterFall2.js";
2
+
3
+ export { C_WaterFall_default as C_WaterFall, DEFAULT_ANIMATION_DURATION, DEFAULT_BREAKPOINTS, DEFAULT_GAP, DEFAULT_SKELETON_COUNT, INFINITE_SCROLL_THRESHOLD, SKELETON_HEIGHT_RANGE, useInfiniteScroll, useResponsiveColumns, useWaterFallLayout };
@@ -0,0 +1,365 @@
1
+ import { t as export_helper_default } from "./export-helper.js";
2
+ import { Fragment, computed, createCommentVNode, createElementBlock, createElementVNode, createVNode, defineComponent, normalizeStyle, onBeforeUnmount, onMounted, openBlock, readonly, ref, renderList, renderSlot, toDisplayString, unref, watch } from "vue";
3
+ import { NSpin } from "naive-ui";
4
+
5
+ //#region src/components/C_WaterFall/constants.ts
6
+ const DEFAULT_GAP = 16;
7
+ const DEFAULT_ANIMATION_DURATION = 300;
8
+ const DEFAULT_SKELETON_COUNT = 8;
9
+ const INFINITE_SCROLL_THRESHOLD = 200;
10
+ const DEFAULT_BREAKPOINTS = [
11
+ {
12
+ minWidth: 1600,
13
+ columns: 6
14
+ },
15
+ {
16
+ minWidth: 1200,
17
+ columns: 5
18
+ },
19
+ {
20
+ minWidth: 992,
21
+ columns: 4
22
+ },
23
+ {
24
+ minWidth: 768,
25
+ columns: 3
26
+ },
27
+ {
28
+ minWidth: 480,
29
+ columns: 2
30
+ },
31
+ {
32
+ minWidth: 0,
33
+ columns: 1
34
+ }
35
+ ];
36
+ const SKELETON_HEIGHT_RANGE = [180, 360];
37
+
38
+ //#endregion
39
+ //#region src/components/C_WaterFall/composables/useResponsiveColumns.ts
40
+ function useResponsiveColumns(containerRef, fixedColumns, breakpoints) {
41
+ const columns = ref(4);
42
+ const containerWidth = ref(0);
43
+ function resolveColumns(width) {
44
+ if (fixedColumns?.value && fixedColumns.value > 0) return fixedColumns.value;
45
+ const sorted = [...breakpoints?.value?.length ? breakpoints.value : DEFAULT_BREAKPOINTS].sort((a, b) => b.minWidth - a.minWidth);
46
+ for (const bp of sorted) if (width >= bp.minWidth) return bp.columns;
47
+ return 1;
48
+ }
49
+ let resizeObserver = null;
50
+ function startObserving() {
51
+ const el = containerRef.value;
52
+ if (!el) return;
53
+ resizeObserver = new ResizeObserver((entries) => {
54
+ for (const entry of entries) {
55
+ const { width } = entry.contentRect;
56
+ containerWidth.value = width;
57
+ columns.value = resolveColumns(width);
58
+ }
59
+ });
60
+ resizeObserver.observe(el);
61
+ const rect = el.getBoundingClientRect();
62
+ containerWidth.value = rect.width;
63
+ columns.value = resolveColumns(rect.width);
64
+ }
65
+ function stopObserving() {
66
+ resizeObserver?.disconnect();
67
+ resizeObserver = null;
68
+ }
69
+ onMounted(startObserving);
70
+ onBeforeUnmount(stopObserving);
71
+ watch(containerRef, () => {
72
+ stopObserving();
73
+ startObserving();
74
+ });
75
+ watch([() => fixedColumns?.value, () => breakpoints?.value], () => {
76
+ if (containerWidth.value > 0) columns.value = resolveColumns(containerWidth.value);
77
+ });
78
+ return {
79
+ columns: readonly(columns),
80
+ containerWidth: readonly(containerWidth)
81
+ };
82
+ }
83
+
84
+ //#endregion
85
+ //#region src/components/C_WaterFall/composables/useWaterFallLayout.ts
86
+ function useWaterFallLayout(items, columns, containerWidth, gap) {
87
+ const layoutItems = ref([]);
88
+ const containerHeight = ref(0);
89
+ const imageHeightCache = /* @__PURE__ */ new Map();
90
+ function cacheImageHeight(id, realHeight) {
91
+ imageHeightCache.set(id, realHeight);
92
+ }
93
+ function calculate() {
94
+ const cols = columns.value;
95
+ const width = containerWidth.value;
96
+ const g = gap.value ?? DEFAULT_GAP;
97
+ if (cols <= 0 || width <= 0 || items.value.length === 0) {
98
+ layoutItems.value = [];
99
+ containerHeight.value = 0;
100
+ return;
101
+ }
102
+ const colWidth = (width - (cols - 1) * g) / cols;
103
+ const columnState = Array.from({ length: cols }, (_, i) => ({
104
+ index: i,
105
+ height: 0
106
+ }));
107
+ const result = [];
108
+ for (const item of items.value) {
109
+ const shortest = columnState.reduce((min, col) => col.height < min.height ? col : min);
110
+ const cached = imageHeightCache.get(item.id);
111
+ const itemHeight = cached ? cached : item.width > 0 ? item.height / item.width * colWidth : colWidth;
112
+ const x = shortest.index * (colWidth + g);
113
+ const y = shortest.height;
114
+ result.push({
115
+ item,
116
+ columnIndex: shortest.index,
117
+ x,
118
+ y,
119
+ width: colWidth,
120
+ height: itemHeight
121
+ });
122
+ shortest.height = y + itemHeight + g;
123
+ }
124
+ layoutItems.value = result;
125
+ containerHeight.value = Math.max(...columnState.map((c) => c.height)) - g;
126
+ }
127
+ watch([
128
+ items,
129
+ () => items.value.length,
130
+ columns,
131
+ containerWidth,
132
+ gap
133
+ ], calculate, { immediate: true });
134
+ return {
135
+ layoutItems: readonly(layoutItems),
136
+ containerHeight: readonly(containerHeight),
137
+ cacheImageHeight,
138
+ relayout: calculate
139
+ };
140
+ }
141
+
142
+ //#endregion
143
+ //#region src/components/C_WaterFall/composables/useInfiniteScroll.ts
144
+ function useInfiniteScroll(sentinelRef, enabled, loading, noMore, onLoadMore) {
145
+ const status = ref("idle");
146
+ let observer = null;
147
+ function handleIntersect(entries) {
148
+ if (!entries[0]?.isIntersecting) return;
149
+ if (!enabled.value || loading.value || noMore.value) return;
150
+ status.value = "loading";
151
+ onLoadMore();
152
+ }
153
+ function startObserving() {
154
+ const el = sentinelRef.value;
155
+ if (!el || !enabled.value) return;
156
+ observer = new IntersectionObserver(handleIntersect, { rootMargin: `${INFINITE_SCROLL_THRESHOLD}px 0px` });
157
+ observer.observe(el);
158
+ }
159
+ function stopObserving() {
160
+ observer?.disconnect();
161
+ observer = null;
162
+ }
163
+ watch([loading, noMore], () => {
164
+ if (noMore.value) status.value = "no-more";
165
+ else if (loading.value) status.value = "loading";
166
+ else status.value = "idle";
167
+ });
168
+ onMounted(startObserving);
169
+ onBeforeUnmount(stopObserving);
170
+ watch([sentinelRef, enabled], () => {
171
+ stopObserving();
172
+ startObserving();
173
+ });
174
+ return { status: readonly(status) };
175
+ }
176
+
177
+ //#endregion
178
+ //#region src/components/C_WaterFall/index.vue?vue&type=script&setup=true&lang.ts
179
+ const _hoisted_1 = ["onClick"];
180
+ const _hoisted_2 = { class: "waterfall__card" };
181
+ const _hoisted_3 = [
182
+ "src",
183
+ "alt",
184
+ "loading",
185
+ "onLoad",
186
+ "onError"
187
+ ];
188
+ const _hoisted_4 = {
189
+ key: 0,
190
+ class: "waterfall__title"
191
+ };
192
+ const _hoisted_5 = {
193
+ key: 0,
194
+ class: "waterfall__footer"
195
+ };
196
+ const _hoisted_6 = {
197
+ key: 0,
198
+ class: "waterfall__status"
199
+ };
200
+ const _hoisted_7 = {
201
+ key: 1,
202
+ class: "waterfall__status waterfall__status--done"
203
+ };
204
+ var index_vue_vue_type_script_setup_true_lang_default = /* @__PURE__ */ defineComponent({
205
+ name: "C_WaterFall",
206
+ __name: "index",
207
+ props: {
208
+ items: {},
209
+ columns: { default: void 0 },
210
+ gap: { default: DEFAULT_GAP },
211
+ lazy: {
212
+ type: Boolean,
213
+ default: true
214
+ },
215
+ infinite: {
216
+ type: Boolean,
217
+ default: false
218
+ },
219
+ skeleton: {
220
+ type: Boolean,
221
+ default: true
222
+ },
223
+ skeletonCount: { default: DEFAULT_SKELETON_COUNT },
224
+ animationDuration: { default: DEFAULT_ANIMATION_DURATION },
225
+ breakpoints: { default: void 0 },
226
+ loading: {
227
+ type: Boolean,
228
+ default: false
229
+ },
230
+ noMore: {
231
+ type: Boolean,
232
+ default: false
233
+ }
234
+ },
235
+ emits: [
236
+ "load-more",
237
+ "item-click",
238
+ "image-loaded",
239
+ "image-error"
240
+ ],
241
+ setup(__props, { expose: __expose, emit: __emit }) {
242
+ const props = __props;
243
+ const emit = __emit;
244
+ const containerRef = ref();
245
+ const sentinelRef = ref();
246
+ const { columns, containerWidth } = useResponsiveColumns(containerRef, computed(() => props.columns), computed(() => props.breakpoints));
247
+ const { layoutItems, containerHeight, cacheImageHeight, relayout } = useWaterFallLayout(computed(() => props.items), columns, containerWidth, computed(() => props.gap));
248
+ const showSkeleton = computed(() => props.skeleton && props.loading && layoutItems.value.length === 0);
249
+ const skeletonItems = computed(() => {
250
+ const cols = columns.value;
251
+ const width = containerWidth.value;
252
+ const g = props.gap;
253
+ if (cols <= 0 || width <= 0) return [];
254
+ const colWidth = (width - (cols - 1) * g) / cols;
255
+ const count = props.skeletonCount ?? DEFAULT_SKELETON_COUNT;
256
+ const colHeights = Array(cols).fill(0);
257
+ const result = [];
258
+ for (let i = 0; i < count; i++) {
259
+ const minIdx = colHeights.indexOf(Math.min(...colHeights));
260
+ const h = SKELETON_HEIGHT_RANGE[0] + Math.random() * (SKELETON_HEIGHT_RANGE[1] - SKELETON_HEIGHT_RANGE[0]);
261
+ result.push({
262
+ x: minIdx * (colWidth + g),
263
+ y: colHeights[minIdx],
264
+ width: colWidth,
265
+ height: h
266
+ });
267
+ colHeights[minIdx] += h + g;
268
+ }
269
+ return result;
270
+ });
271
+ const skeletonHeight = computed(() => {
272
+ if (skeletonItems.value.length === 0) return 0;
273
+ return Math.max(...skeletonItems.value.map((s) => s.y + s.height));
274
+ });
275
+ const bodyHeight = computed(() => showSkeleton.value ? skeletonHeight.value : containerHeight.value);
276
+ const { status: scrollStatus } = useInfiniteScroll(sentinelRef, computed(() => props.infinite), computed(() => props.loading), computed(() => props.noMore), () => emit("load-more"));
277
+ function handleImageLoaded(lay, event) {
278
+ const img = event.target;
279
+ if (img.naturalHeight && img.naturalWidth) {
280
+ const realHeight = img.naturalHeight / img.naturalWidth * lay.width;
281
+ cacheImageHeight(lay.item.id, realHeight);
282
+ }
283
+ emit("image-loaded", lay.item);
284
+ }
285
+ function handleImageError(lay) {
286
+ emit("image-error", lay.item);
287
+ }
288
+ function scrollToTop() {
289
+ containerRef.value?.scrollIntoView({
290
+ behavior: "smooth",
291
+ block: "start"
292
+ });
293
+ }
294
+ __expose({
295
+ relayout,
296
+ scrollToTop,
297
+ getColumns: () => columns.value,
298
+ getContainerHeight: () => containerHeight.value
299
+ });
300
+ return (_ctx, _cache) => {
301
+ return openBlock(), createElementBlock("div", {
302
+ ref_key: "containerRef",
303
+ ref: containerRef,
304
+ class: "c-waterfall"
305
+ }, [createElementVNode("div", {
306
+ class: "waterfall__body",
307
+ style: normalizeStyle({
308
+ height: `${bodyHeight.value}px`,
309
+ position: "relative"
310
+ })
311
+ }, [showSkeleton.value ? (openBlock(true), createElementBlock(Fragment, { key: 0 }, renderList(skeletonItems.value, (sk, idx) => {
312
+ return openBlock(), createElementBlock("div", {
313
+ key: `sk-${idx}`,
314
+ class: "waterfall__skeleton",
315
+ style: normalizeStyle({
316
+ position: "absolute",
317
+ left: `${sk.x}px`,
318
+ top: `${sk.y}px`,
319
+ width: `${sk.width}px`,
320
+ height: `${sk.height}px`,
321
+ transition: `all ${props.animationDuration ?? unref(DEFAULT_ANIMATION_DURATION)}ms ease`
322
+ })
323
+ }, [renderSlot(_ctx.$slots, "skeleton", {}, () => [_cache[0] || (_cache[0] = createElementVNode("div", { class: "waterfall__skeleton-inner" }, null, -1))], true)], 4);
324
+ }), 128)) : createCommentVNode("v-if", true), (openBlock(true), createElementBlock(Fragment, null, renderList(unref(layoutItems), (lay, index) => {
325
+ return openBlock(), createElementBlock("div", {
326
+ key: lay.item.id,
327
+ class: "waterfall__item",
328
+ style: normalizeStyle({
329
+ position: "absolute",
330
+ left: `${lay.x}px`,
331
+ top: `${lay.y}px`,
332
+ width: `${lay.width}px`,
333
+ transition: `all ${props.animationDuration ?? unref(DEFAULT_ANIMATION_DURATION)}ms ease`
334
+ }),
335
+ onClick: ($event) => emit("item-click", lay.item, index)
336
+ }, [renderSlot(_ctx.$slots, "item", {
337
+ item: lay.item,
338
+ index,
339
+ width: lay.width,
340
+ height: lay.height
341
+ }, () => [createElementVNode("div", _hoisted_2, [createElementVNode("img", {
342
+ src: lay.item.src,
343
+ alt: lay.item.title || "",
344
+ loading: props.lazy ? "lazy" : "eager",
345
+ class: "waterfall__image",
346
+ style: normalizeStyle({ height: `${lay.height}px` }),
347
+ onLoad: ($event) => handleImageLoaded(lay, $event),
348
+ onError: ($event) => handleImageError(lay)
349
+ }, null, 44, _hoisted_3), lay.item.title ? (openBlock(), createElementBlock("div", _hoisted_4, toDisplayString(lay.item.title), 1)) : createCommentVNode("v-if", true)])], true)], 12, _hoisted_1);
350
+ }), 128))], 4), props.infinite ? (openBlock(), createElementBlock("div", _hoisted_5, [createElementVNode("div", {
351
+ ref_key: "sentinelRef",
352
+ ref: sentinelRef,
353
+ class: "waterfall__sentinel"
354
+ }, null, 512), renderSlot(_ctx.$slots, "footer", { status: unref(scrollStatus) }, () => [props.loading ? (openBlock(), createElementBlock("div", _hoisted_6, [createVNode(unref(NSpin), { size: "small" }), _cache[1] || (_cache[1] = createElementVNode("span", null, "加载中…", -1))])) : props.noMore ? (openBlock(), createElementBlock("div", _hoisted_7, " 没有更多了 ")) : createCommentVNode("v-if", true)], true)])) : createCommentVNode("v-if", true)], 512);
355
+ };
356
+ }
357
+ });
358
+
359
+ //#endregion
360
+ //#region src/components/C_WaterFall/index.vue
361
+ var C_WaterFall_default = /* @__PURE__ */ export_helper_default(index_vue_vue_type_script_setup_true_lang_default, [["__scopeId", "data-v-d436c326"]]);
362
+
363
+ //#endregion
364
+ export { DEFAULT_ANIMATION_DURATION as a, DEFAULT_SKELETON_COUNT as c, useResponsiveColumns as i, INFINITE_SCROLL_THRESHOLD as l, useInfiniteScroll as n, DEFAULT_BREAKPOINTS as o, useWaterFallLayout as r, DEFAULT_GAP as s, C_WaterFall_default as t, SKELETON_HEIGHT_RANGE as u };
365
+ //# sourceMappingURL=C_WaterFall2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_WaterFall2.js","names":[],"sources":["../src/components/C_WaterFall/constants.ts","../src/components/C_WaterFall/composables/useResponsiveColumns.ts","../src/components/C_WaterFall/composables/useWaterFallLayout.ts","../src/components/C_WaterFall/composables/useInfiniteScroll.ts","../src/components/C_WaterFall/index.vue","../src/components/C_WaterFall/index.vue","../src/components/C_WaterFall/index.vue"],"sourcesContent":["import type { WaterFallBreakpoint } from \"./types\";\r\n\r\nexport const DEFAULT_GAP = 16;\r\nexport const DEFAULT_ANIMATION_DURATION = 300;\r\nexport const DEFAULT_SKELETON_COUNT = 8;\r\nexport const INFINITE_SCROLL_THRESHOLD = 200;\r\n\r\nexport const DEFAULT_BREAKPOINTS: WaterFallBreakpoint[] = [\r\n { minWidth: 1600, columns: 6 },\r\n { minWidth: 1200, columns: 5 },\r\n { minWidth: 992, columns: 4 },\r\n { minWidth: 768, columns: 3 },\r\n { minWidth: 480, columns: 2 },\r\n { minWidth: 0, columns: 1 },\r\n];\r\n\r\nexport const SKELETON_HEIGHT_RANGE: [number, number] = [180, 360];\r\n","import { ref, readonly, watch, onMounted, onBeforeUnmount } from \"vue\";\r\nimport type { Ref } from \"vue\";\r\nimport type { WaterFallBreakpoint } from \"../types\";\r\nimport { DEFAULT_BREAKPOINTS } from \"../constants\";\r\n\r\nexport function useResponsiveColumns(\r\n containerRef: Ref<HTMLElement | undefined>,\r\n fixedColumns?: Ref<number | undefined>,\r\n breakpoints?: Ref<WaterFallBreakpoint[] | undefined>,\r\n) {\r\n const columns = ref(4);\r\n const containerWidth = ref(0);\r\n\r\n function resolveColumns(width: number): number {\r\n if (fixedColumns?.value && fixedColumns.value > 0)\r\n return fixedColumns.value;\r\n const bps = breakpoints?.value?.length\r\n ? breakpoints.value\r\n : DEFAULT_BREAKPOINTS;\r\n const sorted = [...bps].sort((a, b) => b.minWidth - a.minWidth);\r\n for (const bp of sorted) {\r\n if (width >= bp.minWidth) return bp.columns;\r\n }\r\n return 1;\r\n }\r\n\r\n let resizeObserver: ResizeObserver | null = null;\r\n\r\n function startObserving() {\r\n const el = containerRef.value;\r\n if (!el) return;\r\n resizeObserver = new ResizeObserver((entries) => {\r\n for (const entry of entries) {\r\n const { width } = entry.contentRect;\r\n containerWidth.value = width;\r\n columns.value = resolveColumns(width);\r\n }\r\n });\r\n resizeObserver.observe(el);\r\n const rect = el.getBoundingClientRect();\r\n containerWidth.value = rect.width;\r\n columns.value = resolveColumns(rect.width);\r\n }\r\n\r\n function stopObserving() {\r\n resizeObserver?.disconnect();\r\n resizeObserver = null;\r\n }\r\n\r\n onMounted(startObserving);\r\n onBeforeUnmount(stopObserving);\r\n\r\n watch(containerRef, () => {\r\n stopObserving();\r\n startObserving();\r\n });\r\n\r\n watch([() => fixedColumns?.value, () => breakpoints?.value], () => {\r\n if (containerWidth.value > 0) {\r\n columns.value = resolveColumns(containerWidth.value);\r\n }\r\n });\r\n\r\n return {\r\n columns: readonly(columns),\r\n containerWidth: readonly(containerWidth),\r\n };\r\n}\r\n","import { ref, readonly, watch } from \"vue\";\r\nimport type { Ref } from \"vue\";\r\nimport type {\r\n WaterFallItem,\r\n WaterFallLayoutItem,\r\n WaterFallColumn,\r\n} from \"../types\";\r\nimport { DEFAULT_GAP } from \"../constants\";\r\n\r\nexport function useWaterFallLayout(\r\n items: Ref<WaterFallItem[]>,\r\n columns: Readonly<Ref<number>>,\r\n containerWidth: Readonly<Ref<number>>,\r\n gap: Ref<number>,\r\n) {\r\n const layoutItems = ref<WaterFallLayoutItem[]>([]);\r\n const containerHeight = ref(0);\r\n const imageHeightCache = new Map<string | number, number>();\r\n\r\n function cacheImageHeight(id: string | number, realHeight: number) {\r\n imageHeightCache.set(id, realHeight);\r\n }\r\n\r\n function calculate() {\r\n const cols = columns.value;\r\n const width = containerWidth.value;\r\n const g = gap.value ?? DEFAULT_GAP;\r\n\r\n if (cols <= 0 || width <= 0 || items.value.length === 0) {\r\n layoutItems.value = [];\r\n containerHeight.value = 0;\r\n return;\r\n }\r\n\r\n const colWidth = (width - (cols - 1) * g) / cols;\r\n const columnState: WaterFallColumn[] = Array.from(\r\n { length: cols },\r\n (_, i) => ({ index: i, height: 0 }),\r\n );\r\n\r\n const result: WaterFallLayoutItem[] = [];\r\n\r\n for (const item of items.value) {\r\n const shortest = columnState.reduce((min, col) =>\r\n col.height < min.height ? col : min,\r\n );\r\n\r\n const cached = imageHeightCache.get(item.id);\r\n const itemHeight = cached\r\n ? cached\r\n : item.width > 0\r\n ? (item.height / item.width) * colWidth\r\n : colWidth;\r\n\r\n const x = shortest.index * (colWidth + g);\r\n const y = shortest.height;\r\n\r\n result.push({\r\n item,\r\n columnIndex: shortest.index,\r\n x,\r\n y,\r\n width: colWidth,\r\n height: itemHeight,\r\n });\r\n\r\n shortest.height = y + itemHeight + g;\r\n }\r\n\r\n layoutItems.value = result;\r\n containerHeight.value = Math.max(...columnState.map((c) => c.height)) - g;\r\n }\r\n\r\n watch(\r\n [items, () => items.value.length, columns, containerWidth, gap],\r\n calculate,\r\n { immediate: true },\r\n );\r\n\r\n return {\r\n layoutItems: readonly(layoutItems),\r\n containerHeight: readonly(containerHeight),\r\n cacheImageHeight,\r\n relayout: calculate,\r\n };\r\n}\r\n","import { ref, readonly, watch, onMounted, onBeforeUnmount } from \"vue\";\r\nimport type { Ref } from \"vue\";\r\nimport type { InfiniteScrollStatus } from \"../types\";\r\nimport { INFINITE_SCROLL_THRESHOLD } from \"../constants\";\r\n\r\nexport function useInfiniteScroll(\r\n sentinelRef: Ref<HTMLElement | undefined>,\r\n enabled: Ref<boolean>,\r\n loading: Ref<boolean>,\r\n noMore: Ref<boolean>,\r\n onLoadMore: () => void,\r\n) {\r\n const status = ref<InfiniteScrollStatus>(\"idle\");\r\n let observer: IntersectionObserver | null = null;\r\n\r\n function handleIntersect(entries: IntersectionObserverEntry[]) {\r\n const entry = entries[0];\r\n if (!entry?.isIntersecting) return;\r\n if (!enabled.value || loading.value || noMore.value) return;\r\n status.value = \"loading\";\r\n onLoadMore();\r\n }\r\n\r\n function startObserving() {\r\n const el = sentinelRef.value;\r\n if (!el || !enabled.value) return;\r\n observer = new IntersectionObserver(handleIntersect, {\r\n rootMargin: `${INFINITE_SCROLL_THRESHOLD}px 0px`,\r\n });\r\n observer.observe(el);\r\n }\r\n\r\n function stopObserving() {\r\n observer?.disconnect();\r\n observer = null;\r\n }\r\n\r\n watch([loading, noMore], () => {\r\n if (noMore.value) {\r\n status.value = \"no-more\";\r\n } else if (loading.value) {\r\n status.value = \"loading\";\r\n } else {\r\n status.value = \"idle\";\r\n }\r\n });\r\n\r\n onMounted(startObserving);\r\n onBeforeUnmount(stopObserving);\r\n\r\n watch([sentinelRef, enabled], () => {\r\n stopObserving();\r\n startObserving();\r\n });\r\n\r\n return { status: readonly(status) };\r\n}\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2026-02-26\r\n * @Description: 瀑布流布局组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div ref=\"containerRef\" class=\"c-waterfall\">\r\n <div\r\n class=\"waterfall__body\"\r\n :style=\"{ height: `${bodyHeight}px`, position: 'relative' }\"\r\n >\r\n <template v-if=\"showSkeleton\">\r\n <div\r\n v-for=\"(sk, idx) in skeletonItems\"\r\n :key=\"`sk-${idx}`\"\r\n class=\"waterfall__skeleton\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${sk.x}px`,\r\n top: `${sk.y}px`,\r\n width: `${sk.width}px`,\r\n height: `${sk.height}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n >\r\n <slot name=\"skeleton\">\r\n <div class=\"waterfall__skeleton-inner\" />\r\n </slot>\r\n </div>\r\n </template>\r\n\r\n <div\r\n v-for=\"(lay, index) in layoutItems\"\r\n :key=\"lay.item.id\"\r\n class=\"waterfall__item\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${lay.x}px`,\r\n top: `${lay.y}px`,\r\n width: `${lay.width}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n @click=\"emit('item-click', lay.item, index)\"\r\n >\r\n <slot\r\n name=\"item\"\r\n :item=\"lay.item\"\r\n :index=\"index\"\r\n :width=\"lay.width\"\r\n :height=\"lay.height\"\r\n >\r\n <div class=\"waterfall__card\">\r\n <img\r\n :src=\"lay.item.src\"\r\n :alt=\"lay.item.title || ''\"\r\n :loading=\"props.lazy ? 'lazy' : 'eager'\"\r\n class=\"waterfall__image\"\r\n :style=\"{ height: `${lay.height}px` }\"\r\n @load=\"handleImageLoaded(lay, $event)\"\r\n @error=\"handleImageError(lay)\"\r\n />\r\n <div v-if=\"lay.item.title\" class=\"waterfall__title\">\r\n {{ lay.item.title }}\r\n </div>\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n\r\n <div v-if=\"props.infinite\" class=\"waterfall__footer\">\r\n <div ref=\"sentinelRef\" class=\"waterfall__sentinel\" />\r\n <slot name=\"footer\" :status=\"scrollStatus\">\r\n <div v-if=\"props.loading\" class=\"waterfall__status\">\r\n <NSpin size=\"small\" />\r\n <span>加载中…</span>\r\n </div>\r\n <div\r\n v-else-if=\"props.noMore\"\r\n class=\"waterfall__status waterfall__status--done\"\r\n >\r\n 没有更多了\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport type {\r\n WaterFallItem,\r\n WaterFallLayoutItem,\r\n WaterFallProps,\r\n WaterFallExpose,\r\n WaterFallBreakpoint,\r\n} from \"./types\";\r\nimport type { Ref } from \"vue\";\r\nimport {\r\n DEFAULT_GAP,\r\n DEFAULT_ANIMATION_DURATION,\r\n DEFAULT_SKELETON_COUNT,\r\n SKELETON_HEIGHT_RANGE,\r\n} from \"./constants\";\r\nimport { useResponsiveColumns } from \"./composables/useResponsiveColumns\";\r\nimport { useWaterFallLayout } from \"./composables/useWaterFallLayout\";\r\nimport { useInfiniteScroll } from \"./composables/useInfiniteScroll\";\r\n\r\ndefineOptions({ name: \"C_WaterFall\" });\r\n\r\nconst props = withDefaults(defineProps<WaterFallProps>(), {\r\n columns: undefined,\r\n gap: DEFAULT_GAP,\r\n lazy: true,\r\n infinite: false,\r\n skeleton: true,\r\n skeletonCount: DEFAULT_SKELETON_COUNT,\r\n animationDuration: DEFAULT_ANIMATION_DURATION,\r\n breakpoints: undefined,\r\n loading: false,\r\n noMore: false,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"load-more\": [];\r\n \"item-click\": [item: WaterFallItem, index: number];\r\n \"image-loaded\": [item: WaterFallItem];\r\n \"image-error\": [item: WaterFallItem];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement>();\r\nconst sentinelRef = ref<HTMLElement>();\r\n\r\nconst fixedColumns = computed(() => props.columns);\r\nconst breakpointsRef = computed(() => props.breakpoints);\r\nconst { columns, containerWidth } = useResponsiveColumns(\r\n containerRef,\r\n fixedColumns as Ref<number | undefined>,\r\n breakpointsRef as Ref<WaterFallBreakpoint[] | undefined>,\r\n);\r\n\r\nconst itemsRef = computed(() => props.items);\r\nconst gapRef = computed(() => props.gap);\r\nconst { layoutItems, containerHeight, cacheImageHeight, relayout } =\r\n useWaterFallLayout(itemsRef, columns, containerWidth, gapRef);\r\n\r\nconst showSkeleton = computed(\r\n () => props.skeleton && props.loading && layoutItems.value.length === 0,\r\n);\r\n\r\nconst skeletonItems = computed(() => {\r\n const cols = columns.value;\r\n const width = containerWidth.value;\r\n const g = props.gap;\r\n if (cols <= 0 || width <= 0) return [];\r\n\r\n const colWidth = (width - (cols - 1) * g) / cols;\r\n const count = props.skeletonCount ?? DEFAULT_SKELETON_COUNT;\r\n const colHeights = Array(cols).fill(0);\r\n const result: { x: number; y: number; width: number; height: number }[] = [];\r\n\r\n for (let i = 0; i < count; i++) {\r\n const minIdx = colHeights.indexOf(Math.min(...colHeights));\r\n const h =\r\n SKELETON_HEIGHT_RANGE[0] +\r\n Math.random() * (SKELETON_HEIGHT_RANGE[1] - SKELETON_HEIGHT_RANGE[0]);\r\n result.push({\r\n x: minIdx * (colWidth + g),\r\n y: colHeights[minIdx],\r\n width: colWidth,\r\n height: h,\r\n });\r\n colHeights[minIdx] += h + g;\r\n }\r\n return result;\r\n});\r\n\r\nconst skeletonHeight = computed(() => {\r\n if (skeletonItems.value.length === 0) return 0;\r\n return Math.max(...skeletonItems.value.map((s) => s.y + s.height));\r\n});\r\n\r\nconst bodyHeight = computed(() =>\r\n showSkeleton.value ? skeletonHeight.value : containerHeight.value,\r\n);\r\n\r\nconst infiniteEnabled = computed(() => props.infinite);\r\nconst loadingRef = computed(() => props.loading);\r\nconst noMoreRef = computed(() => props.noMore);\r\n\r\nconst { status: scrollStatus } = useInfiniteScroll(\r\n sentinelRef,\r\n infiniteEnabled,\r\n loadingRef,\r\n noMoreRef,\r\n () => emit(\"load-more\"),\r\n);\r\n\r\nfunction handleImageLoaded(lay: WaterFallLayoutItem, event: Event) {\r\n const img = event.target as HTMLImageElement;\r\n if (img.naturalHeight && img.naturalWidth) {\r\n const realHeight = (img.naturalHeight / img.naturalWidth) * lay.width;\r\n cacheImageHeight(lay.item.id, realHeight);\r\n }\r\n emit(\"image-loaded\", lay.item);\r\n}\r\n\r\nfunction handleImageError(lay: WaterFallLayoutItem) {\r\n emit(\"image-error\", lay.item);\r\n}\r\n\r\nfunction scrollToTop() {\r\n containerRef.value?.scrollIntoView({ behavior: \"smooth\", block: \"start\" });\r\n}\r\n\r\ndefineExpose<WaterFallExpose>({\r\n relayout,\r\n scrollToTop,\r\n getColumns: () => columns.value,\r\n getContainerHeight: () => containerHeight.value,\r\n});\r\n</script>\r\n\r\n<style scoped lang=\"scss\">\r\n.c-waterfall {\r\n width: 100%;\r\n}\r\n\r\n.waterfall__body {\r\n overflow: hidden;\r\n}\r\n\r\n.waterfall__item {\r\n cursor: pointer;\r\n}\r\n\r\n.waterfall__card {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n border: 1px solid var(--border-color);\r\n background: var(--card-color);\r\n box-shadow: 0 1px 4px rgba(0, 0, 0, 0.04);\r\n transition:\r\n box-shadow 0.2s ease,\r\n transform 0.2s ease;\r\n\r\n &:hover {\r\n box-shadow: 0 4px 16px rgba(0, 0, 0, 0.1);\r\n transform: translateY(-2px);\r\n }\r\n}\r\n\r\n.waterfall__image {\r\n display: block;\r\n width: 100%;\r\n object-fit: cover;\r\n background: var(--body-color);\r\n}\r\n\r\n.waterfall__title {\r\n padding: 10px 12px;\r\n font-size: 13px;\r\n line-height: 1.5;\r\n color: var(--text-color-1);\r\n word-break: break-all;\r\n}\r\n\r\n.waterfall__skeleton {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n}\r\n\r\n.waterfall__skeleton-inner {\r\n width: 100%;\r\n height: 100%;\r\n border-radius: 8px;\r\n background: linear-gradient(\r\n 90deg,\r\n var(--body-color) 25%,\r\n color-mix(in srgb, var(--body-color) 80%, var(--border-color)) 37%,\r\n var(--body-color) 63%\r\n );\r\n background-size: 400% 100%;\r\n animation: shimmer 1.4s ease infinite;\r\n}\r\n\r\n@keyframes shimmer {\r\n 0% {\r\n background-position: 100% 50%;\r\n }\r\n 100% {\r\n background-position: 0 50%;\r\n }\r\n}\r\n\r\n.waterfall__footer {\r\n padding: 24px 0;\r\n text-align: center;\r\n}\r\n\r\n.waterfall__sentinel {\r\n height: 1px;\r\n}\r\n\r\n.waterfall__status {\r\n display: inline-flex;\r\n gap: 8px;\r\n align-items: center;\r\n font-size: 13px;\r\n color: var(--text-color-3);\r\n\r\n &--done {\r\n color: var(--text-color-4);\r\n }\r\n}\r\n</style>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2026-02-26\r\n * @Description: 瀑布流布局组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div ref=\"containerRef\" class=\"c-waterfall\">\r\n <div\r\n class=\"waterfall__body\"\r\n :style=\"{ height: `${bodyHeight}px`, position: 'relative' }\"\r\n >\r\n <template v-if=\"showSkeleton\">\r\n <div\r\n v-for=\"(sk, idx) in skeletonItems\"\r\n :key=\"`sk-${idx}`\"\r\n class=\"waterfall__skeleton\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${sk.x}px`,\r\n top: `${sk.y}px`,\r\n width: `${sk.width}px`,\r\n height: `${sk.height}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n >\r\n <slot name=\"skeleton\">\r\n <div class=\"waterfall__skeleton-inner\" />\r\n </slot>\r\n </div>\r\n </template>\r\n\r\n <div\r\n v-for=\"(lay, index) in layoutItems\"\r\n :key=\"lay.item.id\"\r\n class=\"waterfall__item\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${lay.x}px`,\r\n top: `${lay.y}px`,\r\n width: `${lay.width}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n @click=\"emit('item-click', lay.item, index)\"\r\n >\r\n <slot\r\n name=\"item\"\r\n :item=\"lay.item\"\r\n :index=\"index\"\r\n :width=\"lay.width\"\r\n :height=\"lay.height\"\r\n >\r\n <div class=\"waterfall__card\">\r\n <img\r\n :src=\"lay.item.src\"\r\n :alt=\"lay.item.title || ''\"\r\n :loading=\"props.lazy ? 'lazy' : 'eager'\"\r\n class=\"waterfall__image\"\r\n :style=\"{ height: `${lay.height}px` }\"\r\n @load=\"handleImageLoaded(lay, $event)\"\r\n @error=\"handleImageError(lay)\"\r\n />\r\n <div v-if=\"lay.item.title\" class=\"waterfall__title\">\r\n {{ lay.item.title }}\r\n </div>\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n\r\n <div v-if=\"props.infinite\" class=\"waterfall__footer\">\r\n <div ref=\"sentinelRef\" class=\"waterfall__sentinel\" />\r\n <slot name=\"footer\" :status=\"scrollStatus\">\r\n <div v-if=\"props.loading\" class=\"waterfall__status\">\r\n <NSpin size=\"small\" />\r\n <span>加载中…</span>\r\n </div>\r\n <div\r\n v-else-if=\"props.noMore\"\r\n class=\"waterfall__status waterfall__status--done\"\r\n >\r\n 没有更多了\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport type {\r\n WaterFallItem,\r\n WaterFallLayoutItem,\r\n WaterFallProps,\r\n WaterFallExpose,\r\n WaterFallBreakpoint,\r\n} from \"./types\";\r\nimport type { Ref } from \"vue\";\r\nimport {\r\n DEFAULT_GAP,\r\n DEFAULT_ANIMATION_DURATION,\r\n DEFAULT_SKELETON_COUNT,\r\n SKELETON_HEIGHT_RANGE,\r\n} from \"./constants\";\r\nimport { useResponsiveColumns } from \"./composables/useResponsiveColumns\";\r\nimport { useWaterFallLayout } from \"./composables/useWaterFallLayout\";\r\nimport { useInfiniteScroll } from \"./composables/useInfiniteScroll\";\r\n\r\ndefineOptions({ name: \"C_WaterFall\" });\r\n\r\nconst props = withDefaults(defineProps<WaterFallProps>(), {\r\n columns: undefined,\r\n gap: DEFAULT_GAP,\r\n lazy: true,\r\n infinite: false,\r\n skeleton: true,\r\n skeletonCount: DEFAULT_SKELETON_COUNT,\r\n animationDuration: DEFAULT_ANIMATION_DURATION,\r\n breakpoints: undefined,\r\n loading: false,\r\n noMore: false,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"load-more\": [];\r\n \"item-click\": [item: WaterFallItem, index: number];\r\n \"image-loaded\": [item: WaterFallItem];\r\n \"image-error\": [item: WaterFallItem];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement>();\r\nconst sentinelRef = ref<HTMLElement>();\r\n\r\nconst fixedColumns = computed(() => props.columns);\r\nconst breakpointsRef = computed(() => props.breakpoints);\r\nconst { columns, containerWidth } = useResponsiveColumns(\r\n containerRef,\r\n fixedColumns as Ref<number | undefined>,\r\n breakpointsRef as Ref<WaterFallBreakpoint[] | undefined>,\r\n);\r\n\r\nconst itemsRef = computed(() => props.items);\r\nconst gapRef = computed(() => props.gap);\r\nconst { layoutItems, containerHeight, cacheImageHeight, relayout } =\r\n useWaterFallLayout(itemsRef, columns, containerWidth, gapRef);\r\n\r\nconst showSkeleton = computed(\r\n () => props.skeleton && props.loading && layoutItems.value.length === 0,\r\n);\r\n\r\nconst skeletonItems = computed(() => {\r\n const cols = columns.value;\r\n const width = containerWidth.value;\r\n const g = props.gap;\r\n if (cols <= 0 || width <= 0) return [];\r\n\r\n const colWidth = (width - (cols - 1) * g) / cols;\r\n const count = props.skeletonCount ?? DEFAULT_SKELETON_COUNT;\r\n const colHeights = Array(cols).fill(0);\r\n const result: { x: number; y: number; width: number; height: number }[] = [];\r\n\r\n for (let i = 0; i < count; i++) {\r\n const minIdx = colHeights.indexOf(Math.min(...colHeights));\r\n const h =\r\n SKELETON_HEIGHT_RANGE[0] +\r\n Math.random() * (SKELETON_HEIGHT_RANGE[1] - SKELETON_HEIGHT_RANGE[0]);\r\n result.push({\r\n x: minIdx * (colWidth + g),\r\n y: colHeights[minIdx],\r\n width: colWidth,\r\n height: h,\r\n });\r\n colHeights[minIdx] += h + g;\r\n }\r\n return result;\r\n});\r\n\r\nconst skeletonHeight = computed(() => {\r\n if (skeletonItems.value.length === 0) return 0;\r\n return Math.max(...skeletonItems.value.map((s) => s.y + s.height));\r\n});\r\n\r\nconst bodyHeight = computed(() =>\r\n showSkeleton.value ? skeletonHeight.value : containerHeight.value,\r\n);\r\n\r\nconst infiniteEnabled = computed(() => props.infinite);\r\nconst loadingRef = computed(() => props.loading);\r\nconst noMoreRef = computed(() => props.noMore);\r\n\r\nconst { status: scrollStatus } = useInfiniteScroll(\r\n sentinelRef,\r\n infiniteEnabled,\r\n loadingRef,\r\n noMoreRef,\r\n () => emit(\"load-more\"),\r\n);\r\n\r\nfunction handleImageLoaded(lay: WaterFallLayoutItem, event: Event) {\r\n const img = event.target as HTMLImageElement;\r\n if (img.naturalHeight && img.naturalWidth) {\r\n const realHeight = (img.naturalHeight / img.naturalWidth) * lay.width;\r\n cacheImageHeight(lay.item.id, realHeight);\r\n }\r\n emit(\"image-loaded\", lay.item);\r\n}\r\n\r\nfunction handleImageError(lay: WaterFallLayoutItem) {\r\n emit(\"image-error\", lay.item);\r\n}\r\n\r\nfunction scrollToTop() {\r\n containerRef.value?.scrollIntoView({ behavior: \"smooth\", block: \"start\" });\r\n}\r\n\r\ndefineExpose<WaterFallExpose>({\r\n relayout,\r\n scrollToTop,\r\n getColumns: () => columns.value,\r\n getContainerHeight: () => containerHeight.value,\r\n});\r\n</script>\r\n\r\n<style scoped lang=\"scss\">\r\n.c-waterfall {\r\n width: 100%;\r\n}\r\n\r\n.waterfall__body {\r\n overflow: hidden;\r\n}\r\n\r\n.waterfall__item {\r\n cursor: pointer;\r\n}\r\n\r\n.waterfall__card {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n border: 1px solid var(--border-color);\r\n background: var(--card-color);\r\n box-shadow: 0 1px 4px rgba(0, 0, 0, 0.04);\r\n transition:\r\n box-shadow 0.2s ease,\r\n transform 0.2s ease;\r\n\r\n &:hover {\r\n box-shadow: 0 4px 16px rgba(0, 0, 0, 0.1);\r\n transform: translateY(-2px);\r\n }\r\n}\r\n\r\n.waterfall__image {\r\n display: block;\r\n width: 100%;\r\n object-fit: cover;\r\n background: var(--body-color);\r\n}\r\n\r\n.waterfall__title {\r\n padding: 10px 12px;\r\n font-size: 13px;\r\n line-height: 1.5;\r\n color: var(--text-color-1);\r\n word-break: break-all;\r\n}\r\n\r\n.waterfall__skeleton {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n}\r\n\r\n.waterfall__skeleton-inner {\r\n width: 100%;\r\n height: 100%;\r\n border-radius: 8px;\r\n background: linear-gradient(\r\n 90deg,\r\n var(--body-color) 25%,\r\n color-mix(in srgb, var(--body-color) 80%, var(--border-color)) 37%,\r\n var(--body-color) 63%\r\n );\r\n background-size: 400% 100%;\r\n animation: shimmer 1.4s ease infinite;\r\n}\r\n\r\n@keyframes shimmer {\r\n 0% {\r\n background-position: 100% 50%;\r\n }\r\n 100% {\r\n background-position: 0 50%;\r\n }\r\n}\r\n\r\n.waterfall__footer {\r\n padding: 24px 0;\r\n text-align: center;\r\n}\r\n\r\n.waterfall__sentinel {\r\n height: 1px;\r\n}\r\n\r\n.waterfall__status {\r\n display: inline-flex;\r\n gap: 8px;\r\n align-items: center;\r\n font-size: 13px;\r\n color: var(--text-color-3);\r\n\r\n &--done {\r\n color: var(--text-color-4);\r\n }\r\n}\r\n</style>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2026-02-26\r\n * @Description: 瀑布流布局组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div ref=\"containerRef\" class=\"c-waterfall\">\r\n <div\r\n class=\"waterfall__body\"\r\n :style=\"{ height: `${bodyHeight}px`, position: 'relative' }\"\r\n >\r\n <template v-if=\"showSkeleton\">\r\n <div\r\n v-for=\"(sk, idx) in skeletonItems\"\r\n :key=\"`sk-${idx}`\"\r\n class=\"waterfall__skeleton\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${sk.x}px`,\r\n top: `${sk.y}px`,\r\n width: `${sk.width}px`,\r\n height: `${sk.height}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n >\r\n <slot name=\"skeleton\">\r\n <div class=\"waterfall__skeleton-inner\" />\r\n </slot>\r\n </div>\r\n </template>\r\n\r\n <div\r\n v-for=\"(lay, index) in layoutItems\"\r\n :key=\"lay.item.id\"\r\n class=\"waterfall__item\"\r\n :style=\"{\r\n position: 'absolute',\r\n left: `${lay.x}px`,\r\n top: `${lay.y}px`,\r\n width: `${lay.width}px`,\r\n transition: `all ${props.animationDuration ?? DEFAULT_ANIMATION_DURATION}ms ease`,\r\n }\"\r\n @click=\"emit('item-click', lay.item, index)\"\r\n >\r\n <slot\r\n name=\"item\"\r\n :item=\"lay.item\"\r\n :index=\"index\"\r\n :width=\"lay.width\"\r\n :height=\"lay.height\"\r\n >\r\n <div class=\"waterfall__card\">\r\n <img\r\n :src=\"lay.item.src\"\r\n :alt=\"lay.item.title || ''\"\r\n :loading=\"props.lazy ? 'lazy' : 'eager'\"\r\n class=\"waterfall__image\"\r\n :style=\"{ height: `${lay.height}px` }\"\r\n @load=\"handleImageLoaded(lay, $event)\"\r\n @error=\"handleImageError(lay)\"\r\n />\r\n <div v-if=\"lay.item.title\" class=\"waterfall__title\">\r\n {{ lay.item.title }}\r\n </div>\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n\r\n <div v-if=\"props.infinite\" class=\"waterfall__footer\">\r\n <div ref=\"sentinelRef\" class=\"waterfall__sentinel\" />\r\n <slot name=\"footer\" :status=\"scrollStatus\">\r\n <div v-if=\"props.loading\" class=\"waterfall__status\">\r\n <NSpin size=\"small\" />\r\n <span>加载中…</span>\r\n </div>\r\n <div\r\n v-else-if=\"props.noMore\"\r\n class=\"waterfall__status waterfall__status--done\"\r\n >\r\n 没有更多了\r\n </div>\r\n </slot>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport type {\r\n WaterFallItem,\r\n WaterFallLayoutItem,\r\n WaterFallProps,\r\n WaterFallExpose,\r\n WaterFallBreakpoint,\r\n} from \"./types\";\r\nimport type { Ref } from \"vue\";\r\nimport {\r\n DEFAULT_GAP,\r\n DEFAULT_ANIMATION_DURATION,\r\n DEFAULT_SKELETON_COUNT,\r\n SKELETON_HEIGHT_RANGE,\r\n} from \"./constants\";\r\nimport { useResponsiveColumns } from \"./composables/useResponsiveColumns\";\r\nimport { useWaterFallLayout } from \"./composables/useWaterFallLayout\";\r\nimport { useInfiniteScroll } from \"./composables/useInfiniteScroll\";\r\n\r\ndefineOptions({ name: \"C_WaterFall\" });\r\n\r\nconst props = withDefaults(defineProps<WaterFallProps>(), {\r\n columns: undefined,\r\n gap: DEFAULT_GAP,\r\n lazy: true,\r\n infinite: false,\r\n skeleton: true,\r\n skeletonCount: DEFAULT_SKELETON_COUNT,\r\n animationDuration: DEFAULT_ANIMATION_DURATION,\r\n breakpoints: undefined,\r\n loading: false,\r\n noMore: false,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"load-more\": [];\r\n \"item-click\": [item: WaterFallItem, index: number];\r\n \"image-loaded\": [item: WaterFallItem];\r\n \"image-error\": [item: WaterFallItem];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement>();\r\nconst sentinelRef = ref<HTMLElement>();\r\n\r\nconst fixedColumns = computed(() => props.columns);\r\nconst breakpointsRef = computed(() => props.breakpoints);\r\nconst { columns, containerWidth } = useResponsiveColumns(\r\n containerRef,\r\n fixedColumns as Ref<number | undefined>,\r\n breakpointsRef as Ref<WaterFallBreakpoint[] | undefined>,\r\n);\r\n\r\nconst itemsRef = computed(() => props.items);\r\nconst gapRef = computed(() => props.gap);\r\nconst { layoutItems, containerHeight, cacheImageHeight, relayout } =\r\n useWaterFallLayout(itemsRef, columns, containerWidth, gapRef);\r\n\r\nconst showSkeleton = computed(\r\n () => props.skeleton && props.loading && layoutItems.value.length === 0,\r\n);\r\n\r\nconst skeletonItems = computed(() => {\r\n const cols = columns.value;\r\n const width = containerWidth.value;\r\n const g = props.gap;\r\n if (cols <= 0 || width <= 0) return [];\r\n\r\n const colWidth = (width - (cols - 1) * g) / cols;\r\n const count = props.skeletonCount ?? DEFAULT_SKELETON_COUNT;\r\n const colHeights = Array(cols).fill(0);\r\n const result: { x: number; y: number; width: number; height: number }[] = [];\r\n\r\n for (let i = 0; i < count; i++) {\r\n const minIdx = colHeights.indexOf(Math.min(...colHeights));\r\n const h =\r\n SKELETON_HEIGHT_RANGE[0] +\r\n Math.random() * (SKELETON_HEIGHT_RANGE[1] - SKELETON_HEIGHT_RANGE[0]);\r\n result.push({\r\n x: minIdx * (colWidth + g),\r\n y: colHeights[minIdx],\r\n width: colWidth,\r\n height: h,\r\n });\r\n colHeights[minIdx] += h + g;\r\n }\r\n return result;\r\n});\r\n\r\nconst skeletonHeight = computed(() => {\r\n if (skeletonItems.value.length === 0) return 0;\r\n return Math.max(...skeletonItems.value.map((s) => s.y + s.height));\r\n});\r\n\r\nconst bodyHeight = computed(() =>\r\n showSkeleton.value ? skeletonHeight.value : containerHeight.value,\r\n);\r\n\r\nconst infiniteEnabled = computed(() => props.infinite);\r\nconst loadingRef = computed(() => props.loading);\r\nconst noMoreRef = computed(() => props.noMore);\r\n\r\nconst { status: scrollStatus } = useInfiniteScroll(\r\n sentinelRef,\r\n infiniteEnabled,\r\n loadingRef,\r\n noMoreRef,\r\n () => emit(\"load-more\"),\r\n);\r\n\r\nfunction handleImageLoaded(lay: WaterFallLayoutItem, event: Event) {\r\n const img = event.target as HTMLImageElement;\r\n if (img.naturalHeight && img.naturalWidth) {\r\n const realHeight = (img.naturalHeight / img.naturalWidth) * lay.width;\r\n cacheImageHeight(lay.item.id, realHeight);\r\n }\r\n emit(\"image-loaded\", lay.item);\r\n}\r\n\r\nfunction handleImageError(lay: WaterFallLayoutItem) {\r\n emit(\"image-error\", lay.item);\r\n}\r\n\r\nfunction scrollToTop() {\r\n containerRef.value?.scrollIntoView({ behavior: \"smooth\", block: \"start\" });\r\n}\r\n\r\ndefineExpose<WaterFallExpose>({\r\n relayout,\r\n scrollToTop,\r\n getColumns: () => columns.value,\r\n getContainerHeight: () => containerHeight.value,\r\n});\r\n</script>\r\n\r\n<style scoped lang=\"scss\">\r\n.c-waterfall {\r\n width: 100%;\r\n}\r\n\r\n.waterfall__body {\r\n overflow: hidden;\r\n}\r\n\r\n.waterfall__item {\r\n cursor: pointer;\r\n}\r\n\r\n.waterfall__card {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n border: 1px solid var(--border-color);\r\n background: var(--card-color);\r\n box-shadow: 0 1px 4px rgba(0, 0, 0, 0.04);\r\n transition:\r\n box-shadow 0.2s ease,\r\n transform 0.2s ease;\r\n\r\n &:hover {\r\n box-shadow: 0 4px 16px rgba(0, 0, 0, 0.1);\r\n transform: translateY(-2px);\r\n }\r\n}\r\n\r\n.waterfall__image {\r\n display: block;\r\n width: 100%;\r\n object-fit: cover;\r\n background: var(--body-color);\r\n}\r\n\r\n.waterfall__title {\r\n padding: 10px 12px;\r\n font-size: 13px;\r\n line-height: 1.5;\r\n color: var(--text-color-1);\r\n word-break: break-all;\r\n}\r\n\r\n.waterfall__skeleton {\r\n overflow: hidden;\r\n border-radius: 8px;\r\n}\r\n\r\n.waterfall__skeleton-inner {\r\n width: 100%;\r\n height: 100%;\r\n border-radius: 8px;\r\n background: linear-gradient(\r\n 90deg,\r\n var(--body-color) 25%,\r\n color-mix(in srgb, var(--body-color) 80%, var(--border-color)) 37%,\r\n var(--body-color) 63%\r\n );\r\n background-size: 400% 100%;\r\n animation: shimmer 1.4s ease infinite;\r\n}\r\n\r\n@keyframes shimmer {\r\n 0% {\r\n background-position: 100% 50%;\r\n }\r\n 100% {\r\n background-position: 0 50%;\r\n }\r\n}\r\n\r\n.waterfall__footer {\r\n padding: 24px 0;\r\n text-align: center;\r\n}\r\n\r\n.waterfall__sentinel {\r\n height: 1px;\r\n}\r\n\r\n.waterfall__status {\r\n display: inline-flex;\r\n gap: 8px;\r\n align-items: center;\r\n font-size: 13px;\r\n color: var(--text-color-3);\r\n\r\n &--done {\r\n color: var(--text-color-4);\r\n }\r\n}\r\n</style>\r\n"],"mappings":"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"}