@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_Signature2.js","names":["showToolbar","disabled","readonly","width","height","backgroundColor"],"sources":["../src/components/C_Signature/composables/useSignatureCanvas.ts","../src/components/C_Signature/composables/useSignatureHistory.ts","../src/components/C_Signature/composables/useSignatureExport.ts","../src/components/C_Signature/data.ts","../src/components/C_Signature/index.vue","../src/components/C_Signature/index.vue","../src/components/C_Signature/index.vue"],"sourcesContent":["import type { Ref } from \"vue\";\r\nimport { ref, readonly } from \"vue\";\r\nimport type {\r\n PenConfig,\r\n PenMode,\r\n SignaturePoint,\r\n SignatureStroke,\r\n} from \"../types\";\r\n\r\ninterface UseSignatureCanvasOptions {\r\n canvasRef: Ref<HTMLCanvasElement | null>;\r\n penConfig: Ref<PenConfig>;\r\n mode: Ref<PenMode>;\r\n disabled: Ref<boolean>;\r\n onStrokeComplete: (stroke: SignatureStroke) => void;\r\n onDrawStart: () => void;\r\n onDrawing: (point: SignaturePoint) => void;\r\n}\r\n\r\nexport function useSignatureCanvas(options: UseSignatureCanvasOptions) {\r\n const {\r\n canvasRef,\r\n penConfig,\r\n mode,\r\n disabled,\r\n onStrokeComplete,\r\n onDrawStart,\r\n onDrawing,\r\n } = options;\r\n\r\n const isDrawing = ref(false);\r\n const currentStroke = ref<SignaturePoint[]>([]);\r\n let ctx: CanvasRenderingContext2D | null = null;\r\n\r\n const initCanvas = () => {\r\n const canvas = canvasRef.value;\r\n if (!canvas) return;\r\n ctx = canvas.getContext(\"2d\", { willReadFrequently: true });\r\n if (!ctx) return;\r\n const dpr = window.devicePixelRatio || 1;\r\n const rect = canvas.getBoundingClientRect();\r\n canvas.width = rect.width * dpr;\r\n canvas.height = rect.height * dpr;\r\n canvas.style.width = `${rect.width}px`;\r\n canvas.style.height = `${rect.height}px`;\r\n ctx.scale(dpr, dpr);\r\n ctx.lineCap = \"round\";\r\n ctx.lineJoin = \"round\";\r\n };\r\n\r\n const getPointFromEvent = (e: MouseEvent | TouchEvent): SignaturePoint => {\r\n const canvas = canvasRef.value;\r\n if (!canvas) return { x: 0, y: 0 };\r\n const rect = canvas.getBoundingClientRect();\r\n let clientX: number;\r\n let clientY: number;\r\n if (\"touches\" in e && e.touches.length > 0) {\r\n ({ clientX, clientY } = e.touches[0]);\r\n } else if (\"clientX\" in e) {\r\n ({ clientX, clientY } = e);\r\n } else {\r\n return { x: 0, y: 0 };\r\n }\r\n return {\r\n x: clientX - rect.left,\r\n y: clientY - rect.top,\r\n timestamp: Date.now(),\r\n };\r\n };\r\n\r\n const drawStroke = (\r\n from: SignaturePoint,\r\n to: SignaturePoint,\r\n config: PenConfig,\r\n eraserMode = false,\r\n ) => {\r\n if (!ctx) return;\r\n ctx.save();\r\n if (eraserMode) {\r\n ctx.globalCompositeOperation = \"destination-out\";\r\n ctx.lineWidth = 20;\r\n } else {\r\n ctx.globalCompositeOperation = \"source-over\";\r\n ctx.strokeStyle = config.color;\r\n ctx.lineWidth = config.width;\r\n ctx.globalAlpha = config.opacity;\r\n }\r\n ctx.beginPath();\r\n ctx.moveTo(from.x, from.y);\r\n ctx.lineTo(to.x, to.y);\r\n ctx.stroke();\r\n ctx.restore();\r\n };\r\n\r\n const startDrawing = (e: MouseEvent | TouchEvent) => {\r\n if (disabled.value) return;\r\n e.preventDefault();\r\n isDrawing.value = true;\r\n currentStroke.value = [];\r\n const point = getPointFromEvent(e);\r\n currentStroke.value.push(point);\r\n onDrawStart();\r\n };\r\n\r\n const draw = (e: MouseEvent | TouchEvent) => {\r\n if (!isDrawing.value || disabled.value) return;\r\n e.preventDefault();\r\n const point = getPointFromEvent(e);\r\n const lastPoint = currentStroke.value[currentStroke.value.length - 1];\r\n if (lastPoint) {\r\n drawStroke(lastPoint, point, penConfig.value, mode.value === \"eraser\");\r\n }\r\n currentStroke.value.push(point);\r\n onDrawing(point);\r\n };\r\n\r\n const endDrawing = (e: MouseEvent | TouchEvent) => {\r\n if (!isDrawing.value || disabled.value) return;\r\n e.preventDefault();\r\n isDrawing.value = false;\r\n if (currentStroke.value.length > 0) {\r\n const stroke: SignatureStroke = {\r\n points: [...currentStroke.value],\r\n color: penConfig.value.color,\r\n width: penConfig.value.width,\r\n opacity: penConfig.value.opacity,\r\n mode: mode.value,\r\n };\r\n onStrokeComplete(stroke);\r\n currentStroke.value = [];\r\n }\r\n };\r\n\r\n const redrawStrokes = (strokes: SignatureStroke[]) => {\r\n if (!ctx || !canvasRef.value) return;\r\n const canvas = canvasRef.value;\r\n ctx.clearRect(0, 0, canvas.width, canvas.height);\r\n strokes.forEach((stroke) => {\r\n for (let i = 1; i < stroke.points.length; i++) {\r\n drawStroke(\r\n stroke.points[i - 1],\r\n stroke.points[i],\r\n { color: stroke.color, width: stroke.width, opacity: stroke.opacity },\r\n stroke.mode === \"eraser\",\r\n );\r\n }\r\n });\r\n };\r\n\r\n const clearCanvas = () => {\r\n if (!ctx || !canvasRef.value) return;\r\n ctx.clearRect(0, 0, canvasRef.value.width, canvasRef.value.height);\r\n };\r\n\r\n const bindEvents = () => {\r\n const canvas = canvasRef.value;\r\n if (!canvas) return;\r\n canvas.addEventListener(\"mousedown\", startDrawing);\r\n canvas.addEventListener(\"mousemove\", draw);\r\n canvas.addEventListener(\"mouseup\", endDrawing);\r\n canvas.addEventListener(\"mouseleave\", endDrawing);\r\n canvas.addEventListener(\"touchstart\", startDrawing, { passive: false });\r\n canvas.addEventListener(\"touchmove\", draw, { passive: false });\r\n canvas.addEventListener(\"touchend\", endDrawing);\r\n canvas.addEventListener(\"touchcancel\", endDrawing);\r\n };\r\n\r\n const unbindEvents = () => {\r\n const canvas = canvasRef.value;\r\n if (!canvas) return;\r\n canvas.removeEventListener(\"mousedown\", startDrawing);\r\n canvas.removeEventListener(\"mousemove\", draw);\r\n canvas.removeEventListener(\"mouseup\", endDrawing);\r\n canvas.removeEventListener(\"mouseleave\", endDrawing);\r\n canvas.removeEventListener(\"touchstart\", startDrawing);\r\n canvas.removeEventListener(\"touchmove\", draw);\r\n canvas.removeEventListener(\"touchend\", endDrawing);\r\n canvas.removeEventListener(\"touchcancel\", endDrawing);\r\n };\r\n\r\n return {\r\n isDrawing: readonly(isDrawing),\r\n initCanvas,\r\n bindEvents,\r\n unbindEvents,\r\n redrawStrokes,\r\n clearCanvas,\r\n };\r\n}\r\n","import { ref, computed, readonly } from \"vue\";\r\nimport type { SignatureStroke } from \"../types\";\r\n\r\ninterface UseSignatureHistoryOptions {\r\n maxHistory?: number;\r\n onChange?: (strokes: SignatureStroke[]) => void;\r\n}\r\n\r\nexport function useSignatureHistory(options: UseSignatureHistoryOptions = {}) {\r\n const { maxHistory = 50, onChange } = options;\r\n\r\n const strokes = ref<SignatureStroke[]>([]);\r\n const historyStack = ref<SignatureStroke[][]>([]);\r\n const historyIndex = ref(-1);\r\n\r\n const canUndo = computed(() => historyIndex.value > 0);\r\n const canRedo = computed(\r\n () => historyIndex.value < historyStack.value.length - 1,\r\n );\r\n\r\n const addStroke = (stroke: SignatureStroke) => {\r\n strokes.value.push(stroke);\r\n saveToHistory();\r\n };\r\n\r\n const saveToHistory = () => {\r\n historyStack.value = historyStack.value.slice(0, historyIndex.value + 1);\r\n historyStack.value.push([...strokes.value]);\r\n historyIndex.value++;\r\n if (historyStack.value.length > maxHistory) {\r\n historyStack.value.shift();\r\n historyIndex.value--;\r\n }\r\n onChange?.(strokes.value);\r\n };\r\n\r\n const undo = (): boolean => {\r\n if (!canUndo.value) return false;\r\n historyIndex.value--;\r\n strokes.value = [...historyStack.value[historyIndex.value]];\r\n onChange?.(strokes.value);\r\n return true;\r\n };\r\n\r\n const redo = (): boolean => {\r\n if (!canRedo.value) return false;\r\n historyIndex.value++;\r\n strokes.value = [...historyStack.value[historyIndex.value]];\r\n onChange?.(strokes.value);\r\n return true;\r\n };\r\n\r\n const clear = () => {\r\n strokes.value = [];\r\n historyStack.value = [[]];\r\n historyIndex.value = 0;\r\n onChange?.(strokes.value);\r\n };\r\n\r\n const loadData = (data: SignatureStroke[]) => {\r\n strokes.value = [...data];\r\n historyStack.value = [[...data]];\r\n historyIndex.value = 0;\r\n onChange?.(strokes.value);\r\n };\r\n\r\n const isEmpty = computed(() => strokes.value.length === 0);\r\n\r\n // 初始化历史栈\r\n historyStack.value = [[]];\r\n historyIndex.value = 0;\r\n\r\n return {\r\n strokes: readonly(strokes),\r\n canUndo: readonly(canUndo),\r\n canRedo: readonly(canRedo),\r\n isEmpty: readonly(isEmpty),\r\n addStroke,\r\n undo,\r\n redo,\r\n clear,\r\n loadData,\r\n };\r\n}\r\n","import type { Ref } from 'vue'\r\nimport type { ExportOptions, WatermarkConfig } from '../types'\r\n\r\ninterface UseSignatureExportOptions {\r\n canvasRef: Ref<HTMLCanvasElement | null>\r\n watermark?: Ref<Partial<WatermarkConfig>>\r\n}\r\n\r\nexport function useSignatureExport(options: UseSignatureExportOptions) {\r\n const { canvasRef, watermark } = options\r\n\r\n const getWatermarkPosition = (\r\n canvas: HTMLCanvasElement,\r\n textWidth: number,\r\n fontSize: number,\r\n position: string\r\n ): { x: number; y: number } => {\r\n const padding = 10\r\n const positions = {\r\n 'top-left': { x: padding, y: fontSize + padding },\r\n 'top-right': { x: canvas.width - textWidth - padding, y: fontSize + padding },\r\n 'bottom-left': { x: padding, y: canvas.height - padding },\r\n 'bottom-right': { x: canvas.width - textWidth - padding, y: canvas.height - padding },\r\n }\r\n return positions[position as keyof typeof positions] || positions['bottom-right']\r\n }\r\n\r\n const drawWatermark = (canvas: HTMLCanvasElement, text: string): void => {\r\n const ctx = canvas.getContext('2d')\r\n if (!ctx || !watermark?.value?.show) return\r\n const config = {\r\n fontSize: watermark.value.fontSize || 12,\r\n color: watermark.value.color || '#999999',\r\n position: watermark.value.position || 'bottom-right',\r\n }\r\n ctx.save()\r\n ctx.font = `${config.fontSize}px Arial`\r\n ctx.fillStyle = config.color\r\n ctx.textBaseline = 'bottom'\r\n const textWidth = ctx.measureText(text).width\r\n const { x, y } = getWatermarkPosition(canvas, textWidth, config.fontSize, config.position)\r\n ctx.fillText(text, x, y)\r\n ctx.restore()\r\n }\r\n\r\n const createTempCanvas = (\r\n sourceCanvas: HTMLCanvasElement,\r\n includeBackground: boolean,\r\n backgroundColor: string\r\n ) => {\r\n const tempCanvas = document.createElement('canvas')\r\n tempCanvas.width = sourceCanvas.width\r\n tempCanvas.height = sourceCanvas.height\r\n const tempCtx = tempCanvas.getContext('2d')\r\n if (!tempCtx) throw new Error('无法创建临时 Canvas')\r\n if (includeBackground) {\r\n tempCtx.fillStyle = backgroundColor\r\n tempCtx.fillRect(0, 0, tempCanvas.width, tempCanvas.height)\r\n }\r\n tempCtx.drawImage(sourceCanvas, 0, 0)\r\n return { canvas: tempCanvas, ctx: tempCtx }\r\n }\r\n\r\n const prepareExportCanvas = (canvas: HTMLCanvasElement, options: ExportOptions): HTMLCanvasElement => {\r\n const { includeBackground = false, backgroundColor = '#FFFFFF', includeWatermark = false } = options\r\n const { canvas: tempCanvas } = createTempCanvas(canvas, includeBackground, backgroundColor)\r\n if (includeWatermark && watermark?.value?.show && watermark.value.text) {\r\n drawWatermark(tempCanvas, watermark.value.text)\r\n }\r\n return tempCanvas\r\n }\r\n\r\n const exportToBlob = (canvas: HTMLCanvasElement, quality: number): Promise<Blob> => {\r\n return new Promise<Blob>((resolve, reject) => {\r\n canvas.toBlob(\r\n blob => (blob ? resolve(blob) : reject(new Error('导出失败'))),\r\n 'image/png',\r\n quality\r\n )\r\n })\r\n }\r\n\r\n const exportSignature = async (options: ExportOptions = {}): Promise<string | Blob> => {\r\n const canvas = canvasRef.value\r\n if (!canvas) throw new Error('Canvas 未初始化')\r\n const { format = 'png', quality = 0.92 } = options\r\n if (format === 'svg') {\r\n return 'data:image/svg+xml;charset=utf-8,<svg xmlns=\"http://www.w3.org/2000/svg\"><text>SVG export placeholder</text></svg>'\r\n }\r\n const tempCanvas = prepareExportCanvas(canvas, options)\r\n if (format === 'blob') return exportToBlob(tempCanvas, quality)\r\n const mimeType = format === 'jpeg' ? 'image/jpeg' : 'image/png'\r\n return tempCanvas.toDataURL(mimeType, quality)\r\n }\r\n\r\n const download = async (filename = 'signature', options: ExportOptions = {}): Promise<void> => {\r\n const format = options.format || 'png'\r\n const result = await exportSignature({ ...options, format })\r\n if (result instanceof Blob) {\r\n const url = URL.createObjectURL(result)\r\n const link = document.createElement('a')\r\n link.href = url\r\n link.download = `${filename}.${format}`\r\n link.click()\r\n URL.revokeObjectURL(url)\r\n } else if (typeof result === 'string') {\r\n const link = document.createElement('a')\r\n link.href = result\r\n link.download = `${filename}.${format}`\r\n link.click()\r\n }\r\n }\r\n\r\n const loadImage = async (imageUrl: string): Promise<void> => {\r\n const canvas = canvasRef.value\r\n if (!canvas) throw new Error('Canvas 未初始化')\r\n const ctx = canvas.getContext('2d')\r\n if (!ctx) throw new Error('无法获取 Canvas 上下文')\r\n return new Promise((resolve, reject) => {\r\n const img = new Image()\r\n img.crossOrigin = 'anonymous'\r\n img.onload = () => {\r\n ctx.clearRect(0, 0, canvas.width, canvas.height)\r\n ctx.drawImage(img, 0, 0, canvas.width, canvas.height)\r\n resolve()\r\n }\r\n img.onerror = () => reject(new Error(`图片加载失败: ${imageUrl}`))\r\n img.src = imageUrl\r\n })\r\n }\r\n\r\n return { exportSignature, download, loadImage }\r\n}\r\n","import type { EraserConfig, PenConfig, WatermarkConfig } from \"./types\";\r\n\r\nexport const DEFAULT_PEN_CONFIG: PenConfig = {\r\n color: \"#000000\",\r\n width: 2,\r\n opacity: 1,\r\n};\r\n\r\nexport const DEFAULT_ERASER_CONFIG: EraserConfig = { size: 20 };\r\n\r\nexport const DEFAULT_WATERMARK_CONFIG: WatermarkConfig = {\r\n show: false,\r\n text: \"\",\r\n fontSize: 12,\r\n color: \"#999999\",\r\n position: \"bottom-right\",\r\n};\r\n\r\nexport const PRESET_COLORS = [\r\n \"#000000\",\r\n \"#FF0000\",\r\n \"#0000FF\",\r\n \"#00AA00\",\r\n \"#FF6600\",\r\n \"#9900FF\",\r\n];\r\n\r\nexport const PRESET_WIDTHS = [1, 2, 3, 5, 8];\r\n\r\nexport const EXPORT_FORMAT_OPTIONS = [\r\n { label: \"PNG\", value: \"png\" },\r\n { label: \"JPEG\", value: \"jpeg\" },\r\n { label: \"SVG\", value: \"svg\" },\r\n { label: \"Blob\", value: \"blob\" },\r\n] as const;\r\n\r\nexport const WATERMARK_POSITION_OPTIONS = [\r\n { label: \"左上\", value: \"top-left\" },\r\n { label: \"右上\", value: \"top-right\" },\r\n { label: \"左下\", value: \"bottom-left\" },\r\n { label: \"右下\", value: \"bottom-right\" },\r\n] as const;\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2026-02-25\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 class=\"c-signature\">\r\n <div v-if=\"showToolbar\" class=\"signature-toolbar\">\r\n <div class=\"toolbar-section\">\r\n <NButtonGroup>\r\n <NButton\r\n :type=\"currentMode === 'pen' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'pen'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:draw\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 画笔\r\n </NButton>\r\n <NButton\r\n :type=\"currentMode === 'eraser' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'eraser'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:eraser\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 橡皮擦\r\n </NButton>\r\n </NButtonGroup>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">颜色</span>\r\n <NColorPicker\r\n v-model:value=\"currentPenConfig.color\"\r\n :show-alpha=\"false\"\r\n size=\"small\"\r\n :swatches=\"PRESET_COLORS\"\r\n />\r\n </div>\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">粗细</span>\r\n <NInputNumber\r\n v-model:value=\"currentPenConfig.width\"\r\n :min=\"1\"\r\n :max=\"20\"\r\n size=\"small\"\r\n style=\"width: 80px\"\r\n />\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton size=\"small\" :disabled=\"!canUndo\" @click=\"handleUndo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:undo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 撤销\r\n </NButton>\r\n <NButton size=\"small\" :disabled=\"!canRedo\" @click=\"handleRedo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:redo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 重做\r\n </NButton>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton\r\n size=\"small\"\r\n type=\"error\"\r\n :disabled=\"isEmpty\"\r\n @click=\"handleClear\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:delete-outline\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 清空\r\n </NButton>\r\n </div>\r\n </div>\r\n\r\n <div\r\n class=\"signature-canvas-wrapper\"\r\n :class=\"{ disabled, readonly }\"\r\n :style=\"{\r\n width: typeof width === 'number' ? `${width}px` : width,\r\n height: typeof height === 'number' ? `${height}px` : height,\r\n backgroundColor: backgroundColor || 'transparent',\r\n }\"\r\n >\r\n <canvas\r\n ref=\"canvasRef\"\r\n class=\"signature-canvas\"\r\n :class=\"{ disabled, readonly }\"\r\n />\r\n <div v-if=\"isEmpty && !disabled && !readonly\" class=\"canvas-placeholder\">\r\n 请在此处签名\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, reactive, toRef, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton, NButtonGroup, NColorPicker, NInputNumber } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSignatureCanvas } from \"./composables/useSignatureCanvas\";\r\nimport { useSignatureHistory } from \"./composables/useSignatureHistory\";\r\nimport { useSignatureExport } from \"./composables/useSignatureExport\";\r\nimport {\r\n DEFAULT_PEN_CONFIG,\r\n DEFAULT_WATERMARK_CONFIG,\r\n PRESET_COLORS,\r\n} from \"./data\";\r\nimport type {\r\n ExportOptions,\r\n PenConfig,\r\n PenMode,\r\n SignatureExpose,\r\n SignaturePoint,\r\n SignatureProps,\r\n SignatureStroke,\r\n WatermarkConfig,\r\n} from \"./types\";\r\n\r\ndefineOptions({ name: \"C_Signature\" });\r\n\r\nconst props = withDefaults(defineProps<SignatureProps>(), {\r\n width: \"100%\",\r\n height: 300,\r\n disabled: false,\r\n readonly: false,\r\n showToolbar: true,\r\n maxHistory: 50,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"start-draw\": [];\r\n drawing: [point: SignaturePoint];\r\n \"end-draw\": [stroke: SignatureStroke];\r\n clear: [];\r\n undo: [];\r\n redo: [];\r\n change: [data: SignatureStroke[]];\r\n}>();\r\n\r\nconst canvasRef = ref<HTMLCanvasElement | null>(null);\r\nconst currentMode = ref<PenMode>(\"pen\");\r\nconst currentPenConfig = reactive<PenConfig>({\r\n ...DEFAULT_PEN_CONFIG,\r\n ...props.penConfig,\r\n});\r\nconst currentWatermark = reactive<WatermarkConfig>({\r\n ...DEFAULT_WATERMARK_CONFIG,\r\n ...props.watermark,\r\n});\r\n\r\nconst {\r\n strokes,\r\n canUndo,\r\n canRedo,\r\n isEmpty,\r\n addStroke,\r\n undo,\r\n redo,\r\n clear,\r\n loadData,\r\n} = useSignatureHistory({\r\n maxHistory: props.maxHistory,\r\n onChange: (data) => {\r\n canvasInstance.redrawStrokes(data as SignatureStroke[]);\r\n emit(\"change\", data as SignatureStroke[]);\r\n },\r\n});\r\n\r\nconst canvasInstance = useSignatureCanvas({\r\n canvasRef,\r\n penConfig: toRef(currentPenConfig),\r\n mode: currentMode,\r\n disabled: toRef(props, \"disabled\"),\r\n onStrokeComplete: (stroke: SignatureStroke) => {\r\n addStroke(stroke);\r\n emit(\"end-draw\", stroke);\r\n },\r\n onDrawStart: () => emit(\"start-draw\"),\r\n onDrawing: (point: SignaturePoint) => emit(\"drawing\", point),\r\n});\r\n\r\nconst exportInstance = useSignatureExport({\r\n canvasRef,\r\n watermark: toRef(currentWatermark),\r\n});\r\n\r\nconst handleUndo = (): boolean => {\r\n const result = undo();\r\n if (result) emit(\"undo\");\r\n return result;\r\n};\r\n\r\nconst handleRedo = (): boolean => {\r\n const result = redo();\r\n if (result) emit(\"redo\");\r\n return result;\r\n};\r\n\r\nconst handleClear = () => {\r\n clear();\r\n canvasInstance.clearCanvas();\r\n emit(\"clear\");\r\n};\r\n\r\nconst exportSignature = async (\r\n options?: ExportOptions,\r\n): Promise<string | Blob> => {\r\n return exportInstance.exportSignature(options);\r\n};\r\n\r\nconst download = async (\r\n filename?: string,\r\n options?: ExportOptions,\r\n): Promise<void> => {\r\n return exportInstance.download(filename, options);\r\n};\r\n\r\nconst loadImage = async (imageUrl: string): Promise<void> => {\r\n await exportInstance.loadImage(imageUrl);\r\n clear();\r\n};\r\n\r\nconst getSignatureData = (): SignatureStroke[] =>\r\n strokes.value as SignatureStroke[];\r\nconst loadSignatureData = (data: SignatureStroke[]): void => loadData(data);\r\nconst isSignatureEmpty = (): boolean => isEmpty.value;\r\n\r\nonMounted(() => {\r\n canvasInstance.initCanvas();\r\n canvasInstance.bindEvents();\r\n if (props.backgroundImage) loadImage(props.backgroundImage);\r\n});\r\n\r\nonUnmounted(() => canvasInstance.unbindEvents());\r\n\r\ndefineExpose<SignatureExpose>({\r\n clear: handleClear,\r\n undo: handleUndo,\r\n redo: handleRedo,\r\n export: exportSignature,\r\n download,\r\n loadImage,\r\n getSignatureData,\r\n loadSignatureData,\r\n isEmpty: isSignatureEmpty,\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: 2026-02-25\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 class=\"c-signature\">\r\n <div v-if=\"showToolbar\" class=\"signature-toolbar\">\r\n <div class=\"toolbar-section\">\r\n <NButtonGroup>\r\n <NButton\r\n :type=\"currentMode === 'pen' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'pen'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:draw\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 画笔\r\n </NButton>\r\n <NButton\r\n :type=\"currentMode === 'eraser' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'eraser'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:eraser\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 橡皮擦\r\n </NButton>\r\n </NButtonGroup>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">颜色</span>\r\n <NColorPicker\r\n v-model:value=\"currentPenConfig.color\"\r\n :show-alpha=\"false\"\r\n size=\"small\"\r\n :swatches=\"PRESET_COLORS\"\r\n />\r\n </div>\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">粗细</span>\r\n <NInputNumber\r\n v-model:value=\"currentPenConfig.width\"\r\n :min=\"1\"\r\n :max=\"20\"\r\n size=\"small\"\r\n style=\"width: 80px\"\r\n />\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton size=\"small\" :disabled=\"!canUndo\" @click=\"handleUndo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:undo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 撤销\r\n </NButton>\r\n <NButton size=\"small\" :disabled=\"!canRedo\" @click=\"handleRedo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:redo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 重做\r\n </NButton>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton\r\n size=\"small\"\r\n type=\"error\"\r\n :disabled=\"isEmpty\"\r\n @click=\"handleClear\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:delete-outline\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 清空\r\n </NButton>\r\n </div>\r\n </div>\r\n\r\n <div\r\n class=\"signature-canvas-wrapper\"\r\n :class=\"{ disabled, readonly }\"\r\n :style=\"{\r\n width: typeof width === 'number' ? `${width}px` : width,\r\n height: typeof height === 'number' ? `${height}px` : height,\r\n backgroundColor: backgroundColor || 'transparent',\r\n }\"\r\n >\r\n <canvas\r\n ref=\"canvasRef\"\r\n class=\"signature-canvas\"\r\n :class=\"{ disabled, readonly }\"\r\n />\r\n <div v-if=\"isEmpty && !disabled && !readonly\" class=\"canvas-placeholder\">\r\n 请在此处签名\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, reactive, toRef, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton, NButtonGroup, NColorPicker, NInputNumber } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSignatureCanvas } from \"./composables/useSignatureCanvas\";\r\nimport { useSignatureHistory } from \"./composables/useSignatureHistory\";\r\nimport { useSignatureExport } from \"./composables/useSignatureExport\";\r\nimport {\r\n DEFAULT_PEN_CONFIG,\r\n DEFAULT_WATERMARK_CONFIG,\r\n PRESET_COLORS,\r\n} from \"./data\";\r\nimport type {\r\n ExportOptions,\r\n PenConfig,\r\n PenMode,\r\n SignatureExpose,\r\n SignaturePoint,\r\n SignatureProps,\r\n SignatureStroke,\r\n WatermarkConfig,\r\n} from \"./types\";\r\n\r\ndefineOptions({ name: \"C_Signature\" });\r\n\r\nconst props = withDefaults(defineProps<SignatureProps>(), {\r\n width: \"100%\",\r\n height: 300,\r\n disabled: false,\r\n readonly: false,\r\n showToolbar: true,\r\n maxHistory: 50,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"start-draw\": [];\r\n drawing: [point: SignaturePoint];\r\n \"end-draw\": [stroke: SignatureStroke];\r\n clear: [];\r\n undo: [];\r\n redo: [];\r\n change: [data: SignatureStroke[]];\r\n}>();\r\n\r\nconst canvasRef = ref<HTMLCanvasElement | null>(null);\r\nconst currentMode = ref<PenMode>(\"pen\");\r\nconst currentPenConfig = reactive<PenConfig>({\r\n ...DEFAULT_PEN_CONFIG,\r\n ...props.penConfig,\r\n});\r\nconst currentWatermark = reactive<WatermarkConfig>({\r\n ...DEFAULT_WATERMARK_CONFIG,\r\n ...props.watermark,\r\n});\r\n\r\nconst {\r\n strokes,\r\n canUndo,\r\n canRedo,\r\n isEmpty,\r\n addStroke,\r\n undo,\r\n redo,\r\n clear,\r\n loadData,\r\n} = useSignatureHistory({\r\n maxHistory: props.maxHistory,\r\n onChange: (data) => {\r\n canvasInstance.redrawStrokes(data as SignatureStroke[]);\r\n emit(\"change\", data as SignatureStroke[]);\r\n },\r\n});\r\n\r\nconst canvasInstance = useSignatureCanvas({\r\n canvasRef,\r\n penConfig: toRef(currentPenConfig),\r\n mode: currentMode,\r\n disabled: toRef(props, \"disabled\"),\r\n onStrokeComplete: (stroke: SignatureStroke) => {\r\n addStroke(stroke);\r\n emit(\"end-draw\", stroke);\r\n },\r\n onDrawStart: () => emit(\"start-draw\"),\r\n onDrawing: (point: SignaturePoint) => emit(\"drawing\", point),\r\n});\r\n\r\nconst exportInstance = useSignatureExport({\r\n canvasRef,\r\n watermark: toRef(currentWatermark),\r\n});\r\n\r\nconst handleUndo = (): boolean => {\r\n const result = undo();\r\n if (result) emit(\"undo\");\r\n return result;\r\n};\r\n\r\nconst handleRedo = (): boolean => {\r\n const result = redo();\r\n if (result) emit(\"redo\");\r\n return result;\r\n};\r\n\r\nconst handleClear = () => {\r\n clear();\r\n canvasInstance.clearCanvas();\r\n emit(\"clear\");\r\n};\r\n\r\nconst exportSignature = async (\r\n options?: ExportOptions,\r\n): Promise<string | Blob> => {\r\n return exportInstance.exportSignature(options);\r\n};\r\n\r\nconst download = async (\r\n filename?: string,\r\n options?: ExportOptions,\r\n): Promise<void> => {\r\n return exportInstance.download(filename, options);\r\n};\r\n\r\nconst loadImage = async (imageUrl: string): Promise<void> => {\r\n await exportInstance.loadImage(imageUrl);\r\n clear();\r\n};\r\n\r\nconst getSignatureData = (): SignatureStroke[] =>\r\n strokes.value as SignatureStroke[];\r\nconst loadSignatureData = (data: SignatureStroke[]): void => loadData(data);\r\nconst isSignatureEmpty = (): boolean => isEmpty.value;\r\n\r\nonMounted(() => {\r\n canvasInstance.initCanvas();\r\n canvasInstance.bindEvents();\r\n if (props.backgroundImage) loadImage(props.backgroundImage);\r\n});\r\n\r\nonUnmounted(() => canvasInstance.unbindEvents());\r\n\r\ndefineExpose<SignatureExpose>({\r\n clear: handleClear,\r\n undo: handleUndo,\r\n redo: handleRedo,\r\n export: exportSignature,\r\n download,\r\n loadImage,\r\n getSignatureData,\r\n loadSignatureData,\r\n isEmpty: isSignatureEmpty,\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: 2026-02-25\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 class=\"c-signature\">\r\n <div v-if=\"showToolbar\" class=\"signature-toolbar\">\r\n <div class=\"toolbar-section\">\r\n <NButtonGroup>\r\n <NButton\r\n :type=\"currentMode === 'pen' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'pen'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:draw\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 画笔\r\n </NButton>\r\n <NButton\r\n :type=\"currentMode === 'eraser' ? 'primary' : 'default'\"\r\n size=\"small\"\r\n @click=\"currentMode = 'eraser'\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:eraser\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 橡皮擦\r\n </NButton>\r\n </NButtonGroup>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">颜色</span>\r\n <NColorPicker\r\n v-model:value=\"currentPenConfig.color\"\r\n :show-alpha=\"false\"\r\n size=\"small\"\r\n :swatches=\"PRESET_COLORS\"\r\n />\r\n </div>\r\n\r\n <div v-if=\"currentMode === 'pen'\" class=\"toolbar-section\">\r\n <span class=\"section-label\">粗细</span>\r\n <NInputNumber\r\n v-model:value=\"currentPenConfig.width\"\r\n :min=\"1\"\r\n :max=\"20\"\r\n size=\"small\"\r\n style=\"width: 80px\"\r\n />\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton size=\"small\" :disabled=\"!canUndo\" @click=\"handleUndo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:undo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 撤销\r\n </NButton>\r\n <NButton size=\"small\" :disabled=\"!canRedo\" @click=\"handleRedo\">\r\n <template #icon>\r\n <C_Icon name=\"mdi:redo\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 重做\r\n </NButton>\r\n </div>\r\n\r\n <div class=\"toolbar-section divider\" />\r\n\r\n <div class=\"toolbar-section\">\r\n <NButton\r\n size=\"small\"\r\n type=\"error\"\r\n :disabled=\"isEmpty\"\r\n @click=\"handleClear\"\r\n >\r\n <template #icon>\r\n <C_Icon name=\"mdi:delete-outline\" :size=\"16\" color=\"currentColor\" />\r\n </template>\r\n 清空\r\n </NButton>\r\n </div>\r\n </div>\r\n\r\n <div\r\n class=\"signature-canvas-wrapper\"\r\n :class=\"{ disabled, readonly }\"\r\n :style=\"{\r\n width: typeof width === 'number' ? `${width}px` : width,\r\n height: typeof height === 'number' ? `${height}px` : height,\r\n backgroundColor: backgroundColor || 'transparent',\r\n }\"\r\n >\r\n <canvas\r\n ref=\"canvasRef\"\r\n class=\"signature-canvas\"\r\n :class=\"{ disabled, readonly }\"\r\n />\r\n <div v-if=\"isEmpty && !disabled && !readonly\" class=\"canvas-placeholder\">\r\n 请在此处签名\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, reactive, toRef, onMounted, onUnmounted } from \"vue\";\r\nimport { NButton, NButtonGroup, NColorPicker, NInputNumber } from \"naive-ui\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSignatureCanvas } from \"./composables/useSignatureCanvas\";\r\nimport { useSignatureHistory } from \"./composables/useSignatureHistory\";\r\nimport { useSignatureExport } from \"./composables/useSignatureExport\";\r\nimport {\r\n DEFAULT_PEN_CONFIG,\r\n DEFAULT_WATERMARK_CONFIG,\r\n PRESET_COLORS,\r\n} from \"./data\";\r\nimport type {\r\n ExportOptions,\r\n PenConfig,\r\n PenMode,\r\n SignatureExpose,\r\n SignaturePoint,\r\n SignatureProps,\r\n SignatureStroke,\r\n WatermarkConfig,\r\n} from \"./types\";\r\n\r\ndefineOptions({ name: \"C_Signature\" });\r\n\r\nconst props = withDefaults(defineProps<SignatureProps>(), {\r\n width: \"100%\",\r\n height: 300,\r\n disabled: false,\r\n readonly: false,\r\n showToolbar: true,\r\n maxHistory: 50,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n \"start-draw\": [];\r\n drawing: [point: SignaturePoint];\r\n \"end-draw\": [stroke: SignatureStroke];\r\n clear: [];\r\n undo: [];\r\n redo: [];\r\n change: [data: SignatureStroke[]];\r\n}>();\r\n\r\nconst canvasRef = ref<HTMLCanvasElement | null>(null);\r\nconst currentMode = ref<PenMode>(\"pen\");\r\nconst currentPenConfig = reactive<PenConfig>({\r\n ...DEFAULT_PEN_CONFIG,\r\n ...props.penConfig,\r\n});\r\nconst currentWatermark = reactive<WatermarkConfig>({\r\n ...DEFAULT_WATERMARK_CONFIG,\r\n ...props.watermark,\r\n});\r\n\r\nconst {\r\n strokes,\r\n canUndo,\r\n canRedo,\r\n isEmpty,\r\n addStroke,\r\n undo,\r\n redo,\r\n clear,\r\n loadData,\r\n} = useSignatureHistory({\r\n maxHistory: props.maxHistory,\r\n onChange: (data) => {\r\n canvasInstance.redrawStrokes(data as SignatureStroke[]);\r\n emit(\"change\", data as SignatureStroke[]);\r\n },\r\n});\r\n\r\nconst canvasInstance = useSignatureCanvas({\r\n canvasRef,\r\n penConfig: toRef(currentPenConfig),\r\n mode: currentMode,\r\n disabled: toRef(props, \"disabled\"),\r\n onStrokeComplete: (stroke: SignatureStroke) => {\r\n addStroke(stroke);\r\n emit(\"end-draw\", stroke);\r\n },\r\n onDrawStart: () => emit(\"start-draw\"),\r\n onDrawing: (point: SignaturePoint) => emit(\"drawing\", point),\r\n});\r\n\r\nconst exportInstance = useSignatureExport({\r\n canvasRef,\r\n watermark: toRef(currentWatermark),\r\n});\r\n\r\nconst handleUndo = (): boolean => {\r\n const result = undo();\r\n if (result) emit(\"undo\");\r\n return result;\r\n};\r\n\r\nconst handleRedo = (): boolean => {\r\n const result = redo();\r\n if (result) emit(\"redo\");\r\n return result;\r\n};\r\n\r\nconst handleClear = () => {\r\n clear();\r\n canvasInstance.clearCanvas();\r\n emit(\"clear\");\r\n};\r\n\r\nconst exportSignature = async (\r\n options?: ExportOptions,\r\n): Promise<string | Blob> => {\r\n return exportInstance.exportSignature(options);\r\n};\r\n\r\nconst download = async (\r\n filename?: string,\r\n options?: ExportOptions,\r\n): Promise<void> => {\r\n return exportInstance.download(filename, options);\r\n};\r\n\r\nconst loadImage = async (imageUrl: string): Promise<void> => {\r\n await exportInstance.loadImage(imageUrl);\r\n clear();\r\n};\r\n\r\nconst getSignatureData = (): SignatureStroke[] =>\r\n strokes.value as SignatureStroke[];\r\nconst loadSignatureData = (data: SignatureStroke[]): void => loadData(data);\r\nconst isSignatureEmpty = (): boolean => isEmpty.value;\r\n\r\nonMounted(() => {\r\n canvasInstance.initCanvas();\r\n canvasInstance.bindEvents();\r\n if (props.backgroundImage) loadImage(props.backgroundImage);\r\n});\r\n\r\nonUnmounted(() => canvasInstance.unbindEvents());\r\n\r\ndefineExpose<SignatureExpose>({\r\n clear: handleClear,\r\n undo: handleUndo,\r\n redo: handleRedo,\r\n export: exportSignature,\r\n download,\r\n loadImage,\r\n getSignatureData,\r\n loadSignatureData,\r\n isEmpty: isSignatureEmpty,\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_SplitPane-C6sBsfKY.css","names":[],"sources":["../src/components/C_SplitPane/index.vue?vue&type=style&index=0&scoped=80043f11&lang.scss"],"sourcesContent":["/* 分割面板组件样式 */\n.c-split-pane[data-v-80043f11] {\n display: flex;\n width: 100%;\n height: 100%;\n overflow: hidden;\n position: relative;\n /* ─── 方向 ────────────────────────────────── */\n /* ─── 拖拽中 ──────────────────────────────── */\n /* ─── 禁用 ────────────────────────────────── */\n /* ─── 折叠过渡 ────────────────────────────── */\n}\n.c-split-pane--horizontal[data-v-80043f11] {\n flex-direction: row;\n}\n.c-split-pane--vertical[data-v-80043f11] {\n flex-direction: column;\n}\n.c-split-pane--dragging .split-panel__content[data-v-80043f11] {\n pointer-events: none;\n user-select: none;\n}\n.c-split-pane--disabled .split-gutter[data-v-80043f11] {\n cursor: default !important;\n opacity: 0.5;\n}\n.c-split-pane--disabled .split-gutter[data-v-80043f11]:hover {\n background: transparent;\n}\n.c-split-pane--collapsed-first .split-panel[data-v-80043f11], .c-split-pane--collapsed-second .split-panel[data-v-80043f11] {\n transition: width 0.3s ease, height 0.3s ease;\n}\n\n/* ─── 面板 ──────────────────────────────────── */\n.split-panel[data-v-80043f11] {\n overflow: hidden;\n position: relative;\n}\n.split-panel__content[data-v-80043f11] {\n width: 100%;\n height: 100%;\n overflow: auto;\n}\n\n/* ─── 分割线手柄 ────────────────────────────── */\n.split-gutter[data-v-80043f11] {\n position: relative;\n display: flex;\n align-items: center;\n justify-content: center;\n flex-shrink: 0;\n z-index: 1;\n background: var(--c-border, #e1e4e8);\n transition: background-color 0.2s ease;\n outline: none;\n /* 水平方向 */\n /* 垂直方向 */\n /* focus 轮廓 */\n /* ─── 视觉指示条 ──────────────────────────── */\n /* ─── 折叠按钮 ────────────────────────────── */\n}\n.c-split-pane--horizontal .split-gutter[data-v-80043f11] {\n cursor: col-resize;\n min-width: 4px;\n}\n.c-split-pane--horizontal .split-gutter[data-v-80043f11]:hover, .c-split-pane--horizontal .split-gutter[data-v-80043f11]:focus-visible {\n background: var(--c-primary, #409eff);\n}\n.c-split-pane--vertical .split-gutter[data-v-80043f11] {\n cursor: row-resize;\n min-height: 4px;\n}\n.c-split-pane--vertical .split-gutter[data-v-80043f11]:hover, .c-split-pane--vertical .split-gutter[data-v-80043f11]:focus-visible {\n background: var(--c-primary, #409eff);\n}\n.split-gutter[data-v-80043f11]:focus-visible {\n box-shadow: 0 0 0 2px rgba(64, 158, 255, 0.3);\n}\n.split-gutter__visual[data-v-80043f11] {\n display: flex;\n align-items: center;\n justify-content: center;\n}\n.split-gutter__dots[data-v-80043f11] {\n display: flex;\n gap: 2px;\n}\n.c-split-pane--horizontal .split-gutter__dots[data-v-80043f11] {\n flex-direction: column;\n}\n.c-split-pane--vertical .split-gutter__dots[data-v-80043f11] {\n flex-direction: row;\n}\n.split-gutter__dot[data-v-80043f11] {\n width: 3px;\n height: 3px;\n border-radius: 50%;\n background: var(--c-text-4, #ccc);\n transition: background-color 0.2s ease;\n}\n.split-gutter:hover .split-gutter__dot[data-v-80043f11] {\n background: #fff;\n}\n.split-gutter__collapse-btn[data-v-80043f11] {\n position: absolute;\n display: flex;\n align-items: center;\n justify-content: center;\n width: 16px;\n height: 16px;\n padding: 0;\n border: 1px solid var(--c-border, #e1e4e8);\n border-radius: 50%;\n background: var(--c-bg-card, #fff);\n color: var(--c-text-3, #999);\n font-size: 12px;\n cursor: pointer;\n opacity: 0;\n transition: opacity 0.2s ease, color 0.2s ease, background-color 0.2s ease;\n z-index: 2;\n /* 水平方向的定位 */\n /* 垂直方向的定位 */\n}\n.split-gutter:hover .split-gutter__collapse-btn[data-v-80043f11] {\n opacity: 1;\n}\n.split-gutter__collapse-btn[data-v-80043f11]:hover {\n color: var(--c-primary, #409eff);\n background: var(--c-bg, #fff);\n}\n.c-split-pane--horizontal .split-gutter__collapse-btn[data-v-80043f11] {\n left: 50%;\n transform: translateX(-50%);\n}\n.c-split-pane--horizontal .split-gutter__collapse-btn--first[data-v-80043f11] {\n top: 8px;\n}\n.c-split-pane--horizontal .split-gutter__collapse-btn--second[data-v-80043f11] {\n bottom: 8px;\n}\n.c-split-pane--vertical .split-gutter__collapse-btn[data-v-80043f11] {\n top: 50%;\n transform: translateY(-50%);\n}\n.c-split-pane--vertical .split-gutter__collapse-btn--first[data-v-80043f11] {\n left: 8px;\n}\n.c-split-pane--vertical .split-gutter__collapse-btn--second[data-v-80043f11] {\n right: 8px;\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;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;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;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,6 @@
|
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
|
2
|
+
require('./C_Icon.js');
|
|
3
|
+
const require_C_SplitPane = require('./C_SplitPane.js');
|
|
4
|
+
|
|
5
|
+
exports.C_SplitPane = require_C_SplitPane.C_SplitPane_default;
|
|
6
|
+
exports.useSplitResize = require_C_SplitPane.useSplitResize;
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import { a as SplitDirection, c as SplitPaneProps, i as PanelInfo, n as _default, o as SplitPaneEmits, r as CollapseTarget, s as SplitPaneExpose, t as useSplitResize } from "./useSplitResize.js";
|
|
2
|
+
export { _default as C_SplitPane, type CollapseTarget, type PanelInfo, type SplitDirection, type SplitPaneEmits, type SplitPaneExpose, type SplitPaneProps, useSplitResize };
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import { a as SplitDirection, c as SplitPaneProps, i as PanelInfo, n as _default, o as SplitPaneEmits, r as CollapseTarget, s as SplitPaneExpose, t as useSplitResize } from "./useSplitResize.js";
|
|
2
|
+
export { _default as C_SplitPane, type CollapseTarget, type PanelInfo, type SplitDirection, type SplitPaneEmits, type SplitPaneExpose, type SplitPaneProps, useSplitResize };
|
|
@@ -0,0 +1,356 @@
|
|
|
1
|
+
import { t as C_Icon_default } from "./C_Icon2.js";
|
|
2
|
+
import { t as export_helper_default } from "./export-helper.js";
|
|
3
|
+
import { Fragment, computed, createCommentVNode, createElementBlock, createElementVNode, createVNode, defineComponent, normalizeClass, normalizeStyle, onUnmounted, openBlock, ref, renderList, renderSlot, toRef, unref, withModifiers } from "vue";
|
|
4
|
+
|
|
5
|
+
//#region src/components/C_SplitPane/composables/useSplitResize.ts
|
|
6
|
+
/**
|
|
7
|
+
* 分割面板拖拽调整逻辑
|
|
8
|
+
*/
|
|
9
|
+
function useSplitResize(options) {
|
|
10
|
+
const { containerRef, direction, defaultSize, minSize, maxSize, disabled, collapsible, step, onResize, onCollapse, onExpand, onDragStart, onDragEnd } = options;
|
|
11
|
+
/** 首面板当前大小(百分比) */
|
|
12
|
+
const panelSize = ref(defaultSize);
|
|
13
|
+
/** 是否正在拖拽 */
|
|
14
|
+
const isDragging = ref(false);
|
|
15
|
+
/** 折叠状态 */
|
|
16
|
+
const collapsedTarget = ref(null);
|
|
17
|
+
/** 折叠前记住的面板大小 */
|
|
18
|
+
const sizeBeforeCollapse = ref(defaultSize);
|
|
19
|
+
/** 首面板是否折叠 */
|
|
20
|
+
const isFirstCollapsed = computed(() => collapsedTarget.value === "first");
|
|
21
|
+
/** 第二面板是否折叠 */
|
|
22
|
+
const isSecondCollapsed = computed(() => collapsedTarget.value === "second");
|
|
23
|
+
/**
|
|
24
|
+
* 将大小限制在 min/max 范围内
|
|
25
|
+
*/
|
|
26
|
+
const clampSize = (size) => {
|
|
27
|
+
return Math.min(maxSize, Math.max(minSize, size));
|
|
28
|
+
};
|
|
29
|
+
/**
|
|
30
|
+
* 获取容器在分割方向上的总长度(px)
|
|
31
|
+
*/
|
|
32
|
+
const getContainerLength = () => {
|
|
33
|
+
const container = containerRef.value;
|
|
34
|
+
if (!container) return 0;
|
|
35
|
+
return direction.value === "horizontal" ? container.offsetWidth : container.offsetHeight;
|
|
36
|
+
};
|
|
37
|
+
/**
|
|
38
|
+
* 获取容器在分割方向上的起始坐标
|
|
39
|
+
*/
|
|
40
|
+
const getContainerStart = () => {
|
|
41
|
+
const container = containerRef.value;
|
|
42
|
+
if (!container) return 0;
|
|
43
|
+
const rect = container.getBoundingClientRect();
|
|
44
|
+
return direction.value === "horizontal" ? rect.left : rect.top;
|
|
45
|
+
};
|
|
46
|
+
/**
|
|
47
|
+
* 根据鼠标/触摸位置计算面板大小(百分比)
|
|
48
|
+
*/
|
|
49
|
+
const calcSizeFromPosition = (clientPos) => {
|
|
50
|
+
const containerLength = getContainerLength();
|
|
51
|
+
if (containerLength === 0) return panelSize.value;
|
|
52
|
+
return clampSize((clientPos - getContainerStart()) / containerLength * 100);
|
|
53
|
+
};
|
|
54
|
+
/**
|
|
55
|
+
* 获取事件的位置坐标
|
|
56
|
+
*/
|
|
57
|
+
const getClientPos = (e) => {
|
|
58
|
+
if (e instanceof TouchEvent) return direction.value === "horizontal" ? e.touches[0].clientX : e.touches[0].clientY;
|
|
59
|
+
return direction.value === "horizontal" ? e.clientX : e.clientY;
|
|
60
|
+
};
|
|
61
|
+
/**
|
|
62
|
+
* 更新面板大小
|
|
63
|
+
*/
|
|
64
|
+
const updateSize = (newSize) => {
|
|
65
|
+
const clamped = clampSize(newSize);
|
|
66
|
+
panelSize.value = clamped;
|
|
67
|
+
onResize?.(clamped, 100 - clamped);
|
|
68
|
+
};
|
|
69
|
+
const handleMouseMove = (e) => {
|
|
70
|
+
if (!isDragging.value || disabled.value) return;
|
|
71
|
+
e.preventDefault();
|
|
72
|
+
const newSize = calcSizeFromPosition(getClientPos(e));
|
|
73
|
+
if (collapsedTarget.value) collapsedTarget.value = null;
|
|
74
|
+
updateSize(newSize);
|
|
75
|
+
};
|
|
76
|
+
const handleMouseUp = () => {
|
|
77
|
+
if (!isDragging.value) return;
|
|
78
|
+
isDragging.value = false;
|
|
79
|
+
document.body.style.cursor = "";
|
|
80
|
+
document.body.style.userSelect = "";
|
|
81
|
+
onDragEnd?.(panelSize.value);
|
|
82
|
+
document.removeEventListener("mousemove", handleMouseMove);
|
|
83
|
+
document.removeEventListener("mouseup", handleMouseUp);
|
|
84
|
+
document.removeEventListener("touchmove", handleMouseMove);
|
|
85
|
+
document.removeEventListener("touchend", handleMouseUp);
|
|
86
|
+
};
|
|
87
|
+
const handleMouseDown = (e) => {
|
|
88
|
+
if (disabled.value) return;
|
|
89
|
+
e.preventDefault();
|
|
90
|
+
e.currentTarget?.focus();
|
|
91
|
+
isDragging.value = true;
|
|
92
|
+
document.body.style.cursor = direction.value === "horizontal" ? "col-resize" : "row-resize";
|
|
93
|
+
document.body.style.userSelect = "none";
|
|
94
|
+
onDragStart?.(panelSize.value);
|
|
95
|
+
document.addEventListener("mousemove", handleMouseMove);
|
|
96
|
+
document.addEventListener("mouseup", handleMouseUp);
|
|
97
|
+
document.addEventListener("touchmove", handleMouseMove, { passive: false });
|
|
98
|
+
document.addEventListener("touchend", handleMouseUp);
|
|
99
|
+
};
|
|
100
|
+
const handleDoubleClick = () => {
|
|
101
|
+
if (disabled.value || !collapsible.value) return;
|
|
102
|
+
toggle("first");
|
|
103
|
+
};
|
|
104
|
+
const collapse = (target = "first") => {
|
|
105
|
+
if (!collapsible.value || collapsedTarget.value === target) return;
|
|
106
|
+
if (!collapsedTarget.value) sizeBeforeCollapse.value = panelSize.value;
|
|
107
|
+
collapsedTarget.value = target;
|
|
108
|
+
if (target === "first") panelSize.value = 0;
|
|
109
|
+
else panelSize.value = 100;
|
|
110
|
+
onCollapse?.(target);
|
|
111
|
+
onResize?.(panelSize.value, 100 - panelSize.value);
|
|
112
|
+
};
|
|
113
|
+
const expand = () => {
|
|
114
|
+
if (!collapsedTarget.value) return;
|
|
115
|
+
const target = collapsedTarget.value;
|
|
116
|
+
collapsedTarget.value = null;
|
|
117
|
+
panelSize.value = sizeBeforeCollapse.value;
|
|
118
|
+
onExpand?.(target);
|
|
119
|
+
onResize?.(panelSize.value, 100 - panelSize.value);
|
|
120
|
+
};
|
|
121
|
+
const toggle = (target = "first") => {
|
|
122
|
+
if (collapsedTarget.value) expand();
|
|
123
|
+
else collapse(target);
|
|
124
|
+
};
|
|
125
|
+
/**
|
|
126
|
+
* 根据按键获取增量
|
|
127
|
+
*/
|
|
128
|
+
const getKeyDelta = (key, isHorizontal) => {
|
|
129
|
+
return {
|
|
130
|
+
ArrowLeft: isHorizontal ? -step : 0,
|
|
131
|
+
ArrowRight: isHorizontal ? step : 0,
|
|
132
|
+
ArrowUp: !isHorizontal ? -step : 0,
|
|
133
|
+
ArrowDown: !isHorizontal ? step : 0
|
|
134
|
+
}[key] ?? null;
|
|
135
|
+
};
|
|
136
|
+
const handleKeyDown = (e) => {
|
|
137
|
+
if (disabled.value) return;
|
|
138
|
+
if (e.key === "Home") {
|
|
139
|
+
updateSize(minSize);
|
|
140
|
+
e.preventDefault();
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
if (e.key === "End") {
|
|
144
|
+
updateSize(maxSize);
|
|
145
|
+
e.preventDefault();
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
const delta = getKeyDelta(e.key, direction.value === "horizontal");
|
|
149
|
+
if (delta === null || delta === 0) return;
|
|
150
|
+
e.preventDefault();
|
|
151
|
+
if (collapsedTarget.value) collapsedTarget.value = null;
|
|
152
|
+
updateSize(panelSize.value + delta);
|
|
153
|
+
};
|
|
154
|
+
const resetSize = () => {
|
|
155
|
+
collapsedTarget.value = null;
|
|
156
|
+
panelSize.value = defaultSize;
|
|
157
|
+
onResize?.(defaultSize, 100 - defaultSize);
|
|
158
|
+
};
|
|
159
|
+
const setSize = (size) => {
|
|
160
|
+
collapsedTarget.value = null;
|
|
161
|
+
updateSize(size);
|
|
162
|
+
};
|
|
163
|
+
const getPanelInfo = () => ({
|
|
164
|
+
first: {
|
|
165
|
+
size: panelSize.value,
|
|
166
|
+
collapsed: isFirstCollapsed.value
|
|
167
|
+
},
|
|
168
|
+
second: {
|
|
169
|
+
size: 100 - panelSize.value,
|
|
170
|
+
collapsed: isSecondCollapsed.value
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
onUnmounted(() => {
|
|
174
|
+
document.removeEventListener("mousemove", handleMouseMove);
|
|
175
|
+
document.removeEventListener("mouseup", handleMouseUp);
|
|
176
|
+
document.removeEventListener("touchmove", handleMouseMove);
|
|
177
|
+
document.removeEventListener("touchend", handleMouseUp);
|
|
178
|
+
});
|
|
179
|
+
return {
|
|
180
|
+
panelSize,
|
|
181
|
+
isDragging,
|
|
182
|
+
collapsedTarget,
|
|
183
|
+
isFirstCollapsed,
|
|
184
|
+
isSecondCollapsed,
|
|
185
|
+
handleMouseDown,
|
|
186
|
+
handleDoubleClick,
|
|
187
|
+
handleKeyDown,
|
|
188
|
+
collapse,
|
|
189
|
+
expand,
|
|
190
|
+
toggle,
|
|
191
|
+
resetSize,
|
|
192
|
+
setSize,
|
|
193
|
+
getPanelInfo
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
//#endregion
|
|
198
|
+
//#region src/components/C_SplitPane/index.vue?vue&type=script&setup=true&lang.ts
|
|
199
|
+
const _hoisted_1 = { class: "split-panel__content" };
|
|
200
|
+
const _hoisted_2 = ["aria-orientation", "aria-valuenow"];
|
|
201
|
+
const _hoisted_3 = { class: "split-gutter__visual" };
|
|
202
|
+
const _hoisted_4 = { class: "split-gutter__dots" };
|
|
203
|
+
const _hoisted_5 = ["title"];
|
|
204
|
+
const _hoisted_6 = ["title"];
|
|
205
|
+
const _hoisted_7 = { class: "split-panel__content" };
|
|
206
|
+
var index_vue_vue_type_script_setup_true_lang_default = /* @__PURE__ */ defineComponent({
|
|
207
|
+
name: "C_SplitPane",
|
|
208
|
+
__name: "index",
|
|
209
|
+
props: {
|
|
210
|
+
direction: { default: "horizontal" },
|
|
211
|
+
defaultSize: { default: 50 },
|
|
212
|
+
minSize: { default: 0 },
|
|
213
|
+
maxSize: { default: 100 },
|
|
214
|
+
disabled: {
|
|
215
|
+
type: Boolean,
|
|
216
|
+
default: false
|
|
217
|
+
},
|
|
218
|
+
collapsible: {
|
|
219
|
+
type: Boolean,
|
|
220
|
+
default: true
|
|
221
|
+
},
|
|
222
|
+
showCollapseButton: {
|
|
223
|
+
type: Boolean,
|
|
224
|
+
default: true
|
|
225
|
+
},
|
|
226
|
+
gutterSize: { default: 6 },
|
|
227
|
+
step: { default: 2 }
|
|
228
|
+
},
|
|
229
|
+
emits: [
|
|
230
|
+
"resize",
|
|
231
|
+
"collapse",
|
|
232
|
+
"expand",
|
|
233
|
+
"drag-start",
|
|
234
|
+
"drag-end"
|
|
235
|
+
],
|
|
236
|
+
setup(__props, { expose: __expose, emit: __emit }) {
|
|
237
|
+
const props = __props;
|
|
238
|
+
const emit = __emit;
|
|
239
|
+
const containerRef = ref(null);
|
|
240
|
+
const { panelSize, isDragging, isFirstCollapsed, isSecondCollapsed, handleMouseDown, handleDoubleClick, handleKeyDown, collapse, expand, toggle, resetSize, setSize, getPanelInfo } = useSplitResize({
|
|
241
|
+
containerRef,
|
|
242
|
+
direction: toRef(props, "direction"),
|
|
243
|
+
defaultSize: props.defaultSize,
|
|
244
|
+
minSize: props.minSize,
|
|
245
|
+
maxSize: props.maxSize,
|
|
246
|
+
disabled: toRef(props, "disabled"),
|
|
247
|
+
collapsible: toRef(props, "collapsible"),
|
|
248
|
+
step: props.step,
|
|
249
|
+
onResize: (first, second) => emit("resize", first, second),
|
|
250
|
+
onCollapse: (target) => emit("collapse", target),
|
|
251
|
+
onExpand: (target) => emit("expand", target),
|
|
252
|
+
onDragStart: (size) => emit("drag-start", size),
|
|
253
|
+
onDragEnd: (size) => emit("drag-end", size)
|
|
254
|
+
});
|
|
255
|
+
const firstPanelStyle = computed(() => {
|
|
256
|
+
const sizeValue = `calc(${panelSize.value}% - ${props.gutterSize / 2}px)`;
|
|
257
|
+
return props.direction === "horizontal" ? { width: sizeValue } : { height: sizeValue };
|
|
258
|
+
});
|
|
259
|
+
const secondPanelStyle = computed(() => {
|
|
260
|
+
const sizeValue = `calc(${100 - panelSize.value}% - ${props.gutterSize / 2}px)`;
|
|
261
|
+
return props.direction === "horizontal" ? { width: sizeValue } : { height: sizeValue };
|
|
262
|
+
});
|
|
263
|
+
const gutterStyle = computed(() => {
|
|
264
|
+
return props.direction === "horizontal" ? { width: `${props.gutterSize}px` } : { height: `${props.gutterSize}px` };
|
|
265
|
+
});
|
|
266
|
+
const collapseFirstIcon = computed(() => {
|
|
267
|
+
if (props.direction === "horizontal") return isFirstCollapsed.value ? "mdi:chevron-right" : "mdi:chevron-left";
|
|
268
|
+
return isFirstCollapsed.value ? "mdi:chevron-down" : "mdi:chevron-up";
|
|
269
|
+
});
|
|
270
|
+
const collapseSecondIcon = computed(() => {
|
|
271
|
+
if (props.direction === "horizontal") return isSecondCollapsed.value ? "mdi:chevron-left" : "mdi:chevron-right";
|
|
272
|
+
return isSecondCollapsed.value ? "mdi:chevron-up" : "mdi:chevron-down";
|
|
273
|
+
});
|
|
274
|
+
__expose({
|
|
275
|
+
collapse,
|
|
276
|
+
expand,
|
|
277
|
+
toggle,
|
|
278
|
+
resetSize,
|
|
279
|
+
getPanelInfo,
|
|
280
|
+
setSize
|
|
281
|
+
});
|
|
282
|
+
return (_ctx, _cache) => {
|
|
283
|
+
return openBlock(), createElementBlock("div", {
|
|
284
|
+
ref_key: "containerRef",
|
|
285
|
+
ref: containerRef,
|
|
286
|
+
class: normalizeClass(["c-split-pane", [`c-split-pane--${_ctx.direction}`, {
|
|
287
|
+
"c-split-pane--dragging": unref(isDragging),
|
|
288
|
+
"c-split-pane--disabled": _ctx.disabled,
|
|
289
|
+
"c-split-pane--collapsed-first": unref(isFirstCollapsed),
|
|
290
|
+
"c-split-pane--collapsed-second": unref(isSecondCollapsed)
|
|
291
|
+
}]])
|
|
292
|
+
}, [
|
|
293
|
+
createCommentVNode(" 首面板 "),
|
|
294
|
+
createElementVNode("div", {
|
|
295
|
+
class: "split-panel split-panel--first",
|
|
296
|
+
style: normalizeStyle(firstPanelStyle.value)
|
|
297
|
+
}, [createElementVNode("div", _hoisted_1, [renderSlot(_ctx.$slots, "first", {}, void 0, true)])], 4),
|
|
298
|
+
createCommentVNode(" 分割线手柄 "),
|
|
299
|
+
createElementVNode("div", {
|
|
300
|
+
class: "split-gutter",
|
|
301
|
+
style: normalizeStyle(gutterStyle.value),
|
|
302
|
+
role: "separator",
|
|
303
|
+
"aria-orientation": _ctx.direction === "horizontal" ? "vertical" : "horizontal",
|
|
304
|
+
"aria-valuenow": Math.round(unref(panelSize)),
|
|
305
|
+
"aria-valuemin": 0,
|
|
306
|
+
"aria-valuemax": 100,
|
|
307
|
+
tabindex: "0",
|
|
308
|
+
onMousedown: _cache[4] || (_cache[4] = (...args) => unref(handleMouseDown) && unref(handleMouseDown)(...args)),
|
|
309
|
+
onTouchstart: _cache[5] || (_cache[5] = withModifiers((...args) => unref(handleMouseDown) && unref(handleMouseDown)(...args), ["prevent"])),
|
|
310
|
+
onDblclick: _cache[6] || (_cache[6] = (...args) => unref(handleDoubleClick) && unref(handleDoubleClick)(...args)),
|
|
311
|
+
onKeydown: _cache[7] || (_cache[7] = (...args) => unref(handleKeyDown) && unref(handleKeyDown)(...args))
|
|
312
|
+
}, [
|
|
313
|
+
createElementVNode("div", _hoisted_3, [createElementVNode("div", _hoisted_4, [(openBlock(), createElementBlock(Fragment, null, renderList(3, (i) => {
|
|
314
|
+
return createElementVNode("span", {
|
|
315
|
+
key: i,
|
|
316
|
+
class: "split-gutter__dot"
|
|
317
|
+
});
|
|
318
|
+
}), 64))])]),
|
|
319
|
+
createCommentVNode(" 折叠按钮 "),
|
|
320
|
+
_ctx.showCollapseButton && _ctx.collapsible ? (openBlock(), createElementBlock(Fragment, { key: 0 }, [createElementVNode("button", {
|
|
321
|
+
class: "split-gutter__collapse-btn split-gutter__collapse-btn--first",
|
|
322
|
+
title: unref(isFirstCollapsed) ? "展开首面板" : "折叠首面板",
|
|
323
|
+
onClick: _cache[0] || (_cache[0] = withModifiers(($event) => unref(toggle)("first"), ["stop"])),
|
|
324
|
+
onMousedown: _cache[1] || (_cache[1] = withModifiers(() => {}, ["stop"]))
|
|
325
|
+
}, [createVNode(C_Icon_default, {
|
|
326
|
+
name: collapseFirstIcon.value,
|
|
327
|
+
size: 16,
|
|
328
|
+
color: "currentColor"
|
|
329
|
+
}, null, 8, ["name"])], 40, _hoisted_5), createElementVNode("button", {
|
|
330
|
+
class: "split-gutter__collapse-btn split-gutter__collapse-btn--second",
|
|
331
|
+
title: unref(isSecondCollapsed) ? "展开第二面板" : "折叠第二面板",
|
|
332
|
+
onClick: _cache[2] || (_cache[2] = withModifiers(($event) => unref(toggle)("second"), ["stop"])),
|
|
333
|
+
onMousedown: _cache[3] || (_cache[3] = withModifiers(() => {}, ["stop"]))
|
|
334
|
+
}, [createVNode(C_Icon_default, {
|
|
335
|
+
name: collapseSecondIcon.value,
|
|
336
|
+
size: 16,
|
|
337
|
+
color: "currentColor"
|
|
338
|
+
}, null, 8, ["name"])], 40, _hoisted_6)], 64)) : createCommentVNode("v-if", true)
|
|
339
|
+
], 44, _hoisted_2),
|
|
340
|
+
createCommentVNode(" 第二面板 "),
|
|
341
|
+
createElementVNode("div", {
|
|
342
|
+
class: "split-panel split-panel--second",
|
|
343
|
+
style: normalizeStyle(secondPanelStyle.value)
|
|
344
|
+
}, [createElementVNode("div", _hoisted_7, [renderSlot(_ctx.$slots, "second", {}, void 0, true)])], 4)
|
|
345
|
+
], 2);
|
|
346
|
+
};
|
|
347
|
+
}
|
|
348
|
+
});
|
|
349
|
+
|
|
350
|
+
//#endregion
|
|
351
|
+
//#region src/components/C_SplitPane/index.vue
|
|
352
|
+
var C_SplitPane_default = /* @__PURE__ */ export_helper_default(index_vue_vue_type_script_setup_true_lang_default, [["__scopeId", "data-v-80043f11"]]);
|
|
353
|
+
|
|
354
|
+
//#endregion
|
|
355
|
+
export { useSplitResize as n, C_SplitPane_default as t };
|
|
356
|
+
//# sourceMappingURL=C_SplitPane2.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"C_SplitPane2.js","names":["direction","disabled","showCollapseButton","collapsible"],"sources":["../src/components/C_SplitPane/composables/useSplitResize.ts","../src/components/C_SplitPane/index.vue","../src/components/C_SplitPane/index.vue","../src/components/C_SplitPane/index.vue"],"sourcesContent":["import { ref, computed, onUnmounted } from \"vue\";\r\nimport type { Ref } from \"vue\";\r\nimport type { CollapseTarget, SplitDirection } from \"../types\";\r\n\r\ninterface UseSplitResizeOptions {\r\n /** 容器引用 */\r\n containerRef: Ref<HTMLElement | null>;\r\n /** 分割方向 */\r\n direction: Ref<SplitDirection>;\r\n /** 首面板默认大小(百分比) */\r\n defaultSize: number;\r\n /** 首面板最小大小(百分比) */\r\n minSize: number;\r\n /** 首面板最大大小(百分比) */\r\n maxSize: number;\r\n /** 是否禁用 */\r\n disabled: Ref<boolean>;\r\n /** 是否可折叠 */\r\n collapsible: Ref<boolean>;\r\n /** 键盘微调步长(百分比) */\r\n step: number;\r\n /** resize 回调 */\r\n onResize?: (firstSize: number, secondSize: number) => void;\r\n /** 折叠回调 */\r\n onCollapse?: (target: CollapseTarget) => void;\r\n /** 展开回调 */\r\n onExpand?: (target: CollapseTarget) => void;\r\n /** 拖拽开始回调 */\r\n onDragStart?: (size: number) => void;\r\n /** 拖拽结束回调 */\r\n onDragEnd?: (size: number) => void;\r\n}\r\n\r\n/**\r\n * 分割面板拖拽调整逻辑\r\n */\r\nexport function useSplitResize(options: UseSplitResizeOptions) {\r\n const {\r\n containerRef,\r\n direction,\r\n defaultSize,\r\n minSize,\r\n maxSize,\r\n disabled,\r\n collapsible,\r\n step,\r\n onResize,\r\n onCollapse,\r\n onExpand,\r\n onDragStart,\r\n onDragEnd,\r\n } = options;\r\n\r\n /** 首面板当前大小(百分比) */\r\n const panelSize = ref(defaultSize);\r\n /** 是否正在拖拽 */\r\n const isDragging = ref(false);\r\n /** 折叠状态 */\r\n const collapsedTarget = ref<CollapseTarget | null>(null);\r\n /** 折叠前记住的面板大小 */\r\n const sizeBeforeCollapse = ref(defaultSize);\r\n\r\n /** 首面板是否折叠 */\r\n const isFirstCollapsed = computed(() => collapsedTarget.value === \"first\");\r\n /** 第二面板是否折叠 */\r\n const isSecondCollapsed = computed(() => collapsedTarget.value === \"second\");\r\n\r\n /**\r\n * 将大小限制在 min/max 范围内\r\n */\r\n const clampSize = (size: number): number => {\r\n return Math.min(maxSize, Math.max(minSize, size));\r\n };\r\n\r\n /**\r\n * 获取容器在分割方向上的总长度(px)\r\n */\r\n const getContainerLength = (): number => {\r\n const container = containerRef.value;\r\n if (!container) return 0;\r\n return direction.value === \"horizontal\"\r\n ? container.offsetWidth\r\n : container.offsetHeight;\r\n };\r\n\r\n /**\r\n * 获取容器在分割方向上的起始坐标\r\n */\r\n const getContainerStart = (): number => {\r\n const container = containerRef.value;\r\n if (!container) return 0;\r\n const rect = container.getBoundingClientRect();\r\n return direction.value === \"horizontal\" ? rect.left : rect.top;\r\n };\r\n\r\n /**\r\n * 根据鼠标/触摸位置计算面板大小(百分比)\r\n */\r\n const calcSizeFromPosition = (clientPos: number): number => {\r\n const containerLength = getContainerLength();\r\n if (containerLength === 0) return panelSize.value;\r\n\r\n const containerStart = getContainerStart();\r\n const offset = clientPos - containerStart;\r\n const percentage = (offset / containerLength) * 100;\r\n\r\n return clampSize(percentage);\r\n };\r\n\r\n /**\r\n * 获取事件的位置坐标\r\n */\r\n const getClientPos = (e: MouseEvent | TouchEvent): number => {\r\n if (e instanceof TouchEvent) {\r\n return direction.value === \"horizontal\"\r\n ? e.touches[0].clientX\r\n : e.touches[0].clientY;\r\n }\r\n return direction.value === \"horizontal\" ? e.clientX : e.clientY;\r\n };\r\n\r\n /**\r\n * 更新面板大小\r\n */\r\n const updateSize = (newSize: number) => {\r\n const clamped = clampSize(newSize);\r\n panelSize.value = clamped;\r\n onResize?.(clamped, 100 - clamped);\r\n };\r\n\r\n /* ─── 拖拽逻辑 ──────────────────────────────── */\r\n const handleMouseMove = (e: MouseEvent | TouchEvent) => {\r\n if (!isDragging.value || disabled.value) return;\r\n\r\n e.preventDefault();\r\n const pos = getClientPos(e);\r\n const newSize = calcSizeFromPosition(pos);\r\n\r\n /* 拖拽过程中取消折叠状态 */\r\n if (collapsedTarget.value) {\r\n collapsedTarget.value = null;\r\n }\r\n updateSize(newSize);\r\n };\r\n\r\n const handleMouseUp = () => {\r\n if (!isDragging.value) return;\r\n\r\n isDragging.value = false;\r\n document.body.style.cursor = \"\";\r\n document.body.style.userSelect = \"\";\r\n\r\n onDragEnd?.(panelSize.value);\r\n\r\n document.removeEventListener(\"mousemove\", handleMouseMove);\r\n document.removeEventListener(\"mouseup\", handleMouseUp);\r\n document.removeEventListener(\"touchmove\", handleMouseMove);\r\n document.removeEventListener(\"touchend\", handleMouseUp);\r\n };\r\n\r\n const handleMouseDown = (e: MouseEvent | TouchEvent) => {\r\n if (disabled.value) return;\r\n\r\n e.preventDefault();\r\n /* preventDefault 会阻止默认聚焦行为,需手动 focus 以支持键盘控制 */\r\n (e.currentTarget as HTMLElement)?.focus();\r\n isDragging.value = true;\r\n\r\n document.body.style.cursor =\r\n direction.value === \"horizontal\" ? \"col-resize\" : \"row-resize\";\r\n document.body.style.userSelect = \"none\";\r\n\r\n onDragStart?.(panelSize.value);\r\n\r\n document.addEventListener(\"mousemove\", handleMouseMove);\r\n document.addEventListener(\"mouseup\", handleMouseUp);\r\n document.addEventListener(\"touchmove\", handleMouseMove, { passive: false });\r\n document.addEventListener(\"touchend\", handleMouseUp);\r\n };\r\n\r\n /* ─── 双击折叠 ──────────────────────────────── */\r\n const handleDoubleClick = () => {\r\n if (disabled.value || !collapsible.value) return;\r\n toggle(\"first\");\r\n };\r\n\r\n /* ─── 折叠 / 展开 ───────────────────────────── */\r\n const collapse = (target: CollapseTarget = \"first\") => {\r\n if (!collapsible.value || collapsedTarget.value === target) return;\r\n\r\n /* 记住折叠前的大小 */\r\n if (!collapsedTarget.value) {\r\n sizeBeforeCollapse.value = panelSize.value;\r\n }\r\n\r\n collapsedTarget.value = target;\r\n\r\n if (target === \"first\") {\r\n panelSize.value = 0;\r\n } else {\r\n panelSize.value = 100;\r\n }\r\n\r\n onCollapse?.(target);\r\n onResize?.(panelSize.value, 100 - panelSize.value);\r\n };\r\n\r\n const expand = () => {\r\n if (!collapsedTarget.value) return;\r\n\r\n const target = collapsedTarget.value;\r\n collapsedTarget.value = null;\r\n panelSize.value = sizeBeforeCollapse.value;\r\n\r\n onExpand?.(target);\r\n onResize?.(panelSize.value, 100 - panelSize.value);\r\n };\r\n\r\n const toggle = (target: CollapseTarget = \"first\") => {\r\n if (collapsedTarget.value) {\r\n expand();\r\n } else {\r\n collapse(target);\r\n }\r\n };\r\n\r\n /* ─── 键盘微调 ──────────────────────────────── */\r\n /**\r\n * 根据按键获取增量\r\n */\r\n const getKeyDelta = (key: string, isHorizontal: boolean): number | null => {\r\n const keyMap: Record<string, number | null> = {\r\n ArrowLeft: isHorizontal ? -step : 0,\r\n ArrowRight: isHorizontal ? step : 0,\r\n ArrowUp: !isHorizontal ? -step : 0,\r\n ArrowDown: !isHorizontal ? step : 0,\r\n };\r\n return keyMap[key] ?? null;\r\n };\r\n\r\n const handleKeyDown = (e: KeyboardEvent) => {\r\n if (disabled.value) return;\r\n\r\n /* Home / End 直接跳到极值 */\r\n if (e.key === \"Home\") {\r\n updateSize(minSize);\r\n e.preventDefault();\r\n return;\r\n }\r\n if (e.key === \"End\") {\r\n updateSize(maxSize);\r\n e.preventDefault();\r\n return;\r\n }\r\n\r\n const delta = getKeyDelta(e.key, direction.value === \"horizontal\");\r\n if (delta === null || delta === 0) return;\r\n\r\n e.preventDefault();\r\n if (collapsedTarget.value) collapsedTarget.value = null;\r\n updateSize(panelSize.value + delta);\r\n };\r\n\r\n /* ─── 重置 / 查询 ───────────────────────────── */\r\n const resetSize = () => {\r\n collapsedTarget.value = null;\r\n panelSize.value = defaultSize;\r\n onResize?.(defaultSize, 100 - defaultSize);\r\n };\r\n\r\n const setSize = (size: number) => {\r\n collapsedTarget.value = null;\r\n updateSize(size);\r\n };\r\n\r\n const getPanelInfo = () => ({\r\n first: {\r\n size: panelSize.value,\r\n collapsed: isFirstCollapsed.value,\r\n },\r\n second: {\r\n size: 100 - panelSize.value,\r\n collapsed: isSecondCollapsed.value,\r\n },\r\n });\r\n\r\n /* 清理 */\r\n onUnmounted(() => {\r\n document.removeEventListener(\"mousemove\", handleMouseMove);\r\n document.removeEventListener(\"mouseup\", handleMouseUp);\r\n document.removeEventListener(\"touchmove\", handleMouseMove);\r\n document.removeEventListener(\"touchend\", handleMouseUp);\r\n });\r\n\r\n return {\r\n panelSize,\r\n isDragging,\r\n collapsedTarget,\r\n isFirstCollapsed,\r\n isSecondCollapsed,\r\n handleMouseDown,\r\n handleDoubleClick,\r\n handleKeyDown,\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n setSize,\r\n getPanelInfo,\r\n };\r\n}\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-06-16\r\n * @Description: 分割面板组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"containerRef\"\r\n class=\"c-split-pane\"\r\n :class=\"[\r\n `c-split-pane--${direction}`,\r\n {\r\n 'c-split-pane--dragging': isDragging,\r\n 'c-split-pane--disabled': disabled,\r\n 'c-split-pane--collapsed-first': isFirstCollapsed,\r\n 'c-split-pane--collapsed-second': isSecondCollapsed,\r\n },\r\n ]\"\r\n >\r\n <!-- 首面板 -->\r\n <div class=\"split-panel split-panel--first\" :style=\"firstPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"first\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 分割线手柄 -->\r\n <div\r\n class=\"split-gutter\"\r\n :style=\"gutterStyle\"\r\n role=\"separator\"\r\n :aria-orientation=\"direction === 'horizontal' ? 'vertical' : 'horizontal'\"\r\n :aria-valuenow=\"Math.round(panelSize)\"\r\n :aria-valuemin=\"0\"\r\n :aria-valuemax=\"100\"\r\n tabindex=\"0\"\r\n @mousedown=\"handleMouseDown\"\r\n @touchstart.prevent=\"handleMouseDown\"\r\n @dblclick=\"handleDoubleClick\"\r\n @keydown=\"handleKeyDown\"\r\n >\r\n <div class=\"split-gutter__visual\">\r\n <div class=\"split-gutter__dots\">\r\n <span v-for=\"i in 3\" :key=\"i\" class=\"split-gutter__dot\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 折叠按钮 -->\r\n <template v-if=\"showCollapseButton && collapsible\">\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--first\"\r\n :title=\"isFirstCollapsed ? '展开首面板' : '折叠首面板'\"\r\n @click.stop=\"toggle('first')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseFirstIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--second\"\r\n :title=\"isSecondCollapsed ? '展开第二面板' : '折叠第二面板'\"\r\n @click.stop=\"toggle('second')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseSecondIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n </template>\r\n </div>\r\n\r\n <!-- 第二面板 -->\r\n <div class=\"split-panel split-panel--second\" :style=\"secondPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"second\" />\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, toRef } from \"vue\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSplitResize } from \"./composables/useSplitResize\";\r\nimport type { CollapseTarget, SplitDirection, SplitPaneExpose } from \"./types\";\r\n\r\ndefineOptions({ name: \"C_SplitPane\" });\r\n\r\ninterface Props {\r\n /** 分割方向 */\r\n direction?: SplitDirection;\r\n /** 首面板默认大小(百分比 0-100) */\r\n defaultSize?: number;\r\n /** 首面板最小大小(百分比 0-100) */\r\n minSize?: number;\r\n /** 首面板最大大小(百分比 0-100) */\r\n maxSize?: number;\r\n /** 是否禁用拖拽调整 */\r\n disabled?: boolean;\r\n /** 是否可折叠 */\r\n collapsible?: boolean;\r\n /** 是否显示折叠按钮 */\r\n showCollapseButton?: boolean;\r\n /** 分割线宽度(px) */\r\n gutterSize?: number;\r\n /** 键盘微调步长(百分比) */\r\n step?: number;\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n direction: \"horizontal\",\r\n defaultSize: 50,\r\n minSize: 0,\r\n maxSize: 100,\r\n disabled: false,\r\n collapsible: true,\r\n showCollapseButton: true,\r\n gutterSize: 6,\r\n step: 2,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n resize: [firstSize: number, secondSize: number];\r\n collapse: [target: CollapseTarget];\r\n expand: [target: CollapseTarget];\r\n \"drag-start\": [size: number];\r\n \"drag-end\": [size: number];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement | null>(null);\r\n\r\nconst {\r\n panelSize,\r\n isDragging,\r\n isFirstCollapsed,\r\n isSecondCollapsed,\r\n handleMouseDown,\r\n handleDoubleClick,\r\n handleKeyDown,\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n setSize,\r\n getPanelInfo,\r\n} = useSplitResize({\r\n containerRef,\r\n direction: toRef(props, \"direction\"),\r\n defaultSize: props.defaultSize,\r\n minSize: props.minSize,\r\n maxSize: props.maxSize,\r\n disabled: toRef(props, \"disabled\"),\r\n collapsible: toRef(props, \"collapsible\"),\r\n step: props.step,\r\n onResize: (first, second) => emit(\"resize\", first, second),\r\n onCollapse: (target) => emit(\"collapse\", target),\r\n onExpand: (target) => emit(\"expand\", target),\r\n onDragStart: (size) => emit(\"drag-start\", size),\r\n onDragEnd: (size) => emit(\"drag-end\", size),\r\n});\r\n\r\n/* ─── 计算样式 ──────────────────────────────── */\r\nconst firstPanelStyle = computed(() => {\r\n const sizeValue = `calc(${panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst secondPanelStyle = computed(() => {\r\n const sizeValue = `calc(${100 - panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst gutterStyle = computed(() => {\r\n return props.direction === \"horizontal\"\r\n ? { width: `${props.gutterSize}px` }\r\n : { height: `${props.gutterSize}px` };\r\n});\r\n\r\n/* ─── 折叠按钮图标 ──────────────────────────── */\r\nconst collapseFirstIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isFirstCollapsed.value ? \"mdi:chevron-right\" : \"mdi:chevron-left\";\r\n }\r\n return isFirstCollapsed.value ? \"mdi:chevron-down\" : \"mdi:chevron-up\";\r\n});\r\n\r\nconst collapseSecondIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isSecondCollapsed.value ? \"mdi:chevron-left\" : \"mdi:chevron-right\";\r\n }\r\n return isSecondCollapsed.value ? \"mdi:chevron-up\" : \"mdi:chevron-down\";\r\n});\r\n\r\n/* ─── 暴露方法 ──────────────────────────────── */\r\ndefineExpose<SplitPaneExpose>({\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n getPanelInfo,\r\n setSize,\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-06-16\r\n * @Description: 分割面板组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"containerRef\"\r\n class=\"c-split-pane\"\r\n :class=\"[\r\n `c-split-pane--${direction}`,\r\n {\r\n 'c-split-pane--dragging': isDragging,\r\n 'c-split-pane--disabled': disabled,\r\n 'c-split-pane--collapsed-first': isFirstCollapsed,\r\n 'c-split-pane--collapsed-second': isSecondCollapsed,\r\n },\r\n ]\"\r\n >\r\n <!-- 首面板 -->\r\n <div class=\"split-panel split-panel--first\" :style=\"firstPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"first\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 分割线手柄 -->\r\n <div\r\n class=\"split-gutter\"\r\n :style=\"gutterStyle\"\r\n role=\"separator\"\r\n :aria-orientation=\"direction === 'horizontal' ? 'vertical' : 'horizontal'\"\r\n :aria-valuenow=\"Math.round(panelSize)\"\r\n :aria-valuemin=\"0\"\r\n :aria-valuemax=\"100\"\r\n tabindex=\"0\"\r\n @mousedown=\"handleMouseDown\"\r\n @touchstart.prevent=\"handleMouseDown\"\r\n @dblclick=\"handleDoubleClick\"\r\n @keydown=\"handleKeyDown\"\r\n >\r\n <div class=\"split-gutter__visual\">\r\n <div class=\"split-gutter__dots\">\r\n <span v-for=\"i in 3\" :key=\"i\" class=\"split-gutter__dot\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 折叠按钮 -->\r\n <template v-if=\"showCollapseButton && collapsible\">\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--first\"\r\n :title=\"isFirstCollapsed ? '展开首面板' : '折叠首面板'\"\r\n @click.stop=\"toggle('first')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseFirstIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--second\"\r\n :title=\"isSecondCollapsed ? '展开第二面板' : '折叠第二面板'\"\r\n @click.stop=\"toggle('second')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseSecondIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n </template>\r\n </div>\r\n\r\n <!-- 第二面板 -->\r\n <div class=\"split-panel split-panel--second\" :style=\"secondPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"second\" />\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, toRef } from \"vue\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSplitResize } from \"./composables/useSplitResize\";\r\nimport type { CollapseTarget, SplitDirection, SplitPaneExpose } from \"./types\";\r\n\r\ndefineOptions({ name: \"C_SplitPane\" });\r\n\r\ninterface Props {\r\n /** 分割方向 */\r\n direction?: SplitDirection;\r\n /** 首面板默认大小(百分比 0-100) */\r\n defaultSize?: number;\r\n /** 首面板最小大小(百分比 0-100) */\r\n minSize?: number;\r\n /** 首面板最大大小(百分比 0-100) */\r\n maxSize?: number;\r\n /** 是否禁用拖拽调整 */\r\n disabled?: boolean;\r\n /** 是否可折叠 */\r\n collapsible?: boolean;\r\n /** 是否显示折叠按钮 */\r\n showCollapseButton?: boolean;\r\n /** 分割线宽度(px) */\r\n gutterSize?: number;\r\n /** 键盘微调步长(百分比) */\r\n step?: number;\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n direction: \"horizontal\",\r\n defaultSize: 50,\r\n minSize: 0,\r\n maxSize: 100,\r\n disabled: false,\r\n collapsible: true,\r\n showCollapseButton: true,\r\n gutterSize: 6,\r\n step: 2,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n resize: [firstSize: number, secondSize: number];\r\n collapse: [target: CollapseTarget];\r\n expand: [target: CollapseTarget];\r\n \"drag-start\": [size: number];\r\n \"drag-end\": [size: number];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement | null>(null);\r\n\r\nconst {\r\n panelSize,\r\n isDragging,\r\n isFirstCollapsed,\r\n isSecondCollapsed,\r\n handleMouseDown,\r\n handleDoubleClick,\r\n handleKeyDown,\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n setSize,\r\n getPanelInfo,\r\n} = useSplitResize({\r\n containerRef,\r\n direction: toRef(props, \"direction\"),\r\n defaultSize: props.defaultSize,\r\n minSize: props.minSize,\r\n maxSize: props.maxSize,\r\n disabled: toRef(props, \"disabled\"),\r\n collapsible: toRef(props, \"collapsible\"),\r\n step: props.step,\r\n onResize: (first, second) => emit(\"resize\", first, second),\r\n onCollapse: (target) => emit(\"collapse\", target),\r\n onExpand: (target) => emit(\"expand\", target),\r\n onDragStart: (size) => emit(\"drag-start\", size),\r\n onDragEnd: (size) => emit(\"drag-end\", size),\r\n});\r\n\r\n/* ─── 计算样式 ──────────────────────────────── */\r\nconst firstPanelStyle = computed(() => {\r\n const sizeValue = `calc(${panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst secondPanelStyle = computed(() => {\r\n const sizeValue = `calc(${100 - panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst gutterStyle = computed(() => {\r\n return props.direction === \"horizontal\"\r\n ? { width: `${props.gutterSize}px` }\r\n : { height: `${props.gutterSize}px` };\r\n});\r\n\r\n/* ─── 折叠按钮图标 ──────────────────────────── */\r\nconst collapseFirstIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isFirstCollapsed.value ? \"mdi:chevron-right\" : \"mdi:chevron-left\";\r\n }\r\n return isFirstCollapsed.value ? \"mdi:chevron-down\" : \"mdi:chevron-up\";\r\n});\r\n\r\nconst collapseSecondIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isSecondCollapsed.value ? \"mdi:chevron-left\" : \"mdi:chevron-right\";\r\n }\r\n return isSecondCollapsed.value ? \"mdi:chevron-up\" : \"mdi:chevron-down\";\r\n});\r\n\r\n/* ─── 暴露方法 ──────────────────────────────── */\r\ndefineExpose<SplitPaneExpose>({\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n getPanelInfo,\r\n setSize,\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-06-16\r\n * @Description: 分割面板组件\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"containerRef\"\r\n class=\"c-split-pane\"\r\n :class=\"[\r\n `c-split-pane--${direction}`,\r\n {\r\n 'c-split-pane--dragging': isDragging,\r\n 'c-split-pane--disabled': disabled,\r\n 'c-split-pane--collapsed-first': isFirstCollapsed,\r\n 'c-split-pane--collapsed-second': isSecondCollapsed,\r\n },\r\n ]\"\r\n >\r\n <!-- 首面板 -->\r\n <div class=\"split-panel split-panel--first\" :style=\"firstPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"first\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 分割线手柄 -->\r\n <div\r\n class=\"split-gutter\"\r\n :style=\"gutterStyle\"\r\n role=\"separator\"\r\n :aria-orientation=\"direction === 'horizontal' ? 'vertical' : 'horizontal'\"\r\n :aria-valuenow=\"Math.round(panelSize)\"\r\n :aria-valuemin=\"0\"\r\n :aria-valuemax=\"100\"\r\n tabindex=\"0\"\r\n @mousedown=\"handleMouseDown\"\r\n @touchstart.prevent=\"handleMouseDown\"\r\n @dblclick=\"handleDoubleClick\"\r\n @keydown=\"handleKeyDown\"\r\n >\r\n <div class=\"split-gutter__visual\">\r\n <div class=\"split-gutter__dots\">\r\n <span v-for=\"i in 3\" :key=\"i\" class=\"split-gutter__dot\" />\r\n </div>\r\n </div>\r\n\r\n <!-- 折叠按钮 -->\r\n <template v-if=\"showCollapseButton && collapsible\">\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--first\"\r\n :title=\"isFirstCollapsed ? '展开首面板' : '折叠首面板'\"\r\n @click.stop=\"toggle('first')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseFirstIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n <button\r\n class=\"split-gutter__collapse-btn split-gutter__collapse-btn--second\"\r\n :title=\"isSecondCollapsed ? '展开第二面板' : '折叠第二面板'\"\r\n @click.stop=\"toggle('second')\"\r\n @mousedown.stop\r\n >\r\n <C_Icon :name=\"collapseSecondIcon\" :size=\"16\" color=\"currentColor\" />\r\n </button>\r\n </template>\r\n </div>\r\n\r\n <!-- 第二面板 -->\r\n <div class=\"split-panel split-panel--second\" :style=\"secondPanelStyle\">\r\n <div class=\"split-panel__content\">\r\n <slot name=\"second\" />\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, toRef } from \"vue\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport { useSplitResize } from \"./composables/useSplitResize\";\r\nimport type { CollapseTarget, SplitDirection, SplitPaneExpose } from \"./types\";\r\n\r\ndefineOptions({ name: \"C_SplitPane\" });\r\n\r\ninterface Props {\r\n /** 分割方向 */\r\n direction?: SplitDirection;\r\n /** 首面板默认大小(百分比 0-100) */\r\n defaultSize?: number;\r\n /** 首面板最小大小(百分比 0-100) */\r\n minSize?: number;\r\n /** 首面板最大大小(百分比 0-100) */\r\n maxSize?: number;\r\n /** 是否禁用拖拽调整 */\r\n disabled?: boolean;\r\n /** 是否可折叠 */\r\n collapsible?: boolean;\r\n /** 是否显示折叠按钮 */\r\n showCollapseButton?: boolean;\r\n /** 分割线宽度(px) */\r\n gutterSize?: number;\r\n /** 键盘微调步长(百分比) */\r\n step?: number;\r\n}\r\n\r\nconst props = withDefaults(defineProps<Props>(), {\r\n direction: \"horizontal\",\r\n defaultSize: 50,\r\n minSize: 0,\r\n maxSize: 100,\r\n disabled: false,\r\n collapsible: true,\r\n showCollapseButton: true,\r\n gutterSize: 6,\r\n step: 2,\r\n});\r\n\r\nconst emit = defineEmits<{\r\n resize: [firstSize: number, secondSize: number];\r\n collapse: [target: CollapseTarget];\r\n expand: [target: CollapseTarget];\r\n \"drag-start\": [size: number];\r\n \"drag-end\": [size: number];\r\n}>();\r\n\r\nconst containerRef = ref<HTMLElement | null>(null);\r\n\r\nconst {\r\n panelSize,\r\n isDragging,\r\n isFirstCollapsed,\r\n isSecondCollapsed,\r\n handleMouseDown,\r\n handleDoubleClick,\r\n handleKeyDown,\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n setSize,\r\n getPanelInfo,\r\n} = useSplitResize({\r\n containerRef,\r\n direction: toRef(props, \"direction\"),\r\n defaultSize: props.defaultSize,\r\n minSize: props.minSize,\r\n maxSize: props.maxSize,\r\n disabled: toRef(props, \"disabled\"),\r\n collapsible: toRef(props, \"collapsible\"),\r\n step: props.step,\r\n onResize: (first, second) => emit(\"resize\", first, second),\r\n onCollapse: (target) => emit(\"collapse\", target),\r\n onExpand: (target) => emit(\"expand\", target),\r\n onDragStart: (size) => emit(\"drag-start\", size),\r\n onDragEnd: (size) => emit(\"drag-end\", size),\r\n});\r\n\r\n/* ─── 计算样式 ──────────────────────────────── */\r\nconst firstPanelStyle = computed(() => {\r\n const sizeValue = `calc(${panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst secondPanelStyle = computed(() => {\r\n const sizeValue = `calc(${100 - panelSize.value}% - ${props.gutterSize / 2}px)`;\r\n return props.direction === \"horizontal\"\r\n ? { width: sizeValue }\r\n : { height: sizeValue };\r\n});\r\n\r\nconst gutterStyle = computed(() => {\r\n return props.direction === \"horizontal\"\r\n ? { width: `${props.gutterSize}px` }\r\n : { height: `${props.gutterSize}px` };\r\n});\r\n\r\n/* ─── 折叠按钮图标 ──────────────────────────── */\r\nconst collapseFirstIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isFirstCollapsed.value ? \"mdi:chevron-right\" : \"mdi:chevron-left\";\r\n }\r\n return isFirstCollapsed.value ? \"mdi:chevron-down\" : \"mdi:chevron-up\";\r\n});\r\n\r\nconst collapseSecondIcon = computed(() => {\r\n if (props.direction === \"horizontal\") {\r\n return isSecondCollapsed.value ? \"mdi:chevron-left\" : \"mdi:chevron-right\";\r\n }\r\n return isSecondCollapsed.value ? \"mdi:chevron-up\" : \"mdi:chevron-down\";\r\n});\r\n\r\n/* ─── 暴露方法 ──────────────────────────────── */\r\ndefineExpose<SplitPaneExpose>({\r\n collapse,\r\n expand,\r\n toggle,\r\n resetSize,\r\n getPanelInfo,\r\n setSize,\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_Steps-CODHN5Hs.css","names":[],"sources":["../src/components/C_Steps/index.vue?vue&type=style&index=0&scoped=0ccfa846&lang.scss"],"sourcesContent":[".c-steps[data-v-0ccfa846] {\n --step-gap: 24px;\n --icon-size: 32px;\n --line-width: 2px;\n --primary: var(--c-primary, #18a058);\n --success: var(--c-success, #18a058);\n --warning: var(--c-warning, #f0a020);\n --error: var(--c-error, #d03050);\n --border: var(--c-border, rgba(0, 0, 0, 0.09));\n --bg-hover: rgba(0, 0, 0, 0.04);\n display: flex;\n}\n.steps-horizontal[data-v-0ccfa846] {\n flex-direction: row;\n align-items: flex-start;\n}\n.steps-vertical[data-v-0ccfa846] {\n flex-direction: column;\n}\n.step-item[data-v-0ccfa846] {\n display: flex;\n position: relative;\n flex: 1;\n gap: 12px;\n}\n.steps-horizontal .step-item[data-v-0ccfa846] {\n flex-direction: column;\n align-items: center;\n text-align: center;\n min-width: 0;\n}\n.steps-vertical .step-item[data-v-0ccfa846] {\n flex-direction: row;\n min-height: 72px;\n flex: none;\n width: 100%;\n}\n.step-indicator[data-v-0ccfa846] {\n position: relative;\n display: flex;\n align-items: center;\n flex-shrink: 0;\n}\n.steps-horizontal .step-indicator[data-v-0ccfa846] {\n width: 100%;\n justify-content: center;\n}\n.steps-vertical .step-indicator[data-v-0ccfa846] {\n flex-direction: column;\n}\n.step-icon[data-v-0ccfa846] {\n width: var(--icon-size);\n height: var(--icon-size);\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n border: var(--line-width) solid var(--border);\n background: var(--c-bg-card, #fff);\n position: relative;\n z-index: 1;\n transition: all 0.2s ease;\n font-size: 16px;\n}\n.step-finish .step-icon[data-v-0ccfa846] {\n border-color: var(--success);\n color: var(--success);\n}\n.step-process .step-icon[data-v-0ccfa846] {\n border-color: var(--primary);\n background: var(--primary);\n color: white;\n}\n.step-error .step-icon[data-v-0ccfa846] {\n border-color: var(--error);\n color: var(--error);\n}\n.step-index[data-v-0ccfa846] {\n font-size: 14px;\n font-weight: 500;\n}\n.step-line[data-v-0ccfa846] {\n background: var(--border);\n position: absolute;\n transition: background 0.2s ease;\n}\n.steps-horizontal .step-line[data-v-0ccfa846] {\n height: var(--line-width);\n left: calc(50% + var(--icon-size) / 2);\n right: calc(-50% + var(--icon-size) / 2 + var(--step-gap));\n top: calc(var(--icon-size) / 2);\n}\n.steps-vertical .step-line[data-v-0ccfa846] {\n width: var(--line-width);\n top: calc(var(--icon-size) + 4px);\n bottom: calc(-72px + var(--icon-size) + 4px);\n left: calc(var(--icon-size) / 2 - 1px);\n}\n.step-finish .step-line[data-v-0ccfa846] {\n background: var(--success);\n}\n.step-content[data-v-0ccfa846] {\n flex: 1;\n min-width: 0;\n}\n.steps-vertical .step-content[data-v-0ccfa846] {\n padding-top: 4px;\n}\n.step-title[data-v-0ccfa846] {\n font-size: 14px;\n font-weight: 500;\n line-height: 22px;\n margin-bottom: 2px;\n word-break: break-word;\n}\n.step-time[data-v-0ccfa846] {\n font-size: 12px;\n line-height: 20px;\n margin-bottom: 2px;\n opacity: 0.65;\n}\n.step-finish .step-time[data-v-0ccfa846] {\n color: var(--success);\n}\n.step-description[data-v-0ccfa846] {\n font-size: 12px;\n line-height: 20px;\n opacity: 0.65;\n}\n.step-wait .step-title[data-v-0ccfa846] {\n opacity: 0.65;\n}\n.step-process .step-title[data-v-0ccfa846] {\n color: var(--primary);\n font-weight: 600;\n}\n.step-error .step-title[data-v-0ccfa846] {\n color: var(--error);\n}\n.step-clickable[data-v-0ccfa846] {\n cursor: pointer;\n}\n.step-clickable:hover .step-icon[data-v-0ccfa846] {\n transform: scale(1.05);\n}\n.step-clickable:not(.step-process):hover .step-icon[data-v-0ccfa846] {\n background: var(--bg-hover);\n}\n.step-clickable:hover .step-title[data-v-0ccfa846] {\n color: var(--primary);\n}\n@media (max-width: 768px) {\n.steps-horizontal[data-v-0ccfa846] {\n flex-wrap: wrap;\n}\n.steps-horizontal .step-item[data-v-0ccfa846] {\n flex: 0 0 50%;\n margin-bottom: 24px;\n}\n.steps-horizontal .step-line[data-v-0ccfa846] {\n display: none;\n}\n}\n@media (prefers-reduced-motion: reduce) {\n.step-icon[data-v-0ccfa846],\n .step-line[data-v-0ccfa846] {\n transition: none;\n}\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;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;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"}
|