@robot-admin/naive-ui-components 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (352) hide show
  1. package/README.md +257 -0
  2. package/dist/C_ActionBar-DWN-woTc.css.map +1 -0
  3. package/dist/C_ActionBar.cjs +5 -0
  4. package/dist/C_ActionBar.d.cts +2 -0
  5. package/dist/C_ActionBar.d.ts +2 -0
  6. package/dist/C_ActionBar.js +4 -0
  7. package/dist/C_ActionBar2.js +196 -0
  8. package/dist/C_ActionBar2.js.map +1 -0
  9. package/dist/C_AntV-AFKyK6hH.css.map +1 -0
  10. package/dist/C_AntV.cjs +8 -0
  11. package/dist/C_AntV.d.cts +2 -0
  12. package/dist/C_AntV.d.ts +2 -0
  13. package/dist/C_AntV.js +4 -0
  14. package/dist/C_AntV2.js +3150 -0
  15. package/dist/C_AntV2.js.map +1 -0
  16. package/dist/C_Barcode-P_EFj8dC.css.map +1 -0
  17. package/dist/C_Barcode.cjs +4 -0
  18. package/dist/C_Barcode.d.cts +2 -0
  19. package/dist/C_Barcode.d.ts +2 -0
  20. package/dist/C_Barcode.js +3 -0
  21. package/dist/C_Barcode2.js +68 -0
  22. package/dist/C_Barcode2.js.map +1 -0
  23. package/dist/C_Captcha-C-ef41xw.css.map +1 -0
  24. package/dist/C_Captcha.cjs +4 -0
  25. package/dist/C_Captcha.d.cts +2 -0
  26. package/dist/C_Captcha.d.ts +2 -0
  27. package/dist/C_Captcha.js +3 -0
  28. package/dist/C_Captcha2.js +155 -0
  29. package/dist/C_Captcha2.js.map +1 -0
  30. package/dist/C_Cascade-D9kNsjsV.css.map +1 -0
  31. package/dist/C_Cascade.cjs +4 -0
  32. package/dist/C_Cascade.d.cts +2 -0
  33. package/dist/C_Cascade.d.ts +2 -0
  34. package/dist/C_Cascade.js +3 -0
  35. package/dist/C_Cascade2.js +103 -0
  36. package/dist/C_Cascade2.js.map +1 -0
  37. package/dist/C_City-BCQ4ipiK.css.map +1 -0
  38. package/dist/C_City.cjs +4 -0
  39. package/dist/C_City.d.cts +2 -0
  40. package/dist/C_City.d.ts +2 -0
  41. package/dist/C_City.js +3 -0
  42. package/dist/C_City2.js +841 -0
  43. package/dist/C_City2.js.map +1 -0
  44. package/dist/C_Code-C9kvvEmO.css.map +1 -0
  45. package/dist/C_Code.cjs +5 -0
  46. package/dist/C_Code.d.cts +2 -0
  47. package/dist/C_Code.d.ts +2 -0
  48. package/dist/C_Code.js +4 -0
  49. package/dist/C_Code2.js +346 -0
  50. package/dist/C_Code2.js.map +1 -0
  51. package/dist/C_CollapsePanel-BUJHuYcU.css.map +1 -0
  52. package/dist/C_CollapsePanel.cjs +6 -0
  53. package/dist/C_CollapsePanel.d.cts +2 -0
  54. package/dist/C_CollapsePanel.d.ts +2 -0
  55. package/dist/C_CollapsePanel.js +4 -0
  56. package/dist/C_CollapsePanel2.js +319 -0
  57. package/dist/C_CollapsePanel2.js.map +1 -0
  58. package/dist/C_Cron-yx2Ob4Jl.css.map +1 -0
  59. package/dist/C_Cron.cjs +15 -0
  60. package/dist/C_Cron.d.cts +2 -0
  61. package/dist/C_Cron.d.ts +2 -0
  62. package/dist/C_Cron.js +4 -0
  63. package/dist/C_Cron2.js +1209 -0
  64. package/dist/C_Cron2.js.map +1 -0
  65. package/dist/C_Date.cjs +4 -0
  66. package/dist/C_Date.d.cts +2 -0
  67. package/dist/C_Date.d.ts +2 -0
  68. package/dist/C_Date.js +3 -0
  69. package/dist/C_Date2.js +219 -0
  70. package/dist/C_Date2.js.map +1 -0
  71. package/dist/C_Draggable-C483syRC.css.map +1 -0
  72. package/dist/C_Draggable.cjs +5 -0
  73. package/dist/C_Draggable.d.cts +2 -0
  74. package/dist/C_Draggable.d.ts +2 -0
  75. package/dist/C_Draggable.js +3 -0
  76. package/dist/C_Draggable2.js +295 -0
  77. package/dist/C_Draggable2.js.map +1 -0
  78. package/dist/C_Editor-Bp0SyIEw.css.map +1 -0
  79. package/dist/C_Editor.cjs +4 -0
  80. package/dist/C_Editor.d.cts +2 -0
  81. package/dist/C_Editor.d.ts +2 -0
  82. package/dist/C_Editor.js +3 -0
  83. package/dist/C_Editor2.js +160 -0
  84. package/dist/C_Editor2.js.map +1 -0
  85. package/dist/C_FilePreview-CPqvhoCy.css.map +1 -0
  86. package/dist/C_FilePreview.cjs +6 -0
  87. package/dist/C_FilePreview.d.cts +2 -0
  88. package/dist/C_FilePreview.d.ts +2 -0
  89. package/dist/C_FilePreview.js +3 -0
  90. package/dist/C_FilePreview2.js +1031 -0
  91. package/dist/C_FilePreview2.js.map +1 -0
  92. package/dist/C_Form-Jx7PY3sT.css.map +1 -0
  93. package/dist/C_Form.cjs +15 -0
  94. package/dist/C_Form.d.cts +2 -0
  95. package/dist/C_Form.d.ts +2 -0
  96. package/dist/C_Form.js +4 -0
  97. package/dist/C_Form2.js +2510 -0
  98. package/dist/C_Form2.js.map +1 -0
  99. package/dist/C_FormSearch-DvRgxlRn.css.map +1 -0
  100. package/dist/C_FormSearch.cjs +6 -0
  101. package/dist/C_FormSearch.d.cts +2 -0
  102. package/dist/C_FormSearch.d.ts +2 -0
  103. package/dist/C_FormSearch.js +3 -0
  104. package/dist/C_FormSearch2.js +356 -0
  105. package/dist/C_FormSearch2.js.map +1 -0
  106. package/dist/C_FormulaEditor-DtGkt4T_.css.map +1 -0
  107. package/dist/C_FormulaEditor.cjs +13 -0
  108. package/dist/C_FormulaEditor.d.cts +2 -0
  109. package/dist/C_FormulaEditor.d.ts +2 -0
  110. package/dist/C_FormulaEditor.js +4 -0
  111. package/dist/C_FormulaEditor2.js +1433 -0
  112. package/dist/C_FormulaEditor2.js.map +1 -0
  113. package/dist/C_FullCalendar-BF7H0YIx.css.map +1 -0
  114. package/dist/C_FullCalendar.cjs +9 -0
  115. package/dist/C_FullCalendar.d.cts +2 -0
  116. package/dist/C_FullCalendar.d.ts +2 -0
  117. package/dist/C_FullCalendar.js +3 -0
  118. package/dist/C_FullCalendar2.js +377 -0
  119. package/dist/C_FullCalendar2.js.map +1 -0
  120. package/dist/C_Guide.cjs +4 -0
  121. package/dist/C_Guide.d.cts +2 -0
  122. package/dist/C_Guide.d.ts +2 -0
  123. package/dist/C_Guide.js +3 -0
  124. package/dist/C_Guide2.js +58 -0
  125. package/dist/C_Guide2.js.map +1 -0
  126. package/dist/C_Icon.cjs +4 -0
  127. package/dist/C_Icon.d.cts +2 -0
  128. package/dist/C_Icon.d.ts +2 -0
  129. package/dist/C_Icon.js +3 -0
  130. package/dist/C_Icon2.js +286 -0
  131. package/dist/C_Icon2.js.map +1 -0
  132. package/dist/C_ImageCropper-BVJfUufl.css.map +1 -0
  133. package/dist/C_ImageCropper.cjs +6 -0
  134. package/dist/C_ImageCropper.d.cts +2 -0
  135. package/dist/C_ImageCropper.d.ts +2 -0
  136. package/dist/C_ImageCropper.js +4 -0
  137. package/dist/C_ImageCropper2.js +723 -0
  138. package/dist/C_ImageCropper2.js.map +1 -0
  139. package/dist/C_Language.cjs +4 -0
  140. package/dist/C_Language.d.cts +2 -0
  141. package/dist/C_Language.d.ts +2 -0
  142. package/dist/C_Language.js +3 -0
  143. package/dist/C_Language2.js +72 -0
  144. package/dist/C_Language2.js.map +1 -0
  145. package/dist/C_Map-DpzeuWdX.css.map +1 -0
  146. package/dist/C_Map.cjs +7 -0
  147. package/dist/C_Map.d.cts +2 -0
  148. package/dist/C_Map.d.ts +2 -0
  149. package/dist/C_Map.js +3 -0
  150. package/dist/C_Map2.js +199 -0
  151. package/dist/C_Map2.js.map +1 -0
  152. package/dist/C_Markdown-BEjxknqd.css.map +1 -0
  153. package/dist/C_Markdown.cjs +4 -0
  154. package/dist/C_Markdown.d.cts +2 -0
  155. package/dist/C_Markdown.d.ts +2 -0
  156. package/dist/C_Markdown.js +3 -0
  157. package/dist/C_Markdown2.js +186 -0
  158. package/dist/C_Markdown2.js.map +1 -0
  159. package/dist/C_NotificationCenter-0l3TY2Gn.css.map +1 -0
  160. package/dist/C_NotificationCenter.cjs +20 -0
  161. package/dist/C_NotificationCenter.d.cts +2 -0
  162. package/dist/C_NotificationCenter.d.ts +2 -0
  163. package/dist/C_NotificationCenter.js +4 -0
  164. package/dist/C_NotificationCenter2.js +1383 -0
  165. package/dist/C_NotificationCenter2.js.map +1 -0
  166. package/dist/C_Progress.cjs +4 -0
  167. package/dist/C_Progress.d.cts +2 -0
  168. package/dist/C_Progress.d.ts +2 -0
  169. package/dist/C_Progress.js +3 -0
  170. package/dist/C_Progress2.js +103 -0
  171. package/dist/C_Progress2.js.map +1 -0
  172. package/dist/C_QRCode-DbdiAIPg.css.map +1 -0
  173. package/dist/C_QRCode.cjs +5 -0
  174. package/dist/C_QRCode.d.cts +2 -0
  175. package/dist/C_QRCode.d.ts +2 -0
  176. package/dist/C_QRCode.js +3 -0
  177. package/dist/C_QRCode2.js +218 -0
  178. package/dist/C_QRCode2.js.map +1 -0
  179. package/dist/C_Signature-zhHCbra9.css.map +1 -0
  180. package/dist/C_Signature.cjs +8 -0
  181. package/dist/C_Signature.d.cts +2 -0
  182. package/dist/C_Signature.d.ts +2 -0
  183. package/dist/C_Signature.js +4 -0
  184. package/dist/C_Signature2.js +618 -0
  185. package/dist/C_Signature2.js.map +1 -0
  186. package/dist/C_SplitPane-C6sBsfKY.css.map +1 -0
  187. package/dist/C_SplitPane.cjs +6 -0
  188. package/dist/C_SplitPane.d.cts +2 -0
  189. package/dist/C_SplitPane.d.ts +2 -0
  190. package/dist/C_SplitPane.js +4 -0
  191. package/dist/C_SplitPane2.js +356 -0
  192. package/dist/C_SplitPane2.js.map +1 -0
  193. package/dist/C_Steps-CODHN5Hs.css.map +1 -0
  194. package/dist/C_Steps.cjs +4 -0
  195. package/dist/C_Steps.d.cts +2 -0
  196. package/dist/C_Steps.d.ts +2 -0
  197. package/dist/C_Steps.js +3 -0
  198. package/dist/C_Steps2.js +82 -0
  199. package/dist/C_Steps2.js.map +1 -0
  200. package/dist/C_Table-DSNsntmT.css.map +1 -0
  201. package/dist/C_Table.cjs +19 -0
  202. package/dist/C_Table.d.cts +2 -0
  203. package/dist/C_Table.d.ts +2 -0
  204. package/dist/C_Table.js +5 -0
  205. package/dist/C_Table2.js +3009 -0
  206. package/dist/C_Table2.js.map +1 -0
  207. package/dist/C_Theme.cjs +4 -0
  208. package/dist/C_Theme.d.cts +2 -0
  209. package/dist/C_Theme.d.ts +2 -0
  210. package/dist/C_Theme.js +3 -0
  211. package/dist/C_Theme2.js +60 -0
  212. package/dist/C_Theme2.js.map +1 -0
  213. package/dist/C_Time-BvZLYraL.css.map +1 -0
  214. package/dist/C_Time.cjs +5 -0
  215. package/dist/C_Time.d.cts +2 -0
  216. package/dist/C_Time.d.ts +2 -0
  217. package/dist/C_Time.js +3 -0
  218. package/dist/C_Time2.js +199 -0
  219. package/dist/C_Time2.js.map +1 -0
  220. package/dist/C_Tree-0GDv--jX.css.map +1 -0
  221. package/dist/C_Tree.cjs +7 -0
  222. package/dist/C_Tree.d.cts +2 -0
  223. package/dist/C_Tree.d.ts +2 -0
  224. package/dist/C_Tree.js +4 -0
  225. package/dist/C_Tree2.js +441 -0
  226. package/dist/C_Tree2.js.map +1 -0
  227. package/dist/C_Upload-BXd3YYLx.css.map +1 -0
  228. package/dist/C_Upload.cjs +12 -0
  229. package/dist/C_Upload.d.cts +2 -0
  230. package/dist/C_Upload.d.ts +2 -0
  231. package/dist/C_Upload.js +4 -0
  232. package/dist/C_Upload2.js +1388 -0
  233. package/dist/C_Upload2.js.map +1 -0
  234. package/dist/C_VideoPlayer-DYG3RL0Q.css.map +1 -0
  235. package/dist/C_VideoPlayer.cjs +23 -0
  236. package/dist/C_VideoPlayer.d.cts +2 -0
  237. package/dist/C_VideoPlayer.d.ts +2 -0
  238. package/dist/C_VideoPlayer.js +3 -0
  239. package/dist/C_VideoPlayer2.js +1932 -0
  240. package/dist/C_VideoPlayer2.js.map +1 -0
  241. package/dist/C_VtableGantt-fhItIiHE.css.map +1 -0
  242. package/dist/C_VtableGantt.cjs +6 -0
  243. package/dist/C_VtableGantt.d.cts +2 -0
  244. package/dist/C_VtableGantt.d.ts +2 -0
  245. package/dist/C_VtableGantt.js +4 -0
  246. package/dist/C_VtableGantt2.js +873 -0
  247. package/dist/C_VtableGantt2.js.map +1 -0
  248. package/dist/C_WaterFall-8sQDFXKg.css.map +1 -0
  249. package/dist/C_WaterFall.cjs +13 -0
  250. package/dist/C_WaterFall.d.cts +2 -0
  251. package/dist/C_WaterFall.d.ts +2 -0
  252. package/dist/C_WaterFall.js +3 -0
  253. package/dist/C_WaterFall2.js +365 -0
  254. package/dist/C_WaterFall2.js.map +1 -0
  255. package/dist/C_WorkFlow-J-dyIuh9.css.map +1 -0
  256. package/dist/C_WorkFlow.cjs +8 -0
  257. package/dist/C_WorkFlow.d.cts +2 -0
  258. package/dist/C_WorkFlow.d.ts +2 -0
  259. package/dist/C_WorkFlow.js +4 -0
  260. package/dist/C_WorkFlow2.js +1984 -0
  261. package/dist/C_WorkFlow2.js.map +1 -0
  262. package/dist/chunk.js +22 -0
  263. package/dist/city.js +4817 -0
  264. package/dist/city.js.map +1 -0
  265. package/dist/constants.d.ts +273 -0
  266. package/dist/constants.d.ts.map +1 -0
  267. package/dist/constants2.d.ts +178 -0
  268. package/dist/constants2.d.ts.map +1 -0
  269. package/dist/constants3.d.ts +475 -0
  270. package/dist/constants3.d.ts.map +1 -0
  271. package/dist/constants4.d.ts +430 -0
  272. package/dist/constants4.d.ts.map +1 -0
  273. package/dist/constants5.d.ts +4283 -0
  274. package/dist/constants5.d.ts.map +1 -0
  275. package/dist/data.d.ts +67 -0
  276. package/dist/data.d.ts.map +1 -0
  277. package/dist/export-helper.js +9 -0
  278. package/dist/index.cjs +409 -0
  279. package/dist/index.d.cts +96 -0
  280. package/dist/index.d.cts.map +1 -0
  281. package/dist/index.d.ts +103 -0
  282. package/dist/index.d.ts.map +1 -0
  283. package/dist/index.js +230 -0
  284. package/dist/index.js.map +1 -0
  285. package/dist/index.vue.d.ts +80 -0
  286. package/dist/index.vue.d.ts.map +1 -0
  287. package/dist/index10.vue.d.ts +72 -0
  288. package/dist/index10.vue.d.ts.map +1 -0
  289. package/dist/index11.vue.d.ts +26 -0
  290. package/dist/index11.vue.d.ts.map +1 -0
  291. package/dist/index12.vue.d.ts +81 -0
  292. package/dist/index12.vue.d.ts.map +1 -0
  293. package/dist/index13.vue.d.ts +55 -0
  294. package/dist/index13.vue.d.ts.map +1 -0
  295. package/dist/index14.vue.d.ts +33 -0
  296. package/dist/index14.vue.d.ts.map +1 -0
  297. package/dist/index15.vue.d.ts +18 -0
  298. package/dist/index15.vue.d.ts.map +1 -0
  299. package/dist/index16.vue.d.ts +662 -0
  300. package/dist/index16.vue.d.ts.map +1 -0
  301. package/dist/index2.vue.d.ts +38 -0
  302. package/dist/index2.vue.d.ts.map +1 -0
  303. package/dist/index3.vue.d.ts +45 -0
  304. package/dist/index3.vue.d.ts.map +1 -0
  305. package/dist/index4.vue.d.ts +31 -0
  306. package/dist/index4.vue.d.ts.map +1 -0
  307. package/dist/index5.vue.d.ts +35 -0
  308. package/dist/index5.vue.d.ts.map +1 -0
  309. package/dist/index6.vue.d.ts +48 -0
  310. package/dist/index6.vue.d.ts.map +1 -0
  311. package/dist/index7.vue.d.ts +56 -0
  312. package/dist/index7.vue.d.ts.map +1 -0
  313. package/dist/index8.vue.d.ts +41 -0
  314. package/dist/index8.vue.d.ts.map +1 -0
  315. package/dist/index9.vue.d.ts +30 -0
  316. package/dist/index9.vue.d.ts.map +1 -0
  317. package/dist/storage.js +31 -0
  318. package/dist/storage.js.map +1 -0
  319. package/dist/style.css +7725 -0
  320. package/dist/useCalendarEvents.d.ts +148 -0
  321. package/dist/useCalendarEvents.d.ts.map +1 -0
  322. package/dist/useCollapsePanel.d.ts +132 -0
  323. package/dist/useCollapsePanel.d.ts.map +1 -0
  324. package/dist/useCropperCore.d.ts +102 -0
  325. package/dist/useCropperCore.d.ts.map +1 -0
  326. package/dist/useDraggableLayout.d.ts +194 -0
  327. package/dist/useDraggableLayout.d.ts.map +1 -0
  328. package/dist/useDynamicFormState.d.ts +4248 -0
  329. package/dist/useDynamicFormState.d.ts.map +1 -0
  330. package/dist/useEdgeInteraction.d.ts +7614 -0
  331. package/dist/useEdgeInteraction.d.ts.map +1 -0
  332. package/dist/useFullscreen.d.ts +166 -0
  333. package/dist/useFullscreen.d.ts.map +1 -0
  334. package/dist/useInfiniteScroll.d.ts +169 -0
  335. package/dist/useInfiniteScroll.d.ts.map +1 -0
  336. package/dist/useModalEdit.d.ts +960 -0
  337. package/dist/useModalEdit.d.ts.map +1 -0
  338. package/dist/useQRCode.d.ts +87 -0
  339. package/dist/useQRCode.d.ts.map +1 -0
  340. package/dist/useSearchState.d.ts +180 -0
  341. package/dist/useSearchState.d.ts.map +1 -0
  342. package/dist/useSignatureHistory.d.ts +189 -0
  343. package/dist/useSignatureHistory.d.ts.map +1 -0
  344. package/dist/useSplitResize.d.ts +158 -0
  345. package/dist/useSplitResize.d.ts.map +1 -0
  346. package/dist/useTimeSelection.d.ts +105 -0
  347. package/dist/useTimeSelection.d.ts.map +1 -0
  348. package/dist/useTreeOperations.d.ts +183 -0
  349. package/dist/useTreeOperations.d.ts.map +1 -0
  350. package/dist/useWorkflowValidation.d.ts +1052 -0
  351. package/dist/useWorkflowValidation.d.ts.map +1 -0
  352. package/package.json +342 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_ImageCropper2.js","names":["ratioPresets","currentRatio","$emit","circular","previewData"],"sources":["../src/components/C_ImageCropper/composables/useCropperCore.ts","../src/components/C_ImageCropper/components/CropperToolbar.vue","../src/components/C_ImageCropper/components/CropperToolbar.vue","../src/components/C_ImageCropper/components/CropperToolbar.vue","../src/components/C_ImageCropper/components/CropperPreview.vue","../src/components/C_ImageCropper/components/CropperPreview.vue","../src/components/C_ImageCropper/components/CropperPreview.vue","../src/components/C_ImageCropper/index.vue","../src/components/C_ImageCropper/index.vue","../src/components/C_ImageCropper/index.vue"],"sourcesContent":["import { ref } from \"vue\";\r\nimport type { Ref } from \"vue\";\r\nimport type { CropOutputFormat, CropResult } from \"../types\";\r\n\r\ninterface UseCropperCoreOptions {\r\n format?: Ref<CropOutputFormat>;\r\n quality?: Ref<number>;\r\n maxWidth?: Ref<number>;\r\n maxHeight?: Ref<number>;\r\n}\r\n\r\nexport function useCropperCore(options: UseCropperCoreOptions = {}) {\r\n const cropperRef = ref<any>(null);\r\n\r\n function rotateLeft() {\r\n cropperRef.value?.rotateLeft();\r\n }\r\n\r\n function rotateRight() {\r\n cropperRef.value?.rotateRight();\r\n }\r\n\r\n function rotate(angle: number) {\r\n const steps = Math.round(angle / 90);\r\n const fn = steps > 0 ? rotateRight : rotateLeft;\r\n for (let i = 0; i < Math.abs(steps); i++) fn();\r\n }\r\n\r\n function zoom(scale: number) {\r\n cropperRef.value?.changeScale(scale > 0 ? 1 : -1);\r\n }\r\n\r\n function flipX() {\r\n const el = cropperRef.value?.$refs?.img;\r\n if (!el) return;\r\n const current = el.style.transform || \"\";\r\n if (current.includes(\"scaleX(-1)\")) {\r\n el.style.transform = current.replace(\"scaleX(-1)\", \"scaleX(1)\");\r\n } else {\r\n el.style.transform =\r\n current.replace(/scaleX\\([^)]*\\)/, \"\") + \" scaleX(-1)\";\r\n }\r\n }\r\n\r\n function flipY() {\r\n const el = cropperRef.value?.$refs?.img;\r\n if (!el) return;\r\n const current = el.style.transform || \"\";\r\n if (current.includes(\"scaleY(-1)\")) {\r\n el.style.transform = current.replace(\"scaleY(-1)\", \"scaleY(1)\");\r\n } else {\r\n el.style.transform =\r\n current.replace(/scaleY\\([^)]*\\)/, \"\") + \" scaleY(-1)\";\r\n }\r\n }\r\n\r\n function reset() {\r\n cropperRef.value?.refresh();\r\n }\r\n\r\n function constrainSize(w: number, h: number) {\r\n let ow = w;\r\n let oh = h;\r\n const maxW = options.maxWidth?.value ?? 0;\r\n const maxH = options.maxHeight?.value ?? 0;\r\n\r\n if (maxW > 0 && ow > maxW) {\r\n const ratio = maxW / ow;\r\n ow = maxW;\r\n oh = Math.round(oh * ratio);\r\n }\r\n if (maxH > 0 && oh > maxH) {\r\n const ratio = maxH / oh;\r\n oh = maxH;\r\n ow = Math.round(ow * ratio);\r\n }\r\n return { width: ow, height: oh };\r\n }\r\n\r\n function getCropResult(): Promise<CropResult> {\r\n return new Promise((resolve, reject) => {\r\n const cropper = cropperRef.value;\r\n if (!cropper) return reject(new Error(\"Cropper not initialized\"));\r\n\r\n const format = options.format?.value ?? \"png\";\r\n const quality = options.quality?.value ?? 0.92;\r\n const mime =\r\n format === \"jpeg\"\r\n ? \"image/jpeg\"\r\n : format === \"webp\"\r\n ? \"image/webp\"\r\n : \"image/png\";\r\n\r\n cropper.getCropData((base64: string) => {\r\n cropper.getCropBlob((blob: Blob) => {\r\n const img = new Image();\r\n img.onload = () => {\r\n const { width, height } = constrainSize(img.width, img.height);\r\n\r\n if (width !== img.width || height !== img.height) {\r\n const canvas = document.createElement(\"canvas\");\r\n canvas.width = width;\r\n canvas.height = height;\r\n const ctx = canvas.getContext(\"2d\")!;\r\n ctx.drawImage(img, 0, 0, width, height);\r\n const constrainedBase64 = canvas.toDataURL(mime, quality);\r\n canvas.toBlob(\r\n (constrainedBlob) => {\r\n resolve({\r\n base64: constrainedBase64,\r\n blob: constrainedBlob!,\r\n width,\r\n height,\r\n format,\r\n });\r\n },\r\n mime,\r\n format === \"png\" ? undefined : quality,\r\n );\r\n } else {\r\n resolve({\r\n base64,\r\n blob,\r\n width: img.width,\r\n height: img.height,\r\n format,\r\n });\r\n }\r\n };\r\n img.onerror = () => reject(new Error(\"Failed to load crop result\"));\r\n img.src = base64;\r\n });\r\n });\r\n });\r\n }\r\n\r\n return {\r\n cropperRef,\r\n rotate,\r\n rotateLeft,\r\n rotateRight,\r\n zoom,\r\n flipX,\r\n flipY,\r\n reset,\r\n getCropResult,\r\n };\r\n}\r\n","<template>\r\n <div class=\"cropper-toolbar\">\r\n <NButtonGroup size=\"small\">\r\n <NButton\r\n v-for=\"item in ratioPresets\"\r\n :key=\"item.value\"\r\n :type=\"currentRatio === item.value ? 'primary' : 'default'\"\r\n @click=\"$emit('ratio', item.value)\"\r\n >\r\n {{ item.label }}\r\n </NButton>\r\n </NButtonGroup>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', -90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-left\" /></template>\r\n </NButton>\r\n </template>\r\n 逆时针旋转 90°\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', 90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-right\" /></template>\r\n </NButton>\r\n </template>\r\n 顺时针旋转 90°\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipX')\">\r\n <template #icon><C_Icon name=\"mdi:flip-horizontal\" /></template>\r\n </NButton>\r\n </template>\r\n 水平翻转\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipY')\">\r\n <template #icon><C_Icon name=\"mdi:flip-vertical\" /></template>\r\n </NButton>\r\n </template>\r\n 垂直翻转\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', 0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-plus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 放大\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', -0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-minus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 缩小\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('reset')\">\r\n <template #icon><C_Icon name=\"mdi:refresh\" /></template>\r\n </NButton>\r\n </template>\r\n 重置\r\n </NTooltip>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { NButton, NButtonGroup, NDivider, NTooltip } from \"naive-ui\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\nimport type { AspectRatioPreset } from \"../types\";\r\n\r\ndefineProps<{\r\n currentRatio: number;\r\n ratioPresets: AspectRatioPreset[];\r\n}>();\r\n\r\ndefineEmits<{\r\n ratio: [value: number];\r\n rotate: [angle: number];\r\n flipX: [];\r\n flipY: [];\r\n zoom: [scale: number];\r\n reset: [];\r\n}>();\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-toolbar {\r\n display: flex;\r\n flex-wrap: wrap;\r\n gap: 4px;\r\n align-items: center;\r\n padding: 8px 0;\r\n}\r\n</style>\r\n","<template>\r\n <div class=\"cropper-toolbar\">\r\n <NButtonGroup size=\"small\">\r\n <NButton\r\n v-for=\"item in ratioPresets\"\r\n :key=\"item.value\"\r\n :type=\"currentRatio === item.value ? 'primary' : 'default'\"\r\n @click=\"$emit('ratio', item.value)\"\r\n >\r\n {{ item.label }}\r\n </NButton>\r\n </NButtonGroup>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', -90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-left\" /></template>\r\n </NButton>\r\n </template>\r\n 逆时针旋转 90°\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', 90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-right\" /></template>\r\n </NButton>\r\n </template>\r\n 顺时针旋转 90°\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipX')\">\r\n <template #icon><C_Icon name=\"mdi:flip-horizontal\" /></template>\r\n </NButton>\r\n </template>\r\n 水平翻转\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipY')\">\r\n <template #icon><C_Icon name=\"mdi:flip-vertical\" /></template>\r\n </NButton>\r\n </template>\r\n 垂直翻转\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', 0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-plus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 放大\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', -0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-minus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 缩小\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('reset')\">\r\n <template #icon><C_Icon name=\"mdi:refresh\" /></template>\r\n </NButton>\r\n </template>\r\n 重置\r\n </NTooltip>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { NButton, NButtonGroup, NDivider, NTooltip } from \"naive-ui\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\nimport type { AspectRatioPreset } from \"../types\";\r\n\r\ndefineProps<{\r\n currentRatio: number;\r\n ratioPresets: AspectRatioPreset[];\r\n}>();\r\n\r\ndefineEmits<{\r\n ratio: [value: number];\r\n rotate: [angle: number];\r\n flipX: [];\r\n flipY: [];\r\n zoom: [scale: number];\r\n reset: [];\r\n}>();\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-toolbar {\r\n display: flex;\r\n flex-wrap: wrap;\r\n gap: 4px;\r\n align-items: center;\r\n padding: 8px 0;\r\n}\r\n</style>\r\n","<template>\r\n <div class=\"cropper-toolbar\">\r\n <NButtonGroup size=\"small\">\r\n <NButton\r\n v-for=\"item in ratioPresets\"\r\n :key=\"item.value\"\r\n :type=\"currentRatio === item.value ? 'primary' : 'default'\"\r\n @click=\"$emit('ratio', item.value)\"\r\n >\r\n {{ item.label }}\r\n </NButton>\r\n </NButtonGroup>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', -90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-left\" /></template>\r\n </NButton>\r\n </template>\r\n 逆时针旋转 90°\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('rotate', 90)\">\r\n <template #icon><C_Icon name=\"mdi:rotate-right\" /></template>\r\n </NButton>\r\n </template>\r\n 顺时针旋转 90°\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipX')\">\r\n <template #icon><C_Icon name=\"mdi:flip-horizontal\" /></template>\r\n </NButton>\r\n </template>\r\n 水平翻转\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('flipY')\">\r\n <template #icon><C_Icon name=\"mdi:flip-vertical\" /></template>\r\n </NButton>\r\n </template>\r\n 垂直翻转\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', 0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-plus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 放大\r\n </NTooltip>\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('zoom', -0.1)\">\r\n <template #icon><C_Icon name=\"mdi:magnify-minus-outline\" /></template>\r\n </NButton>\r\n </template>\r\n 缩小\r\n </NTooltip>\r\n\r\n <NDivider vertical />\r\n\r\n <NTooltip>\r\n <template #trigger>\r\n <NButton size=\"small\" quaternary @click=\"$emit('reset')\">\r\n <template #icon><C_Icon name=\"mdi:refresh\" /></template>\r\n </NButton>\r\n </template>\r\n 重置\r\n </NTooltip>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { NButton, NButtonGroup, NDivider, NTooltip } from \"naive-ui\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\nimport type { AspectRatioPreset } from \"../types\";\r\n\r\ndefineProps<{\r\n currentRatio: number;\r\n ratioPresets: AspectRatioPreset[];\r\n}>();\r\n\r\ndefineEmits<{\r\n ratio: [value: number];\r\n rotate: [angle: number];\r\n flipX: [];\r\n flipY: [];\r\n zoom: [scale: number];\r\n reset: [];\r\n}>();\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-toolbar {\r\n display: flex;\r\n flex-wrap: wrap;\r\n gap: 4px;\r\n align-items: center;\r\n padding: 8px 0;\r\n}\r\n</style>\r\n","<template>\r\n <div class=\"cropper-preview\">\r\n <div class=\"preview-label\">\r\n <C_Icon name=\"mdi:eye-outline\" />\r\n <span>裁剪预览</span>\r\n </div>\r\n\r\n <div\r\n ref=\"mainBoxRef\"\r\n class=\"preview-main\"\r\n :class=\"{ 'preview-main--circular': circular }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div class=\"preview-main__viewport\" :style=\"mainViewportStyle\">\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"preview\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n <div v-else class=\"preview-empty\">\r\n <C_Icon\r\n name=\"mdi:image-outline\"\r\n style=\"font-size: 32px; opacity: 0.2\"\r\n />\r\n </div>\r\n </div>\r\n\r\n <div class=\"preview-thumbs\">\r\n <div v-for=\"item in thumbSizes\" :key=\"item.label\" class=\"preview-thumb\">\r\n <div\r\n class=\"preview-thumb__box\"\r\n :class=\"{ 'preview-thumb__box--circular': circular }\"\r\n :style=\"{ width: `${item.size}px`, height: `${item.size}px` }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div\r\n class=\"preview-thumb__viewport\"\r\n :style=\"getThumbViewportStyle(item.size)\"\r\n >\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"thumb\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n </div>\r\n <span class=\"preview-thumb__label\">{{ item.label }}</span>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, onMounted, onBeforeUnmount } from \"vue\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\n\r\nconst props = defineProps<{\r\n previewData: any;\r\n circular?: boolean;\r\n}>();\r\n\r\nconst mainBoxRef = ref<HTMLElement | null>(null);\r\nconst mainBoxWidth = ref(200);\r\n\r\nconst thumbSizes = [\r\n { label: \"80px\", size: 80 },\r\n { label: \"48px\", size: 48 },\r\n { label: \"32px\", size: 32 },\r\n];\r\n\r\nconst hasValidPreview = computed(() => {\r\n const d = props.previewData;\r\n return d && d.url && d.w > 0 && d.h > 0;\r\n});\r\n\r\nconst mainViewportStyle = computed(() => {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n const zoom = mainBoxWidth.value / w;\r\n return { zoom, overflow: \"hidden\" };\r\n});\r\n\r\nfunction getThumbViewportStyle(size: number) {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n return { zoom: size / w, overflow: \"hidden\" };\r\n}\r\n\r\nlet resizeObserver: ResizeObserver | null = null;\r\n\r\nonMounted(() => {\r\n if (mainBoxRef.value) {\r\n mainBoxWidth.value = mainBoxRef.value.clientWidth;\r\n resizeObserver = new ResizeObserver((entries) => {\r\n requestAnimationFrame(() => {\r\n for (const entry of entries) {\r\n mainBoxWidth.value = entry.contentRect.width;\r\n }\r\n });\r\n });\r\n resizeObserver.observe(mainBoxRef.value);\r\n }\r\n});\r\n\r\nonBeforeUnmount(() => {\r\n resizeObserver?.disconnect();\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-preview {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 16px;\r\n\r\n .preview-label {\r\n display: flex;\r\n gap: 6px;\r\n align-items: center;\r\n font-size: 13px;\r\n font-weight: 500;\r\n color: var(--text-color-2);\r\n }\r\n\r\n .preview-main {\r\n width: 100%;\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 6px;\r\n background:\r\n linear-gradient(45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(-45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(45deg, transparent 75%, #f0f0f0 75%),\r\n linear-gradient(-45deg, transparent 75%, #f0f0f0 75%);\r\n background-position:\r\n 0 0,\r\n 0 8px,\r\n 8px -8px,\r\n -8px 0;\r\n background-size: 16px 16px;\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n }\r\n\r\n .preview-empty {\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 100%;\r\n height: 100%;\r\n min-height: 120px;\r\n }\r\n\r\n .preview-thumbs {\r\n display: flex;\r\n gap: 12px;\r\n align-items: flex-end;\r\n justify-content: center;\r\n }\r\n\r\n .preview-thumb {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 4px;\r\n align-items: center;\r\n\r\n &__box {\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 4px;\r\n background: var(--body-color);\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n\r\n &__label {\r\n font-size: 11px;\r\n color: var(--text-color-3);\r\n }\r\n }\r\n}\r\n</style>\r\n","<template>\r\n <div class=\"cropper-preview\">\r\n <div class=\"preview-label\">\r\n <C_Icon name=\"mdi:eye-outline\" />\r\n <span>裁剪预览</span>\r\n </div>\r\n\r\n <div\r\n ref=\"mainBoxRef\"\r\n class=\"preview-main\"\r\n :class=\"{ 'preview-main--circular': circular }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div class=\"preview-main__viewport\" :style=\"mainViewportStyle\">\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"preview\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n <div v-else class=\"preview-empty\">\r\n <C_Icon\r\n name=\"mdi:image-outline\"\r\n style=\"font-size: 32px; opacity: 0.2\"\r\n />\r\n </div>\r\n </div>\r\n\r\n <div class=\"preview-thumbs\">\r\n <div v-for=\"item in thumbSizes\" :key=\"item.label\" class=\"preview-thumb\">\r\n <div\r\n class=\"preview-thumb__box\"\r\n :class=\"{ 'preview-thumb__box--circular': circular }\"\r\n :style=\"{ width: `${item.size}px`, height: `${item.size}px` }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div\r\n class=\"preview-thumb__viewport\"\r\n :style=\"getThumbViewportStyle(item.size)\"\r\n >\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"thumb\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n </div>\r\n <span class=\"preview-thumb__label\">{{ item.label }}</span>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, onMounted, onBeforeUnmount } from \"vue\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\n\r\nconst props = defineProps<{\r\n previewData: any;\r\n circular?: boolean;\r\n}>();\r\n\r\nconst mainBoxRef = ref<HTMLElement | null>(null);\r\nconst mainBoxWidth = ref(200);\r\n\r\nconst thumbSizes = [\r\n { label: \"80px\", size: 80 },\r\n { label: \"48px\", size: 48 },\r\n { label: \"32px\", size: 32 },\r\n];\r\n\r\nconst hasValidPreview = computed(() => {\r\n const d = props.previewData;\r\n return d && d.url && d.w > 0 && d.h > 0;\r\n});\r\n\r\nconst mainViewportStyle = computed(() => {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n const zoom = mainBoxWidth.value / w;\r\n return { zoom, overflow: \"hidden\" };\r\n});\r\n\r\nfunction getThumbViewportStyle(size: number) {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n return { zoom: size / w, overflow: \"hidden\" };\r\n}\r\n\r\nlet resizeObserver: ResizeObserver | null = null;\r\n\r\nonMounted(() => {\r\n if (mainBoxRef.value) {\r\n mainBoxWidth.value = mainBoxRef.value.clientWidth;\r\n resizeObserver = new ResizeObserver((entries) => {\r\n requestAnimationFrame(() => {\r\n for (const entry of entries) {\r\n mainBoxWidth.value = entry.contentRect.width;\r\n }\r\n });\r\n });\r\n resizeObserver.observe(mainBoxRef.value);\r\n }\r\n});\r\n\r\nonBeforeUnmount(() => {\r\n resizeObserver?.disconnect();\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-preview {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 16px;\r\n\r\n .preview-label {\r\n display: flex;\r\n gap: 6px;\r\n align-items: center;\r\n font-size: 13px;\r\n font-weight: 500;\r\n color: var(--text-color-2);\r\n }\r\n\r\n .preview-main {\r\n width: 100%;\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 6px;\r\n background:\r\n linear-gradient(45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(-45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(45deg, transparent 75%, #f0f0f0 75%),\r\n linear-gradient(-45deg, transparent 75%, #f0f0f0 75%);\r\n background-position:\r\n 0 0,\r\n 0 8px,\r\n 8px -8px,\r\n -8px 0;\r\n background-size: 16px 16px;\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n }\r\n\r\n .preview-empty {\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 100%;\r\n height: 100%;\r\n min-height: 120px;\r\n }\r\n\r\n .preview-thumbs {\r\n display: flex;\r\n gap: 12px;\r\n align-items: flex-end;\r\n justify-content: center;\r\n }\r\n\r\n .preview-thumb {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 4px;\r\n align-items: center;\r\n\r\n &__box {\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 4px;\r\n background: var(--body-color);\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n\r\n &__label {\r\n font-size: 11px;\r\n color: var(--text-color-3);\r\n }\r\n }\r\n}\r\n</style>\r\n","<template>\r\n <div class=\"cropper-preview\">\r\n <div class=\"preview-label\">\r\n <C_Icon name=\"mdi:eye-outline\" />\r\n <span>裁剪预览</span>\r\n </div>\r\n\r\n <div\r\n ref=\"mainBoxRef\"\r\n class=\"preview-main\"\r\n :class=\"{ 'preview-main--circular': circular }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div class=\"preview-main__viewport\" :style=\"mainViewportStyle\">\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"preview\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n <div v-else class=\"preview-empty\">\r\n <C_Icon\r\n name=\"mdi:image-outline\"\r\n style=\"font-size: 32px; opacity: 0.2\"\r\n />\r\n </div>\r\n </div>\r\n\r\n <div class=\"preview-thumbs\">\r\n <div v-for=\"item in thumbSizes\" :key=\"item.label\" class=\"preview-thumb\">\r\n <div\r\n class=\"preview-thumb__box\"\r\n :class=\"{ 'preview-thumb__box--circular': circular }\"\r\n :style=\"{ width: `${item.size}px`, height: `${item.size}px` }\"\r\n >\r\n <template v-if=\"hasValidPreview\">\r\n <div\r\n class=\"preview-thumb__viewport\"\r\n :style=\"getThumbViewportStyle(item.size)\"\r\n >\r\n <div :style=\"previewData.div\">\r\n <img\r\n :src=\"previewData.url\"\r\n :style=\"previewData.img\"\r\n alt=\"thumb\"\r\n />\r\n </div>\r\n </div>\r\n </template>\r\n </div>\r\n <span class=\"preview-thumb__label\">{{ item.label }}</span>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, onMounted, onBeforeUnmount } from \"vue\";\r\nimport C_Icon from \"../../C_Icon/index.vue\";\r\n\r\nconst props = defineProps<{\r\n previewData: any;\r\n circular?: boolean;\r\n}>();\r\n\r\nconst mainBoxRef = ref<HTMLElement | null>(null);\r\nconst mainBoxWidth = ref(200);\r\n\r\nconst thumbSizes = [\r\n { label: \"80px\", size: 80 },\r\n { label: \"48px\", size: 48 },\r\n { label: \"32px\", size: 32 },\r\n];\r\n\r\nconst hasValidPreview = computed(() => {\r\n const d = props.previewData;\r\n return d && d.url && d.w > 0 && d.h > 0;\r\n});\r\n\r\nconst mainViewportStyle = computed(() => {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n const zoom = mainBoxWidth.value / w;\r\n return { zoom, overflow: \"hidden\" };\r\n});\r\n\r\nfunction getThumbViewportStyle(size: number) {\r\n if (!hasValidPreview.value) return {};\r\n const { w } = props.previewData;\r\n return { zoom: size / w, overflow: \"hidden\" };\r\n}\r\n\r\nlet resizeObserver: ResizeObserver | null = null;\r\n\r\nonMounted(() => {\r\n if (mainBoxRef.value) {\r\n mainBoxWidth.value = mainBoxRef.value.clientWidth;\r\n resizeObserver = new ResizeObserver((entries) => {\r\n requestAnimationFrame(() => {\r\n for (const entry of entries) {\r\n mainBoxWidth.value = entry.contentRect.width;\r\n }\r\n });\r\n });\r\n resizeObserver.observe(mainBoxRef.value);\r\n }\r\n});\r\n\r\nonBeforeUnmount(() => {\r\n resizeObserver?.disconnect();\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.cropper-preview {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 16px;\r\n\r\n .preview-label {\r\n display: flex;\r\n gap: 6px;\r\n align-items: center;\r\n font-size: 13px;\r\n font-weight: 500;\r\n color: var(--text-color-2);\r\n }\r\n\r\n .preview-main {\r\n width: 100%;\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 6px;\r\n background:\r\n linear-gradient(45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(-45deg, #f0f0f0 25%, transparent 25%),\r\n linear-gradient(45deg, transparent 75%, #f0f0f0 75%),\r\n linear-gradient(-45deg, transparent 75%, #f0f0f0 75%);\r\n background-position:\r\n 0 0,\r\n 0 8px,\r\n 8px -8px,\r\n -8px 0;\r\n background-size: 16px 16px;\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n }\r\n\r\n .preview-empty {\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 100%;\r\n height: 100%;\r\n min-height: 120px;\r\n }\r\n\r\n .preview-thumbs {\r\n display: flex;\r\n gap: 12px;\r\n align-items: flex-end;\r\n justify-content: center;\r\n }\r\n\r\n .preview-thumb {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 4px;\r\n align-items: center;\r\n\r\n &__box {\r\n overflow: hidden;\r\n border: 1px solid var(--border-color);\r\n border-radius: 4px;\r\n background: var(--body-color);\r\n\r\n &--circular {\r\n border-radius: 50%;\r\n }\r\n }\r\n\r\n &__viewport img {\r\n display: block;\r\n max-width: none !important;\r\n }\r\n\r\n &__label {\r\n font-size: 11px;\r\n color: var(--text-color-3);\r\n }\r\n }\r\n}\r\n</style>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2026-02-25\r\n * @Description: 图片裁剪组件(基于 vue-cropper)\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <NModal\r\n v-if=\"props.modal\"\r\n v-model:show=\"modalVisible\"\r\n preset=\"card\"\r\n :title=\"props.modalTitle || '图片裁剪'\"\r\n :style=\"{ width: '860px' }\"\r\n :mask-closable=\"false\"\r\n :closable=\"true\"\r\n >\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"modalCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n <template #footer>\r\n <NSpace justify=\"end\">\r\n <NButton @click=\"handleCancel\">取消</NButton>\r\n <NButton type=\"primary\" :loading=\"exporting\" @click=\"handleConfirm\"\r\n >确认裁剪</NButton\r\n >\r\n </NSpace>\r\n </template>\r\n </NModal>\r\n\r\n <div v-else class=\"c-image-cropper\">\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"inlineCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch } from \"vue\";\r\nimport { NModal, NButton, NSpace } from \"naive-ui\";\r\nimport { VueCropper } from \"vue-cropper\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport type {\r\n AspectRatioPreset,\r\n CropOutputFormat,\r\n ImageCropperExpose,\r\n ImageCropperProps,\r\n} from \"./types\";\r\nimport { useCropperCore } from \"./composables/useCropperCore\";\r\nimport CropperToolbar from \"./components/CropperToolbar.vue\";\r\nimport CropperPreview from \"./components/CropperPreview.vue\";\r\n\r\ndefineOptions({ name: \"C_ImageCropper\" });\r\n\r\nconst props = withDefaults(defineProps<ImageCropperProps>(), {\r\n src: \"\",\r\n aspectRatio: 0,\r\n outputFormat: \"png\",\r\n outputQuality: 0.92,\r\n maxOutputWidth: 0,\r\n maxOutputHeight: 0,\r\n showPreview: true,\r\n showToolbar: true,\r\n circular: false,\r\n disabled: false,\r\n height: \"400px\",\r\n modal: false,\r\n modalTitle: \"图片裁剪\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n crop: [result: any];\r\n ready: [];\r\n error: [error: Event];\r\n confirm: [result: any];\r\n cancel: [];\r\n}>();\r\n\r\nconst imgSrc = ref(props.src);\r\nconst currentRatio = ref(props.aspectRatio);\r\nconst modalVisible = ref(false);\r\nconst exporting = ref(false);\r\nconst previewData = ref<any>(null);\r\n\r\nconst inlineCropperRef = ref<any>(null);\r\nconst modalCropperRef = ref<any>(null);\r\n\r\nconst activeCropperRef = computed(() =>\r\n props.modal ? modalCropperRef.value : inlineCropperRef.value,\r\n);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst vueCropperOutputType = computed(() => {\r\n const map: Record<CropOutputFormat, string> = {\r\n png: \"png\",\r\n jpeg: \"jpeg\",\r\n webp: \"webp\",\r\n };\r\n return map[props.outputFormat as CropOutputFormat] || \"png\";\r\n});\r\n\r\nconst isFixed = computed(() => currentRatio.value > 0);\r\n\r\nconst fixedNumber = computed(() => {\r\n if (currentRatio.value <= 0) return [1, 1];\r\n if (currentRatio.value === 1) return [1, 1];\r\n if (Math.abs(currentRatio.value - 16 / 9) < 0.01) return [16, 9];\r\n if (Math.abs(currentRatio.value - 4 / 3) < 0.01) return [4, 3];\r\n if (Math.abs(currentRatio.value - 3 / 2) < 0.01) return [3, 2];\r\n return [Math.round(currentRatio.value * 100), 100];\r\n});\r\n\r\nconst autoCropSize = computed(() => ({\r\n width: 300,\r\n height: currentRatio.value > 0 ? 300 / currentRatio.value : 200,\r\n}));\r\n\r\nconst ratioPresets: AspectRatioPreset[] = [\r\n { label: \"自由\", value: 0 },\r\n { label: \"1:1\", value: 1 },\r\n { label: \"16:9\", value: 16 / 9 },\r\n { label: \"4:3\", value: 4 / 3 },\r\n { label: \"3:2\", value: 3 / 2 },\r\n];\r\n\r\nconst formatRef = computed(() => props.outputFormat as CropOutputFormat);\r\nconst qualityRef = computed(() => props.outputQuality);\r\nconst maxWRef = computed(() => props.maxOutputWidth);\r\nconst maxHRef = computed(() => props.maxOutputHeight);\r\n\r\nconst core = useCropperCore({\r\n format: formatRef,\r\n quality: qualityRef,\r\n maxWidth: maxWRef,\r\n maxHeight: maxHRef,\r\n});\r\n\r\nwatch(activeCropperRef, (v) => {\r\n core.cropperRef.value = v;\r\n});\r\n\r\nfunction handleRealTimePreview(data: any) {\r\n previewData.value = data;\r\n}\r\nfunction onImgLoad(status: string) {\r\n if (status === \"success\") emit(\"ready\");\r\n}\r\nfunction rotate(angle: number) {\r\n core.rotate(angle);\r\n}\r\nfunction handleRatio(v: number) {\r\n currentRatio.value = v;\r\n}\r\nfunction handleReset() {\r\n core.reset();\r\n}\r\n\r\nasync function handleConfirm() {\r\n exporting.value = true;\r\n try {\r\n const result = await core.getCropResult();\r\n emit(\"confirm\", result);\r\n emit(\"crop\", result);\r\n modalVisible.value = false;\r\n } finally {\r\n exporting.value = false;\r\n }\r\n}\r\n\r\nfunction handleCancel() {\r\n modalVisible.value = false;\r\n emit(\"cancel\");\r\n}\r\n\r\nfunction loadFile(file: File) {\r\n const reader = new FileReader();\r\n reader.onload = (e) => {\r\n imgSrc.value = e.target?.result as string;\r\n };\r\n reader.readAsDataURL(file);\r\n}\r\n\r\nwatch(\r\n () => props.src,\r\n (v) => {\r\n imgSrc.value = v;\r\n },\r\n);\r\nwatch(\r\n () => props.aspectRatio,\r\n (v) => {\r\n currentRatio.value = v;\r\n },\r\n);\r\n\r\ndefineExpose<ImageCropperExpose>({\r\n getCropResult: core.getCropResult,\r\n rotate: core.rotate,\r\n zoom: core.zoom,\r\n flipX: core.flipX,\r\n flipY: core.flipY,\r\n reset: core.reset,\r\n setAspectRatio: (r: number) => {\r\n currentRatio.value = r;\r\n },\r\n loadFile,\r\n open: (src?: string) => {\r\n if (src) imgSrc.value = src;\r\n modalVisible.value = true;\r\n },\r\n close: () => {\r\n modalVisible.value = false;\r\n },\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: 图片裁剪组件(基于 vue-cropper)\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <NModal\r\n v-if=\"props.modal\"\r\n v-model:show=\"modalVisible\"\r\n preset=\"card\"\r\n :title=\"props.modalTitle || '图片裁剪'\"\r\n :style=\"{ width: '860px' }\"\r\n :mask-closable=\"false\"\r\n :closable=\"true\"\r\n >\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"modalCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n <template #footer>\r\n <NSpace justify=\"end\">\r\n <NButton @click=\"handleCancel\">取消</NButton>\r\n <NButton type=\"primary\" :loading=\"exporting\" @click=\"handleConfirm\"\r\n >确认裁剪</NButton\r\n >\r\n </NSpace>\r\n </template>\r\n </NModal>\r\n\r\n <div v-else class=\"c-image-cropper\">\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"inlineCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch } from \"vue\";\r\nimport { NModal, NButton, NSpace } from \"naive-ui\";\r\nimport { VueCropper } from \"vue-cropper\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport type {\r\n AspectRatioPreset,\r\n CropOutputFormat,\r\n ImageCropperExpose,\r\n ImageCropperProps,\r\n} from \"./types\";\r\nimport { useCropperCore } from \"./composables/useCropperCore\";\r\nimport CropperToolbar from \"./components/CropperToolbar.vue\";\r\nimport CropperPreview from \"./components/CropperPreview.vue\";\r\n\r\ndefineOptions({ name: \"C_ImageCropper\" });\r\n\r\nconst props = withDefaults(defineProps<ImageCropperProps>(), {\r\n src: \"\",\r\n aspectRatio: 0,\r\n outputFormat: \"png\",\r\n outputQuality: 0.92,\r\n maxOutputWidth: 0,\r\n maxOutputHeight: 0,\r\n showPreview: true,\r\n showToolbar: true,\r\n circular: false,\r\n disabled: false,\r\n height: \"400px\",\r\n modal: false,\r\n modalTitle: \"图片裁剪\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n crop: [result: any];\r\n ready: [];\r\n error: [error: Event];\r\n confirm: [result: any];\r\n cancel: [];\r\n}>();\r\n\r\nconst imgSrc = ref(props.src);\r\nconst currentRatio = ref(props.aspectRatio);\r\nconst modalVisible = ref(false);\r\nconst exporting = ref(false);\r\nconst previewData = ref<any>(null);\r\n\r\nconst inlineCropperRef = ref<any>(null);\r\nconst modalCropperRef = ref<any>(null);\r\n\r\nconst activeCropperRef = computed(() =>\r\n props.modal ? modalCropperRef.value : inlineCropperRef.value,\r\n);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst vueCropperOutputType = computed(() => {\r\n const map: Record<CropOutputFormat, string> = {\r\n png: \"png\",\r\n jpeg: \"jpeg\",\r\n webp: \"webp\",\r\n };\r\n return map[props.outputFormat as CropOutputFormat] || \"png\";\r\n});\r\n\r\nconst isFixed = computed(() => currentRatio.value > 0);\r\n\r\nconst fixedNumber = computed(() => {\r\n if (currentRatio.value <= 0) return [1, 1];\r\n if (currentRatio.value === 1) return [1, 1];\r\n if (Math.abs(currentRatio.value - 16 / 9) < 0.01) return [16, 9];\r\n if (Math.abs(currentRatio.value - 4 / 3) < 0.01) return [4, 3];\r\n if (Math.abs(currentRatio.value - 3 / 2) < 0.01) return [3, 2];\r\n return [Math.round(currentRatio.value * 100), 100];\r\n});\r\n\r\nconst autoCropSize = computed(() => ({\r\n width: 300,\r\n height: currentRatio.value > 0 ? 300 / currentRatio.value : 200,\r\n}));\r\n\r\nconst ratioPresets: AspectRatioPreset[] = [\r\n { label: \"自由\", value: 0 },\r\n { label: \"1:1\", value: 1 },\r\n { label: \"16:9\", value: 16 / 9 },\r\n { label: \"4:3\", value: 4 / 3 },\r\n { label: \"3:2\", value: 3 / 2 },\r\n];\r\n\r\nconst formatRef = computed(() => props.outputFormat as CropOutputFormat);\r\nconst qualityRef = computed(() => props.outputQuality);\r\nconst maxWRef = computed(() => props.maxOutputWidth);\r\nconst maxHRef = computed(() => props.maxOutputHeight);\r\n\r\nconst core = useCropperCore({\r\n format: formatRef,\r\n quality: qualityRef,\r\n maxWidth: maxWRef,\r\n maxHeight: maxHRef,\r\n});\r\n\r\nwatch(activeCropperRef, (v) => {\r\n core.cropperRef.value = v;\r\n});\r\n\r\nfunction handleRealTimePreview(data: any) {\r\n previewData.value = data;\r\n}\r\nfunction onImgLoad(status: string) {\r\n if (status === \"success\") emit(\"ready\");\r\n}\r\nfunction rotate(angle: number) {\r\n core.rotate(angle);\r\n}\r\nfunction handleRatio(v: number) {\r\n currentRatio.value = v;\r\n}\r\nfunction handleReset() {\r\n core.reset();\r\n}\r\n\r\nasync function handleConfirm() {\r\n exporting.value = true;\r\n try {\r\n const result = await core.getCropResult();\r\n emit(\"confirm\", result);\r\n emit(\"crop\", result);\r\n modalVisible.value = false;\r\n } finally {\r\n exporting.value = false;\r\n }\r\n}\r\n\r\nfunction handleCancel() {\r\n modalVisible.value = false;\r\n emit(\"cancel\");\r\n}\r\n\r\nfunction loadFile(file: File) {\r\n const reader = new FileReader();\r\n reader.onload = (e) => {\r\n imgSrc.value = e.target?.result as string;\r\n };\r\n reader.readAsDataURL(file);\r\n}\r\n\r\nwatch(\r\n () => props.src,\r\n (v) => {\r\n imgSrc.value = v;\r\n },\r\n);\r\nwatch(\r\n () => props.aspectRatio,\r\n (v) => {\r\n currentRatio.value = v;\r\n },\r\n);\r\n\r\ndefineExpose<ImageCropperExpose>({\r\n getCropResult: core.getCropResult,\r\n rotate: core.rotate,\r\n zoom: core.zoom,\r\n flipX: core.flipX,\r\n flipY: core.flipY,\r\n reset: core.reset,\r\n setAspectRatio: (r: number) => {\r\n currentRatio.value = r;\r\n },\r\n loadFile,\r\n open: (src?: string) => {\r\n if (src) imgSrc.value = src;\r\n modalVisible.value = true;\r\n },\r\n close: () => {\r\n modalVisible.value = false;\r\n },\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: 图片裁剪组件(基于 vue-cropper)\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2026 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <NModal\r\n v-if=\"props.modal\"\r\n v-model:show=\"modalVisible\"\r\n preset=\"card\"\r\n :title=\"props.modalTitle || '图片裁剪'\"\r\n :style=\"{ width: '860px' }\"\r\n :mask-closable=\"false\"\r\n :closable=\"true\"\r\n >\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"modalCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n <template #footer>\r\n <NSpace justify=\"end\">\r\n <NButton @click=\"handleCancel\">取消</NButton>\r\n <NButton type=\"primary\" :loading=\"exporting\" @click=\"handleConfirm\"\r\n >确认裁剪</NButton\r\n >\r\n </NSpace>\r\n </template>\r\n </NModal>\r\n\r\n <div v-else class=\"c-image-cropper\">\r\n <div class=\"c-image-cropper__body\">\r\n <CropperToolbar\r\n v-if=\"props.showToolbar\"\r\n :current-ratio=\"currentRatio\"\r\n :ratio-presets=\"ratioPresets\"\r\n @ratio=\"handleRatio\"\r\n @rotate=\"rotate\"\r\n @flip-x=\"core.flipX\"\r\n @flip-y=\"core.flipY\"\r\n @zoom=\"core.zoom\"\r\n @reset=\"handleReset\"\r\n />\r\n <div class=\"c-image-cropper__workspace\">\r\n <div\r\n class=\"c-image-cropper__canvas\"\r\n :style=\"{ height: containerHeight }\"\r\n >\r\n <VueCropper\r\n v-if=\"imgSrc\"\r\n ref=\"inlineCropperRef\"\r\n :img=\"imgSrc\"\r\n :output-size=\"props.outputQuality\"\r\n :output-type=\"vueCropperOutputType\"\r\n :can-scale=\"false\"\r\n :auto-crop=\"true\"\r\n :auto-crop-width=\"autoCropSize.width\"\r\n :auto-crop-height=\"autoCropSize.height\"\r\n :fixed=\"isFixed\"\r\n :fixed-number=\"fixedNumber\"\r\n :center-box=\"true\"\r\n :info=\"true\"\r\n :info-true=\"true\"\r\n :can-move=\"true\"\r\n :can-move-box=\"true\"\r\n :original=\"false\"\r\n :high=\"true\"\r\n :full=\"false\"\r\n :mode=\"'contain'\"\r\n @real-time=\"handleRealTimePreview\"\r\n @img-load=\"onImgLoad\"\r\n />\r\n <div v-else class=\"c-image-cropper__placeholder\">\r\n <C_Icon\r\n name=\"mdi:image-plus-outline\"\r\n style=\"font-size: 48px; opacity: 0.3\"\r\n />\r\n <span>请选择图片</span>\r\n </div>\r\n </div>\r\n <div\r\n v-if=\"props.showPreview && imgSrc\"\r\n class=\"c-image-cropper__preview-panel\"\r\n >\r\n <CropperPreview\r\n :preview-data=\"previewData\"\r\n :circular=\"props.circular\"\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch } from \"vue\";\r\nimport { NModal, NButton, NSpace } from \"naive-ui\";\r\nimport { VueCropper } from \"vue-cropper\";\r\nimport C_Icon from \"../C_Icon/index.vue\";\r\nimport type {\r\n AspectRatioPreset,\r\n CropOutputFormat,\r\n ImageCropperExpose,\r\n ImageCropperProps,\r\n} from \"./types\";\r\nimport { useCropperCore } from \"./composables/useCropperCore\";\r\nimport CropperToolbar from \"./components/CropperToolbar.vue\";\r\nimport CropperPreview from \"./components/CropperPreview.vue\";\r\n\r\ndefineOptions({ name: \"C_ImageCropper\" });\r\n\r\nconst props = withDefaults(defineProps<ImageCropperProps>(), {\r\n src: \"\",\r\n aspectRatio: 0,\r\n outputFormat: \"png\",\r\n outputQuality: 0.92,\r\n maxOutputWidth: 0,\r\n maxOutputHeight: 0,\r\n showPreview: true,\r\n showToolbar: true,\r\n circular: false,\r\n disabled: false,\r\n height: \"400px\",\r\n modal: false,\r\n modalTitle: \"图片裁剪\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n crop: [result: any];\r\n ready: [];\r\n error: [error: Event];\r\n confirm: [result: any];\r\n cancel: [];\r\n}>();\r\n\r\nconst imgSrc = ref(props.src);\r\nconst currentRatio = ref(props.aspectRatio);\r\nconst modalVisible = ref(false);\r\nconst exporting = ref(false);\r\nconst previewData = ref<any>(null);\r\n\r\nconst inlineCropperRef = ref<any>(null);\r\nconst modalCropperRef = ref<any>(null);\r\n\r\nconst activeCropperRef = computed(() =>\r\n props.modal ? modalCropperRef.value : inlineCropperRef.value,\r\n);\r\n\r\nconst containerHeight = computed(() =>\r\n typeof props.height === \"number\" ? `${props.height}px` : props.height,\r\n);\r\n\r\nconst vueCropperOutputType = computed(() => {\r\n const map: Record<CropOutputFormat, string> = {\r\n png: \"png\",\r\n jpeg: \"jpeg\",\r\n webp: \"webp\",\r\n };\r\n return map[props.outputFormat as CropOutputFormat] || \"png\";\r\n});\r\n\r\nconst isFixed = computed(() => currentRatio.value > 0);\r\n\r\nconst fixedNumber = computed(() => {\r\n if (currentRatio.value <= 0) return [1, 1];\r\n if (currentRatio.value === 1) return [1, 1];\r\n if (Math.abs(currentRatio.value - 16 / 9) < 0.01) return [16, 9];\r\n if (Math.abs(currentRatio.value - 4 / 3) < 0.01) return [4, 3];\r\n if (Math.abs(currentRatio.value - 3 / 2) < 0.01) return [3, 2];\r\n return [Math.round(currentRatio.value * 100), 100];\r\n});\r\n\r\nconst autoCropSize = computed(() => ({\r\n width: 300,\r\n height: currentRatio.value > 0 ? 300 / currentRatio.value : 200,\r\n}));\r\n\r\nconst ratioPresets: AspectRatioPreset[] = [\r\n { label: \"自由\", value: 0 },\r\n { label: \"1:1\", value: 1 },\r\n { label: \"16:9\", value: 16 / 9 },\r\n { label: \"4:3\", value: 4 / 3 },\r\n { label: \"3:2\", value: 3 / 2 },\r\n];\r\n\r\nconst formatRef = computed(() => props.outputFormat as CropOutputFormat);\r\nconst qualityRef = computed(() => props.outputQuality);\r\nconst maxWRef = computed(() => props.maxOutputWidth);\r\nconst maxHRef = computed(() => props.maxOutputHeight);\r\n\r\nconst core = useCropperCore({\r\n format: formatRef,\r\n quality: qualityRef,\r\n maxWidth: maxWRef,\r\n maxHeight: maxHRef,\r\n});\r\n\r\nwatch(activeCropperRef, (v) => {\r\n core.cropperRef.value = v;\r\n});\r\n\r\nfunction handleRealTimePreview(data: any) {\r\n previewData.value = data;\r\n}\r\nfunction onImgLoad(status: string) {\r\n if (status === \"success\") emit(\"ready\");\r\n}\r\nfunction rotate(angle: number) {\r\n core.rotate(angle);\r\n}\r\nfunction handleRatio(v: number) {\r\n currentRatio.value = v;\r\n}\r\nfunction handleReset() {\r\n core.reset();\r\n}\r\n\r\nasync function handleConfirm() {\r\n exporting.value = true;\r\n try {\r\n const result = await core.getCropResult();\r\n emit(\"confirm\", result);\r\n emit(\"crop\", result);\r\n modalVisible.value = false;\r\n } finally {\r\n exporting.value = false;\r\n }\r\n}\r\n\r\nfunction handleCancel() {\r\n modalVisible.value = false;\r\n emit(\"cancel\");\r\n}\r\n\r\nfunction loadFile(file: File) {\r\n const reader = new FileReader();\r\n reader.onload = (e) => {\r\n imgSrc.value = e.target?.result as string;\r\n };\r\n reader.readAsDataURL(file);\r\n}\r\n\r\nwatch(\r\n () => props.src,\r\n (v) => {\r\n imgSrc.value = v;\r\n },\r\n);\r\nwatch(\r\n () => props.aspectRatio,\r\n (v) => {\r\n currentRatio.value = v;\r\n },\r\n);\r\n\r\ndefineExpose<ImageCropperExpose>({\r\n getCropResult: core.getCropResult,\r\n rotate: core.rotate,\r\n zoom: core.zoom,\r\n flipX: core.flipX,\r\n flipY: core.flipY,\r\n reset: core.reset,\r\n setAspectRatio: (r: number) => {\r\n currentRatio.value = r;\r\n },\r\n loadFile,\r\n open: (src?: string) => {\r\n if (src) imgSrc.value = src;\r\n modalVisible.value = true;\r\n },\r\n close: () => {\r\n modalVisible.value = false;\r\n },\r\n});\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n@use \"./index.scss\";\r\n</style>\r\n"],"mappings":";;;;;;;AAWA,SAAgB,eAAe,UAAiC,EAAE,EAAE;CAClE,MAAM,aAAa,IAAS,KAAK;CAEjC,SAAS,aAAa;AACpB,aAAW,OAAO,YAAY;;CAGhC,SAAS,cAAc;AACrB,aAAW,OAAO,aAAa;;CAGjC,SAAS,OAAO,OAAe;EAC7B,MAAM,QAAQ,KAAK,MAAM,QAAQ,GAAG;EACpC,MAAM,KAAK,QAAQ,IAAI,cAAc;AACrC,OAAK,IAAI,IAAI,GAAG,IAAI,KAAK,IAAI,MAAM,EAAE,IAAK,KAAI;;CAGhD,SAAS,KAAK,OAAe;AAC3B,aAAW,OAAO,YAAY,QAAQ,IAAI,IAAI,GAAG;;CAGnD,SAAS,QAAQ;EACf,MAAM,KAAK,WAAW,OAAO,OAAO;AACpC,MAAI,CAAC,GAAI;EACT,MAAM,UAAU,GAAG,MAAM,aAAa;AACtC,MAAI,QAAQ,SAAS,aAAa,CAChC,IAAG,MAAM,YAAY,QAAQ,QAAQ,cAAc,YAAY;MAE/D,IAAG,MAAM,YACP,QAAQ,QAAQ,mBAAmB,GAAG,GAAG;;CAI/C,SAAS,QAAQ;EACf,MAAM,KAAK,WAAW,OAAO,OAAO;AACpC,MAAI,CAAC,GAAI;EACT,MAAM,UAAU,GAAG,MAAM,aAAa;AACtC,MAAI,QAAQ,SAAS,aAAa,CAChC,IAAG,MAAM,YAAY,QAAQ,QAAQ,cAAc,YAAY;MAE/D,IAAG,MAAM,YACP,QAAQ,QAAQ,mBAAmB,GAAG,GAAG;;CAI/C,SAAS,QAAQ;AACf,aAAW,OAAO,SAAS;;CAG7B,SAAS,cAAc,GAAW,GAAW;EAC3C,IAAI,KAAK;EACT,IAAI,KAAK;EACT,MAAM,OAAO,QAAQ,UAAU,SAAS;EACxC,MAAM,OAAO,QAAQ,WAAW,SAAS;AAEzC,MAAI,OAAO,KAAK,KAAK,MAAM;GACzB,MAAM,QAAQ,OAAO;AACrB,QAAK;AACL,QAAK,KAAK,MAAM,KAAK,MAAM;;AAE7B,MAAI,OAAO,KAAK,KAAK,MAAM;GACzB,MAAM,QAAQ,OAAO;AACrB,QAAK;AACL,QAAK,KAAK,MAAM,KAAK,MAAM;;AAE7B,SAAO;GAAE,OAAO;GAAI,QAAQ;GAAI;;CAGlC,SAAS,gBAAqC;AAC5C,SAAO,IAAI,SAAS,SAAS,WAAW;GACtC,MAAM,UAAU,WAAW;AAC3B,OAAI,CAAC,QAAS,QAAO,uBAAO,IAAI,MAAM,0BAA0B,CAAC;GAEjE,MAAM,SAAS,QAAQ,QAAQ,SAAS;GACxC,MAAM,UAAU,QAAQ,SAAS,SAAS;GAC1C,MAAM,OACJ,WAAW,SACP,eACA,WAAW,SACT,eACA;AAER,WAAQ,aAAa,WAAmB;AACtC,YAAQ,aAAa,SAAe;KAClC,MAAM,MAAM,IAAI,OAAO;AACvB,SAAI,eAAe;MACjB,MAAM,EAAE,OAAO,WAAW,cAAc,IAAI,OAAO,IAAI,OAAO;AAE9D,UAAI,UAAU,IAAI,SAAS,WAAW,IAAI,QAAQ;OAChD,MAAM,SAAS,SAAS,cAAc,SAAS;AAC/C,cAAO,QAAQ;AACf,cAAO,SAAS;AAEhB,OADY,OAAO,WAAW,KAAK,CAC/B,UAAU,KAAK,GAAG,GAAG,OAAO,OAAO;OACvC,MAAM,oBAAoB,OAAO,UAAU,MAAM,QAAQ;AACzD,cAAO,QACJ,oBAAoB;AACnB,gBAAQ;SACN,QAAQ;SACR,MAAM;SACN;SACA;SACA;SACD,CAAC;UAEJ,MACA,WAAW,QAAQ,SAAY,QAChC;YAED,SAAQ;OACN;OACA;OACA,OAAO,IAAI;OACX,QAAQ,IAAI;OACZ;OACD,CAAC;;AAGN,SAAI,gBAAgB,uBAAO,IAAI,MAAM,6BAA6B,CAAC;AACnE,SAAI,MAAM;MACV;KACF;IACF;;AAGJ,QAAO;EACL;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACD;;;;;;;;;;;;;;;;;;;;;;uBEjJD,mBA+EM,OA/EN,cA+EM;IA9EJ,YASe,MAAA,aAAA,EAAA,EATD,MAAK,SAAO,EAAA;4BAEM,mBAD9B,mBAOU,UAAA,MAAA,WANOA,KAAAA,eAAR,SAAI;0BADb,YAOU,MAAA,QAAA,EAAA;OALP,KAAK,KAAK;OACV,MAAMC,KAAAA,iBAAiB,KAAK,QAAK,YAAA;OACjC,UAAK,WAAEC,KAAAA,MAAK,SAAU,KAAK,MAAK;;8BAEjB,iCAAb,KAAK,MAAK,EAAA,EAAA;;;;;;IAIjB,YAAqB,MAAA,SAAA,EAAA,EAAX,UAAA,IAAQ,CAAA;IAElB,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,UAAA,IAAA;;MACjC,MAAI,cAAkC,CAAjC,YAAiC,gBAAA,EAAzB,MAAK,mBAAiB,CAAA;;;4BAIpD,2CAFa,eAEb,GAAA;;;;IACA,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,UAAA,GAAA;;MACjC,MAAI,cAAmC,CAAlC,YAAkC,gBAAA,EAA1B,MAAK,oBAAkB,CAAA;;;4BAIrD,2CAFa,eAEb,GAAA;;;;IAEA,YAAqB,MAAA,SAAA,EAAA,EAAX,UAAA,IAAQ,CAAA;IAElB,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,QAAA;;MACjC,MAAI,cAAsC,CAArC,YAAqC,gBAAA,EAA7B,MAAK,uBAAqB,CAAA;;;4BAIxD,2CAFa,UAEb,GAAA;;;;IACA,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,QAAA;;MACjC,MAAI,cAAoC,CAAnC,YAAmC,gBAAA,EAA3B,MAAK,qBAAmB,CAAA;;;4BAItD,6CAFa,UAEb,GAAA;;;;IAEA,YAAqB,MAAA,SAAA,EAAA,EAAX,UAAA,IAAQ,CAAA;IAElB,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,QAAA,GAAA;;MACjC,MAAI,cAA2C,CAA1C,YAA0C,gBAAA,EAAlC,MAAK,4BAA0B,CAAA;;;4BAI7D,6CAFa,QAEb,GAAA;;;;IACA,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,QAAA,IAAA;;MACjC,MAAI,cAA4C,CAA3C,YAA2C,gBAAA,EAAnC,MAAK,6BAA2B,CAAA;;;4BAI9D,6CAFa,QAEb,GAAA;;;;IAEA,YAAqB,MAAA,SAAA,EAAA,EAAX,UAAA,IAAQ,CAAA;IAElB,YAOW,MAAA,SAAA,EAAA,MAAA;KANE,SAAO,cAGN,CAFV,YAEU,MAAA,QAAA,EAAA;MAFD,MAAK;MAAQ,YAAA;MAAY,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,QAAA;;MACjC,MAAI,cAA8B,CAA7B,YAA6B,gBAAA,EAArB,MAAK,eAAa,CAAA;;;4BAIhD,6CAFa,QAEb,GAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EGhBJ,MAAM,QAAQ;EAKd,MAAM,aAAa,IAAwB,KAAK;EAChD,MAAM,eAAe,IAAI,IAAI;EAE7B,MAAM,aAAa;GACjB;IAAE,OAAO;IAAQ,MAAM;IAAI;GAC3B;IAAE,OAAO;IAAQ,MAAM;IAAI;GAC3B;IAAE,OAAO;IAAQ,MAAM;IAAI;GAC5B;EAED,MAAM,kBAAkB,eAAe;GACrC,MAAM,IAAI,MAAM;AAChB,UAAO,KAAK,EAAE,OAAO,EAAE,IAAI,KAAK,EAAE,IAAI;IACtC;EAEF,MAAM,oBAAoB,eAAe;AACvC,OAAI,CAAC,gBAAgB,MAAO,QAAO,EAAE;GACrC,MAAM,EAAE,MAAM,MAAM;AAEpB,UAAO;IAAE,MADI,aAAa,QAAQ;IACnB,UAAU;IAAU;IACnC;EAEF,SAAS,sBAAsB,MAAc;AAC3C,OAAI,CAAC,gBAAgB,MAAO,QAAO,EAAE;GACrC,MAAM,EAAE,MAAM,MAAM;AACpB,UAAO;IAAE,MAAM,OAAO;IAAG,UAAU;IAAU;;EAG/C,IAAI,iBAAwC;AAE5C,kBAAgB;AACd,OAAI,WAAW,OAAO;AACpB,iBAAa,QAAQ,WAAW,MAAM;AACtC,qBAAiB,IAAI,gBAAgB,YAAY;AAC/C,iCAA4B;AAC1B,WAAK,MAAM,SAAS,QAClB,cAAa,QAAQ,MAAM,YAAY;OAEzC;MACF;AACF,mBAAe,QAAQ,WAAW,MAAM;;IAE1C;AAEF,wBAAsB;AACpB,mBAAgB,YAAY;IAC5B;;uBAhHA,mBAuDM,OAvDN,cAuDM;IAtDJ,mBAGM,OAHN,cAGM,CAFJ,YAAiC,gBAAA,EAAzB,MAAK,mBAAiB,CAAA,4BAC9B,mBAAiB,QAAA,MAAX,QAAI,GAAA;IAGZ,mBAsBM,OAAA;cArBA;KAAJ,KAAI;KACJ,OAAK,eAAA,CAAC,gBAAc,EAAA,0BACgBC,KAAAA,UAAQ,CAAA,CAAA;QAE5B,gBAAA,sBACd,mBAQM,OAAA;;KARD,OAAM;KAA0B,OAAK,eAAE,kBAAA,MAAiB;QAC3D,mBAMM,OAAA,EANA,OAAK,eAAEC,KAAAA,YAAY,IAAG,KAC1B,mBAIE,OAAA;KAHC,KAAKA,KAAAA,YAAY;KACjB,OAAK,eAAEA,KAAAA,YAAY,IAAG;KACvB,KAAI;0DAKZ,mBAKM,OALN,cAKM,CAJJ,YAGE,gBAAA;KAFA,MAAK;KACL,OAAA;MAAA,aAAA;MAAA,WAAA;MAAqC;;IAK3C,mBAwBM,OAxBN,cAwBM,eAvBJ,mBAsBM,UAAA,MAAA,WAtBc,aAAR,SAAI;YAAhB,mBAsBM,OAAA;MAtB2B,KAAK,KAAK;MAAO,OAAM;SACtD,mBAmBM,OAAA;MAlBJ,OAAK,eAAA,CAAC,sBAAoB,EAAA,gCACgBD,KAAAA,UAAQ,CAAA,CAAA;MACjD,OAAK,eAAA;OAAA,OAAA,GAAc,KAAK,KAAI;OAAA,QAAA,GAAiB,KAAK,KAAI;OAAA,CAAA;SAEvC,gBAAA,sBACd,mBAWM,OAAA;;MAVJ,OAAM;MACL,OAAK,eAAE,sBAAsB,KAAK,KAAI,CAAA;SAEvC,mBAMM,OAAA,EANA,OAAK,eAAEC,KAAAA,YAAY,IAAG,KAC1B,mBAIE,OAAA;MAHC,KAAKA,KAAAA,YAAY;MACjB,OAAK,eAAEA,KAAAA,YAAY,IAAG;MACvB,KAAI;oFAMd,mBAA0D,QAA1D,cAA0D,gBAApB,KAAK,MAAK,EAAA,EAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EGmHxD,MAAM,QAAQ;EAgBd,MAAM,OAAO;EAQb,MAAM,SAAS,IAAI,MAAM,IAAI;EAC7B,MAAM,eAAe,IAAI,MAAM,YAAY;EAC3C,MAAM,eAAe,IAAI,MAAM;EAC/B,MAAM,YAAY,IAAI,MAAM;EAC5B,MAAM,cAAc,IAAS,KAAK;EAElC,MAAM,mBAAmB,IAAS,KAAK;EACvC,MAAM,kBAAkB,IAAS,KAAK;EAEtC,MAAM,mBAAmB,eACvB,MAAM,QAAQ,gBAAgB,QAAQ,iBAAiB,MACxD;EAED,MAAM,kBAAkB,eACtB,OAAO,MAAM,WAAW,WAAW,GAAG,MAAM,OAAO,MAAM,MAAM,OAChE;EAED,MAAM,uBAAuB,eAAe;AAM1C,UAL8C;IAC5C,KAAK;IACL,MAAM;IACN,MAAM;IACP,CACU,MAAM,iBAAqC;IACtD;EAEF,MAAM,UAAU,eAAe,aAAa,QAAQ,EAAE;EAEtD,MAAM,cAAc,eAAe;AACjC,OAAI,aAAa,SAAS,EAAG,QAAO,CAAC,GAAG,EAAE;AAC1C,OAAI,aAAa,UAAU,EAAG,QAAO,CAAC,GAAG,EAAE;AAC3C,OAAI,KAAK,IAAI,aAAa,QAAQ,KAAK,EAAE,GAAG,IAAM,QAAO,CAAC,IAAI,EAAE;AAChE,OAAI,KAAK,IAAI,aAAa,QAAQ,IAAI,EAAE,GAAG,IAAM,QAAO,CAAC,GAAG,EAAE;AAC9D,OAAI,KAAK,IAAI,aAAa,QAAQ,IAAI,EAAE,GAAG,IAAM,QAAO,CAAC,GAAG,EAAE;AAC9D,UAAO,CAAC,KAAK,MAAM,aAAa,QAAQ,IAAI,EAAE,IAAI;IAClD;EAEF,MAAM,eAAe,gBAAgB;GACnC,OAAO;GACP,QAAQ,aAAa,QAAQ,IAAI,MAAM,aAAa,QAAQ;GAC7D,EAAE;EAEH,MAAM,eAAoC;GACxC;IAAE,OAAO;IAAM,OAAO;IAAG;GACzB;IAAE,OAAO;IAAO,OAAO;IAAG;GAC1B;IAAE,OAAO;IAAQ,OAAO,KAAK;IAAG;GAChC;IAAE,OAAO;IAAO,OAAO,IAAI;IAAG;GAC9B;IAAE,OAAO;IAAO,OAAO,IAAI;IAAG;GAC/B;EAOD,MAAM,OAAO,eAAe;GAC1B,QANgB,eAAe,MAAM,aAAiC;GAOtE,SANiB,eAAe,MAAM,cAAc;GAOpD,UANc,eAAe,MAAM,eAAe;GAOlD,WANc,eAAe,MAAM,gBAAgB;GAOpD,CAAC;AAEF,QAAM,mBAAmB,MAAM;AAC7B,QAAK,WAAW,QAAQ;IACxB;EAEF,SAAS,sBAAsB,MAAW;AACxC,eAAY,QAAQ;;EAEtB,SAAS,UAAU,QAAgB;AACjC,OAAI,WAAW,UAAW,MAAK,QAAQ;;EAEzC,SAAS,OAAO,OAAe;AAC7B,QAAK,OAAO,MAAM;;EAEpB,SAAS,YAAY,GAAW;AAC9B,gBAAa,QAAQ;;EAEvB,SAAS,cAAc;AACrB,QAAK,OAAO;;EAGd,eAAe,gBAAgB;AAC7B,aAAU,QAAQ;AAClB,OAAI;IACF,MAAM,SAAS,MAAM,KAAK,eAAe;AACzC,SAAK,WAAW,OAAO;AACvB,SAAK,QAAQ,OAAO;AACpB,iBAAa,QAAQ;aACb;AACR,cAAU,QAAQ;;;EAItB,SAAS,eAAe;AACtB,gBAAa,QAAQ;AACrB,QAAK,SAAS;;EAGhB,SAAS,SAAS,MAAY;GAC5B,MAAM,SAAS,IAAI,YAAY;AAC/B,UAAO,UAAU,MAAM;AACrB,WAAO,QAAQ,EAAE,QAAQ;;AAE3B,UAAO,cAAc,KAAK;;AAG5B,cACQ,MAAM,MACX,MAAM;AACL,UAAO,QAAQ;IAElB;AACD,cACQ,MAAM,cACX,MAAM;AACL,gBAAa,QAAQ;IAExB;AAED,WAAiC;GAC/B,eAAe,KAAK;GACpB,QAAQ,KAAK;GACb,MAAM,KAAK;GACX,OAAO,KAAK;GACZ,OAAO,KAAK;GACZ,OAAO,KAAK;GACZ,iBAAiB,MAAc;AAC7B,iBAAa,QAAQ;;GAEvB;GACA,OAAO,QAAiB;AACtB,QAAI,IAAK,QAAO,QAAQ;AACxB,iBAAa,QAAQ;;GAEvB,aAAa;AACX,iBAAa,QAAQ;;GAExB,CAAC;;UAjUQ,MAAM,sBADd,YA6ES,MAAA,OAAA,EAAA;;IA3EC,MAAM,aAAA;2DAAA,aAAY,QAAA;IAC1B,QAAO;IACN,OAAO,MAAM,cAAU;IACvB,OAAO,EAAA,OAAA,SAAkB;IACzB,iBAAe;IACf,UAAU;;IA8DA,QAAM,cAMN,CALT,YAKS,MAAA,OAAA,EAAA,EALD,SAAQ,OAAK,EAAA;4BACwB,CAA3C,YAA2C,MAAA,QAAA,EAAA,EAAjC,SAAO,cAAY,EAAA;6BAAI,OAAA,OAAA,OAAA,KAAA,iBAAF,MAAE,GAAA;;;SACjC,YAC2B,MAAA,QAAA,EAAA;MADlB,MAAK;MAAW,SAAS,UAAA;MAAY,SAAO;;6BAC9C,OAAA,OAAA,OAAA,KAAA,iBAAJ,QAAI,GAAA;;;;;;2BALL,CA3DN,mBA2DM,OA3DN,YA2DM,CAzDI,MAAM,4BADd,YAUE,wBAAA;;KARC,iBAAe,aAAA;KACf,iBAAe;KACf,SAAO;KACP,UAAQ;KACR,SAAQ,MAAA,KAAI,CAAC;KACb,SAAQ,MAAA,KAAI,CAAC;KACb,QAAM,MAAA,KAAI,CAAC;KACX,SAAO;;;;;;4CAEV,mBA8CM,OA9CN,YA8CM,CA7CJ,mBAmCM,OAAA;KAlCJ,OAAM;KACL,OAAK,eAAA,EAAA,QAAY,gBAAA,OAAe,CAAA;QAGzB,OAAA,sBADR,YAuBE,MAAA,WAAA,EAAA;;cArBI;KAAJ,KAAI;KACH,KAAK,OAAA;KACL,eAAa,MAAM;KACnB,eAAa,qBAAA;KACb,aAAW;KACX,aAAW;KACX,mBAAiB,aAAA,MAAa;KAC9B,oBAAkB,aAAA,MAAa;KAC/B,OAAO,QAAA;KACP,gBAAc,YAAA;KACd,cAAY;KACZ,MAAM;KACN,aAAW;KACX,YAAU;KACV,gBAAc;KACd,UAAU;KACV,MAAM;KACN,MAAM;KACN,MAAM;KACN,YAAW;KACD;;;;;;;;;wBAEb,mBAMM,OANN,YAMM,CALJ,YAGE,gBAAA;KAFA,MAAK;KACL,OAAA;MAAA,aAAA;MAAA,WAAA;MAAqC;kCAEvC,mBAAkB,QAAA,MAAZ,SAAK,GAAA,WAIP,MAAM,eAAe,OAAA,sBAD7B,mBAQM,OARN,YAQM,CAJJ,YAGE,wBAAA;KAFC,gBAAc,YAAA;KACd,UAAU,MAAM;;;6CAe3B,mBA6DM,OA7DN,YA6DM,CA5DJ,mBA2DM,OA3DN,YA2DM,CAzDI,MAAM,4BADd,YAUE,wBAAA;;IARC,iBAAe,aAAA;IACf,iBAAe;IACf,SAAO;IACP,UAAQ;IACR,SAAQ,MAAA,KAAI,CAAC;IACb,SAAQ,MAAA,KAAI,CAAC;IACb,QAAM,MAAA,KAAI,CAAC;IACX,SAAO;;;;;;2CAEV,mBA8CM,OA9CN,YA8CM,CA7CJ,mBAmCM,OAAA;IAlCJ,OAAM;IACL,OAAK,eAAA,EAAA,QAAY,gBAAA,OAAe,CAAA;OAGzB,OAAA,sBADR,YAuBE,MAAA,WAAA,EAAA;;aArBI;IAAJ,KAAI;IACH,KAAK,OAAA;IACL,eAAa,MAAM;IACnB,eAAa,qBAAA;IACb,aAAW;IACX,aAAW;IACX,mBAAiB,aAAA,MAAa;IAC9B,oBAAkB,aAAA,MAAa;IAC/B,OAAO,QAAA;IACP,gBAAc,YAAA;IACd,cAAY;IACZ,MAAM;IACN,aAAW;IACX,YAAU;IACV,gBAAc;IACd,UAAU;IACV,MAAM;IACN,MAAM;IACN,MAAM;IACN,YAAW;IACD;;;;;;;;;uBAEb,mBAMM,OANN,YAMM,CALJ,YAGE,gBAAA;IAFA,MAAK;IACL,OAAA;KAAA,aAAA;KAAA,WAAA;KAAqC;iCAEvC,mBAAkB,QAAA,MAAZ,SAAK,GAAA,WAIP,MAAM,eAAe,OAAA,sBAD7B,mBAQM,OARN,YAQM,CAJJ,YAGE,wBAAA;IAFC,gBAAc,YAAA;IACd,UAAU,MAAM"}
@@ -0,0 +1,4 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_C_Language = require('./C_Language.js');
3
+
4
+ exports.C_Language = require_C_Language.C_Language_default;
@@ -0,0 +1,2 @@
1
+ import { n as _default, t as LanguageOption } from "./index11.vue.js";
2
+ export { _default as C_Language, type LanguageOption };
@@ -0,0 +1,2 @@
1
+ import { n as _default, t as LanguageOption } from "./index11.vue.js";
2
+ export { _default as C_Language, type LanguageOption };
@@ -0,0 +1,3 @@
1
+ import { t as C_Language_default } from "./C_Language2.js";
2
+
3
+ export { C_Language_default as C_Language };
@@ -0,0 +1,72 @@
1
+ import { computed, createBlock, createElementVNode, createVNode, defineComponent, h, openBlock, unref, withCtx } from "vue";
2
+ import { NButton, NDropdown } from "naive-ui";
3
+
4
+ //#region src/components/C_Language/index.vue?vue&type=script&setup=true&lang.ts
5
+ var index_vue_vue_type_script_setup_true_lang_default = /* @__PURE__ */ defineComponent({
6
+ name: "C_Language",
7
+ __name: "index",
8
+ props: {
9
+ modelValue: { default: "zh-cn" },
10
+ options: { default: () => [
11
+ {
12
+ key: "zh-cn",
13
+ label: "简体中文",
14
+ iconClass: "i-mdi:alpha-c"
15
+ },
16
+ {
17
+ key: "en",
18
+ label: "English",
19
+ iconClass: "i-mdi:alpha-u"
20
+ },
21
+ {
22
+ key: "ja",
23
+ label: "日本語",
24
+ iconClass: "i-mdi:alpha-j"
25
+ },
26
+ {
27
+ key: "ko",
28
+ label: "한국어",
29
+ iconClass: "i-mdi:alpha-k"
30
+ }
31
+ ] }
32
+ },
33
+ emits: ["update:modelValue", "change"],
34
+ setup(__props, { emit: __emit }) {
35
+ const props = __props;
36
+ const emit = __emit;
37
+ const finalOptions = computed(() => props.options.map((opt) => ({
38
+ key: opt.key,
39
+ label: opt.label,
40
+ icon: opt.iconClass ? () => h("span", { class: opt.iconClass }) : void 0
41
+ })));
42
+ const handleLanguageChange = (key) => {
43
+ if (key === props.modelValue) return;
44
+ emit("update:modelValue", key);
45
+ emit("change", key);
46
+ };
47
+ return (_ctx, _cache) => {
48
+ return openBlock(), createBlock(unref(NDropdown), {
49
+ size: "small",
50
+ trigger: "hover",
51
+ options: finalOptions.value,
52
+ value: _ctx.modelValue,
53
+ onSelect: handleLanguageChange
54
+ }, {
55
+ default: withCtx(() => [createVNode(unref(NButton), { text: "" }, {
56
+ default: withCtx(() => _cache[0] || (_cache[0] = [createElementVNode("div", { class: "flex items-center" }, [createElementVNode("span", { class: "i-mdi:language" })], -1)])),
57
+ _: 1,
58
+ __: [0]
59
+ })]),
60
+ _: 1
61
+ }, 8, ["options", "value"]);
62
+ };
63
+ }
64
+ });
65
+
66
+ //#endregion
67
+ //#region src/components/C_Language/index.vue
68
+ var C_Language_default = index_vue_vue_type_script_setup_true_lang_default;
69
+
70
+ //#endregion
71
+ export { C_Language_default as t };
72
+ //# sourceMappingURL=C_Language2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_Language2.js","names":["modelValue"],"sources":["../src/components/C_Language/index.vue","../src/components/C_Language/index.vue","../src/components/C_Language/index.vue"],"sourcesContent":["<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-11-19\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 <NDropdown\r\n size=\"small\"\r\n trigger=\"hover\"\r\n :options=\"finalOptions\"\r\n :value=\"modelValue\"\r\n @select=\"handleLanguageChange\"\r\n >\r\n <NButton text>\r\n <div class=\"flex items-center\">\r\n <span class=\"i-mdi:language\"></span>\r\n </div>\r\n </NButton>\r\n </NDropdown>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { computed, h } from \"vue\";\r\nimport { NDropdown, NButton } from \"naive-ui\";\r\n\r\ndefineOptions({ name: \"C_Language\" });\r\n\r\nexport interface LanguageOption {\r\n key: string;\r\n label: string;\r\n iconClass?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n modelValue?: string;\r\n options?: LanguageOption[];\r\n }>(),\r\n {\r\n modelValue: \"zh-cn\",\r\n options: () => [\r\n { key: \"zh-cn\", label: \"简体中文\", iconClass: \"i-mdi:alpha-c\" },\r\n { key: \"en\", label: \"English\", iconClass: \"i-mdi:alpha-u\" },\r\n { key: \"ja\", label: \"日本語\", iconClass: \"i-mdi:alpha-j\" },\r\n { key: \"ko\", label: \"한국어\", iconClass: \"i-mdi:alpha-k\" },\r\n ],\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n \"update:modelValue\": [key: string];\r\n change: [key: string];\r\n}>();\r\n\r\nconst finalOptions = computed(() =>\r\n props.options.map((opt) => ({\r\n key: opt.key,\r\n label: opt.label,\r\n icon: opt.iconClass ? () => h(\"span\", { class: opt.iconClass }) : undefined,\r\n })),\r\n);\r\n\r\nconst handleLanguageChange = (key: string) => {\r\n if (key === props.modelValue) return;\r\n emit(\"update:modelValue\", key);\r\n emit(\"change\", key);\r\n};\r\n</script>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-11-19\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 <NDropdown\r\n size=\"small\"\r\n trigger=\"hover\"\r\n :options=\"finalOptions\"\r\n :value=\"modelValue\"\r\n @select=\"handleLanguageChange\"\r\n >\r\n <NButton text>\r\n <div class=\"flex items-center\">\r\n <span class=\"i-mdi:language\"></span>\r\n </div>\r\n </NButton>\r\n </NDropdown>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { computed, h } from \"vue\";\r\nimport { NDropdown, NButton } from \"naive-ui\";\r\n\r\ndefineOptions({ name: \"C_Language\" });\r\n\r\nexport interface LanguageOption {\r\n key: string;\r\n label: string;\r\n iconClass?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n modelValue?: string;\r\n options?: LanguageOption[];\r\n }>(),\r\n {\r\n modelValue: \"zh-cn\",\r\n options: () => [\r\n { key: \"zh-cn\", label: \"简体中文\", iconClass: \"i-mdi:alpha-c\" },\r\n { key: \"en\", label: \"English\", iconClass: \"i-mdi:alpha-u\" },\r\n { key: \"ja\", label: \"日本語\", iconClass: \"i-mdi:alpha-j\" },\r\n { key: \"ko\", label: \"한국어\", iconClass: \"i-mdi:alpha-k\" },\r\n ],\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n \"update:modelValue\": [key: string];\r\n change: [key: string];\r\n}>();\r\n\r\nconst finalOptions = computed(() =>\r\n props.options.map((opt) => ({\r\n key: opt.key,\r\n label: opt.label,\r\n icon: opt.iconClass ? () => h(\"span\", { class: opt.iconClass }) : undefined,\r\n })),\r\n);\r\n\r\nconst handleLanguageChange = (key: string) => {\r\n if (key === props.modelValue) return;\r\n emit(\"update:modelValue\", key);\r\n emit(\"change\", key);\r\n};\r\n</script>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-11-19\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 <NDropdown\r\n size=\"small\"\r\n trigger=\"hover\"\r\n :options=\"finalOptions\"\r\n :value=\"modelValue\"\r\n @select=\"handleLanguageChange\"\r\n >\r\n <NButton text>\r\n <div class=\"flex items-center\">\r\n <span class=\"i-mdi:language\"></span>\r\n </div>\r\n </NButton>\r\n </NDropdown>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { computed, h } from \"vue\";\r\nimport { NDropdown, NButton } from \"naive-ui\";\r\n\r\ndefineOptions({ name: \"C_Language\" });\r\n\r\nexport interface LanguageOption {\r\n key: string;\r\n label: string;\r\n iconClass?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n modelValue?: string;\r\n options?: LanguageOption[];\r\n }>(),\r\n {\r\n modelValue: \"zh-cn\",\r\n options: () => [\r\n { key: \"zh-cn\", label: \"简体中文\", iconClass: \"i-mdi:alpha-c\" },\r\n { key: \"en\", label: \"English\", iconClass: \"i-mdi:alpha-u\" },\r\n { key: \"ja\", label: \"日本語\", iconClass: \"i-mdi:alpha-j\" },\r\n { key: \"ko\", label: \"한국어\", iconClass: \"i-mdi:alpha-k\" },\r\n ],\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n \"update:modelValue\": [key: string];\r\n change: [key: string];\r\n}>();\r\n\r\nconst finalOptions = computed(() =>\r\n props.options.map((opt) => ({\r\n key: opt.key,\r\n label: opt.label,\r\n icon: opt.iconClass ? () => h(\"span\", { class: opt.iconClass }) : undefined,\r\n })),\r\n);\r\n\r\nconst handleLanguageChange = (key: string) => {\r\n if (key === props.modelValue) return;\r\n emit(\"update:modelValue\", key);\r\n emit(\"change\", key);\r\n};\r\n</script>\r\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECmCA,MAAM,QAAQ;EAgBd,MAAM,OAAO;EAKb,MAAM,eAAe,eACnB,MAAM,QAAQ,KAAK,SAAS;GAC1B,KAAK,IAAI;GACT,OAAO,IAAI;GACX,MAAM,IAAI,kBAAkB,EAAE,QAAQ,EAAE,OAAO,IAAI,WAAW,CAAC,GAAG;GACnE,EAAE,CACJ;EAED,MAAM,wBAAwB,QAAgB;AAC5C,OAAI,QAAQ,MAAM,WAAY;AAC9B,QAAK,qBAAqB,IAAI;AAC9B,QAAK,UAAU,IAAI;;;uBA3DnB,YAYY,MAAA,UAAA,EAAA;IAXV,MAAK;IACL,SAAQ;IACP,SAAS,aAAA;IACT,OAAOA,KAAAA;IACP,UAAQ;;2BAMC,CAJV,YAIU,MAAA,QAAA,EAAA,EAJD,MAAA,IAAI,EAAA;4BAGL,OAAA,OAAA,OAAA,KAAA,CAFN,mBAEM,OAAA,EAFD,OAAM,qBAAmB,EAAA,CAC5B,mBAAoC,QAAA,EAA9B,OAAM,kBAAgB,CAAA"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_Map-DpzeuWdX.css","names":[],"sources":["../src/components/C_Map/index.vue?vue&type=style&index=0&scoped=9017d3a8&lang.scss"],"sourcesContent":[".c-map[data-v-9017d3a8] {\n position: relative;\n width: 100%;\n min-height: 300px;\n background: #f5f5f5;\n border-radius: 4px;\n overflow: hidden;\n}\n.c-map .map-container[data-v-9017d3a8] {\n width: 100%;\n height: 100%;\n border-radius: 4px;\n overflow: hidden;\n}\n.c-map .map-loading[data-v-9017d3a8] {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n display: flex;\n align-items: center;\n justify-content: center;\n background: rgba(255, 255, 255, 0.9);\n border-radius: 4px;\n z-index: 1000;\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"}
package/dist/C_Map.cjs ADDED
@@ -0,0 +1,7 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_C_Map = require('./C_Map.js');
3
+
4
+ exports.C_Map = require_C_Map.C_Map_default;
5
+ exports.DEFAULT_MAP_CONFIG = require_C_Map.DEFAULT_MAP_CONFIG;
6
+ exports.MAP_TYPES = require_C_Map.MAP_TYPES;
7
+ exports.OSM_TILE_CONFIG = require_C_Map.OSM_TILE_CONFIG;
@@ -0,0 +1,2 @@
1
+ import { a as MapType, i as MapMarker, n as MAP_TYPES, o as OSM_TILE_CONFIG, r as MapConfig, s as _default, t as DEFAULT_MAP_CONFIG } from "./data.js";
2
+ export { _default as C_Map, DEFAULT_MAP_CONFIG, MAP_TYPES, type MapConfig, type MapMarker, type MapType, OSM_TILE_CONFIG };
@@ -0,0 +1,2 @@
1
+ import { a as MapType, i as MapMarker, n as MAP_TYPES, o as OSM_TILE_CONFIG, r as MapConfig, s as _default, t as DEFAULT_MAP_CONFIG } from "./data.js";
2
+ export { _default as C_Map, DEFAULT_MAP_CONFIG, MAP_TYPES, type MapConfig, type MapMarker, type MapType, OSM_TILE_CONFIG };
package/dist/C_Map.js ADDED
@@ -0,0 +1,3 @@
1
+ import { i as OSM_TILE_CONFIG, n as DEFAULT_MAP_CONFIG, r as MAP_TYPES, t as C_Map_default } from "./C_Map2.js";
2
+
3
+ export { C_Map_default as C_Map, DEFAULT_MAP_CONFIG, MAP_TYPES, OSM_TILE_CONFIG };
package/dist/C_Map2.js ADDED
@@ -0,0 +1,199 @@
1
+ import { t as export_helper_default } from "./export-helper.js";
2
+ import { createCommentVNode, createElementBlock, createElementVNode, createVNode, defineComponent, nextTick, normalizeStyle, onMounted, onUnmounted, openBlock, ref, unref, watch } from "vue";
3
+ import { NSpin } from "naive-ui";
4
+ import L from "leaflet";
5
+ import "leaflet/dist/leaflet.css";
6
+
7
+ //#region src/components/C_Map/data.ts
8
+ const MAP_TYPES = [{
9
+ label: "OpenStreetMap",
10
+ value: "osm"
11
+ }, {
12
+ label: "高德地图",
13
+ value: "amap"
14
+ }];
15
+ const DEFAULT_MAP_CONFIG = {
16
+ height: "400px",
17
+ center: [39.9042, 116.4074],
18
+ zoom: 10,
19
+ mapType: "osm"
20
+ };
21
+ const OSM_TILE_CONFIG = {
22
+ url: "https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png",
23
+ attribution: "© OpenStreetMap contributors",
24
+ maxZoom: 19,
25
+ minZoom: 1,
26
+ tileSize: 256,
27
+ detectRetina: true
28
+ };
29
+ const AMAP_CONFIG = {
30
+ apiUrl: "https://webapi.amap.com/maps?v=2.0&key=",
31
+ note: "高德地图需要API Key,如需使用请申请:https://lbs.amap.com/api/javascript-api/guide/create/"
32
+ };
33
+ const MAP_ICONS = {
34
+ iconRetinaUrl: "https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon-2x.png",
35
+ iconUrl: "https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon.png",
36
+ shadowUrl: "https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-shadow.png"
37
+ };
38
+
39
+ //#endregion
40
+ //#region src/components/C_Map/index.vue?vue&type=script&setup=true&lang.ts
41
+ const _hoisted_1 = { class: "c-map" };
42
+ const _hoisted_2 = {
43
+ key: 0,
44
+ class: "map-loading"
45
+ };
46
+ var index_vue_vue_type_script_setup_true_lang_default = /* @__PURE__ */ defineComponent({
47
+ name: "C_Map",
48
+ __name: "index",
49
+ props: {
50
+ height: { default: "400px" },
51
+ center: { default: () => [39.9042, 116.4074] },
52
+ zoom: { default: 10 },
53
+ markers: { default: () => [] },
54
+ mapType: { default: "osm" },
55
+ amapKey: { default: "" }
56
+ },
57
+ emits: ["ready", "markerClick"],
58
+ setup(__props, { emit: __emit }) {
59
+ delete L.Icon.Default.prototype._getIconUrl;
60
+ L.Icon.Default.mergeOptions(MAP_ICONS);
61
+ const props = __props;
62
+ const emit = __emit;
63
+ const mapContainer = ref();
64
+ const loading = ref(true);
65
+ let map = null;
66
+ const initOSMMap = async () => {
67
+ if (!mapContainer.value) return;
68
+ try {
69
+ mapContainer.value.innerHTML = "";
70
+ map = L.map(mapContainer.value, {
71
+ center: props.center,
72
+ zoom: props.zoom,
73
+ zoomControl: true,
74
+ preferCanvas: true
75
+ });
76
+ L.tileLayer(OSM_TILE_CONFIG.url, OSM_TILE_CONFIG).addTo(map);
77
+ addMarkers();
78
+ await nextTick();
79
+ requestAnimationFrame(() => {
80
+ map?.invalidateSize({
81
+ reset: true,
82
+ pan: false
83
+ });
84
+ });
85
+ loading.value = false;
86
+ emit("ready", map);
87
+ } catch (error) {
88
+ console.error("OpenStreetMap初始化失败:", error);
89
+ loading.value = false;
90
+ }
91
+ };
92
+ const initAMap = async () => {
93
+ if (!mapContainer.value || !props.amapKey) return;
94
+ try {
95
+ mapContainer.value.innerHTML = "";
96
+ const script = document.createElement("script");
97
+ script.type = "text/javascript";
98
+ script.src = `${AMAP_CONFIG.apiUrl}${props.amapKey}`;
99
+ script.onload = () => {
100
+ if (window.AMap) {
101
+ const amap = new window.AMap.Map(mapContainer.value, {
102
+ zoom: props.zoom,
103
+ center: props.center
104
+ });
105
+ addAMapMarkers(amap);
106
+ loading.value = false;
107
+ emit("ready", amap);
108
+ }
109
+ };
110
+ script.onerror = () => {
111
+ loading.value = false;
112
+ };
113
+ document.head.appendChild(script);
114
+ } catch {
115
+ loading.value = false;
116
+ }
117
+ };
118
+ const addMarkers = () => {
119
+ if (!map || props.mapType !== "osm" || !props.markers) return;
120
+ map.eachLayer((layer) => {
121
+ if (layer instanceof L.Marker) map.removeLayer(layer);
122
+ });
123
+ props.markers.forEach((marker) => {
124
+ const leafletMarker = L.marker([marker.lat, marker.lng]);
125
+ if (marker.popup) {
126
+ leafletMarker.bindPopup(marker.popup);
127
+ leafletMarker.on("click", (event) => {
128
+ emit("markerClick", marker, event);
129
+ });
130
+ }
131
+ leafletMarker.addTo(map);
132
+ });
133
+ };
134
+ const addAMapMarkers = (amap) => {
135
+ if (!amap || props.mapType !== "amap" || !props.markers) return;
136
+ props.markers.forEach((marker) => {
137
+ const amapMarker = new window.AMap.Marker({
138
+ position: [marker.lat, marker.lng],
139
+ title: marker.popup || ""
140
+ });
141
+ if (marker.popup) {
142
+ const infoWindow = new window.AMap.InfoWindow({
143
+ content: marker.popup,
144
+ offset: new window.AMap.Pixel(0, -30)
145
+ });
146
+ amapMarker.on("click", () => {
147
+ infoWindow.open(amap, amapMarker.getPosition());
148
+ emit("markerClick", marker, null);
149
+ });
150
+ }
151
+ amapMarker.setMap(amap);
152
+ });
153
+ };
154
+ watch(() => props.markers, () => {
155
+ if (map && props.mapType === "osm") addMarkers();
156
+ }, { deep: true });
157
+ watch(() => props.mapType, async (newType, oldType) => {
158
+ if (newType === oldType) return;
159
+ if (map) {
160
+ map.remove();
161
+ map = null;
162
+ loading.value = true;
163
+ await nextTick();
164
+ if (newType === "amap" && props.amapKey) await initAMap();
165
+ else await initOSMMap();
166
+ }
167
+ });
168
+ watch([() => props.center, () => props.zoom], () => {
169
+ if (map) map.setView(props.center, props.zoom);
170
+ }, { deep: true });
171
+ onMounted(async () => {
172
+ await nextTick();
173
+ if (props.mapType === "amap" && props.amapKey) await initAMap();
174
+ else await initOSMMap();
175
+ });
176
+ onUnmounted(() => {
177
+ if (map) {
178
+ map.remove();
179
+ map = null;
180
+ }
181
+ });
182
+ return (_ctx, _cache) => {
183
+ return openBlock(), createElementBlock("div", _hoisted_1, [createElementVNode("div", {
184
+ ref_key: "mapContainer",
185
+ ref: mapContainer,
186
+ class: "map-container",
187
+ style: normalizeStyle({ height: _ctx.height })
188
+ }, null, 4), loading.value ? (openBlock(), createElementBlock("div", _hoisted_2, [createVNode(unref(NSpin), { size: "large" })])) : createCommentVNode("v-if", true)]);
189
+ };
190
+ }
191
+ });
192
+
193
+ //#endregion
194
+ //#region src/components/C_Map/index.vue
195
+ var C_Map_default = /* @__PURE__ */ export_helper_default(index_vue_vue_type_script_setup_true_lang_default, [["__scopeId", "data-v-9017d3a8"]]);
196
+
197
+ //#endregion
198
+ export { OSM_TILE_CONFIG as i, DEFAULT_MAP_CONFIG as n, MAP_TYPES as r, C_Map_default as t };
199
+ //# sourceMappingURL=C_Map2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_Map2.js","names":["height"],"sources":["../src/components/C_Map/data.ts","../src/components/C_Map/index.vue","../src/components/C_Map/index.vue","../src/components/C_Map/index.vue"],"sourcesContent":["export const MAP_TYPES = [\r\n { label: \"OpenStreetMap\", value: \"osm\" },\r\n { label: \"高德地图\", value: \"amap\" },\r\n] as const;\r\n\r\nexport type MapType = (typeof MAP_TYPES)[number][\"value\"];\r\n\r\nexport interface MapMarker {\r\n lat: number;\r\n lng: number;\r\n popup?: string;\r\n}\r\n\r\nexport interface MapConfig {\r\n height?: string;\r\n center?: [number, number];\r\n zoom?: number;\r\n markers?: MapMarker[];\r\n mapType?: MapType;\r\n amapKey?: string;\r\n}\r\n\r\nexport const DEFAULT_MAP_CONFIG: Required<\r\n Omit<MapConfig, \"markers\" | \"amapKey\">\r\n> = {\r\n height: \"400px\",\r\n center: [39.9042, 116.4074],\r\n zoom: 10,\r\n mapType: \"osm\",\r\n};\r\n\r\nexport const OSM_TILE_CONFIG = {\r\n url: \"https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png\",\r\n attribution: \"© OpenStreetMap contributors\",\r\n maxZoom: 19,\r\n minZoom: 1,\r\n tileSize: 256,\r\n detectRetina: true,\r\n} as const;\r\n\r\nexport const AMAP_CONFIG = {\r\n apiUrl: \"https://webapi.amap.com/maps?v=2.0&key=\",\r\n note: \"高德地图需要API Key,如需使用请申请:https://lbs.amap.com/api/javascript-api/guide/create/\",\r\n} as const;\r\n\r\nexport const MAP_ICONS = {\r\n iconRetinaUrl:\r\n \"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon-2x.png\",\r\n iconUrl:\r\n \"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-icon.png\",\r\n shadowUrl:\r\n \"https://cdnjs.cloudflare.com/ajax/libs/leaflet/1.9.4/images/marker-shadow.png\",\r\n} as const;\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-12-02\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 class=\"c-map\">\r\n <div\r\n ref=\"mapContainer\"\r\n class=\"map-container\"\r\n :style=\"{ height: height }\"\r\n ></div>\r\n <div v-if=\"loading\" class=\"map-loading\">\r\n <NSpin size=\"large\" />\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport L from \"leaflet\";\r\nimport \"leaflet/dist/leaflet.css\";\r\nimport { OSM_TILE_CONFIG, AMAP_CONFIG, MAP_ICONS } from \"./data\";\r\n\r\ndefineOptions({ name: \"C_Map\" });\r\n\r\ndelete (L.Icon.Default.prototype as any)._getIconUrl;\r\nL.Icon.Default.mergeOptions(MAP_ICONS);\r\n\r\ninterface MapMarker {\r\n lat: number;\r\n lng: number;\r\n popup?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n height?: string;\r\n center?: [number, number];\r\n zoom?: number;\r\n markers?: MapMarker[];\r\n mapType?: \"osm\" | \"amap\";\r\n amapKey?: string;\r\n }>(),\r\n {\r\n height: \"400px\",\r\n center: () => [39.9042, 116.4074],\r\n zoom: 10,\r\n markers: () => [],\r\n mapType: \"osm\",\r\n amapKey: \"\",\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n ready: [map: any];\r\n markerClick: [marker: MapMarker, event: any];\r\n}>();\r\n\r\nconst mapContainer = ref<HTMLElement>();\r\nconst loading = ref(true);\r\nlet map: any = null;\r\n\r\nconst initOSMMap = async () => {\r\n if (!mapContainer.value) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n map = L.map(mapContainer.value, {\r\n center: props.center,\r\n zoom: props.zoom,\r\n zoomControl: true,\r\n preferCanvas: true,\r\n });\r\n const tileLayer = L.tileLayer(OSM_TILE_CONFIG.url, OSM_TILE_CONFIG);\r\n tileLayer.addTo(map);\r\n addMarkers();\r\n await nextTick();\r\n requestAnimationFrame(() => {\r\n map?.invalidateSize({ reset: true, pan: false });\r\n });\r\n loading.value = false;\r\n emit(\"ready\", map);\r\n } catch (error) {\r\n console.error(\"OpenStreetMap初始化失败:\", error);\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst initAMap = async () => {\r\n if (!mapContainer.value || !props.amapKey) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n const script = document.createElement(\"script\");\r\n script.type = \"text/javascript\";\r\n script.src = `${AMAP_CONFIG.apiUrl}${props.amapKey}`;\r\n script.onload = () => {\r\n if ((window as any).AMap) {\r\n const amap = new (window as any).AMap.Map(mapContainer.value, {\r\n zoom: props.zoom,\r\n center: props.center,\r\n });\r\n addAMapMarkers(amap);\r\n loading.value = false;\r\n emit(\"ready\", amap);\r\n }\r\n };\r\n script.onerror = () => {\r\n loading.value = false;\r\n };\r\n document.head.appendChild(script);\r\n } catch {\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst addMarkers = () => {\r\n if (!map || props.mapType !== \"osm\" || !props.markers) return;\r\n map.eachLayer((layer: any) => {\r\n if (layer instanceof L.Marker) {\r\n map.removeLayer(layer);\r\n }\r\n });\r\n props.markers.forEach((marker) => {\r\n const leafletMarker = L.marker([marker.lat, marker.lng]);\r\n if (marker.popup) {\r\n leafletMarker.bindPopup(marker.popup);\r\n leafletMarker.on(\"click\", (event: any) => {\r\n emit(\"markerClick\", marker, event);\r\n });\r\n }\r\n leafletMarker.addTo(map);\r\n });\r\n};\r\n\r\nconst addAMapMarkers = (amap: any) => {\r\n if (!amap || props.mapType !== \"amap\" || !props.markers) return;\r\n props.markers.forEach((marker) => {\r\n const amapMarker = new (window as any).AMap.Marker({\r\n position: [marker.lat, marker.lng],\r\n title: marker.popup || \"\",\r\n });\r\n if (marker.popup) {\r\n const infoWindow = new (window as any).AMap.InfoWindow({\r\n content: marker.popup,\r\n offset: new (window as any).AMap.Pixel(0, -30),\r\n });\r\n amapMarker.on(\"click\", () => {\r\n infoWindow.open(amap, amapMarker.getPosition());\r\n emit(\"markerClick\", marker, null);\r\n });\r\n }\r\n amapMarker.setMap(amap);\r\n });\r\n};\r\n\r\nwatch(\r\n () => props.markers,\r\n () => {\r\n if (map && props.mapType === \"osm\") addMarkers();\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.mapType,\r\n async (newType, oldType) => {\r\n if (newType === oldType) return;\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n loading.value = true;\r\n await nextTick();\r\n if (newType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n }\r\n },\r\n);\r\n\r\nwatch(\r\n [() => props.center, () => props.zoom],\r\n () => {\r\n if (map) map.setView(props.center, props.zoom);\r\n },\r\n { deep: true },\r\n);\r\n\r\nonMounted(async () => {\r\n await nextTick();\r\n if (props.mapType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n});\r\n\r\nonUnmounted(() => {\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n }\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-12-02\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 class=\"c-map\">\r\n <div\r\n ref=\"mapContainer\"\r\n class=\"map-container\"\r\n :style=\"{ height: height }\"\r\n ></div>\r\n <div v-if=\"loading\" class=\"map-loading\">\r\n <NSpin size=\"large\" />\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport L from \"leaflet\";\r\nimport \"leaflet/dist/leaflet.css\";\r\nimport { OSM_TILE_CONFIG, AMAP_CONFIG, MAP_ICONS } from \"./data\";\r\n\r\ndefineOptions({ name: \"C_Map\" });\r\n\r\ndelete (L.Icon.Default.prototype as any)._getIconUrl;\r\nL.Icon.Default.mergeOptions(MAP_ICONS);\r\n\r\ninterface MapMarker {\r\n lat: number;\r\n lng: number;\r\n popup?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n height?: string;\r\n center?: [number, number];\r\n zoom?: number;\r\n markers?: MapMarker[];\r\n mapType?: \"osm\" | \"amap\";\r\n amapKey?: string;\r\n }>(),\r\n {\r\n height: \"400px\",\r\n center: () => [39.9042, 116.4074],\r\n zoom: 10,\r\n markers: () => [],\r\n mapType: \"osm\",\r\n amapKey: \"\",\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n ready: [map: any];\r\n markerClick: [marker: MapMarker, event: any];\r\n}>();\r\n\r\nconst mapContainer = ref<HTMLElement>();\r\nconst loading = ref(true);\r\nlet map: any = null;\r\n\r\nconst initOSMMap = async () => {\r\n if (!mapContainer.value) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n map = L.map(mapContainer.value, {\r\n center: props.center,\r\n zoom: props.zoom,\r\n zoomControl: true,\r\n preferCanvas: true,\r\n });\r\n const tileLayer = L.tileLayer(OSM_TILE_CONFIG.url, OSM_TILE_CONFIG);\r\n tileLayer.addTo(map);\r\n addMarkers();\r\n await nextTick();\r\n requestAnimationFrame(() => {\r\n map?.invalidateSize({ reset: true, pan: false });\r\n });\r\n loading.value = false;\r\n emit(\"ready\", map);\r\n } catch (error) {\r\n console.error(\"OpenStreetMap初始化失败:\", error);\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst initAMap = async () => {\r\n if (!mapContainer.value || !props.amapKey) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n const script = document.createElement(\"script\");\r\n script.type = \"text/javascript\";\r\n script.src = `${AMAP_CONFIG.apiUrl}${props.amapKey}`;\r\n script.onload = () => {\r\n if ((window as any).AMap) {\r\n const amap = new (window as any).AMap.Map(mapContainer.value, {\r\n zoom: props.zoom,\r\n center: props.center,\r\n });\r\n addAMapMarkers(amap);\r\n loading.value = false;\r\n emit(\"ready\", amap);\r\n }\r\n };\r\n script.onerror = () => {\r\n loading.value = false;\r\n };\r\n document.head.appendChild(script);\r\n } catch {\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst addMarkers = () => {\r\n if (!map || props.mapType !== \"osm\" || !props.markers) return;\r\n map.eachLayer((layer: any) => {\r\n if (layer instanceof L.Marker) {\r\n map.removeLayer(layer);\r\n }\r\n });\r\n props.markers.forEach((marker) => {\r\n const leafletMarker = L.marker([marker.lat, marker.lng]);\r\n if (marker.popup) {\r\n leafletMarker.bindPopup(marker.popup);\r\n leafletMarker.on(\"click\", (event: any) => {\r\n emit(\"markerClick\", marker, event);\r\n });\r\n }\r\n leafletMarker.addTo(map);\r\n });\r\n};\r\n\r\nconst addAMapMarkers = (amap: any) => {\r\n if (!amap || props.mapType !== \"amap\" || !props.markers) return;\r\n props.markers.forEach((marker) => {\r\n const amapMarker = new (window as any).AMap.Marker({\r\n position: [marker.lat, marker.lng],\r\n title: marker.popup || \"\",\r\n });\r\n if (marker.popup) {\r\n const infoWindow = new (window as any).AMap.InfoWindow({\r\n content: marker.popup,\r\n offset: new (window as any).AMap.Pixel(0, -30),\r\n });\r\n amapMarker.on(\"click\", () => {\r\n infoWindow.open(amap, amapMarker.getPosition());\r\n emit(\"markerClick\", marker, null);\r\n });\r\n }\r\n amapMarker.setMap(amap);\r\n });\r\n};\r\n\r\nwatch(\r\n () => props.markers,\r\n () => {\r\n if (map && props.mapType === \"osm\") addMarkers();\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.mapType,\r\n async (newType, oldType) => {\r\n if (newType === oldType) return;\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n loading.value = true;\r\n await nextTick();\r\n if (newType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n }\r\n },\r\n);\r\n\r\nwatch(\r\n [() => props.center, () => props.zoom],\r\n () => {\r\n if (map) map.setView(props.center, props.zoom);\r\n },\r\n { deep: true },\r\n);\r\n\r\nonMounted(async () => {\r\n await nextTick();\r\n if (props.mapType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n});\r\n\r\nonUnmounted(() => {\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n }\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-12-02\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 class=\"c-map\">\r\n <div\r\n ref=\"mapContainer\"\r\n class=\"map-container\"\r\n :style=\"{ height: height }\"\r\n ></div>\r\n <div v-if=\"loading\" class=\"map-loading\">\r\n <NSpin size=\"large\" />\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script setup lang=\"ts\">\r\nimport { ref, computed, watch, nextTick, onMounted, onUnmounted } from \"vue\";\r\nimport { NSpin } from \"naive-ui\";\r\nimport L from \"leaflet\";\r\nimport \"leaflet/dist/leaflet.css\";\r\nimport { OSM_TILE_CONFIG, AMAP_CONFIG, MAP_ICONS } from \"./data\";\r\n\r\ndefineOptions({ name: \"C_Map\" });\r\n\r\ndelete (L.Icon.Default.prototype as any)._getIconUrl;\r\nL.Icon.Default.mergeOptions(MAP_ICONS);\r\n\r\ninterface MapMarker {\r\n lat: number;\r\n lng: number;\r\n popup?: string;\r\n}\r\n\r\nconst props = withDefaults(\r\n defineProps<{\r\n height?: string;\r\n center?: [number, number];\r\n zoom?: number;\r\n markers?: MapMarker[];\r\n mapType?: \"osm\" | \"amap\";\r\n amapKey?: string;\r\n }>(),\r\n {\r\n height: \"400px\",\r\n center: () => [39.9042, 116.4074],\r\n zoom: 10,\r\n markers: () => [],\r\n mapType: \"osm\",\r\n amapKey: \"\",\r\n },\r\n);\r\n\r\nconst emit = defineEmits<{\r\n ready: [map: any];\r\n markerClick: [marker: MapMarker, event: any];\r\n}>();\r\n\r\nconst mapContainer = ref<HTMLElement>();\r\nconst loading = ref(true);\r\nlet map: any = null;\r\n\r\nconst initOSMMap = async () => {\r\n if (!mapContainer.value) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n map = L.map(mapContainer.value, {\r\n center: props.center,\r\n zoom: props.zoom,\r\n zoomControl: true,\r\n preferCanvas: true,\r\n });\r\n const tileLayer = L.tileLayer(OSM_TILE_CONFIG.url, OSM_TILE_CONFIG);\r\n tileLayer.addTo(map);\r\n addMarkers();\r\n await nextTick();\r\n requestAnimationFrame(() => {\r\n map?.invalidateSize({ reset: true, pan: false });\r\n });\r\n loading.value = false;\r\n emit(\"ready\", map);\r\n } catch (error) {\r\n console.error(\"OpenStreetMap初始化失败:\", error);\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst initAMap = async () => {\r\n if (!mapContainer.value || !props.amapKey) return;\r\n try {\r\n mapContainer.value.innerHTML = \"\";\r\n const script = document.createElement(\"script\");\r\n script.type = \"text/javascript\";\r\n script.src = `${AMAP_CONFIG.apiUrl}${props.amapKey}`;\r\n script.onload = () => {\r\n if ((window as any).AMap) {\r\n const amap = new (window as any).AMap.Map(mapContainer.value, {\r\n zoom: props.zoom,\r\n center: props.center,\r\n });\r\n addAMapMarkers(amap);\r\n loading.value = false;\r\n emit(\"ready\", amap);\r\n }\r\n };\r\n script.onerror = () => {\r\n loading.value = false;\r\n };\r\n document.head.appendChild(script);\r\n } catch {\r\n loading.value = false;\r\n }\r\n};\r\n\r\nconst addMarkers = () => {\r\n if (!map || props.mapType !== \"osm\" || !props.markers) return;\r\n map.eachLayer((layer: any) => {\r\n if (layer instanceof L.Marker) {\r\n map.removeLayer(layer);\r\n }\r\n });\r\n props.markers.forEach((marker) => {\r\n const leafletMarker = L.marker([marker.lat, marker.lng]);\r\n if (marker.popup) {\r\n leafletMarker.bindPopup(marker.popup);\r\n leafletMarker.on(\"click\", (event: any) => {\r\n emit(\"markerClick\", marker, event);\r\n });\r\n }\r\n leafletMarker.addTo(map);\r\n });\r\n};\r\n\r\nconst addAMapMarkers = (amap: any) => {\r\n if (!amap || props.mapType !== \"amap\" || !props.markers) return;\r\n props.markers.forEach((marker) => {\r\n const amapMarker = new (window as any).AMap.Marker({\r\n position: [marker.lat, marker.lng],\r\n title: marker.popup || \"\",\r\n });\r\n if (marker.popup) {\r\n const infoWindow = new (window as any).AMap.InfoWindow({\r\n content: marker.popup,\r\n offset: new (window as any).AMap.Pixel(0, -30),\r\n });\r\n amapMarker.on(\"click\", () => {\r\n infoWindow.open(amap, amapMarker.getPosition());\r\n emit(\"markerClick\", marker, null);\r\n });\r\n }\r\n amapMarker.setMap(amap);\r\n });\r\n};\r\n\r\nwatch(\r\n () => props.markers,\r\n () => {\r\n if (map && props.mapType === \"osm\") addMarkers();\r\n },\r\n { deep: true },\r\n);\r\n\r\nwatch(\r\n () => props.mapType,\r\n async (newType, oldType) => {\r\n if (newType === oldType) return;\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n loading.value = true;\r\n await nextTick();\r\n if (newType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n }\r\n },\r\n);\r\n\r\nwatch(\r\n [() => props.center, () => props.zoom],\r\n () => {\r\n if (map) map.setView(props.center, props.zoom);\r\n },\r\n { deep: true },\r\n);\r\n\r\nonMounted(async () => {\r\n await nextTick();\r\n if (props.mapType === \"amap\" && props.amapKey) await initAMap();\r\n else await initOSMMap();\r\n});\r\n\r\nonUnmounted(() => {\r\n if (map) {\r\n map.remove();\r\n map = null;\r\n }\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_Markdown-BEjxknqd.css","names":[],"sources":["../src/components/C_Markdown/index.vue?vue&type=style&index=0&scoped=26eea21a&lang.scss"],"sourcesContent":["/* C_Markdown 组件样式文件 */\n.c-markdown-wrapper[data-v-26eea21a] {\n position: relative;\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor {\n border-radius: 6px;\n border: 1px solid var(--c-border);\n background-color: var(--c-bg-card);\n /* 禁用状态样式 */\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor:hover {\n border-color: var(--c-primary);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor:focus-within {\n border-color: var(--c-primary);\n box-shadow: 0 0 0 2px var(--c-primary-opacity, rgba(24, 160, 88, 0.2));\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor.disabled {\n background-color: var(--c-bg-disabled, #fafafc);\n cursor: not-allowed;\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor.disabled .v-md-editor__editor {\n cursor: not-allowed;\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar {\n background-color: var(--c-bg-card);\n border-bottom: 1px solid var(--c-border);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__editor-wrapper {\n background-color: var(--c-bg-card);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__editor {\n background-color: var(--c-bg-card);\n color: var(--c-text-1);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__editor textarea {\n color: var(--c-text-1);\n caret-color: var(--c-text-1);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__preview-wrapper {\n background-color: var(--c-bg-card);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__preview {\n background-color: var(--c-bg-card);\n color: var(--c-text-1);\n}\n.c-markdown-wrapper[data-v-26eea21a] .v-md-editor .v-md-editor__toc-nav-wrapper {\n background-color: var(--c-bg-card);\n border-left: 1px solid var(--c-border);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-item {\n color: #ffffff;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-item:hover {\n background-color: rgba(255, 255, 255, 0.08);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-item.is-active {\n background-color: rgba(255, 255, 255, 0.12);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-item svg {\n fill: #ffffff;\n opacity: 0.9;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-item:hover svg {\n opacity: 1;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-right .v-md-editor__toolbar-item {\n background-color: transparent;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-right .v-md-editor__toolbar-item svg {\n fill: #ffffff;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-right .v-md-editor__toolbar-item:hover {\n background-color: rgba(255, 255, 255, 0.08);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__toolbar-divider {\n background-color: var(--c-border);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__editor::-webkit-scrollbar,\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__preview::-webkit-scrollbar {\n width: 8px;\n height: 8px;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__editor::-webkit-scrollbar-track,\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__preview::-webkit-scrollbar-track {\n background-color: rgba(255, 255, 255, 0.05);\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__editor::-webkit-scrollbar-thumb,\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__preview::-webkit-scrollbar-thumb {\n background-color: rgba(255, 255, 255, 0.2);\n border-radius: 4px;\n}\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__editor::-webkit-scrollbar-thumb:hover,\n.c-markdown-wrapper.is-dark[data-v-26eea21a] .v-md-editor .v-md-editor__preview::-webkit-scrollbar-thumb:hover {\n background-color: rgba(255, 255, 255, 0.3);\n}\n.c-markdown-wrapper .word-count[data-v-26eea21a] {\n position: absolute;\n bottom: 8px;\n right: 12px;\n font-size: 12px;\n color: var(--c-text-4);\n background: var(--c-bg-card);\n padding: 2px 6px;\n border-radius: 4px;\n z-index: 10;\n}\n.c-markdown-wrapper .word-count.exceed[data-v-26eea21a] {\n color: var(--c-error);\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"}
@@ -0,0 +1,4 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_C_Markdown = require('./C_Markdown.js');
3
+
4
+ exports.C_Markdown = require_C_Markdown.C_Markdown_default;
@@ -0,0 +1,2 @@
1
+ import { t as _default } from "./index12.vue.js";
2
+ export { _default as C_Markdown };
@@ -0,0 +1,2 @@
1
+ import { t as _default } from "./index12.vue.js";
2
+ export { _default as C_Markdown };
@@ -0,0 +1,3 @@
1
+ import { t as C_Markdown_default } from "./C_Markdown2.js";
2
+
3
+ export { C_Markdown_default as C_Markdown };