@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.
- package/README.md +257 -0
- package/dist/C_ActionBar-DWN-woTc.css.map +1 -0
- package/dist/C_ActionBar.cjs +5 -0
- package/dist/C_ActionBar.d.cts +2 -0
- package/dist/C_ActionBar.d.ts +2 -0
- package/dist/C_ActionBar.js +4 -0
- package/dist/C_ActionBar2.js +196 -0
- package/dist/C_ActionBar2.js.map +1 -0
- package/dist/C_AntV-AFKyK6hH.css.map +1 -0
- package/dist/C_AntV.cjs +8 -0
- package/dist/C_AntV.d.cts +2 -0
- package/dist/C_AntV.d.ts +2 -0
- package/dist/C_AntV.js +4 -0
- package/dist/C_AntV2.js +3150 -0
- package/dist/C_AntV2.js.map +1 -0
- package/dist/C_Barcode-P_EFj8dC.css.map +1 -0
- package/dist/C_Barcode.cjs +4 -0
- package/dist/C_Barcode.d.cts +2 -0
- package/dist/C_Barcode.d.ts +2 -0
- package/dist/C_Barcode.js +3 -0
- package/dist/C_Barcode2.js +68 -0
- package/dist/C_Barcode2.js.map +1 -0
- package/dist/C_Captcha-C-ef41xw.css.map +1 -0
- package/dist/C_Captcha.cjs +4 -0
- package/dist/C_Captcha.d.cts +2 -0
- package/dist/C_Captcha.d.ts +2 -0
- package/dist/C_Captcha.js +3 -0
- package/dist/C_Captcha2.js +155 -0
- package/dist/C_Captcha2.js.map +1 -0
- package/dist/C_Cascade-D9kNsjsV.css.map +1 -0
- package/dist/C_Cascade.cjs +4 -0
- package/dist/C_Cascade.d.cts +2 -0
- package/dist/C_Cascade.d.ts +2 -0
- package/dist/C_Cascade.js +3 -0
- package/dist/C_Cascade2.js +103 -0
- package/dist/C_Cascade2.js.map +1 -0
- package/dist/C_City-BCQ4ipiK.css.map +1 -0
- package/dist/C_City.cjs +4 -0
- package/dist/C_City.d.cts +2 -0
- package/dist/C_City.d.ts +2 -0
- package/dist/C_City.js +3 -0
- package/dist/C_City2.js +841 -0
- package/dist/C_City2.js.map +1 -0
- package/dist/C_Code-C9kvvEmO.css.map +1 -0
- package/dist/C_Code.cjs +5 -0
- package/dist/C_Code.d.cts +2 -0
- package/dist/C_Code.d.ts +2 -0
- package/dist/C_Code.js +4 -0
- package/dist/C_Code2.js +346 -0
- package/dist/C_Code2.js.map +1 -0
- package/dist/C_CollapsePanel-BUJHuYcU.css.map +1 -0
- package/dist/C_CollapsePanel.cjs +6 -0
- package/dist/C_CollapsePanel.d.cts +2 -0
- package/dist/C_CollapsePanel.d.ts +2 -0
- package/dist/C_CollapsePanel.js +4 -0
- package/dist/C_CollapsePanel2.js +319 -0
- package/dist/C_CollapsePanel2.js.map +1 -0
- package/dist/C_Cron-yx2Ob4Jl.css.map +1 -0
- package/dist/C_Cron.cjs +15 -0
- package/dist/C_Cron.d.cts +2 -0
- package/dist/C_Cron.d.ts +2 -0
- package/dist/C_Cron.js +4 -0
- package/dist/C_Cron2.js +1209 -0
- package/dist/C_Cron2.js.map +1 -0
- package/dist/C_Date.cjs +4 -0
- package/dist/C_Date.d.cts +2 -0
- package/dist/C_Date.d.ts +2 -0
- package/dist/C_Date.js +3 -0
- package/dist/C_Date2.js +219 -0
- package/dist/C_Date2.js.map +1 -0
- package/dist/C_Draggable-C483syRC.css.map +1 -0
- package/dist/C_Draggable.cjs +5 -0
- package/dist/C_Draggable.d.cts +2 -0
- package/dist/C_Draggable.d.ts +2 -0
- package/dist/C_Draggable.js +3 -0
- package/dist/C_Draggable2.js +295 -0
- package/dist/C_Draggable2.js.map +1 -0
- package/dist/C_Editor-Bp0SyIEw.css.map +1 -0
- package/dist/C_Editor.cjs +4 -0
- package/dist/C_Editor.d.cts +2 -0
- package/dist/C_Editor.d.ts +2 -0
- package/dist/C_Editor.js +3 -0
- package/dist/C_Editor2.js +160 -0
- package/dist/C_Editor2.js.map +1 -0
- package/dist/C_FilePreview-CPqvhoCy.css.map +1 -0
- package/dist/C_FilePreview.cjs +6 -0
- package/dist/C_FilePreview.d.cts +2 -0
- package/dist/C_FilePreview.d.ts +2 -0
- package/dist/C_FilePreview.js +3 -0
- package/dist/C_FilePreview2.js +1031 -0
- package/dist/C_FilePreview2.js.map +1 -0
- package/dist/C_Form-Jx7PY3sT.css.map +1 -0
- package/dist/C_Form.cjs +15 -0
- package/dist/C_Form.d.cts +2 -0
- package/dist/C_Form.d.ts +2 -0
- package/dist/C_Form.js +4 -0
- package/dist/C_Form2.js +2510 -0
- package/dist/C_Form2.js.map +1 -0
- package/dist/C_FormSearch-DvRgxlRn.css.map +1 -0
- package/dist/C_FormSearch.cjs +6 -0
- package/dist/C_FormSearch.d.cts +2 -0
- package/dist/C_FormSearch.d.ts +2 -0
- package/dist/C_FormSearch.js +3 -0
- package/dist/C_FormSearch2.js +356 -0
- package/dist/C_FormSearch2.js.map +1 -0
- package/dist/C_FormulaEditor-DtGkt4T_.css.map +1 -0
- package/dist/C_FormulaEditor.cjs +13 -0
- package/dist/C_FormulaEditor.d.cts +2 -0
- package/dist/C_FormulaEditor.d.ts +2 -0
- package/dist/C_FormulaEditor.js +4 -0
- package/dist/C_FormulaEditor2.js +1433 -0
- package/dist/C_FormulaEditor2.js.map +1 -0
- package/dist/C_FullCalendar-BF7H0YIx.css.map +1 -0
- package/dist/C_FullCalendar.cjs +9 -0
- package/dist/C_FullCalendar.d.cts +2 -0
- package/dist/C_FullCalendar.d.ts +2 -0
- package/dist/C_FullCalendar.js +3 -0
- package/dist/C_FullCalendar2.js +377 -0
- package/dist/C_FullCalendar2.js.map +1 -0
- package/dist/C_Guide.cjs +4 -0
- package/dist/C_Guide.d.cts +2 -0
- package/dist/C_Guide.d.ts +2 -0
- package/dist/C_Guide.js +3 -0
- package/dist/C_Guide2.js +58 -0
- package/dist/C_Guide2.js.map +1 -0
- package/dist/C_Icon.cjs +4 -0
- package/dist/C_Icon.d.cts +2 -0
- package/dist/C_Icon.d.ts +2 -0
- package/dist/C_Icon.js +3 -0
- package/dist/C_Icon2.js +286 -0
- package/dist/C_Icon2.js.map +1 -0
- package/dist/C_ImageCropper-BVJfUufl.css.map +1 -0
- package/dist/C_ImageCropper.cjs +6 -0
- package/dist/C_ImageCropper.d.cts +2 -0
- package/dist/C_ImageCropper.d.ts +2 -0
- package/dist/C_ImageCropper.js +4 -0
- package/dist/C_ImageCropper2.js +723 -0
- package/dist/C_ImageCropper2.js.map +1 -0
- package/dist/C_Language.cjs +4 -0
- package/dist/C_Language.d.cts +2 -0
- package/dist/C_Language.d.ts +2 -0
- package/dist/C_Language.js +3 -0
- package/dist/C_Language2.js +72 -0
- package/dist/C_Language2.js.map +1 -0
- package/dist/C_Map-DpzeuWdX.css.map +1 -0
- package/dist/C_Map.cjs +7 -0
- package/dist/C_Map.d.cts +2 -0
- package/dist/C_Map.d.ts +2 -0
- package/dist/C_Map.js +3 -0
- package/dist/C_Map2.js +199 -0
- package/dist/C_Map2.js.map +1 -0
- package/dist/C_Markdown-BEjxknqd.css.map +1 -0
- package/dist/C_Markdown.cjs +4 -0
- package/dist/C_Markdown.d.cts +2 -0
- package/dist/C_Markdown.d.ts +2 -0
- package/dist/C_Markdown.js +3 -0
- package/dist/C_Markdown2.js +186 -0
- package/dist/C_Markdown2.js.map +1 -0
- package/dist/C_NotificationCenter-0l3TY2Gn.css.map +1 -0
- package/dist/C_NotificationCenter.cjs +20 -0
- package/dist/C_NotificationCenter.d.cts +2 -0
- package/dist/C_NotificationCenter.d.ts +2 -0
- package/dist/C_NotificationCenter.js +4 -0
- package/dist/C_NotificationCenter2.js +1383 -0
- package/dist/C_NotificationCenter2.js.map +1 -0
- package/dist/C_Progress.cjs +4 -0
- package/dist/C_Progress.d.cts +2 -0
- package/dist/C_Progress.d.ts +2 -0
- package/dist/C_Progress.js +3 -0
- package/dist/C_Progress2.js +103 -0
- package/dist/C_Progress2.js.map +1 -0
- package/dist/C_QRCode-DbdiAIPg.css.map +1 -0
- package/dist/C_QRCode.cjs +5 -0
- package/dist/C_QRCode.d.cts +2 -0
- package/dist/C_QRCode.d.ts +2 -0
- package/dist/C_QRCode.js +3 -0
- package/dist/C_QRCode2.js +218 -0
- package/dist/C_QRCode2.js.map +1 -0
- package/dist/C_Signature-zhHCbra9.css.map +1 -0
- package/dist/C_Signature.cjs +8 -0
- package/dist/C_Signature.d.cts +2 -0
- package/dist/C_Signature.d.ts +2 -0
- package/dist/C_Signature.js +4 -0
- package/dist/C_Signature2.js +618 -0
- package/dist/C_Signature2.js.map +1 -0
- package/dist/C_SplitPane-C6sBsfKY.css.map +1 -0
- package/dist/C_SplitPane.cjs +6 -0
- package/dist/C_SplitPane.d.cts +2 -0
- package/dist/C_SplitPane.d.ts +2 -0
- package/dist/C_SplitPane.js +4 -0
- package/dist/C_SplitPane2.js +356 -0
- package/dist/C_SplitPane2.js.map +1 -0
- package/dist/C_Steps-CODHN5Hs.css.map +1 -0
- package/dist/C_Steps.cjs +4 -0
- package/dist/C_Steps.d.cts +2 -0
- package/dist/C_Steps.d.ts +2 -0
- package/dist/C_Steps.js +3 -0
- package/dist/C_Steps2.js +82 -0
- package/dist/C_Steps2.js.map +1 -0
- package/dist/C_Table-DSNsntmT.css.map +1 -0
- package/dist/C_Table.cjs +19 -0
- package/dist/C_Table.d.cts +2 -0
- package/dist/C_Table.d.ts +2 -0
- package/dist/C_Table.js +5 -0
- package/dist/C_Table2.js +3009 -0
- package/dist/C_Table2.js.map +1 -0
- package/dist/C_Theme.cjs +4 -0
- package/dist/C_Theme.d.cts +2 -0
- package/dist/C_Theme.d.ts +2 -0
- package/dist/C_Theme.js +3 -0
- package/dist/C_Theme2.js +60 -0
- package/dist/C_Theme2.js.map +1 -0
- package/dist/C_Time-BvZLYraL.css.map +1 -0
- package/dist/C_Time.cjs +5 -0
- package/dist/C_Time.d.cts +2 -0
- package/dist/C_Time.d.ts +2 -0
- package/dist/C_Time.js +3 -0
- package/dist/C_Time2.js +199 -0
- package/dist/C_Time2.js.map +1 -0
- package/dist/C_Tree-0GDv--jX.css.map +1 -0
- package/dist/C_Tree.cjs +7 -0
- package/dist/C_Tree.d.cts +2 -0
- package/dist/C_Tree.d.ts +2 -0
- package/dist/C_Tree.js +4 -0
- package/dist/C_Tree2.js +441 -0
- package/dist/C_Tree2.js.map +1 -0
- package/dist/C_Upload-BXd3YYLx.css.map +1 -0
- package/dist/C_Upload.cjs +12 -0
- package/dist/C_Upload.d.cts +2 -0
- package/dist/C_Upload.d.ts +2 -0
- package/dist/C_Upload.js +4 -0
- package/dist/C_Upload2.js +1388 -0
- package/dist/C_Upload2.js.map +1 -0
- package/dist/C_VideoPlayer-DYG3RL0Q.css.map +1 -0
- package/dist/C_VideoPlayer.cjs +23 -0
- package/dist/C_VideoPlayer.d.cts +2 -0
- package/dist/C_VideoPlayer.d.ts +2 -0
- package/dist/C_VideoPlayer.js +3 -0
- package/dist/C_VideoPlayer2.js +1932 -0
- package/dist/C_VideoPlayer2.js.map +1 -0
- package/dist/C_VtableGantt-fhItIiHE.css.map +1 -0
- package/dist/C_VtableGantt.cjs +6 -0
- package/dist/C_VtableGantt.d.cts +2 -0
- package/dist/C_VtableGantt.d.ts +2 -0
- package/dist/C_VtableGantt.js +4 -0
- package/dist/C_VtableGantt2.js +873 -0
- package/dist/C_VtableGantt2.js.map +1 -0
- package/dist/C_WaterFall-8sQDFXKg.css.map +1 -0
- package/dist/C_WaterFall.cjs +13 -0
- package/dist/C_WaterFall.d.cts +2 -0
- package/dist/C_WaterFall.d.ts +2 -0
- package/dist/C_WaterFall.js +3 -0
- package/dist/C_WaterFall2.js +365 -0
- package/dist/C_WaterFall2.js.map +1 -0
- package/dist/C_WorkFlow-J-dyIuh9.css.map +1 -0
- package/dist/C_WorkFlow.cjs +8 -0
- package/dist/C_WorkFlow.d.cts +2 -0
- package/dist/C_WorkFlow.d.ts +2 -0
- package/dist/C_WorkFlow.js +4 -0
- package/dist/C_WorkFlow2.js +1984 -0
- package/dist/C_WorkFlow2.js.map +1 -0
- package/dist/chunk.js +22 -0
- package/dist/city.js +4817 -0
- package/dist/city.js.map +1 -0
- package/dist/constants.d.ts +273 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants2.d.ts +178 -0
- package/dist/constants2.d.ts.map +1 -0
- package/dist/constants3.d.ts +475 -0
- package/dist/constants3.d.ts.map +1 -0
- package/dist/constants4.d.ts +430 -0
- package/dist/constants4.d.ts.map +1 -0
- package/dist/constants5.d.ts +4283 -0
- package/dist/constants5.d.ts.map +1 -0
- package/dist/data.d.ts +67 -0
- package/dist/data.d.ts.map +1 -0
- package/dist/export-helper.js +9 -0
- package/dist/index.cjs +409 -0
- package/dist/index.d.cts +96 -0
- package/dist/index.d.cts.map +1 -0
- package/dist/index.d.ts +103 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +230 -0
- package/dist/index.js.map +1 -0
- package/dist/index.vue.d.ts +80 -0
- package/dist/index.vue.d.ts.map +1 -0
- package/dist/index10.vue.d.ts +72 -0
- package/dist/index10.vue.d.ts.map +1 -0
- package/dist/index11.vue.d.ts +26 -0
- package/dist/index11.vue.d.ts.map +1 -0
- package/dist/index12.vue.d.ts +81 -0
- package/dist/index12.vue.d.ts.map +1 -0
- package/dist/index13.vue.d.ts +55 -0
- package/dist/index13.vue.d.ts.map +1 -0
- package/dist/index14.vue.d.ts +33 -0
- package/dist/index14.vue.d.ts.map +1 -0
- package/dist/index15.vue.d.ts +18 -0
- package/dist/index15.vue.d.ts.map +1 -0
- package/dist/index16.vue.d.ts +662 -0
- package/dist/index16.vue.d.ts.map +1 -0
- package/dist/index2.vue.d.ts +38 -0
- package/dist/index2.vue.d.ts.map +1 -0
- package/dist/index3.vue.d.ts +45 -0
- package/dist/index3.vue.d.ts.map +1 -0
- package/dist/index4.vue.d.ts +31 -0
- package/dist/index4.vue.d.ts.map +1 -0
- package/dist/index5.vue.d.ts +35 -0
- package/dist/index5.vue.d.ts.map +1 -0
- package/dist/index6.vue.d.ts +48 -0
- package/dist/index6.vue.d.ts.map +1 -0
- package/dist/index7.vue.d.ts +56 -0
- package/dist/index7.vue.d.ts.map +1 -0
- package/dist/index8.vue.d.ts +41 -0
- package/dist/index8.vue.d.ts.map +1 -0
- package/dist/index9.vue.d.ts +30 -0
- package/dist/index9.vue.d.ts.map +1 -0
- package/dist/storage.js +31 -0
- package/dist/storage.js.map +1 -0
- package/dist/style.css +7725 -0
- package/dist/useCalendarEvents.d.ts +148 -0
- package/dist/useCalendarEvents.d.ts.map +1 -0
- package/dist/useCollapsePanel.d.ts +132 -0
- package/dist/useCollapsePanel.d.ts.map +1 -0
- package/dist/useCropperCore.d.ts +102 -0
- package/dist/useCropperCore.d.ts.map +1 -0
- package/dist/useDraggableLayout.d.ts +194 -0
- package/dist/useDraggableLayout.d.ts.map +1 -0
- package/dist/useDynamicFormState.d.ts +4248 -0
- package/dist/useDynamicFormState.d.ts.map +1 -0
- package/dist/useEdgeInteraction.d.ts +7614 -0
- package/dist/useEdgeInteraction.d.ts.map +1 -0
- package/dist/useFullscreen.d.ts +166 -0
- package/dist/useFullscreen.d.ts.map +1 -0
- package/dist/useInfiniteScroll.d.ts +169 -0
- package/dist/useInfiniteScroll.d.ts.map +1 -0
- package/dist/useModalEdit.d.ts +960 -0
- package/dist/useModalEdit.d.ts.map +1 -0
- package/dist/useQRCode.d.ts +87 -0
- package/dist/useQRCode.d.ts.map +1 -0
- package/dist/useSearchState.d.ts +180 -0
- package/dist/useSearchState.d.ts.map +1 -0
- package/dist/useSignatureHistory.d.ts +189 -0
- package/dist/useSignatureHistory.d.ts.map +1 -0
- package/dist/useSplitResize.d.ts +158 -0
- package/dist/useSplitResize.d.ts.map +1 -0
- package/dist/useTimeSelection.d.ts +105 -0
- package/dist/useTimeSelection.d.ts.map +1 -0
- package/dist/useTreeOperations.d.ts +183 -0
- package/dist/useTreeOperations.d.ts.map +1 -0
- package/dist/useWorkflowValidation.d.ts +1052 -0
- package/dist/useWorkflowValidation.d.ts.map +1 -0
- 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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"}
|