@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,286 @@
1
+ import { Fragment, computed, createBlock, createCommentVNode, createElementBlock, createElementVNode, defineComponent, mergeProps, nextTick, normalizeClass, normalizeStyle, onMounted, openBlock, readonly, ref, resolveDynamicComponent, unref, watch } from "vue";
2
+ import { Icon } from "@iconify/vue";
3
+
4
+ //#region src/hooks/useImage/index.ts
5
+ /**
6
+ * @description: 图片路径解析函数(组件库版本)
7
+ * 在组件库中,图片路径由宿主应用管理。
8
+ * 如果传入的是完整 URL(http/https/data:)则直接返回;
9
+ * 否则可通过 imageResolver 配置自定义解析逻辑。
10
+ * @param {string} imagePath 图片路径或完整 URL
11
+ * @param {Function} resolver 可选的自定义解析器
12
+ * @return {Promise<string>} 处理后图片路径
13
+ */
14
+ const useImage = async (imagePath, resolver) => {
15
+ if (!imagePath) return "";
16
+ if (imagePath.startsWith("http://") || imagePath.startsWith("https://") || imagePath.startsWith("data:") || imagePath.startsWith("blob:") || imagePath.startsWith("/")) return imagePath;
17
+ if (resolver) try {
18
+ return await resolver(imagePath);
19
+ } catch (e) {
20
+ console.error(`[useImage] 自定义解析器处理失败: ${imagePath}`, e);
21
+ return "";
22
+ }
23
+ return imagePath;
24
+ };
25
+
26
+ //#endregion
27
+ //#region src/components/C_Icon/index.vue?vue&type=script&setup=true&lang.ts
28
+ const _hoisted_1 = [
29
+ "title",
30
+ "aria-label",
31
+ "role",
32
+ "tabindex"
33
+ ];
34
+ const _hoisted_2 = [
35
+ "width",
36
+ "height",
37
+ "viewBox",
38
+ "fill"
39
+ ];
40
+ const _hoisted_3 = ["d", "fill"];
41
+ const _hoisted_4 = [
42
+ "src",
43
+ "alt",
44
+ "width",
45
+ "height"
46
+ ];
47
+ const _hoisted_5 = ["title"];
48
+ var index_vue_vue_type_script_setup_true_lang_default = /* @__PURE__ */ defineComponent({
49
+ __name: "index",
50
+ props: {
51
+ name: { default: void 0 },
52
+ type: { default: "iconify" },
53
+ color: { default: "currentColor" },
54
+ size: { default: 18 },
55
+ svgPath: { default: "" },
56
+ viewBox: { default: "0 0 24 24" },
57
+ alt: { default: "" },
58
+ clickable: {
59
+ type: Boolean,
60
+ default: false
61
+ },
62
+ loading: {
63
+ type: Boolean,
64
+ default: false
65
+ },
66
+ fallbackIcon: { default: "" },
67
+ title: { default: "" },
68
+ ariaLabel: { default: "" },
69
+ customClass: { default: "" },
70
+ rotate: { default: 0 },
71
+ flip: { default: void 0 },
72
+ componentProps: { default: () => ({}) }
73
+ },
74
+ emits: [
75
+ "click",
76
+ "error",
77
+ "load"
78
+ ],
79
+ setup(__props, { expose: __expose, emit: __emit }) {
80
+ const props = __props;
81
+ const emit = __emit;
82
+ const iconRef = ref();
83
+ const hasError = ref(false);
84
+ const errorMessage = ref("");
85
+ const imageSrc = ref("");
86
+ const imageLoading = ref(false);
87
+ const rootStyle = computed(() => ({
88
+ fontSize: typeof props.size === "number" ? `${props.size}px` : props.size,
89
+ color: props.color,
90
+ cursor: props.clickable ? "pointer" : "default"
91
+ }));
92
+ const iconStyle = computed(() => {
93
+ const style = {
94
+ display: "inline-flex",
95
+ justifyContent: "center",
96
+ alignItems: "center",
97
+ width: typeof props.size === "number" ? `${props.size}px` : props.size,
98
+ height: typeof props.size === "number" ? `${props.size}px` : props.size
99
+ };
100
+ if (props.rotate) style.transform = `rotate(${props.rotate}deg)`;
101
+ if (props.flip) {
102
+ const scaleX = props.flip === "horizontal" || props.flip === "both" ? -1 : 1;
103
+ const scaleY = props.flip === "vertical" || props.flip === "both" ? -1 : 1;
104
+ style.transform = `${style.transform || ""} scaleX(${scaleX}) scaleY(${scaleY})`.trim();
105
+ }
106
+ return style;
107
+ });
108
+ const iconDisplayName = computed(() => {
109
+ if (typeof props.name === "string") return props.name;
110
+ if (props.name && props.name.__name) return props.name.__name;
111
+ if (props.name && props.name.name) return props.name.name;
112
+ return "Component Icon";
113
+ });
114
+ const iconClasses = computed(() => [props.customClass, {
115
+ "c-icon__rotatable": props.rotate !== 0,
116
+ "c-icon__flippable": !!props.flip
117
+ }]);
118
+ const resolvedComponent = computed(() => {
119
+ if (props.type !== "component" || !props.name) return null;
120
+ return props.name;
121
+ });
122
+ const handleError = (type, message, originalError) => {
123
+ console.warn(`[C_Icon] ${message}:`, iconDisplayName.value, originalError);
124
+ hasError.value = true;
125
+ errorMessage.value = message;
126
+ imageLoading.value = false;
127
+ emit("error", type, originalError);
128
+ if (props.fallbackIcon && !hasError.value) {}
129
+ };
130
+ const handleClick = (event) => {
131
+ if (!props.clickable || props.loading) return;
132
+ emit("click", event);
133
+ };
134
+ const handleKeydown = (event) => {
135
+ if (!props.clickable || props.loading) return;
136
+ if (event.key === "Enter" || event.key === " ") {
137
+ event.preventDefault();
138
+ emit("click", event);
139
+ }
140
+ };
141
+ const handleImageLoad = () => {
142
+ imageLoading.value = false;
143
+ emit("load");
144
+ };
145
+ const loadImageSrc = async () => {
146
+ if (props.type !== "image" || typeof props.name !== "string") {
147
+ imageSrc.value = "";
148
+ return;
149
+ }
150
+ if (props.name.startsWith("http://") || props.name.startsWith("https://") || props.name.startsWith("//") || props.name.startsWith("/")) {
151
+ imageSrc.value = props.name;
152
+ return;
153
+ }
154
+ try {
155
+ imageLoading.value = true;
156
+ const imageUrl = await useImage(props.name);
157
+ if (imageUrl) imageSrc.value = imageUrl;
158
+ else handleError("image", "图片路径解析失败");
159
+ } catch (error) {
160
+ handleError("image", "图片加载失败", error);
161
+ } finally {
162
+ imageLoading.value = false;
163
+ }
164
+ };
165
+ const validationRules = {
166
+ iconify: () => {
167
+ if (!props.name.includes(":")) console.warn("[C_Icon] Iconify图标名称格式应为 \"prefix:name\",如 \"mdi:home\"");
168
+ return null;
169
+ },
170
+ unocss: () => {
171
+ if (typeof props.name !== "string" || !props.name.startsWith("i-")) console.warn("[C_Icon] UnoCSS图标名称应以 \"i-\" 开头");
172
+ return null;
173
+ },
174
+ component: () => !resolvedComponent.value ? "无法解析组件" : null,
175
+ svg: () => !props.svgPath ? "SVG路径不能为空" : null,
176
+ image: () => {
177
+ return null;
178
+ }
179
+ };
180
+ const validateProps = () => {
181
+ hasError.value = false;
182
+ errorMessage.value = "";
183
+ if (!props.name) return handleError("validation", "图标名称不能为空");
184
+ const validator = validationRules[props.type];
185
+ if (!validator) return handleError("type", "不支持的图标类型");
186
+ const errorMsg = validator();
187
+ if (errorMsg) handleError(props.type, errorMsg);
188
+ if (props.type === "image") loadImageSrc();
189
+ };
190
+ watch(() => [
191
+ props.name,
192
+ props.type,
193
+ props.svgPath
194
+ ], () => {
195
+ nextTick(validateProps);
196
+ }, { immediate: true });
197
+ onMounted(() => {
198
+ validateProps();
199
+ if (!hasError.value && props.type !== "image") emit("load");
200
+ });
201
+ __expose({
202
+ validate: validateProps,
203
+ hasError: readonly(hasError),
204
+ errorMessage: readonly(errorMessage),
205
+ el: iconRef
206
+ });
207
+ return (_ctx, _cache) => {
208
+ return openBlock(), createElementBlock("div", {
209
+ ref_key: "iconRef",
210
+ ref: iconRef,
211
+ class: normalizeClass(["c-icon", [`c-icon--${_ctx.type}`, {
212
+ "c-icon--clickable": _ctx.clickable,
213
+ "c-icon--loading": _ctx.loading,
214
+ "c-icon--error": hasError.value
215
+ }]]),
216
+ style: normalizeStyle(rootStyle.value),
217
+ title: _ctx.title || iconDisplayName.value,
218
+ "aria-label": _ctx.ariaLabel || iconDisplayName.value,
219
+ role: _ctx.clickable ? "button" : "img",
220
+ tabindex: _ctx.clickable ? 0 : -1,
221
+ onClick: handleClick,
222
+ onKeydown: handleKeydown
223
+ }, [
224
+ createCommentVNode(" 方式1:Iconify图标(默认方式) "),
225
+ (_ctx.type === "iconify" || !_ctx.type) && !hasError.value && typeof _ctx.name === "string" ? (openBlock(), createBlock(unref(Icon), {
226
+ key: 0,
227
+ icon: _ctx.name,
228
+ width: _ctx.size,
229
+ height: _ctx.size,
230
+ color: _ctx.color,
231
+ style: normalizeStyle(iconStyle.value),
232
+ onError: _cache[0] || (_cache[0] = () => handleError("iconify", "Iconify图标加载失败"))
233
+ }, null, 8, [
234
+ "icon",
235
+ "width",
236
+ "height",
237
+ "color",
238
+ "style"
239
+ ])) : _ctx.type === "unocss" && !hasError.value && typeof _ctx.name === "string" ? (openBlock(), createElementBlock(Fragment, { key: 1 }, [createCommentVNode(" 方式2:UnoCSS图标 "), createElementVNode("div", {
240
+ class: normalizeClass([_ctx.name, iconClasses.value]),
241
+ style: normalizeStyle(iconStyle.value),
242
+ onError: _cache[1] || (_cache[1] = () => handleError("unocss", "UnoCSS图标渲染失败"))
243
+ }, null, 38)], 2112)) : _ctx.type === "component" && resolvedComponent.value && !hasError.value ? (openBlock(), createElementBlock(Fragment, { key: 2 }, [createCommentVNode(" 方式3:组件挂载方式 "), (openBlock(), createBlock(resolveDynamicComponent(resolvedComponent.value), mergeProps({ style: iconStyle.value }, _ctx.componentProps, { onError: _cache[2] || (_cache[2] = () => handleError("component", "组件图标渲染失败")) }), null, 16, ["style"]))], 2112)) : _ctx.type === "svg" && _ctx.svgPath && !hasError.value ? (openBlock(), createElementBlock(Fragment, { key: 3 }, [createCommentVNode(" 方式4:SVG路径方式 "), (openBlock(), createElementBlock("svg", {
244
+ width: _ctx.size,
245
+ height: _ctx.size,
246
+ viewBox: _ctx.viewBox,
247
+ style: normalizeStyle(iconStyle.value),
248
+ fill: _ctx.color,
249
+ onError: _cache[3] || (_cache[3] = () => handleError("svg", "SVG图标渲染失败"))
250
+ }, [createElementVNode("path", {
251
+ d: _ctx.svgPath,
252
+ fill: _ctx.color
253
+ }, null, 8, _hoisted_3)], 44, _hoisted_2))], 2112)) : _ctx.type === "image" && imageSrc.value && !hasError.value && !imageLoading.value ? (openBlock(), createElementBlock(Fragment, { key: 4 }, [createCommentVNode(" 方式5:图片方式 - 修复异步加载 "), createElementVNode("img", {
254
+ src: imageSrc.value,
255
+ alt: _ctx.alt || iconDisplayName.value,
256
+ width: _ctx.size,
257
+ height: _ctx.size,
258
+ style: {
259
+ "display": "inline-block",
260
+ "vertical-align": "middle"
261
+ },
262
+ onError: _cache[4] || (_cache[4] = () => handleError("image", "图片图标加载失败")),
263
+ onLoad: handleImageLoad
264
+ }, null, 40, _hoisted_4)], 2112)) : createCommentVNode("v-if", true),
265
+ createCommentVNode(" 图片加载状态 "),
266
+ (_ctx.loading || imageLoading.value) && !hasError.value ? (openBlock(), createElementBlock("div", {
267
+ key: 5,
268
+ class: "c-icon__loading",
269
+ style: normalizeStyle(iconStyle.value)
270
+ }, _cache[5] || (_cache[5] = [createElementVNode("div", { class: "c-icon__spinner" }, null, -1)]), 4)) : hasError.value ? (openBlock(), createElementBlock(Fragment, { key: 6 }, [createCommentVNode(" 统一错误状态显示 "), createElementVNode("div", {
271
+ class: "c-icon__error",
272
+ style: normalizeStyle(iconStyle.value),
273
+ title: `图标解析失败: ${iconDisplayName.value} (${errorMessage.value})${_ctx.fallbackIcon ? ` (fallback: ${_ctx.fallbackIcon})` : ""}`
274
+ }, " ⚠️ ", 12, _hoisted_5)], 2112)) : createCommentVNode("v-if", true)
275
+ ], 46, _hoisted_1);
276
+ };
277
+ }
278
+ });
279
+
280
+ //#endregion
281
+ //#region src/components/C_Icon/index.vue
282
+ var C_Icon_default = index_vue_vue_type_script_setup_true_lang_default;
283
+
284
+ //#endregion
285
+ export { C_Icon_default as t };
286
+ //# sourceMappingURL=C_Icon2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_Icon2.js","names":["type","clickable","loading","title","ariaLabel","name","size","color","componentProps","svgPath","viewBox","alt","fallbackIcon"],"sources":["../src/hooks/useImage/index.ts","../src/components/C_Icon/index.vue","../src/components/C_Icon/index.vue","../src/components/C_Icon/index.vue"],"sourcesContent":["/**\r\n * @description: 图片路径解析函数(组件库版本)\r\n * 在组件库中,图片路径由宿主应用管理。\r\n * 如果传入的是完整 URL(http/https/data:)则直接返回;\r\n * 否则可通过 imageResolver 配置自定义解析逻辑。\r\n * @param {string} imagePath 图片路径或完整 URL\r\n * @param {Function} resolver 可选的自定义解析器\r\n * @return {Promise<string>} 处理后图片路径\r\n */\r\nexport const useImage = async (\r\n imagePath: string,\r\n resolver?: (path: string) => string | Promise<string>,\r\n): Promise<string> => {\r\n if (!imagePath) return \"\";\r\n\r\n // 完整 URL 直接返回\r\n if (\r\n imagePath.startsWith(\"http://\") ||\r\n imagePath.startsWith(\"https://\") ||\r\n imagePath.startsWith(\"data:\") ||\r\n imagePath.startsWith(\"blob:\") ||\r\n imagePath.startsWith(\"/\")\r\n ) {\r\n return imagePath;\r\n }\r\n\r\n // 如果提供了自定义解析器,使用自定义解析器\r\n if (resolver) {\r\n try {\r\n return await resolver(imagePath);\r\n } catch (e) {\r\n console.error(`[useImage] 自定义解析器处理失败: ${imagePath}`, e);\r\n return \"\";\r\n }\r\n }\r\n\r\n // 默认:直接返回路径(宿主应用应保证路径可访问)\r\n return imagePath;\r\n};\r\n\r\n/**\r\n * @description: 同步版本的图片路径处理函数\r\n * @param {string} imagePath 图片路径或完整 URL\r\n * @return {string} 图片URL\r\n */\r\nexport const useImageSync = (imagePath: string): string => {\r\n if (!imagePath) return \"\";\r\n\r\n // 完整 URL 直接返回\r\n if (\r\n imagePath.startsWith(\"http://\") ||\r\n imagePath.startsWith(\"https://\") ||\r\n imagePath.startsWith(\"data:\") ||\r\n imagePath.startsWith(\"blob:\") ||\r\n imagePath.startsWith(\"/\")\r\n ) {\r\n return imagePath;\r\n }\r\n\r\n return imagePath;\r\n};\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-05-14\r\n * @Description: 支持多种图标使用方式,默认使用Iconify图标,统一错误处理\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"iconRef\"\r\n class=\"c-icon\"\r\n :class=\"[\r\n `c-icon--${type}`,\r\n {\r\n 'c-icon--clickable': clickable,\r\n 'c-icon--loading': loading,\r\n 'c-icon--error': hasError,\r\n },\r\n ]\"\r\n :style=\"rootStyle\"\r\n :title=\"title || iconDisplayName\"\r\n :aria-label=\"ariaLabel || iconDisplayName\"\r\n :role=\"clickable ? 'button' : 'img'\"\r\n :tabindex=\"clickable ? 0 : -1\"\r\n @click=\"handleClick\"\r\n @keydown=\"handleKeydown\"\r\n >\r\n <!-- 方式1:Iconify图标(默认方式) -->\r\n <Icon\r\n v-if=\"\r\n (type === 'iconify' || !type) && !hasError && typeof name === 'string'\r\n \"\r\n :icon=\"name\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :color=\"color\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('iconify', 'Iconify图标加载失败')\"\r\n />\r\n\r\n <!-- 方式2:UnoCSS图标 -->\r\n <div\r\n v-else-if=\"type === 'unocss' && !hasError && typeof name === 'string'\"\r\n :class=\"[name, iconClasses]\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('unocss', 'UnoCSS图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式3:组件挂载方式 -->\r\n <component\r\n v-else-if=\"type === 'component' && resolvedComponent && !hasError\"\r\n :is=\"resolvedComponent\"\r\n :style=\"iconStyle\"\r\n v-bind=\"componentProps\"\r\n @error=\"() => handleError('component', '组件图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式4:SVG路径方式 -->\r\n <svg\r\n v-else-if=\"type === 'svg' && svgPath && !hasError\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :viewBox=\"viewBox\"\r\n :style=\"iconStyle\"\r\n :fill=\"color\"\r\n @error=\"() => handleError('svg', 'SVG图标渲染失败')\"\r\n >\r\n <path :d=\"svgPath\" :fill=\"color\" />\r\n </svg>\r\n\r\n <!-- 方式5:图片方式 - 修复异步加载 -->\r\n <img\r\n v-else-if=\"type === 'image' && imageSrc && !hasError && !imageLoading\"\r\n :src=\"imageSrc\"\r\n :alt=\"alt || iconDisplayName\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n style=\"display: inline-block; vertical-align: middle\"\r\n @error=\"() => handleError('image', '图片图标加载失败')\"\r\n @load=\"handleImageLoad\"\r\n />\r\n\r\n <!-- 图片加载状态 -->\r\n <div\r\n v-if=\"(loading || imageLoading) && !hasError\"\r\n class=\"c-icon__loading\"\r\n :style=\"iconStyle\"\r\n >\r\n <div class=\"c-icon__spinner\" />\r\n </div>\r\n\r\n <!-- 统一错误状态显示 -->\r\n <div\r\n v-else-if=\"hasError\"\r\n class=\"c-icon__error\"\r\n :style=\"iconStyle\"\r\n :title=\"`图标解析失败: ${iconDisplayName} (${errorMessage})${\r\n fallbackIcon ? ` (fallback: ${fallbackIcon})` : ''\r\n }`\"\r\n >\r\n ⚠️\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script lang=\"ts\" setup>\r\nimport { Icon } from \"@iconify/vue\";\r\nimport { useImage } from \"../../hooks/useImage\";\r\nimport { computed, nextTick, onMounted, readonly, ref, watch } from \"vue\";\r\n\r\nexport interface IconProps {\r\n /** 图标名称/路径/组件实例 */\r\n name: string | any;\r\n /** 图标类型 */\r\n type?: \"iconify\" | \"unocss\" | \"component\" | \"svg\" | \"image\";\r\n /** 图标颜色 */\r\n color?: string;\r\n /** 图标大小(px) */\r\n size?: number | string;\r\n /** SVG路径数据(仅type=svg时使用) */\r\n svgPath?: string;\r\n /** SVG viewBox(仅type=svg时使用) */\r\n viewBox?: string;\r\n /** 图片alt属性(仅type=image时使用) */\r\n alt?: string;\r\n /** 是否可点击 */\r\n clickable?: boolean;\r\n /** 加载状态 */\r\n loading?: boolean;\r\n /** 错误时的回退图标 */\r\n fallbackIcon?: string;\r\n /** 工具提示 */\r\n title?: string;\r\n /** 无障碍标签 */\r\n ariaLabel?: string;\r\n /** 自定义样式类 */\r\n customClass?: string;\r\n /** 旋转角度 */\r\n rotate?: number;\r\n /** 是否翻转 */\r\n flip?: \"horizontal\" | \"vertical\" | \"both\";\r\n /** 传递给组件的额外属性(仅type=component时使用) */\r\n componentProps?: Record<string, any>;\r\n}\r\n\r\nconst props = withDefaults(defineProps<IconProps>(), {\r\n name: undefined,\r\n type: \"iconify\",\r\n color: \"currentColor\",\r\n size: 18,\r\n svgPath: \"\",\r\n viewBox: \"0 0 24 24\",\r\n alt: \"\",\r\n clickable: false,\r\n loading: false,\r\n fallbackIcon: \"\",\r\n title: \"\",\r\n ariaLabel: \"\",\r\n customClass: \"\",\r\n rotate: 0,\r\n flip: undefined,\r\n componentProps: () => ({}),\r\n});\r\n\r\nconst emit = defineEmits<{\r\n click: [event: MouseEvent];\r\n error: [type: string, error?: any];\r\n load: [];\r\n}>();\r\n\r\nconst iconRef = ref<HTMLElement>();\r\nconst hasError = ref(false);\r\nconst errorMessage = ref(\"\");\r\nconst imageSrc = ref<string>(\"\"); // 存储异步加载的图片URL\r\nconst imageLoading = ref(false); // 图片加载状态\r\n\r\n// 计算根元素样式\r\nconst rootStyle = computed(() => ({\r\n fontSize: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n color: props.color,\r\n cursor: props.clickable ? \"pointer\" : \"default\",\r\n}));\r\n\r\n// 计算图标样式\r\nconst iconStyle = computed(() => {\r\n const style: Record<string, any> = {\r\n display: \"inline-flex\",\r\n justifyContent: \"center\",\r\n alignItems: \"center\",\r\n width: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n height: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n };\r\n\r\n // 旋转\r\n if (props.rotate) {\r\n style.transform = `rotate(${props.rotate}deg)`;\r\n }\r\n\r\n // 翻转\r\n if (props.flip) {\r\n const scaleX =\r\n props.flip === \"horizontal\" || props.flip === \"both\" ? -1 : 1;\r\n const scaleY = props.flip === \"vertical\" || props.flip === \"both\" ? -1 : 1;\r\n style.transform = `${\r\n style.transform || \"\"\r\n } scaleX(${scaleX}) scaleY(${scaleY})`.trim();\r\n }\r\n\r\n return style;\r\n});\r\n\r\n// 获取图标名称用于显示\r\nconst iconDisplayName = computed(() => {\r\n if (typeof props.name === \"string\") {\r\n return props.name;\r\n }\r\n if (props.name && props.name.__name) {\r\n return props.name.__name;\r\n }\r\n if (props.name && props.name.name) {\r\n return props.name.name;\r\n }\r\n return \"Component Icon\";\r\n});\r\n\r\n// 图标类名\r\nconst iconClasses = computed(() => [\r\n props.customClass,\r\n {\r\n \"c-icon__rotatable\": props.rotate !== 0,\r\n \"c-icon__flippable\": !!props.flip,\r\n },\r\n]);\r\n\r\n// 解析组件名称(纯计算属性,无副作用)\r\nconst resolvedComponent = computed(() => {\r\n if (props.type !== \"component\" || !props.name) return null;\r\n return props.name;\r\n});\r\n\r\n// 统一错误处理函数\r\nconst handleError = (type: string, message: string, originalError?: any) => {\r\n console.warn(`[C_Icon] ${message}:`, iconDisplayName.value, originalError);\r\n hasError.value = true;\r\n errorMessage.value = message;\r\n imageLoading.value = false; // 错误时停止加载状态\r\n emit(\"error\", type, originalError);\r\n\r\n // 如果有回退图标,可以在这里尝试使用回退方案\r\n if (props.fallbackIcon && !hasError.value) {\r\n // 这里可以实现回退逻辑\r\n // 比如尝试使用fallbackIcon作为iconify图标\r\n }\r\n};\r\n\r\n// 处理点击事件\r\nconst handleClick = (event: MouseEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n emit(\"click\", event);\r\n};\r\n\r\n// 处理键盘事件\r\nconst handleKeydown = (event: KeyboardEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n if (event.key === \"Enter\" || event.key === \" \") {\r\n event.preventDefault();\r\n emit(\"click\", event as any);\r\n }\r\n};\r\n\r\n// 处理图片加载成功\r\nconst handleImageLoad = () => {\r\n imageLoading.value = false;\r\n emit(\"load\");\r\n};\r\n\r\n// 异步加载图片 - 修复关键函数\r\nconst loadImageSrc = async () => {\r\n if (props.type !== \"image\" || typeof props.name !== \"string\") {\r\n imageSrc.value = \"\";\r\n return;\r\n }\r\n\r\n // 如果是完整URL,直接设置\r\n if (\r\n props.name.startsWith(\"http://\") ||\r\n props.name.startsWith(\"https://\") ||\r\n props.name.startsWith(\"//\") ||\r\n props.name.startsWith(\"/\")\r\n ) {\r\n imageSrc.value = props.name;\r\n return;\r\n }\r\n\r\n // 其他情况都当作本地图片处理\r\n try {\r\n imageLoading.value = true;\r\n const imageUrl = await useImage(props.name);\r\n if (imageUrl) {\r\n imageSrc.value = imageUrl;\r\n } else {\r\n handleError(\"image\", \"图片路径解析失败\");\r\n }\r\n } catch (error) {\r\n handleError(\"image\", \"图片加载失败\", error);\r\n } finally {\r\n imageLoading.value = false;\r\n }\r\n};\r\n\r\n// 验证规则映射\r\nconst validationRules = {\r\n iconify: (): string | null => {\r\n if (!props.name.includes(\":\")) {\r\n console.warn(\r\n '[C_Icon] Iconify图标名称格式应为 \"prefix:name\",如 \"mdi:home\"',\r\n );\r\n }\r\n return null;\r\n },\r\n unocss: (): string | null => {\r\n if (typeof props.name !== \"string\" || !props.name.startsWith(\"i-\")) {\r\n console.warn('[C_Icon] UnoCSS图标名称应以 \"i-\" 开头');\r\n }\r\n return null;\r\n },\r\n component: (): string | null =>\r\n !resolvedComponent.value ? \"无法解析组件\" : null,\r\n svg: (): string | null => (!props.svgPath ? \"SVG路径不能为空\" : null),\r\n image: (): string | null => {\r\n // 对于image类型,异步验证在loadImageSrc中处理\r\n return null;\r\n },\r\n};\r\n\r\n// 验证配置\r\nconst validateProps = () => {\r\n hasError.value = false;\r\n errorMessage.value = \"\";\r\n\r\n // 基础验证\r\n if (!props.name) {\r\n return handleError(\"validation\", \"图标名称不能为空\");\r\n }\r\n\r\n // 类型验证\r\n const validator = validationRules[props.type];\r\n if (!validator) {\r\n return handleError(\"type\", \"不支持的图标类型\");\r\n }\r\n\r\n const errorMsg = validator();\r\n if (errorMsg) {\r\n handleError(props.type, errorMsg);\r\n }\r\n\r\n // 如果是图片类型,异步加载图片\r\n if (props.type === \"image\") {\r\n loadImageSrc();\r\n }\r\n};\r\n\r\n// 监听关键属性变化\r\nwatch(\r\n () => [props.name, props.type, props.svgPath],\r\n () => {\r\n nextTick(validateProps);\r\n },\r\n { immediate: true },\r\n);\r\n\r\n// 组件挂载后验证\r\nonMounted(() => {\r\n validateProps();\r\n // 非image类型在这里触发load事件\r\n if (!hasError.value && props.type !== \"image\") {\r\n emit(\"load\");\r\n }\r\n});\r\n\r\n// 暴露方法给父组件\r\ndefineExpose({\r\n /** 重新验证配置 */\r\n validate: validateProps,\r\n /** 是否有错误 */\r\n hasError: readonly(hasError),\r\n /** 错误信息 */\r\n errorMessage: readonly(errorMessage),\r\n /** DOM引用 */\r\n el: iconRef,\r\n});\r\n</script>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-05-14\r\n * @Description: 支持多种图标使用方式,默认使用Iconify图标,统一错误处理\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"iconRef\"\r\n class=\"c-icon\"\r\n :class=\"[\r\n `c-icon--${type}`,\r\n {\r\n 'c-icon--clickable': clickable,\r\n 'c-icon--loading': loading,\r\n 'c-icon--error': hasError,\r\n },\r\n ]\"\r\n :style=\"rootStyle\"\r\n :title=\"title || iconDisplayName\"\r\n :aria-label=\"ariaLabel || iconDisplayName\"\r\n :role=\"clickable ? 'button' : 'img'\"\r\n :tabindex=\"clickable ? 0 : -1\"\r\n @click=\"handleClick\"\r\n @keydown=\"handleKeydown\"\r\n >\r\n <!-- 方式1:Iconify图标(默认方式) -->\r\n <Icon\r\n v-if=\"\r\n (type === 'iconify' || !type) && !hasError && typeof name === 'string'\r\n \"\r\n :icon=\"name\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :color=\"color\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('iconify', 'Iconify图标加载失败')\"\r\n />\r\n\r\n <!-- 方式2:UnoCSS图标 -->\r\n <div\r\n v-else-if=\"type === 'unocss' && !hasError && typeof name === 'string'\"\r\n :class=\"[name, iconClasses]\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('unocss', 'UnoCSS图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式3:组件挂载方式 -->\r\n <component\r\n v-else-if=\"type === 'component' && resolvedComponent && !hasError\"\r\n :is=\"resolvedComponent\"\r\n :style=\"iconStyle\"\r\n v-bind=\"componentProps\"\r\n @error=\"() => handleError('component', '组件图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式4:SVG路径方式 -->\r\n <svg\r\n v-else-if=\"type === 'svg' && svgPath && !hasError\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :viewBox=\"viewBox\"\r\n :style=\"iconStyle\"\r\n :fill=\"color\"\r\n @error=\"() => handleError('svg', 'SVG图标渲染失败')\"\r\n >\r\n <path :d=\"svgPath\" :fill=\"color\" />\r\n </svg>\r\n\r\n <!-- 方式5:图片方式 - 修复异步加载 -->\r\n <img\r\n v-else-if=\"type === 'image' && imageSrc && !hasError && !imageLoading\"\r\n :src=\"imageSrc\"\r\n :alt=\"alt || iconDisplayName\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n style=\"display: inline-block; vertical-align: middle\"\r\n @error=\"() => handleError('image', '图片图标加载失败')\"\r\n @load=\"handleImageLoad\"\r\n />\r\n\r\n <!-- 图片加载状态 -->\r\n <div\r\n v-if=\"(loading || imageLoading) && !hasError\"\r\n class=\"c-icon__loading\"\r\n :style=\"iconStyle\"\r\n >\r\n <div class=\"c-icon__spinner\" />\r\n </div>\r\n\r\n <!-- 统一错误状态显示 -->\r\n <div\r\n v-else-if=\"hasError\"\r\n class=\"c-icon__error\"\r\n :style=\"iconStyle\"\r\n :title=\"`图标解析失败: ${iconDisplayName} (${errorMessage})${\r\n fallbackIcon ? ` (fallback: ${fallbackIcon})` : ''\r\n }`\"\r\n >\r\n ⚠️\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script lang=\"ts\" setup>\r\nimport { Icon } from \"@iconify/vue\";\r\nimport { useImage } from \"../../hooks/useImage\";\r\nimport { computed, nextTick, onMounted, readonly, ref, watch } from \"vue\";\r\n\r\nexport interface IconProps {\r\n /** 图标名称/路径/组件实例 */\r\n name: string | any;\r\n /** 图标类型 */\r\n type?: \"iconify\" | \"unocss\" | \"component\" | \"svg\" | \"image\";\r\n /** 图标颜色 */\r\n color?: string;\r\n /** 图标大小(px) */\r\n size?: number | string;\r\n /** SVG路径数据(仅type=svg时使用) */\r\n svgPath?: string;\r\n /** SVG viewBox(仅type=svg时使用) */\r\n viewBox?: string;\r\n /** 图片alt属性(仅type=image时使用) */\r\n alt?: string;\r\n /** 是否可点击 */\r\n clickable?: boolean;\r\n /** 加载状态 */\r\n loading?: boolean;\r\n /** 错误时的回退图标 */\r\n fallbackIcon?: string;\r\n /** 工具提示 */\r\n title?: string;\r\n /** 无障碍标签 */\r\n ariaLabel?: string;\r\n /** 自定义样式类 */\r\n customClass?: string;\r\n /** 旋转角度 */\r\n rotate?: number;\r\n /** 是否翻转 */\r\n flip?: \"horizontal\" | \"vertical\" | \"both\";\r\n /** 传递给组件的额外属性(仅type=component时使用) */\r\n componentProps?: Record<string, any>;\r\n}\r\n\r\nconst props = withDefaults(defineProps<IconProps>(), {\r\n name: undefined,\r\n type: \"iconify\",\r\n color: \"currentColor\",\r\n size: 18,\r\n svgPath: \"\",\r\n viewBox: \"0 0 24 24\",\r\n alt: \"\",\r\n clickable: false,\r\n loading: false,\r\n fallbackIcon: \"\",\r\n title: \"\",\r\n ariaLabel: \"\",\r\n customClass: \"\",\r\n rotate: 0,\r\n flip: undefined,\r\n componentProps: () => ({}),\r\n});\r\n\r\nconst emit = defineEmits<{\r\n click: [event: MouseEvent];\r\n error: [type: string, error?: any];\r\n load: [];\r\n}>();\r\n\r\nconst iconRef = ref<HTMLElement>();\r\nconst hasError = ref(false);\r\nconst errorMessage = ref(\"\");\r\nconst imageSrc = ref<string>(\"\"); // 存储异步加载的图片URL\r\nconst imageLoading = ref(false); // 图片加载状态\r\n\r\n// 计算根元素样式\r\nconst rootStyle = computed(() => ({\r\n fontSize: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n color: props.color,\r\n cursor: props.clickable ? \"pointer\" : \"default\",\r\n}));\r\n\r\n// 计算图标样式\r\nconst iconStyle = computed(() => {\r\n const style: Record<string, any> = {\r\n display: \"inline-flex\",\r\n justifyContent: \"center\",\r\n alignItems: \"center\",\r\n width: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n height: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n };\r\n\r\n // 旋转\r\n if (props.rotate) {\r\n style.transform = `rotate(${props.rotate}deg)`;\r\n }\r\n\r\n // 翻转\r\n if (props.flip) {\r\n const scaleX =\r\n props.flip === \"horizontal\" || props.flip === \"both\" ? -1 : 1;\r\n const scaleY = props.flip === \"vertical\" || props.flip === \"both\" ? -1 : 1;\r\n style.transform = `${\r\n style.transform || \"\"\r\n } scaleX(${scaleX}) scaleY(${scaleY})`.trim();\r\n }\r\n\r\n return style;\r\n});\r\n\r\n// 获取图标名称用于显示\r\nconst iconDisplayName = computed(() => {\r\n if (typeof props.name === \"string\") {\r\n return props.name;\r\n }\r\n if (props.name && props.name.__name) {\r\n return props.name.__name;\r\n }\r\n if (props.name && props.name.name) {\r\n return props.name.name;\r\n }\r\n return \"Component Icon\";\r\n});\r\n\r\n// 图标类名\r\nconst iconClasses = computed(() => [\r\n props.customClass,\r\n {\r\n \"c-icon__rotatable\": props.rotate !== 0,\r\n \"c-icon__flippable\": !!props.flip,\r\n },\r\n]);\r\n\r\n// 解析组件名称(纯计算属性,无副作用)\r\nconst resolvedComponent = computed(() => {\r\n if (props.type !== \"component\" || !props.name) return null;\r\n return props.name;\r\n});\r\n\r\n// 统一错误处理函数\r\nconst handleError = (type: string, message: string, originalError?: any) => {\r\n console.warn(`[C_Icon] ${message}:`, iconDisplayName.value, originalError);\r\n hasError.value = true;\r\n errorMessage.value = message;\r\n imageLoading.value = false; // 错误时停止加载状态\r\n emit(\"error\", type, originalError);\r\n\r\n // 如果有回退图标,可以在这里尝试使用回退方案\r\n if (props.fallbackIcon && !hasError.value) {\r\n // 这里可以实现回退逻辑\r\n // 比如尝试使用fallbackIcon作为iconify图标\r\n }\r\n};\r\n\r\n// 处理点击事件\r\nconst handleClick = (event: MouseEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n emit(\"click\", event);\r\n};\r\n\r\n// 处理键盘事件\r\nconst handleKeydown = (event: KeyboardEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n if (event.key === \"Enter\" || event.key === \" \") {\r\n event.preventDefault();\r\n emit(\"click\", event as any);\r\n }\r\n};\r\n\r\n// 处理图片加载成功\r\nconst handleImageLoad = () => {\r\n imageLoading.value = false;\r\n emit(\"load\");\r\n};\r\n\r\n// 异步加载图片 - 修复关键函数\r\nconst loadImageSrc = async () => {\r\n if (props.type !== \"image\" || typeof props.name !== \"string\") {\r\n imageSrc.value = \"\";\r\n return;\r\n }\r\n\r\n // 如果是完整URL,直接设置\r\n if (\r\n props.name.startsWith(\"http://\") ||\r\n props.name.startsWith(\"https://\") ||\r\n props.name.startsWith(\"//\") ||\r\n props.name.startsWith(\"/\")\r\n ) {\r\n imageSrc.value = props.name;\r\n return;\r\n }\r\n\r\n // 其他情况都当作本地图片处理\r\n try {\r\n imageLoading.value = true;\r\n const imageUrl = await useImage(props.name);\r\n if (imageUrl) {\r\n imageSrc.value = imageUrl;\r\n } else {\r\n handleError(\"image\", \"图片路径解析失败\");\r\n }\r\n } catch (error) {\r\n handleError(\"image\", \"图片加载失败\", error);\r\n } finally {\r\n imageLoading.value = false;\r\n }\r\n};\r\n\r\n// 验证规则映射\r\nconst validationRules = {\r\n iconify: (): string | null => {\r\n if (!props.name.includes(\":\")) {\r\n console.warn(\r\n '[C_Icon] Iconify图标名称格式应为 \"prefix:name\",如 \"mdi:home\"',\r\n );\r\n }\r\n return null;\r\n },\r\n unocss: (): string | null => {\r\n if (typeof props.name !== \"string\" || !props.name.startsWith(\"i-\")) {\r\n console.warn('[C_Icon] UnoCSS图标名称应以 \"i-\" 开头');\r\n }\r\n return null;\r\n },\r\n component: (): string | null =>\r\n !resolvedComponent.value ? \"无法解析组件\" : null,\r\n svg: (): string | null => (!props.svgPath ? \"SVG路径不能为空\" : null),\r\n image: (): string | null => {\r\n // 对于image类型,异步验证在loadImageSrc中处理\r\n return null;\r\n },\r\n};\r\n\r\n// 验证配置\r\nconst validateProps = () => {\r\n hasError.value = false;\r\n errorMessage.value = \"\";\r\n\r\n // 基础验证\r\n if (!props.name) {\r\n return handleError(\"validation\", \"图标名称不能为空\");\r\n }\r\n\r\n // 类型验证\r\n const validator = validationRules[props.type];\r\n if (!validator) {\r\n return handleError(\"type\", \"不支持的图标类型\");\r\n }\r\n\r\n const errorMsg = validator();\r\n if (errorMsg) {\r\n handleError(props.type, errorMsg);\r\n }\r\n\r\n // 如果是图片类型,异步加载图片\r\n if (props.type === \"image\") {\r\n loadImageSrc();\r\n }\r\n};\r\n\r\n// 监听关键属性变化\r\nwatch(\r\n () => [props.name, props.type, props.svgPath],\r\n () => {\r\n nextTick(validateProps);\r\n },\r\n { immediate: true },\r\n);\r\n\r\n// 组件挂载后验证\r\nonMounted(() => {\r\n validateProps();\r\n // 非image类型在这里触发load事件\r\n if (!hasError.value && props.type !== \"image\") {\r\n emit(\"load\");\r\n }\r\n});\r\n\r\n// 暴露方法给父组件\r\ndefineExpose({\r\n /** 重新验证配置 */\r\n validate: validateProps,\r\n /** 是否有错误 */\r\n hasError: readonly(hasError),\r\n /** 错误信息 */\r\n errorMessage: readonly(errorMessage),\r\n /** DOM引用 */\r\n el: iconRef,\r\n});\r\n</script>\r\n","<!--\r\n * @Author: ChenYu ycyplus@gmail.com\r\n * @Date: 2025-05-14\r\n * @Description: 支持多种图标使用方式,默认使用Iconify图标,统一错误处理\r\n * @Migration: naive-ui-components 组件库迁移版本\r\n * Copyright (c) 2025 by CHENY, All Rights Reserved.\r\n-->\r\n<template>\r\n <div\r\n ref=\"iconRef\"\r\n class=\"c-icon\"\r\n :class=\"[\r\n `c-icon--${type}`,\r\n {\r\n 'c-icon--clickable': clickable,\r\n 'c-icon--loading': loading,\r\n 'c-icon--error': hasError,\r\n },\r\n ]\"\r\n :style=\"rootStyle\"\r\n :title=\"title || iconDisplayName\"\r\n :aria-label=\"ariaLabel || iconDisplayName\"\r\n :role=\"clickable ? 'button' : 'img'\"\r\n :tabindex=\"clickable ? 0 : -1\"\r\n @click=\"handleClick\"\r\n @keydown=\"handleKeydown\"\r\n >\r\n <!-- 方式1:Iconify图标(默认方式) -->\r\n <Icon\r\n v-if=\"\r\n (type === 'iconify' || !type) && !hasError && typeof name === 'string'\r\n \"\r\n :icon=\"name\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :color=\"color\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('iconify', 'Iconify图标加载失败')\"\r\n />\r\n\r\n <!-- 方式2:UnoCSS图标 -->\r\n <div\r\n v-else-if=\"type === 'unocss' && !hasError && typeof name === 'string'\"\r\n :class=\"[name, iconClasses]\"\r\n :style=\"iconStyle\"\r\n @error=\"() => handleError('unocss', 'UnoCSS图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式3:组件挂载方式 -->\r\n <component\r\n v-else-if=\"type === 'component' && resolvedComponent && !hasError\"\r\n :is=\"resolvedComponent\"\r\n :style=\"iconStyle\"\r\n v-bind=\"componentProps\"\r\n @error=\"() => handleError('component', '组件图标渲染失败')\"\r\n />\r\n\r\n <!-- 方式4:SVG路径方式 -->\r\n <svg\r\n v-else-if=\"type === 'svg' && svgPath && !hasError\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n :viewBox=\"viewBox\"\r\n :style=\"iconStyle\"\r\n :fill=\"color\"\r\n @error=\"() => handleError('svg', 'SVG图标渲染失败')\"\r\n >\r\n <path :d=\"svgPath\" :fill=\"color\" />\r\n </svg>\r\n\r\n <!-- 方式5:图片方式 - 修复异步加载 -->\r\n <img\r\n v-else-if=\"type === 'image' && imageSrc && !hasError && !imageLoading\"\r\n :src=\"imageSrc\"\r\n :alt=\"alt || iconDisplayName\"\r\n :width=\"size\"\r\n :height=\"size\"\r\n style=\"display: inline-block; vertical-align: middle\"\r\n @error=\"() => handleError('image', '图片图标加载失败')\"\r\n @load=\"handleImageLoad\"\r\n />\r\n\r\n <!-- 图片加载状态 -->\r\n <div\r\n v-if=\"(loading || imageLoading) && !hasError\"\r\n class=\"c-icon__loading\"\r\n :style=\"iconStyle\"\r\n >\r\n <div class=\"c-icon__spinner\" />\r\n </div>\r\n\r\n <!-- 统一错误状态显示 -->\r\n <div\r\n v-else-if=\"hasError\"\r\n class=\"c-icon__error\"\r\n :style=\"iconStyle\"\r\n :title=\"`图标解析失败: ${iconDisplayName} (${errorMessage})${\r\n fallbackIcon ? ` (fallback: ${fallbackIcon})` : ''\r\n }`\"\r\n >\r\n ⚠️\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script lang=\"ts\" setup>\r\nimport { Icon } from \"@iconify/vue\";\r\nimport { useImage } from \"../../hooks/useImage\";\r\nimport { computed, nextTick, onMounted, readonly, ref, watch } from \"vue\";\r\n\r\nexport interface IconProps {\r\n /** 图标名称/路径/组件实例 */\r\n name: string | any;\r\n /** 图标类型 */\r\n type?: \"iconify\" | \"unocss\" | \"component\" | \"svg\" | \"image\";\r\n /** 图标颜色 */\r\n color?: string;\r\n /** 图标大小(px) */\r\n size?: number | string;\r\n /** SVG路径数据(仅type=svg时使用) */\r\n svgPath?: string;\r\n /** SVG viewBox(仅type=svg时使用) */\r\n viewBox?: string;\r\n /** 图片alt属性(仅type=image时使用) */\r\n alt?: string;\r\n /** 是否可点击 */\r\n clickable?: boolean;\r\n /** 加载状态 */\r\n loading?: boolean;\r\n /** 错误时的回退图标 */\r\n fallbackIcon?: string;\r\n /** 工具提示 */\r\n title?: string;\r\n /** 无障碍标签 */\r\n ariaLabel?: string;\r\n /** 自定义样式类 */\r\n customClass?: string;\r\n /** 旋转角度 */\r\n rotate?: number;\r\n /** 是否翻转 */\r\n flip?: \"horizontal\" | \"vertical\" | \"both\";\r\n /** 传递给组件的额外属性(仅type=component时使用) */\r\n componentProps?: Record<string, any>;\r\n}\r\n\r\nconst props = withDefaults(defineProps<IconProps>(), {\r\n name: undefined,\r\n type: \"iconify\",\r\n color: \"currentColor\",\r\n size: 18,\r\n svgPath: \"\",\r\n viewBox: \"0 0 24 24\",\r\n alt: \"\",\r\n clickable: false,\r\n loading: false,\r\n fallbackIcon: \"\",\r\n title: \"\",\r\n ariaLabel: \"\",\r\n customClass: \"\",\r\n rotate: 0,\r\n flip: undefined,\r\n componentProps: () => ({}),\r\n});\r\n\r\nconst emit = defineEmits<{\r\n click: [event: MouseEvent];\r\n error: [type: string, error?: any];\r\n load: [];\r\n}>();\r\n\r\nconst iconRef = ref<HTMLElement>();\r\nconst hasError = ref(false);\r\nconst errorMessage = ref(\"\");\r\nconst imageSrc = ref<string>(\"\"); // 存储异步加载的图片URL\r\nconst imageLoading = ref(false); // 图片加载状态\r\n\r\n// 计算根元素样式\r\nconst rootStyle = computed(() => ({\r\n fontSize: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n color: props.color,\r\n cursor: props.clickable ? \"pointer\" : \"default\",\r\n}));\r\n\r\n// 计算图标样式\r\nconst iconStyle = computed(() => {\r\n const style: Record<string, any> = {\r\n display: \"inline-flex\",\r\n justifyContent: \"center\",\r\n alignItems: \"center\",\r\n width: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n height: typeof props.size === \"number\" ? `${props.size}px` : props.size,\r\n };\r\n\r\n // 旋转\r\n if (props.rotate) {\r\n style.transform = `rotate(${props.rotate}deg)`;\r\n }\r\n\r\n // 翻转\r\n if (props.flip) {\r\n const scaleX =\r\n props.flip === \"horizontal\" || props.flip === \"both\" ? -1 : 1;\r\n const scaleY = props.flip === \"vertical\" || props.flip === \"both\" ? -1 : 1;\r\n style.transform = `${\r\n style.transform || \"\"\r\n } scaleX(${scaleX}) scaleY(${scaleY})`.trim();\r\n }\r\n\r\n return style;\r\n});\r\n\r\n// 获取图标名称用于显示\r\nconst iconDisplayName = computed(() => {\r\n if (typeof props.name === \"string\") {\r\n return props.name;\r\n }\r\n if (props.name && props.name.__name) {\r\n return props.name.__name;\r\n }\r\n if (props.name && props.name.name) {\r\n return props.name.name;\r\n }\r\n return \"Component Icon\";\r\n});\r\n\r\n// 图标类名\r\nconst iconClasses = computed(() => [\r\n props.customClass,\r\n {\r\n \"c-icon__rotatable\": props.rotate !== 0,\r\n \"c-icon__flippable\": !!props.flip,\r\n },\r\n]);\r\n\r\n// 解析组件名称(纯计算属性,无副作用)\r\nconst resolvedComponent = computed(() => {\r\n if (props.type !== \"component\" || !props.name) return null;\r\n return props.name;\r\n});\r\n\r\n// 统一错误处理函数\r\nconst handleError = (type: string, message: string, originalError?: any) => {\r\n console.warn(`[C_Icon] ${message}:`, iconDisplayName.value, originalError);\r\n hasError.value = true;\r\n errorMessage.value = message;\r\n imageLoading.value = false; // 错误时停止加载状态\r\n emit(\"error\", type, originalError);\r\n\r\n // 如果有回退图标,可以在这里尝试使用回退方案\r\n if (props.fallbackIcon && !hasError.value) {\r\n // 这里可以实现回退逻辑\r\n // 比如尝试使用fallbackIcon作为iconify图标\r\n }\r\n};\r\n\r\n// 处理点击事件\r\nconst handleClick = (event: MouseEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n emit(\"click\", event);\r\n};\r\n\r\n// 处理键盘事件\r\nconst handleKeydown = (event: KeyboardEvent) => {\r\n if (!props.clickable || props.loading) return;\r\n if (event.key === \"Enter\" || event.key === \" \") {\r\n event.preventDefault();\r\n emit(\"click\", event as any);\r\n }\r\n};\r\n\r\n// 处理图片加载成功\r\nconst handleImageLoad = () => {\r\n imageLoading.value = false;\r\n emit(\"load\");\r\n};\r\n\r\n// 异步加载图片 - 修复关键函数\r\nconst loadImageSrc = async () => {\r\n if (props.type !== \"image\" || typeof props.name !== \"string\") {\r\n imageSrc.value = \"\";\r\n return;\r\n }\r\n\r\n // 如果是完整URL,直接设置\r\n if (\r\n props.name.startsWith(\"http://\") ||\r\n props.name.startsWith(\"https://\") ||\r\n props.name.startsWith(\"//\") ||\r\n props.name.startsWith(\"/\")\r\n ) {\r\n imageSrc.value = props.name;\r\n return;\r\n }\r\n\r\n // 其他情况都当作本地图片处理\r\n try {\r\n imageLoading.value = true;\r\n const imageUrl = await useImage(props.name);\r\n if (imageUrl) {\r\n imageSrc.value = imageUrl;\r\n } else {\r\n handleError(\"image\", \"图片路径解析失败\");\r\n }\r\n } catch (error) {\r\n handleError(\"image\", \"图片加载失败\", error);\r\n } finally {\r\n imageLoading.value = false;\r\n }\r\n};\r\n\r\n// 验证规则映射\r\nconst validationRules = {\r\n iconify: (): string | null => {\r\n if (!props.name.includes(\":\")) {\r\n console.warn(\r\n '[C_Icon] Iconify图标名称格式应为 \"prefix:name\",如 \"mdi:home\"',\r\n );\r\n }\r\n return null;\r\n },\r\n unocss: (): string | null => {\r\n if (typeof props.name !== \"string\" || !props.name.startsWith(\"i-\")) {\r\n console.warn('[C_Icon] UnoCSS图标名称应以 \"i-\" 开头');\r\n }\r\n return null;\r\n },\r\n component: (): string | null =>\r\n !resolvedComponent.value ? \"无法解析组件\" : null,\r\n svg: (): string | null => (!props.svgPath ? \"SVG路径不能为空\" : null),\r\n image: (): string | null => {\r\n // 对于image类型,异步验证在loadImageSrc中处理\r\n return null;\r\n },\r\n};\r\n\r\n// 验证配置\r\nconst validateProps = () => {\r\n hasError.value = false;\r\n errorMessage.value = \"\";\r\n\r\n // 基础验证\r\n if (!props.name) {\r\n return handleError(\"validation\", \"图标名称不能为空\");\r\n }\r\n\r\n // 类型验证\r\n const validator = validationRules[props.type];\r\n if (!validator) {\r\n return handleError(\"type\", \"不支持的图标类型\");\r\n }\r\n\r\n const errorMsg = validator();\r\n if (errorMsg) {\r\n handleError(props.type, errorMsg);\r\n }\r\n\r\n // 如果是图片类型,异步加载图片\r\n if (props.type === \"image\") {\r\n loadImageSrc();\r\n }\r\n};\r\n\r\n// 监听关键属性变化\r\nwatch(\r\n () => [props.name, props.type, props.svgPath],\r\n () => {\r\n nextTick(validateProps);\r\n },\r\n { immediate: true },\r\n);\r\n\r\n// 组件挂载后验证\r\nonMounted(() => {\r\n validateProps();\r\n // 非image类型在这里触发load事件\r\n if (!hasError.value && props.type !== \"image\") {\r\n emit(\"load\");\r\n }\r\n});\r\n\r\n// 暴露方法给父组件\r\ndefineExpose({\r\n /** 重新验证配置 */\r\n validate: validateProps,\r\n /** 是否有错误 */\r\n hasError: readonly(hasError),\r\n /** 错误信息 */\r\n errorMessage: readonly(errorMessage),\r\n /** DOM引用 */\r\n el: iconRef,\r\n});\r\n</script>\r\n"],"mappings":";;;;;;;;;;;;;AASA,MAAa,WAAW,OACtB,WACA,aACoB;AACpB,KAAI,CAAC,UAAW,QAAO;AAGvB,KACE,UAAU,WAAW,UAAU,IAC/B,UAAU,WAAW,WAAW,IAChC,UAAU,WAAW,QAAQ,IAC7B,UAAU,WAAW,QAAQ,IAC7B,UAAU,WAAW,IAAI,CAEzB,QAAO;AAIT,KAAI,SACF,KAAI;AACF,SAAO,MAAM,SAAS,UAAU;UACzB,GAAG;AACV,UAAQ,MAAM,0BAA0B,aAAa,EAAE;AACvD,SAAO;;AAKX,QAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EE4GT,MAAM,QAAQ;EAmBd,MAAM,OAAO;EAMb,MAAM,UAAU,KAAkB;EAClC,MAAM,WAAW,IAAI,MAAM;EAC3B,MAAM,eAAe,IAAI,GAAG;EAC5B,MAAM,WAAW,IAAY,GAAG;EAChC,MAAM,eAAe,IAAI,MAAM;EAG/B,MAAM,YAAY,gBAAgB;GAChC,UAAU,OAAO,MAAM,SAAS,WAAW,GAAG,MAAM,KAAK,MAAM,MAAM;GACrE,OAAO,MAAM;GACb,QAAQ,MAAM,YAAY,YAAY;GACvC,EAAE;EAGH,MAAM,YAAY,eAAe;GAC/B,MAAM,QAA6B;IACjC,SAAS;IACT,gBAAgB;IAChB,YAAY;IACZ,OAAO,OAAO,MAAM,SAAS,WAAW,GAAG,MAAM,KAAK,MAAM,MAAM;IAClE,QAAQ,OAAO,MAAM,SAAS,WAAW,GAAG,MAAM,KAAK,MAAM,MAAM;IACpE;AAGD,OAAI,MAAM,OACR,OAAM,YAAY,UAAU,MAAM,OAAO;AAI3C,OAAI,MAAM,MAAM;IACd,MAAM,SACJ,MAAM,SAAS,gBAAgB,MAAM,SAAS,SAAS,KAAK;IAC9D,MAAM,SAAS,MAAM,SAAS,cAAc,MAAM,SAAS,SAAS,KAAK;AACzE,UAAM,YAAY,GAChB,MAAM,aAAa,GACpB,UAAU,OAAO,WAAW,OAAO,GAAG,MAAM;;AAG/C,UAAO;IACP;EAGF,MAAM,kBAAkB,eAAe;AACrC,OAAI,OAAO,MAAM,SAAS,SACxB,QAAO,MAAM;AAEf,OAAI,MAAM,QAAQ,MAAM,KAAK,OAC3B,QAAO,MAAM,KAAK;AAEpB,OAAI,MAAM,QAAQ,MAAM,KAAK,KAC3B,QAAO,MAAM,KAAK;AAEpB,UAAO;IACP;EAGF,MAAM,cAAc,eAAe,CACjC,MAAM,aACN;GACE,qBAAqB,MAAM,WAAW;GACtC,qBAAqB,CAAC,CAAC,MAAM;GAC9B,CACF,CAAC;EAGF,MAAM,oBAAoB,eAAe;AACvC,OAAI,MAAM,SAAS,eAAe,CAAC,MAAM,KAAM,QAAO;AACtD,UAAO,MAAM;IACb;EAGF,MAAM,eAAe,MAAc,SAAiB,kBAAwB;AAC1E,WAAQ,KAAK,YAAY,QAAQ,IAAI,gBAAgB,OAAO,cAAc;AAC1E,YAAS,QAAQ;AACjB,gBAAa,QAAQ;AACrB,gBAAa,QAAQ;AACrB,QAAK,SAAS,MAAM,cAAc;AAGlC,OAAI,MAAM,gBAAgB,CAAC,SAAS,OAAO;;EAO7C,MAAM,eAAe,UAAsB;AACzC,OAAI,CAAC,MAAM,aAAa,MAAM,QAAS;AACvC,QAAK,SAAS,MAAM;;EAItB,MAAM,iBAAiB,UAAyB;AAC9C,OAAI,CAAC,MAAM,aAAa,MAAM,QAAS;AACvC,OAAI,MAAM,QAAQ,WAAW,MAAM,QAAQ,KAAK;AAC9C,UAAM,gBAAgB;AACtB,SAAK,SAAS,MAAa;;;EAK/B,MAAM,wBAAwB;AAC5B,gBAAa,QAAQ;AACrB,QAAK,OAAO;;EAId,MAAM,eAAe,YAAY;AAC/B,OAAI,MAAM,SAAS,WAAW,OAAO,MAAM,SAAS,UAAU;AAC5D,aAAS,QAAQ;AACjB;;AAIF,OACE,MAAM,KAAK,WAAW,UAAU,IAChC,MAAM,KAAK,WAAW,WAAW,IACjC,MAAM,KAAK,WAAW,KAAK,IAC3B,MAAM,KAAK,WAAW,IAAI,EAC1B;AACA,aAAS,QAAQ,MAAM;AACvB;;AAIF,OAAI;AACF,iBAAa,QAAQ;IACrB,MAAM,WAAW,MAAM,SAAS,MAAM,KAAK;AAC3C,QAAI,SACF,UAAS,QAAQ;QAEjB,aAAY,SAAS,WAAW;YAE3B,OAAO;AACd,gBAAY,SAAS,UAAU,MAAM;aAC7B;AACR,iBAAa,QAAQ;;;EAKzB,MAAM,kBAAkB;GACtB,eAA8B;AAC5B,QAAI,CAAC,MAAM,KAAK,SAAS,IAAI,CAC3B,SAAQ,KACN,0DACD;AAEH,WAAO;;GAET,cAA6B;AAC3B,QAAI,OAAO,MAAM,SAAS,YAAY,CAAC,MAAM,KAAK,WAAW,KAAK,CAChE,SAAQ,KAAK,kCAAgC;AAE/C,WAAO;;GAET,iBACE,CAAC,kBAAkB,QAAQ,WAAW;GACxC,WAA2B,CAAC,MAAM,UAAU,cAAc;GAC1D,aAA4B;AAE1B,WAAO;;GAEV;EAGD,MAAM,sBAAsB;AAC1B,YAAS,QAAQ;AACjB,gBAAa,QAAQ;AAGrB,OAAI,CAAC,MAAM,KACT,QAAO,YAAY,cAAc,WAAW;GAI9C,MAAM,YAAY,gBAAgB,MAAM;AACxC,OAAI,CAAC,UACH,QAAO,YAAY,QAAQ,WAAW;GAGxC,MAAM,WAAW,WAAW;AAC5B,OAAI,SACF,aAAY,MAAM,MAAM,SAAS;AAInC,OAAI,MAAM,SAAS,QACjB,eAAc;;AAKlB,cACQ;GAAC,MAAM;GAAM,MAAM;GAAM,MAAM;GAAQ,QACvC;AACJ,YAAS,cAAc;KAEzB,EAAE,WAAW,MAAM,CACpB;AAGD,kBAAgB;AACd,kBAAe;AAEf,OAAI,CAAC,SAAS,SAAS,MAAM,SAAS,QACpC,MAAK,OAAO;IAEd;AAGF,WAAa;GAEX,UAAU;GAEV,UAAU,SAAS,SAAS;GAE5B,cAAc,SAAS,aAAa;GAEpC,IAAI;GACL,CAAC;;uBA9XA,mBA8FM,OAAA;aA7FA;IAAJ,KAAI;IACJ,OAAK,eAAA,CAAC,UAAQ,YACcA,KAAAA;0BAA+CC,KAAAA;wBAAuCC,KAAAA;sBAAmC,SAAA;;IAQpJ,OAAK,eAAE,UAAA,MAAS;IAChB,OAAOC,KAAAA,SAAS,gBAAA;IAChB,cAAYC,KAAAA,aAAa,gBAAA;IACzB,MAAMH,KAAAA,YAAS,WAAA;IACf,UAAUA,KAAAA,YAAS,IAAA;IACnB,SAAO;IACP,WAAS;;IAEV,mBAAA,wBAA4B;KAETD,KAAAA,SAAI,aAAA,CAAmBA,KAAAA,SAAI,CAAM,SAAA,SAAQ,OAAWK,KAAAA,SAAI,yBAD3E,YAUE,MAAA,KAAA,EAAA;;KANC,MAAMA,KAAAA;KACN,OAAOC,KAAAA;KACP,QAAQA,KAAAA;KACR,OAAOC,KAAAA;KACP,OAAK,eAAE,UAAA,MAAS;KAChB,SAAK,OAAA,OAAA,OAAA,WAAQ,YAAW,WAAA,gBAAA;;;;;;;UAKdP,KAAAA,SAAI,YAAA,CAAkB,SAAA,SAAQ,OAAWK,KAAAA,SAAI,yBAD1D,mBAKE,UAAA,EAAA,KAAA,GAAA,EAAA,CANF,mBAAA,iBAAqB,EACrB,mBAKE,OAAA;KAHC,OAAK,eAAA,CAAGA,KAAAA,MAAM,YAAA,MAAW,CAAA;KACzB,OAAK,eAAE,UAAA,MAAS;KAChB,SAAK,OAAA,OAAA,OAAA,WAAQ,YAAW,UAAA,eAAA;4BAKdL,KAAAA,SAAI,eAAoB,kBAAA,SAAiB,CAAK,SAAA,sBAD3D,mBAME,UAAA,EAAA,KAAA,GAAA,EAAA,CAPF,mBAAA,eAAmB,gBACnB,YAME,wBAJK,kBAAA,MAAiB,EAFxB,WAME,EAHC,OAAO,UAAA,OAAS,EACTQ,KAAAA,gBAAc,EACrB,SAAK,OAAA,OAAA,OAAA,WAAQ,YAAW,aAAA,WAAA,uCAKdR,KAAAA,SAAI,SAAcS,KAAAA,WAAO,CAAK,SAAA,sBAD3C,mBAUM,UAAA,EAAA,KAAA,GAAA,EAAA,CAXN,mBAAA,gBAAoB,gBACpB,mBAUM,OAAA;KARH,OAAOH,KAAAA;KACP,QAAQA,KAAAA;KACR,SAASI,KAAAA;KACT,OAAK,eAAE,UAAA,MAAS;KAChB,MAAMH,KAAAA;KACN,SAAK,OAAA,OAAA,OAAA,WAAQ,YAAW,OAAA,YAAA;QAEzB,mBAAmC,QAAA;KAA5B,GAAGE,KAAAA;KAAU,MAAMF,KAAAA;0DAKfP,KAAAA,SAAI,WAAgB,SAAA,SAAQ,CAAK,SAAA,SAAQ,CAAK,aAAA,sBAD3D,mBASE,UAAA,EAAA,KAAA,GAAA,EAAA,CAVF,mBAAA,sBAA0B,EAC1B,mBASE,OAAA;KAPC,KAAK,SAAA;KACL,KAAKW,KAAAA,OAAO,gBAAA;KACZ,OAAOL,KAAAA;KACP,QAAQA,KAAAA;KACT,OAAA;MAAA,WAAA;MAAA,kBAAA;MAAqD;KACpD,SAAK,OAAA,OAAA,OAAA,WAAQ,YAAW,SAAA,WAAA;KACxB,QAAM;;IAGT,mBAAA,WAAe;KAENJ,KAAAA,WAAW,aAAA,UAAY,CAAM,SAAA,sBADtC,mBAMM,OAAA;;KAJJ,OAAM;KACL,OAAK,eAAE,UAAA,MAAS;kCAEjB,mBAA+B,OAAA,EAA1B,OAAM,mBAAiB,EAAA,MAAA,GAAA,UAKjB,SAAA,sBADb,mBASM,UAAA,EAAA,KAAA,GAAA,EAAA,CAVN,mBAAA,aAAiB,EACjB,mBASM,OAAA;KAPJ,OAAM;KACL,OAAK,eAAE,UAAA,MAAS;KAChB,OAAK,WAAa,gBAAA,MAAe,IAAK,aAAA,MAAY,GAAcU,KAAAA,eAAY,eAAkBA,KAAAA,aAAY,KAAA;OAG5G,QAED,IAAA,WAAA"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"C_ImageCropper-BVJfUufl.css","names":[],"sources":["../src/components/C_ImageCropper/components/CropperToolbar.vue?vue&type=style&index=0&scoped=499ffd01&lang.scss","../src/components/C_ImageCropper/components/CropperPreview.vue?vue&type=style&index=0&scoped=cefefdc0&lang.scss","../src/components/C_ImageCropper/index.vue?vue&type=style&index=0&scoped=692ad231&lang.scss"],"sourcesContent":[".cropper-toolbar[data-v-499ffd01] {\n display: flex;\n flex-wrap: wrap;\n gap: 4px;\n align-items: center;\n padding: 8px 0;\n}",".cropper-preview[data-v-cefefdc0] {\n display: flex;\n flex-direction: column;\n gap: 16px;\n}\n.cropper-preview .preview-label[data-v-cefefdc0] {\n display: flex;\n gap: 6px;\n align-items: center;\n font-size: 13px;\n font-weight: 500;\n color: var(--text-color-2);\n}\n.cropper-preview .preview-main[data-v-cefefdc0] {\n width: 100%;\n overflow: hidden;\n border: 1px solid var(--border-color);\n border-radius: 6px;\n background: linear-gradient(45deg, #f0f0f0 25%, transparent 25%), linear-gradient(-45deg, #f0f0f0 25%, transparent 25%), linear-gradient(45deg, transparent 75%, #f0f0f0 75%), linear-gradient(-45deg, transparent 75%, #f0f0f0 75%);\n background-position: 0 0, 0 8px, 8px -8px, -8px 0;\n background-size: 16px 16px;\n}\n.cropper-preview .preview-main--circular[data-v-cefefdc0] {\n border-radius: 50%;\n}\n.cropper-preview .preview-main__viewport img[data-v-cefefdc0] {\n display: block;\n max-width: none !important;\n}\n.cropper-preview .preview-empty[data-v-cefefdc0] {\n display: flex;\n align-items: center;\n justify-content: center;\n width: 100%;\n height: 100%;\n min-height: 120px;\n}\n.cropper-preview .preview-thumbs[data-v-cefefdc0] {\n display: flex;\n gap: 12px;\n align-items: flex-end;\n justify-content: center;\n}\n.cropper-preview .preview-thumb[data-v-cefefdc0] {\n display: flex;\n flex-direction: column;\n gap: 4px;\n align-items: center;\n}\n.cropper-preview .preview-thumb__box[data-v-cefefdc0] {\n overflow: hidden;\n border: 1px solid var(--border-color);\n border-radius: 4px;\n background: var(--body-color);\n}\n.cropper-preview .preview-thumb__box--circular[data-v-cefefdc0] {\n border-radius: 50%;\n}\n.cropper-preview .preview-thumb__viewport img[data-v-cefefdc0] {\n display: block;\n max-width: none !important;\n}\n.cropper-preview .preview-thumb__label[data-v-cefefdc0] {\n font-size: 11px;\n color: var(--text-color-3);\n}","/* C_ImageCropper styles */\n.c-image-cropper[data-v-692ad231] {\n width: 100%;\n}\n.c-image-cropper__body[data-v-692ad231] {\n display: flex;\n flex-direction: column;\n gap: 8px;\n}\n.c-image-cropper__workspace[data-v-692ad231] {\n display: flex;\n gap: 16px;\n align-items: stretch;\n}\n.c-image-cropper__canvas[data-v-692ad231] {\n position: relative;\n flex: 1;\n min-width: 0;\n overflow: hidden;\n border: 1px solid var(--border-color);\n border-radius: 6px;\n background: var(--body-color);\n}\n.c-image-cropper__canvas[data-v-692ad231] .vue-cropper {\n width: 100% !important;\n height: 100% !important;\n}\n.c-image-cropper__preview-panel[data-v-692ad231] {\n flex: 0 1 280px;\n min-width: 200px;\n padding: 16px;\n overflow: hidden;\n border: 1px solid var(--border-color);\n border-radius: 6px;\n background: var(--card-color);\n}\n.c-image-cropper__placeholder[data-v-692ad231] {\n display: flex;\n flex-direction: column;\n gap: 8px;\n align-items: center;\n justify-content: center;\n height: 100%;\n color: var(--text-color-3);\n font-size: 14px;\n}\n@media (width <= 768px) {\n.c-image-cropper__workspace[data-v-692ad231] {\n flex-direction: column;\n}\n.c-image-cropper__preview-panel[data-v-692ad231] {\n max-width: none;\n}\n}"],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA"}
@@ -0,0 +1,6 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ require('./C_Icon.js');
3
+ const require_C_ImageCropper = require('./C_ImageCropper.js');
4
+
5
+ exports.C_ImageCropper = require_C_ImageCropper.C_ImageCropper_default;
6
+ exports.useCropperCore = require_C_ImageCropper.useCropperCore;
@@ -0,0 +1,2 @@
1
+ import { a as CropResult, c as ImageCropperProps, i as CropOutputFormat, n as _default, o as ImageCropperEmits, r as AspectRatioPreset, s as ImageCropperExpose, t as useCropperCore } from "./useCropperCore.js";
2
+ export { type AspectRatioPreset, _default as C_ImageCropper, type CropOutputFormat, type CropResult, type ImageCropperEmits, type ImageCropperExpose, type ImageCropperProps, useCropperCore };
@@ -0,0 +1,2 @@
1
+ import { a as CropResult, c as ImageCropperProps, i as CropOutputFormat, n as _default, o as ImageCropperEmits, r as AspectRatioPreset, s as ImageCropperExpose, t as useCropperCore } from "./useCropperCore.js";
2
+ export { type AspectRatioPreset, _default as C_ImageCropper, type CropOutputFormat, type CropResult, type ImageCropperEmits, type ImageCropperExpose, type ImageCropperProps, useCropperCore };
@@ -0,0 +1,4 @@
1
+ import "./C_Icon2.js";
2
+ import { n as useCropperCore, t as C_ImageCropper_default } from "./C_ImageCropper2.js";
3
+
4
+ export { C_ImageCropper_default as C_ImageCropper, useCropperCore };