@yoka-ui/ui 1.0.4 → 1.0.6

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 (260) hide show
  1. package/@Docs-yoka/exports.generated.md +68 -63
  2. package/README.md +44 -15
  3. package/dist/es/business/AiChat/index.js.map +1 -1
  4. package/dist/es/business/AiChat/intentRecognizer.js.map +1 -1
  5. package/dist/es/business/AiChat/navigationManager.js +6 -6
  6. package/dist/es/business/AiChat/navigationManager.js.map +2 -2
  7. package/dist/es/business/DrawerPageInfo/index.js +2 -2
  8. package/dist/es/business/DrawerPageInfo/index.js.map +2 -2
  9. package/dist/es/business/Editor/index.d.ts +1 -1
  10. package/dist/es/business/Editor/index.js.map +2 -2
  11. package/dist/es/business/Empty/index.js +1 -1
  12. package/dist/es/business/Empty/index.js.map +1 -1
  13. package/dist/es/business/ModCommonFilter/components/PopoverContent/Category.js +2 -2
  14. package/dist/es/business/ModCommonFilter/components/PopoverContent/Category.js.map +2 -2
  15. package/dist/es/business/ModCommonFilter/components/PopoverContent/Content.js +3 -3
  16. package/dist/es/business/ModCommonFilter/components/PopoverContent/Content.js.map +2 -2
  17. package/dist/es/business/ModCommonFilter/components/PopoverContent/Selected.js +2 -2
  18. package/dist/es/business/ModCommonFilter/components/PopoverContent/Selected.js.map +2 -2
  19. package/dist/es/business/ModCommonFilter/index.d.ts +1 -1
  20. package/dist/es/business/ModCommonFilter/index.js.map +2 -2
  21. package/dist/es/business/YkLoginModule/SmsLoginForm.d.ts +25 -0
  22. package/dist/es/business/YkLoginModule/SmsLoginForm.js +178 -0
  23. package/dist/es/business/YkLoginModule/SmsLoginForm.js.map +7 -0
  24. package/dist/es/business/YkLoginModule/index.d.ts +48 -0
  25. package/dist/es/business/YkLoginModule/index.js +198 -0
  26. package/dist/es/business/YkLoginModule/index.js.map +7 -0
  27. package/dist/es/business/YkLoginModule/styles.module.less +169 -0
  28. package/dist/es/business/YkPorjectSelect/index.d.ts +1 -1
  29. package/dist/es/business/YkPorjectSelect/index.js +2 -2
  30. package/dist/es/business/YkPorjectSelect/index.js.map +2 -2
  31. package/dist/es/business/YkSqlEdit/code-mirror-custom.module.less +154 -0
  32. package/dist/es/business/YkSqlEdit/index.d.ts +20 -0
  33. package/dist/es/business/YkSqlEdit/index.js +180 -0
  34. package/dist/es/business/YkSqlEdit/index.js.map +7 -0
  35. package/dist/es/business/YkSqlEdit/sql-language.d.ts +11 -0
  36. package/dist/es/business/YkSqlEdit/sql-language.js +1460 -0
  37. package/dist/es/business/YkSqlEdit/sql-language.js.map +7 -0
  38. package/dist/es/components/DebounceInput/index.js.map +2 -2
  39. package/dist/es/components/MultipleSelect/index.d.ts +14 -0
  40. package/dist/es/components/MultipleSelect/index.js +1 -1
  41. package/dist/es/components/MultipleSelect/index.js.map +2 -2
  42. package/dist/es/components/RefreshButton/index.js.map +2 -2
  43. package/dist/es/components/SearchWithHistory/index.js +1 -1
  44. package/dist/es/components/SearchWithHistory/index.js.map +2 -2
  45. package/dist/es/components/TextWithToolTip/index.d.ts +1 -1
  46. package/dist/es/components/TextWithToolTip/index.js.map +2 -2
  47. package/dist/es/components/TreeTransfer/components/TreeTransferPanel/index.d.ts +1 -24
  48. package/dist/es/components/TreeTransfer/components/TreeTransferPanel/index.js +2 -2
  49. package/dist/es/components/TreeTransfer/components/TreeTransferPanel/index.js.map +2 -2
  50. package/dist/es/components/TreeTransfer/index.d.ts +1 -24
  51. package/dist/es/components/TreeTransfer/index.js +8 -8
  52. package/dist/es/components/TreeTransfer/index.js.map +2 -2
  53. package/dist/es/components/TreeTransfer/utils/index.d.ts +1 -1
  54. package/dist/es/components/TreeTransfer/utils/index.js.map +2 -2
  55. package/dist/es/components/YkDateRangePicker/index.d.ts +1 -1
  56. package/dist/es/components/YkDateRangePicker/index.js +1 -1
  57. package/dist/es/components/YkDateRangePicker/index.js.map +2 -2
  58. package/dist/es/components/YkDateRangePicker/index.module.less +2 -1
  59. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.d.ts +1 -1
  60. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.js +3 -2
  61. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.js.map +2 -2
  62. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSRange.d.ts +1 -1
  63. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSRange.js +60 -9
  64. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSRange.js.map +3 -3
  65. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSSelect.d.ts +1 -1
  66. package/dist/es/components/YkRangeDateWithVS/YkRangeDateWithVSSelect.js.map +2 -2
  67. package/dist/es/components/YkRangeDateWithVS/index.d.ts +4 -4
  68. package/dist/es/components/YkRangeDateWithVS/index.js +2 -3
  69. package/dist/es/components/YkRangeDateWithVS/index.js.map +2 -2
  70. package/dist/es/components/YkRangeDateWithVS/index.module.less +23 -4
  71. package/dist/es/components/YkRangeTimeWithRecent/index.d.ts +1 -18
  72. package/dist/es/components/YkRangeTimeWithRecent/index.js +27 -7
  73. package/dist/es/components/YkRangeTimeWithRecent/index.js.map +2 -2
  74. package/dist/es/creative/ArcCheckbox/index.d.ts +12 -0
  75. package/dist/es/creative/ArcCheckbox/index.js +49 -0
  76. package/dist/es/creative/ArcCheckbox/index.js.map +7 -0
  77. package/dist/es/creative/ArcCheckbox/index.module.less +102 -0
  78. package/dist/es/creative/ButtonRadioWithInfo/index.js.map +1 -1
  79. package/dist/es/creative/ButtonWithProgress/index.d.ts +1 -1
  80. package/dist/es/creative/ButtonWithProgress/index.js.map +2 -2
  81. package/dist/es/creative/GlassSegmentedRadio/index.d.ts +24 -0
  82. package/dist/es/creative/GlassSegmentedRadio/index.js +75 -0
  83. package/dist/es/creative/GlassSegmentedRadio/index.js.map +7 -0
  84. package/dist/es/creative/GlassSegmentedRadio/index.module.less +241 -0
  85. package/dist/es/index.d.ts +32 -26
  86. package/dist/es/index.js +90 -82
  87. package/dist/es/index.js.map +2 -2
  88. package/dist/es/index.less +5 -1
  89. package/dist/es/layout/FlexGrid/index.d.ts +1 -1
  90. package/dist/es/layout/FlexGrid/index.js.map +2 -2
  91. package/dist/es/layout/YkContainer/index.js.map +1 -1
  92. package/dist/es/layout/YkDrawer/index.d.ts +1 -1
  93. package/dist/es/layout/YkDrawer/index.js +2 -1
  94. package/dist/es/layout/YkDrawer/index.js.map +2 -2
  95. package/dist/es/ui/LabelSelect/demo.js +1 -1
  96. package/dist/es/ui/LabelSelect/demo.js.map +2 -2
  97. package/dist/es/ui/LabelSelect/index.d.ts +1 -1
  98. package/dist/es/ui/LabelSelect/index.js +1 -1
  99. package/dist/es/ui/LabelSelect/index.js.map +2 -2
  100. package/dist/es/ui/LogicOperator/index.d.ts +1 -1
  101. package/dist/es/ui/LogicOperator/index.js.map +2 -2
  102. package/dist/es/ui/YkButton/index.d.ts +1 -1
  103. package/dist/es/ui/YkButton/index.js.map +2 -2
  104. package/dist/es/ui/YkCard/index.d.ts +1 -1
  105. package/dist/es/ui/YkCard/index.js +1 -1
  106. package/dist/es/ui/YkCard/index.js.map +2 -2
  107. package/dist/es/ui/YkCheckbox/index.d.ts +1 -1
  108. package/dist/es/ui/YkCheckbox/index.js.map +2 -2
  109. package/dist/es/ui/YkDescriptions/index.d.ts +1 -1
  110. package/dist/es/ui/YkDescriptions/index.js.map +2 -2
  111. package/dist/es/ui/YkPagination/index.d.ts +1 -1
  112. package/dist/es/ui/YkPagination/index.js.map +2 -2
  113. package/dist/es/ui/YkRadio/index.d.ts +1 -1
  114. package/dist/es/ui/YkRadio/index.js.map +2 -2
  115. package/dist/es/ui/YkSegmented/index.d.ts +1 -1
  116. package/dist/es/ui/YkSegmented/index.js.map +2 -2
  117. package/dist/es/ui/YkSelect/index.d.ts +1 -1
  118. package/dist/es/ui/YkSelect/index.js.map +2 -2
  119. package/dist/es/ui/YkSpin/index.d.ts +1 -1
  120. package/dist/es/ui/YkSpin/index.js.map +2 -2
  121. package/dist/es/ui/YkStatistic/index.d.ts +1 -1
  122. package/dist/es/ui/YkStatistic/index.js.map +2 -2
  123. package/dist/es/ui/YkSwitch/index.d.ts +1 -1
  124. package/dist/es/ui/YkSwitch/index.js.map +2 -2
  125. package/dist/es/ui/YkTabs/index.d.ts +1 -1
  126. package/dist/es/ui/YkTabs/index.js.map +2 -2
  127. package/dist/es/ui/YkTooltip/index.d.ts +1 -1
  128. package/dist/es/ui/YkTooltip/index.js.map +2 -2
  129. package/dist/es/utils/styleUtils.js.map +2 -2
  130. package/dist/es/utils/ykStorybookDoc.js +1 -1
  131. package/dist/es/utils/ykStorybookDoc.js.map +1 -1
  132. package/dist/lib/business/AiChat/index.js.map +1 -1
  133. package/dist/lib/business/AiChat/intentRecognizer.js.map +1 -1
  134. package/dist/lib/business/AiChat/navigationManager.js +6 -6
  135. package/dist/lib/business/AiChat/navigationManager.js.map +2 -2
  136. package/dist/lib/business/DrawerPageInfo/index.js +1 -1
  137. package/dist/lib/business/DrawerPageInfo/index.js.map +2 -2
  138. package/dist/lib/business/Editor/index.d.ts +1 -1
  139. package/dist/lib/business/Editor/index.js.map +2 -2
  140. package/dist/lib/business/Empty/index.js +1 -1
  141. package/dist/lib/business/Empty/index.js.map +1 -1
  142. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Category.js +3 -3
  143. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Category.js.map +2 -2
  144. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Content.js +4 -4
  145. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Content.js.map +2 -2
  146. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Selected.js +3 -3
  147. package/dist/lib/business/ModCommonFilter/components/PopoverContent/Selected.js.map +2 -2
  148. package/dist/lib/business/ModCommonFilter/index.d.ts +1 -1
  149. package/dist/lib/business/ModCommonFilter/index.js.map +2 -2
  150. package/dist/lib/business/YkLoginModule/SmsLoginForm.d.ts +25 -0
  151. package/dist/lib/business/YkLoginModule/SmsLoginForm.js +171 -0
  152. package/dist/lib/business/YkLoginModule/SmsLoginForm.js.map +7 -0
  153. package/dist/lib/business/YkLoginModule/index.d.ts +48 -0
  154. package/dist/lib/business/YkLoginModule/index.js +206 -0
  155. package/dist/lib/business/YkLoginModule/index.js.map +7 -0
  156. package/dist/lib/business/YkLoginModule/styles.module.less +169 -0
  157. package/dist/lib/business/YkPorjectSelect/index.d.ts +1 -1
  158. package/dist/lib/business/YkPorjectSelect/index.js +3 -3
  159. package/dist/lib/business/YkPorjectSelect/index.js.map +2 -2
  160. package/dist/lib/business/YkSqlEdit/code-mirror-custom.module.less +154 -0
  161. package/dist/lib/business/YkSqlEdit/index.d.ts +20 -0
  162. package/dist/lib/business/YkSqlEdit/index.js +202 -0
  163. package/dist/lib/business/YkSqlEdit/index.js.map +7 -0
  164. package/dist/lib/business/YkSqlEdit/sql-language.d.ts +11 -0
  165. package/dist/lib/business/YkSqlEdit/sql-language.js +1493 -0
  166. package/dist/lib/business/YkSqlEdit/sql-language.js.map +7 -0
  167. package/dist/lib/components/DebounceInput/index.js.map +2 -2
  168. package/dist/lib/components/MultipleSelect/index.d.ts +14 -0
  169. package/dist/lib/components/MultipleSelect/index.js +1 -1
  170. package/dist/lib/components/MultipleSelect/index.js.map +2 -2
  171. package/dist/lib/components/RefreshButton/index.js.map +2 -2
  172. package/dist/lib/components/SearchWithHistory/index.js +1 -1
  173. package/dist/lib/components/SearchWithHistory/index.js.map +2 -2
  174. package/dist/lib/components/TextWithToolTip/index.d.ts +1 -1
  175. package/dist/lib/components/TextWithToolTip/index.js.map +2 -2
  176. package/dist/lib/components/TreeTransfer/components/TreeTransferPanel/index.d.ts +1 -24
  177. package/dist/lib/components/TreeTransfer/components/TreeTransferPanel/index.js +2 -2
  178. package/dist/lib/components/TreeTransfer/components/TreeTransferPanel/index.js.map +2 -2
  179. package/dist/lib/components/TreeTransfer/index.d.ts +1 -24
  180. package/dist/lib/components/TreeTransfer/index.js +3 -3
  181. package/dist/lib/components/TreeTransfer/index.js.map +2 -2
  182. package/dist/lib/components/TreeTransfer/utils/index.d.ts +1 -1
  183. package/dist/lib/components/TreeTransfer/utils/index.js.map +2 -2
  184. package/dist/lib/components/YkDateRangePicker/index.d.ts +1 -1
  185. package/dist/lib/components/YkDateRangePicker/index.js +1 -1
  186. package/dist/lib/components/YkDateRangePicker/index.js.map +2 -2
  187. package/dist/lib/components/YkDateRangePicker/index.module.less +2 -1
  188. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.d.ts +1 -1
  189. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.js +3 -2
  190. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSCompare.js.map +2 -2
  191. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSRange.d.ts +1 -1
  192. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSRange.js +59 -8
  193. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSRange.js.map +3 -3
  194. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSSelect.d.ts +1 -1
  195. package/dist/lib/components/YkRangeDateWithVS/YkRangeDateWithVSSelect.js.map +2 -2
  196. package/dist/lib/components/YkRangeDateWithVS/index.d.ts +4 -4
  197. package/dist/lib/components/YkRangeDateWithVS/index.js +4 -5
  198. package/dist/lib/components/YkRangeDateWithVS/index.js.map +3 -3
  199. package/dist/lib/components/YkRangeDateWithVS/index.module.less +23 -4
  200. package/dist/lib/components/YkRangeTimeWithRecent/index.d.ts +1 -18
  201. package/dist/lib/components/YkRangeTimeWithRecent/index.js +27 -7
  202. package/dist/lib/components/YkRangeTimeWithRecent/index.js.map +2 -2
  203. package/dist/lib/creative/ArcCheckbox/index.d.ts +12 -0
  204. package/dist/lib/creative/ArcCheckbox/index.js +50 -0
  205. package/dist/lib/creative/ArcCheckbox/index.js.map +7 -0
  206. package/dist/lib/creative/ArcCheckbox/index.module.less +102 -0
  207. package/dist/lib/creative/ButtonRadioWithInfo/index.js.map +1 -1
  208. package/dist/lib/creative/ButtonWithProgress/index.d.ts +1 -1
  209. package/dist/lib/creative/ButtonWithProgress/index.js.map +2 -2
  210. package/dist/lib/creative/GlassSegmentedRadio/index.d.ts +24 -0
  211. package/dist/lib/creative/GlassSegmentedRadio/index.js +78 -0
  212. package/dist/lib/creative/GlassSegmentedRadio/index.js.map +7 -0
  213. package/dist/lib/creative/GlassSegmentedRadio/index.module.less +241 -0
  214. package/dist/lib/index.d.ts +32 -26
  215. package/dist/lib/index.js +37 -25
  216. package/dist/lib/index.js.map +2 -2
  217. package/dist/lib/index.less +5 -1
  218. package/dist/lib/layout/FlexGrid/index.d.ts +1 -1
  219. package/dist/lib/layout/FlexGrid/index.js.map +2 -2
  220. package/dist/lib/layout/YkContainer/index.js.map +1 -1
  221. package/dist/lib/layout/YkDrawer/index.d.ts +1 -1
  222. package/dist/lib/layout/YkDrawer/index.js +2 -1
  223. package/dist/lib/layout/YkDrawer/index.js.map +2 -2
  224. package/dist/lib/ui/LabelSelect/demo.js +1 -1
  225. package/dist/lib/ui/LabelSelect/demo.js.map +2 -2
  226. package/dist/lib/ui/LabelSelect/index.d.ts +1 -1
  227. package/dist/lib/ui/LabelSelect/index.js +1 -1
  228. package/dist/lib/ui/LabelSelect/index.js.map +2 -2
  229. package/dist/lib/ui/LogicOperator/index.d.ts +1 -1
  230. package/dist/lib/ui/LogicOperator/index.js.map +2 -2
  231. package/dist/lib/ui/YkButton/index.d.ts +1 -1
  232. package/dist/lib/ui/YkButton/index.js.map +2 -2
  233. package/dist/lib/ui/YkCard/index.d.ts +1 -1
  234. package/dist/lib/ui/YkCard/index.js.map +2 -2
  235. package/dist/lib/ui/YkCheckbox/index.d.ts +1 -1
  236. package/dist/lib/ui/YkCheckbox/index.js.map +2 -2
  237. package/dist/lib/ui/YkDescriptions/index.d.ts +1 -1
  238. package/dist/lib/ui/YkDescriptions/index.js.map +2 -2
  239. package/dist/lib/ui/YkPagination/index.d.ts +1 -1
  240. package/dist/lib/ui/YkPagination/index.js.map +2 -2
  241. package/dist/lib/ui/YkRadio/index.d.ts +1 -1
  242. package/dist/lib/ui/YkRadio/index.js.map +2 -2
  243. package/dist/lib/ui/YkSegmented/index.d.ts +1 -1
  244. package/dist/lib/ui/YkSegmented/index.js.map +2 -2
  245. package/dist/lib/ui/YkSelect/index.d.ts +1 -1
  246. package/dist/lib/ui/YkSelect/index.js.map +2 -2
  247. package/dist/lib/ui/YkSpin/index.d.ts +1 -1
  248. package/dist/lib/ui/YkSpin/index.js.map +2 -2
  249. package/dist/lib/ui/YkStatistic/index.d.ts +1 -1
  250. package/dist/lib/ui/YkStatistic/index.js.map +2 -2
  251. package/dist/lib/ui/YkSwitch/index.d.ts +1 -1
  252. package/dist/lib/ui/YkSwitch/index.js.map +2 -2
  253. package/dist/lib/ui/YkTabs/index.d.ts +1 -1
  254. package/dist/lib/ui/YkTabs/index.js.map +2 -2
  255. package/dist/lib/ui/YkTooltip/index.d.ts +1 -1
  256. package/dist/lib/ui/YkTooltip/index.js.map +2 -2
  257. package/dist/lib/utils/styleUtils.js.map +2 -2
  258. package/dist/lib/utils/ykStorybookDoc.js +1 -1
  259. package/dist/lib/utils/ykStorybookDoc.js.map +1 -1
  260. package/package.json +43 -45
@@ -1,93 +1,98 @@
1
1
  # @yoka-ui/ui 具名导出表
2
2
 
3
3
  > **自动生成**:`pnpm run generate:yoka-llms`(源文件 `src/index.tsx`)
4
- > **包版本**:`1.0.4`
5
- > **生成时间**:`2026-04-09T11:51:12.277Z`
4
+ > **包版本**:`1.0.6`
5
+ > **生成时间**:`2026-04-20T11:13:15.402Z`
6
6
  > 请勿手动编辑本文件;修改导出请改 `src/index.tsx` 后重新运行上述命令。
7
7
 
8
8
  ## 使用约定(给 LLM / 业务开发者)
9
9
 
10
- - **包名**:`@yoka-ui/ui`(npm 公网);可选 Git 安装:`@yoka-ui/ui`: `git+http://gitlab.sh.com/web/yoka-ui.git#v1.0.4` 等,以实际 tag 为准
11
- - **导入**:仅使用下表中的**具名**导出,与发布产物 `dist/es/index.d.ts` 一致;勿臆造未列出符号。
10
+ - **包名**:`@yoka-ui/ui`(npm 公网);可选 Git 安装:`@yoka-ui/ui`: `git+http://gitlab.sh.com/web/yoka-ui.git#v1.0.6` 等,以实际 tag 为准
11
+ - **导入**:仅使用下表中的**具名**导出(组件/值与 `export type` 类型),与发布产物 `dist/es/index.d.ts` 一致;勿臆造未列出符号。
12
12
  - **全局样式(常用)**:业务入口引入一行,例如 `import '@yoka-ui/ui/dist/index.less'`(若以 README / 实际产物路径为准)。
13
13
  - **技术栈**:React 18、Ant Design 5;复杂表单/表格优先使用库内封装(`InputTheme`、`TableTheme`、`ModCommonFilter` 等),需要原生 antd API 时查阅 [Ant Design 文档](https://ant.design) 或本仓库 `@Docs/llms.txt`(克隆源码时可用)。
14
14
 
15
15
  ## 导出清单(按 `src/index.tsx` 分区)
16
16
 
17
- ### Export all UI components
17
+ ### Export business components
18
18
 
19
- | 导出符号 | 源码 `from` 路径 |
20
- | --- | --- |
21
- | `LabelSelect` | `./ui/LabelSelect` |
22
- | `LogicOperator` | `./ui/LogicOperator` |
23
- | `YkButton` | `./ui/YkButton` |
24
- | `YkCard` | `./ui/YkCard` |
25
- | `YkCheckbox` | `./ui/YkCheckbox` |
26
- | `YkDescriptions` | `./ui/YkDescriptions` |
27
- | `YkPagination` | `./ui/YkPagination` |
28
- | `YkRadio` | `./ui/YkRadio` |
29
- | `YkRadioBtnSpecial` | `./ui/YkRadioBtnSpecial` |
30
- | `YkSegmented` | `./ui/YkSegmented` |
31
- | `YkSelect` | `./ui/YkSelect` |
32
- | `YkSpin` | `./ui/YkSpin` |
33
- | `YkStatistic` | `./ui/YkStatistic` |
34
- | `YkSwitch` | `./ui/YkSwitch` |
35
- | `YkTabs` | `./ui/YkTabs` |
36
- | `YkTooltip` | `./ui/YkTooltip` |
19
+ | 导出符号 | 类别 | 源码 `from` 路径 |
20
+ | --- | --- | --- |
21
+ | `AiChat` | 组件 | `./business/AiChat` |
22
+ | `DrawerPageInfo` | 组件 | `./business/DrawerPageInfo` |
23
+ | `Editor` | 组件 | `./business/Editor` |
24
+ | `Empty` | 组件 | `./business/Empty` |
25
+ | `ModCommonFilter` | 组件 | `./business/ModCommonFilter` |
26
+ | `YkLoginModule` | 组件 | `./business/YkLoginModule` |
27
+ | `YkPorjectSelect` | 组件 | `./business/YkPorjectSelect` |
28
+ | `YkSqlEdit` | 组件 | `./business/YkSqlEdit` |
37
29
 
38
- ### Export business components
30
+ ### Export common components
39
31
 
40
- | 导出符号 | 源码 `from` 路径 |
41
- | --- | --- |
42
- | `DrawerPageInfo` | `./business/DrawerPageInfo` |
43
- | `Editor` | `./business/Editor` |
44
- | `Empty` | `./business/Empty` |
45
- | `ModCommonFilter` | `./business/ModCommonFilter` |
46
- | `YkPorjectSelect` | `./business/YkPorjectSelect` |
47
- | `YkDateRangePicker` | `./components/YkDateRangePicker` |
48
- | `YkRangeDateWithVS` | `./components/YkRangeDateWithVS` |
49
- | `YkRangeTimeWithRecent` | `./components/YkRangeTimeWithRecent` |
32
+ | 导出符号 | 类别 | 源码 `from` 路径 |
33
+ | --- | --- | --- |
34
+ | `Clock` | 组件 | `./components/Clock` |
35
+ | `DebounceInput` | 组件 | `./components/DebounceInput` |
36
+ | `MultipleSelect` | 组件 | `./components/MultipleSelect` |
37
+ | `NumericInput` | 组件 | `./components/NumericInput` |
38
+ | `RefreshButton` | 组件 | `./components/RefreshButton` |
39
+ | `SearchWithHistory` | 组件 | `./components/SearchWithHistory` |
40
+ | `TextWithInput` | 组件 | `./components/TextWithInput` |
41
+ | `TextWithToolTip` | 组件 | `./components/TextWithToolTip` |
42
+ | `TreeTransfer` | 组件 | `./components/TreeTransfer` |
43
+ | `YkDateRangePicker` | 组件 | `./components/YkDateRangePicker` |
44
+ | `YkRangeDateWithVS` | 组件 | `./components/YkRangeDateWithVS` |
45
+ | `YkRangeTimeWithRecent` | 组件 | `./components/YkRangeTimeWithRecent` |
50
46
 
51
47
  ### Export creative components
52
48
 
53
- | 导出符号 | 源码 `from` 路径 |
54
- | --- | --- |
55
- | `ButtonRadioWithInfo` | `./creative/ButtonRadioWithInfo` |
56
- | `ButtonWithProgress` | `./creative/ButtonWithProgress` |
49
+ | 导出符号 | 类别 | 源码 `from` 路径 |
50
+ | --- | --- | --- |
51
+ | `ArcCheckboxProps` | 类型 | `./creative/ArcCheckbox` |
52
+ | `ArcCheckbox` | 组件 | `./creative/ArcCheckbox` |
53
+ | `ButtonRadioWithInfo` | 组件 | `./creative/ButtonRadioWithInfo` |
54
+ | `ButtonWithProgress` | 组件 | `./creative/ButtonWithProgress` |
55
+ | `GlassSegmentedRadio` | 组件 | `./creative/GlassSegmentedRadio` |
57
56
 
58
- ### Export theme components
57
+ ### Export layout
59
58
 
60
- | 导出符号 | 源码 `from` 路径 |
61
- | --- | --- |
62
- | `InputTheme` | `./Themes/InputTheme` |
63
- | `TableTheme` | `./Themes/TableTheme` |
59
+ | 导出符号 | 类别 | 源码 `from` 路径 |
60
+ | --- | --- | --- |
61
+ | `FlexGrid` | 组件 | `./layout/FlexGrid` |
62
+ | `YkContainer` | 组件 | `./layout/YkContainer` |
63
+ | `YkDrawer` | 组件 | `./layout/YkDrawer` |
64
64
 
65
- ### Export layout
65
+ ### Export theme components
66
66
 
67
- | 导出符号 | 源码 `from` 路径 |
68
- | --- | --- |
69
- | `FlexGrid` | `./layout/FlexGrid` |
70
- | `YkContainer` | `./layout/YkContainer` |
71
- | `YkDrawer` | `./layout/YkDrawer` |
67
+ | 导出符号 | 类别 | 源码 `from` 路径 |
68
+ | --- | --- | --- |
69
+ | `InputTheme` | 组件 | `./Themes/InputTheme` |
70
+ | `TableTheme` | 组件 | `./Themes/TableTheme` |
72
71
 
73
- ### Export common components
72
+ ### Export all UI components
74
73
 
75
- | 导出符号 | 源码 `from` 路径 |
76
- | --- | --- |
77
- | `Clock` | `./components/Clock` |
78
- | `DebounceInput` | `./components/DebounceInput` |
79
- | `MultipleSelect` | `./components/MultipleSelect` |
80
- | `NumericInput` | `./components/NumericInput` |
81
- | `RefreshButton` | `./components/RefreshButton` |
82
- | `SearchWithHistory` | `./components/SearchWithHistory` |
83
- | `TextWithInput` | `./components/TextWithInput` |
84
- | `TextWithToolTip` | `./components/TextWithToolTip` |
85
- | `TreeTransfer` | `./components/TreeTransfer` |
86
- | `AiChat` | `./business/AiChat` |
74
+ | 导出符号 | 类别 | 源码 `from` 路径 |
75
+ | --- | --- | --- |
76
+ | `LabelSelect` | 组件 | `./ui/LabelSelect` |
77
+ | `LogicOperator` | 组件 | `./ui/LogicOperator` |
78
+ | `YkButton` | 组件 | `./ui/YkButton` |
79
+ | `YkCard` | 组件 | `./ui/YkCard` |
80
+ | `YkCheckbox` | 组件 | `./ui/YkCheckbox` |
81
+ | `YkDescriptions` | 组件 | `./ui/YkDescriptions` |
82
+ | `YkPagination` | 组件 | `./ui/YkPagination` |
83
+ | `YkRadio` | 组件 | `./ui/YkRadio` |
84
+ | `YkRadioBtnSpecial` | 组件 | `./ui/YkRadioBtnSpecial` |
85
+ | `YkSegmented` | 组件 | `./ui/YkSegmented` |
86
+ | `YkSelect` | 组件 | `./ui/YkSelect` |
87
+ | `YkSpin` | 组件 | `./ui/YkSpin` |
88
+ | `YkStatistic` | 组件 | `./ui/YkStatistic` |
89
+ | `YkSwitch` | 组件 | `./ui/YkSwitch` |
90
+ | `YkTabs` | 组件 | `./ui/YkTabs` |
91
+ | `YkTooltip` | 组件 | `./ui/YkTooltip` |
87
92
 
88
93
  ## 纯符号列表(便于检索)
89
94
 
90
95
  ```
91
- LabelSelect, LogicOperator, YkButton, YkCard, YkCheckbox, YkDescriptions, YkPagination, YkRadio, YkRadioBtnSpecial, YkSegmented, YkSelect, YkSpin, YkStatistic, YkSwitch, YkTabs, YkTooltip, DrawerPageInfo, Editor, Empty, ModCommonFilter, YkPorjectSelect, YkDateRangePicker, YkRangeDateWithVS, YkRangeTimeWithRecent, ButtonRadioWithInfo, ButtonWithProgress, InputTheme, TableTheme, FlexGrid, YkContainer, YkDrawer, Clock, DebounceInput, MultipleSelect, NumericInput, RefreshButton, SearchWithHistory, TextWithInput, TextWithToolTip, TreeTransfer, AiChat
96
+ AiChat, DrawerPageInfo, Editor, Empty, ModCommonFilter, YkLoginModule, YkPorjectSelect, YkSqlEdit, Clock, DebounceInput, MultipleSelect, NumericInput, RefreshButton, SearchWithHistory, TextWithInput, TextWithToolTip, TreeTransfer, YkDateRangePicker, YkRangeDateWithVS, YkRangeTimeWithRecent, ArcCheckboxProps, ArcCheckbox, ButtonRadioWithInfo, ButtonWithProgress, GlassSegmentedRadio, FlexGrid, YkContainer, YkDrawer, InputTheme, TableTheme, LabelSelect, LogicOperator, YkButton, YkCard, YkCheckbox, YkDescriptions, YkPagination, YkRadio, YkRadioBtnSpecial, YkSegmented, YkSelect, YkSpin, YkStatistic, YkSwitch, YkTabs, YkTooltip
92
97
  ```
93
98
 
package/README.md CHANGED
@@ -1,16 +1,40 @@
1
- ## 安装使用
1
+ ## 安装与使用
2
2
 
3
3
  ### 安装组件库
4
4
 
5
5
  ```bash
6
+ # npm
7
+ npm i @yoka-ui/ui
8
+
9
+ # 或 pnpm
6
10
  pnpm add @yoka-ui/ui
7
11
  ```
8
12
 
13
+ ### (推荐)配置 Cursor / IDE 规则
14
+
15
+ 在项目的 Cursor rules 中创建文件 `yoka-ui.mdc`,用于约束导入方式与文档入口(导出清单、LLM 索引等):
16
+
17
+ ```md
18
+ ---
19
+ description: 使用 @yoka/ui 时的导入与文档约定
20
+ globs: "**/*.{ts,tsx}"
21
+ alwaysApply: false
22
+ ---
23
+
24
+ # @yoka/ui
25
+
26
+ - **导入**:仅从 `@yoka/ui` 做**具名** `import`,勿使用未在导出表中出现的符号。
27
+ - **导出清单(权威)**:以安装包内 `node_modules/@yoka/ui/@Docs-yoka/exports.generated.md` 为准,与发布产物类型一致;该文件由 yoka-ui 在导出变更后重新生成。
28
+ - **LLM / 助手入口**:同目录 `node_modules/@yoka/ui/@Docs-yoka/llms.txt`(索引、外链与使用规则摘要)。
29
+ - **API**:不确定 props 时查组件库 Storybook / 源码示例或 README,勿臆造 API;原生 antd 行为对照 [Ant Design 文档](https://ant.design)。
30
+ - **全局样式**:业务入口按需引入,常见为 `import '@yoka/ui/dist/index.less'`(以 `@yoka/ui` 的 README 与当前产物路径为准)。
31
+ ```
32
+
9
33
  ### 基础用法
10
34
 
11
35
  ```tsx
12
- import { YkButton, YkCard } from '@yoka-ui/ui';
13
- import '@yoka-ui/ui/dist/index.less'; // 引入样式
36
+ import { YkButton, YkCard } from "@yoka-ui/ui";
37
+ import "@yoka-ui/ui/dist/index.less"; // 引入样式
14
38
 
15
39
  function App() {
16
40
  return (
@@ -21,13 +45,14 @@ function App() {
21
45
  }
22
46
  ```
23
47
 
24
- ### 按需引入
48
+ ### 导入约定(重要)
25
49
 
26
50
  ```tsx
27
- // 完整引入
28
- import { YkButton } from '@yoka-ui/ui';
51
+ // 仅从包入口做“具名导入”
52
+ import { YkPorjectSelect } from "@yoka-ui/ui";
29
53
 
30
- // tree-shaking 会自动排除未使用的组件
54
+ // 具体有哪些符号可用,以 @Docs-yoka/exports.generated.md 为准
55
+ // node_modules/@yoka-ui/ui/@Docs-yoka/exports.generated.md
31
56
  ```
32
57
 
33
58
  ## 组件列表
@@ -96,18 +121,22 @@ import { YkButton } from '@yoka-ui/ui';
96
121
 
97
122
  ### 业务组件 (`src/business/`)
98
123
 
99
- | 组件 | 说明 |
100
- | --------------- | --------------------------------------------------- |
101
- | DrawerPageInfo | 抽屉页面信息 |
102
- | Editor | 富文本编辑器 |
103
- | Empty | 空状态 |
104
- | ModCommonFilter | 通用筛选器 |
105
- | YkPorjectSelect | 项目选择器(搜索、关注、在营/关服 Tab、最近访问等) |
124
+ | 组件 | 说明 |
125
+ | --------------- | -------------------------------------------------------------------- |
126
+ | DrawerPageInfo | 抽屉页面信息 |
127
+ | Editor | 富文本编辑器 |
128
+ | Empty | 空状态 |
129
+ | ModCommonFilter | 通用筛选器 |
130
+ | YkLoginModule | 登录组件(当前支持:扫码 / 短信验证码;Tabs 由 `tabs` 配置) |
131
+ | YkPorjectSelect | 项目选择器(搜索、关注、在营/关服 Tab、最近访问等) |
132
+ | YkSqlEdit | SQL 编辑器(CodeMirror 6:动态参数 `{{...}}` 高亮、关键词/函数补全) |
106
133
 
107
134
  **导出说明**:业务项目里 `import { ... } from '@yoka-ui/ui'` 可用的符号以 `[src/index.tsx](src/index.tsx)` 为准。若某组件已在源码目录中实现但尚未在入口文件中 `export`,需先在组件库补齐导出并发布后,再在业务侧引用。
108
135
 
109
136
  ### LLM / 助手文档(`@Docs-yoka`)
110
137
 
111
138
  - 入口索引:[`@Docs-yoka/llms.txt`](@Docs-yoka/llms.txt)(导航 + 使用约定,结构对齐仓库内 Ant Design [@Docs/llms.txt](@Docs/llms.txt) 思路)。
112
- - **具名导出表**:[`@Docs-yoka/exports.generated.md`](@Docs-yoka/exports.generated.md) 由脚本根据 `src/index.tsx` 自动生成;修改导出后请执行 `pnpm run generate:yoka-llms`(发版时 `prepack` 会执行一次以保证包内为最新)。
139
+ - **具名导出表**:[`@Docs-yoka/exports.generated.md`](@Docs-yoka/exports.generated.md) 由脚本根据 `src/index.tsx` 自动生成;修改导出后请执行
140
+ #### `pnpm run generate:yoka-llms`
141
+ (发版时 `prepack` 会执行一次以保证包内为最新)。
113
142
  - 安装 npm 包后路径示例:`node_modules/@yoka-ui/ui/@Docs-yoka/llms.txt`(需 `package.json` 的 `files` 包含 `@Docs-yoka`)。
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/business/AiChat/index.tsx"],
4
- "sourcesContent": ["import {\n CloseOutlined,\n CompressOutlined,\n CopyOutlined,\n DeleteOutlined,\n ExpandOutlined,\n FileImageOutlined,\n MenuOutlined,\n MessageOutlined,\n PaperClipOutlined,\n PlusOutlined,\n ReloadOutlined,\n SendOutlined,\n StopOutlined,\n} from '@ant-design/icons';\nimport type { UploadFile } from 'antd';\nimport { Button, Input, List, message, Popconfirm, Space, Tooltip, Typography, Upload } from 'antd';\nimport type { CSSProperties, FC, ReactNode } from 'react';\nimport React, { useCallback, useEffect, useRef, useState } from 'react';\nimport { generateSuggestions, isTaskCreationIntent, recognizeIntent } from './intentRecognizer';\nimport MarkdownRender from './MarkdownRender';\nimport type { AiChatConfig, ChatSession, EnhancedAiChatConfig, Message } from './type';\nimport { useAiChat } from './useAiChat';\nimport { useTaskWorkflow } from './useTaskWorkflow';\n\nconst { TextArea } = Input;\nconst { Title } = Typography;\n\n// 合并Props类型\nexport interface AiChatProps extends AiChatConfig, Partial<EnhancedAiChatConfig> {\n className?: string;\n style?: CSSProperties;\n renderTrigger?: ReactNode;\n renderHeader?: ReactNode;\n renderFooter?: ReactNode;\n}\n\nconst AiChat: FC<AiChatProps> = ({\n apiPath = '/api/ai/chat-stream',\n storageKey = 'ai_chat_sessions',\n position: initialPosition = { bottom: 24, right: 24 },\n buttonSize = 56,\n zIndex = 1080,\n showSessionPanel = true,\n onBeforeSend,\n onError,\n onSuccess,\n onFetchSessions,\n onFetchSessionDetail,\n onCreateSession,\n onSendMessage,\n onUploadFile,\n className,\n style,\n renderTrigger,\n renderHeader,\n renderFooter,\n // 增强配置\n taskWorkflow,\n intentRecognition: intentConfig,\n navigation: navConfig,\n}) => {\n const {\n sessions,\n activeSession,\n loading,\n visible,\n createNewSession,\n switchSession,\n deleteSession,\n sendQuestion,\n regenerateAnswer,\n deleteSingleMsg,\n cancelRequest,\n toggleVisible,\n } = useAiChat({\n apiPath,\n storageKey,\n onBeforeSend,\n onError,\n onSuccess,\n onFetchSessions,\n onFetchSessionDetail,\n onCreateSession,\n onSendMessage,\n onUploadFile,\n });\n\n // 任务工作流\n const {\n tasks,\n createTask,\n updateTaskStatus,\n deleteTask,\n createTaskFromNaturalLanguage,\n updateUrlWithTaskId,\n refreshCurrentPage,\n executeTaskCreationWorkflow,\n } = useTaskWorkflow({\n enabled: taskWorkflow?.enabled,\n createTaskApi: taskWorkflow?.createTaskApi,\n supportedTaskTypes: taskWorkflow?.supportedTaskTypes,\n autoNavigate: taskWorkflow?.autoNavigate,\n refreshDelay: navConfig?.refreshDelay,\n taskIdParamName: taskWorkflow?.taskIdParamName,\n taskTypeParamName: taskWorkflow?.taskTypeParamName,\n onTaskCreated: (taskId, taskData) => {\n // 添加系统消息到会话\n if (activeSession) {\n const systemMsg: Message = {\n id: `${Date.now()}_sys`,\n role: 'system',\n content: `任务创建成功,ID: ${taskId}`,\n taskId,\n };\n // 这里可以添加到消息列表的逻辑\n }\n },\n onError,\n });\n\n const [inputValue, setInputValue] = useState('');\n const [showSessions, setShowSessions] = useState(false);\n const [suggestions, setSuggestions] = useState<string[]>([]);\n const [fileList, setFileList] = useState<UploadFile<any>[]>([]);\n const [isMaximized, setIsMaximized] = useState(false);\n const [dialogPosition, setDialogPosition] = useState({ x: 0, y: 0 });\n const [isDragging, setIsDragging] = useState(false);\n const dragOffset = useRef({ x: 0, y: 0 });\n const messagesEndRef = useRef<HTMLDivElement>(null);\n const triggerRef = useRef<HTMLDivElement>(null);\n const fileInputRef = useRef<HTMLInputElement>(null);\n\n useEffect(() => {\n messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' });\n }, [activeSession?.messages]);\n\n // 拖拽处理 - 拖拽整个组件位置\n const handleTriggerMouseDown = useCallback(\n (e: React.MouseEvent) => {\n if (visible) return; // 对话框展开时不处理\n const target = e.target as HTMLElement;\n if (target.closest('.ant-upload, .ant-btn')) return; // 排除按钮和上传元素\n\n e.preventDefault();\n const triggerEl = triggerRef.current;\n if (!triggerEl) return;\n\n const rect = triggerEl.getBoundingClientRect();\n dragOffset.current = {\n x: e.clientX - rect.left,\n y: e.clientY - rect.top,\n };\n setIsDragging(true);\n },\n [visible],\n );\n\n const handleTriggerMouseMove = useCallback(\n (e: MouseEvent) => {\n if (!isDragging || visible) return;\n setDialogPosition({\n x: e.clientX - dragOffset.current.x,\n y: e.clientY - dragOffset.current.y,\n });\n },\n [isDragging, visible],\n );\n\n const handleTriggerMouseUp = useCallback(() => {\n setIsDragging(false);\n }, []);\n\n useEffect(() => {\n if (isDragging && !visible) {\n document.addEventListener('mousemove', handleTriggerMouseMove);\n document.addEventListener('mouseup', handleTriggerMouseUp);\n return () => {\n document.removeEventListener('mousemove', handleTriggerMouseMove);\n document.removeEventListener('mouseup', handleTriggerMouseUp);\n };\n }\n return undefined;\n }, [isDragging, visible, handleTriggerMouseMove, handleTriggerMouseUp]);\n\n // 实时分析输入内容生成建议\n const handleInputChange = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {\n const value = e.target.value;\n setInputValue(value);\n\n if (value.length > 3) {\n const intent = recognizeIntent(value);\n const newSuggestions = generateSuggestions(value, intent);\n setSuggestions(newSuggestions);\n } else {\n setSuggestions([]);\n }\n }, []);\n\n // 文件上传处理\n const handleFileChange = useCallback(\n async (info: { fileList: UploadFile<any>[]; file?: UploadFile<any> }) => {\n const uploadFile = info.file;\n if (!uploadFile || !uploadFile.originFileObj) {\n setFileList(info.fileList);\n return;\n }\n\n const file = uploadFile.originFileObj as File;\n const isImage = file.type.startsWith('image/');\n const fileType: 'image' | 'document' = isImage ? 'image' : 'document';\n\n try {\n // 调用 onUploadFile 回调\n if (onUploadFile && activeSession) {\n const result = await onUploadFile({\n sessionId: activeSession.id,\n file: file as any,\n type: fileType,\n });\n\n // 将上传结果添加到文件列表\n const uploadedFile: UploadFile = {\n uid: file.name,\n name: result.name || file.name,\n status: 'done',\n url: result.url,\n };\n setFileList([...fileList, uploadedFile]);\n message.success(`${fileType === 'image' ? '图片' : '文件'}上传成功`);\n } else {\n // 没有回调时,仅本地记录\n setFileList([...fileList, uploadFile]);\n }\n } catch (err) {\n console.error('文件上传失败:', err);\n message.error('文件上传失败');\n setFileList(fileList.filter((f) => f.uid !== file.name));\n }\n },\n [fileList, activeSession, onUploadFile],\n );\n\n // 判断是否为任务创建意图并执行工作流\n const handleSend = useCallback(async () => {\n if (!inputValue.trim() && fileList.length === 0) {\n message.warning('请输入问题或上传文件');\n return;\n }\n\n let content = inputValue.trim();\n\n // 如果有文件,附加文件信息\n if (fileList.length > 0) {\n const fileInfo = fileList.map((f) => f.name).join(', ');\n content += `\\n[附件: ${fileInfo}]`;\n }\n\n setInputValue('');\n setSuggestions([]);\n setFileList([]);\n\n // 检查是否为任务创建意图\n const intentEnabled = intentConfig?.enabled !== false;\n if (intentEnabled && isTaskCreationIntent(content)) {\n // 执行任务创建工作流\n await executeTaskCreationWorkflow(content);\n return;\n }\n\n // 普通问答\n sendQuestion(content);\n }, [inputValue, fileList, intentConfig?.enabled, executeTaskCreationWorkflow, sendQuestion]);\n\n const copyMsg = (c: string) => {\n navigator.clipboard.writeText(c);\n message.success('复制成功');\n };\n\n const toggleMaximize = useCallback(() => {\n setIsMaximized((prev) => !prev);\n }, []);\n\n const renderMsg = (item: Message) => (\n <List.Item\n key={item.id}\n style={{\n justifyContent: item.role === 'user' ? 'flex-end' : 'flex-start',\n marginBottom: 12,\n padding: '4px 0',\n }}\n >\n <div\n style={{\n maxWidth: '85%',\n padding: '10px 14px',\n borderRadius: 12,\n backgroundColor: item.role === 'user' ? '#1677ff' : item.role === 'system' ? '#f6ffed' : '#fafafa',\n color: item.role === 'user' ? '#fff' : '#333',\n border: item.role === 'ai' ? '1px solid #f0f0f0' : item.role === 'system' ? '1px solid #b7eb8f' : 'none',\n boxShadow:\n item.role === 'user'\n ? '0 2px 8px rgba(22, 119, 255, 0.3)'\n : item.role === 'system'\n ? '0 1px 4px rgba(82, 196, 26, 0.1)'\n : '0 1px 4px rgba(0,0,0,0.05)',\n }}\n >\n {item.role === 'user' ? (\n <div style={{ display: 'flex', alignItems: 'flex-start', gap: 8 }}>\n <span>{item.content}</span>\n <CopyOutlined\n onClick={() => copyMsg(item.content)}\n style={{\n fontSize: 12,\n color: 'rgba(255,255,255,0.7)',\n cursor: 'pointer',\n flexShrink: 0,\n }}\n />\n </div>\n ) : item.role === 'system' ? (\n <div\n style={{\n display: 'flex',\n alignItems: 'center',\n gap: 8,\n fontSize: 13,\n }}\n >\n <span style={{ color: '#52c41a' }}>✓</span>\n <span>{item.content}</span>\n {item.taskId && (\n <Button\n type=\"link\"\n size=\"small\"\n style={{ padding: 0, fontSize: 12 }}\n onClick={() => updateUrlWithTaskId(item.taskId!)}\n >\n 查看任务\n </Button>\n )}\n </div>\n ) : (\n <>\n <div style={{ minHeight: 20 }}>\n <MarkdownRender content={item.content} msgId={item.id} isAi={true} onDelete={deleteSingleMsg} />\n </div>\n {item.loading && <span style={{ color: '#1677ff', fontSize: 12 }}> 思考中...</span>}\n {!item.loading && (\n <Space size={8} style={{ fontSize: 12, marginTop: 6, display: 'flex' }}>\n <Tooltip title=\"复制\">\n <CopyOutlined onClick={() => copyMsg(item.content)} style={{ cursor: 'pointer', color: '#999' }} />\n </Tooltip>\n <Tooltip title=\"重新生成\">\n <ReloadOutlined onClick={() => regenerateAnswer(item)} style={{ cursor: 'pointer', color: '#999' }} />\n </Tooltip>\n </Space>\n )}\n </>\n )}\n </div>\n </List.Item>\n );\n\n const renderSession = (s: ChatSession) => (\n <List.Item\n key={s.id}\n onClick={(e) => {\n if ((e.target as HTMLElement).closest('.ant-popover, .ant-popconfirm')) {\n return;\n }\n switchSession(s.id);\n }}\n style={{\n padding: '10px 12px',\n borderRadius: 8,\n cursor: 'pointer',\n backgroundColor: activeSession?.id === s.id ? '#e6f7ff' : '#fff',\n transition: 'all 0.2s',\n marginBottom: 8,\n }}\n >\n <div\n style={{\n flex: 1,\n overflow: 'hidden',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n }}\n >\n <div style={{ flex: 1, overflow: 'hidden' }}>\n <div\n style={{\n fontSize: 14,\n fontWeight: 500,\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {s.title}\n </div>\n <div style={{ fontSize: 12, color: '#999', marginTop: 2 }}>\n {new Date(s.createTime).toLocaleString().slice(0, 16)}\n </div>\n </div>\n <Popconfirm\n title=\"确定删除?\"\n onConfirm={(e: any) => {\n deleteSession(s.id, e as React.MouseEvent);\n }}\n >\n <DeleteOutlined\n style={{ color: '#ff4d4f', cursor: 'pointer', marginLeft: 8 }}\n onClick={(e) => e.stopPropagation()}\n />\n </Popconfirm>\n </div>\n </List.Item>\n );\n\n // 计算对话框位置和大小\n const getDialogStyle = (): CSSProperties => {\n if (isMaximized) {\n return {\n position: 'fixed',\n top: 0,\n left: 0,\n width: '100vw',\n height: '100vh',\n borderRadius: 0,\n };\n }\n\n const baseStyle: CSSProperties = {\n position: 'relative',\n width: showSessions ? 960 : 720,\n height: 680,\n borderRadius: 12,\n };\n\n if (dialogPosition.x !== 0 || dialogPosition.y !== 0) {\n return {\n ...baseStyle,\n transform: `translate(${dialogPosition.x}px, ${dialogPosition.y}px)`,\n };\n }\n\n return baseStyle;\n };\n\n return (\n <div\n className={className}\n style={{\n position: 'fixed',\n bottom: initialPosition.bottom,\n right: initialPosition.right,\n zIndex,\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'flex-end',\n transform:\n dialogPosition.x !== 0 || dialogPosition.y !== 0\n ? `translate(${dialogPosition.x}px, ${dialogPosition.y}px)`\n : undefined,\n ...style,\n }}\n >\n {!visible ? (\n renderTrigger ? (\n <div\n ref={triggerRef}\n onMouseDown={handleTriggerMouseDown}\n onClick={() => !isDragging && toggleVisible(true)}\n style={{ cursor: isDragging ? 'grabbing' : 'grab' }}\n >\n {renderTrigger}\n </div>\n ) : (\n <div\n ref={triggerRef}\n onMouseDown={handleTriggerMouseDown}\n onClick={() => !isDragging && toggleVisible(true)}\n style={{ cursor: isDragging ? 'grabbing' : 'grab' }}\n >\n <Button\n type=\"primary\"\n shape=\"circle\"\n size=\"large\"\n icon={<MessageOutlined />}\n style={{\n width: buttonSize,\n height: buttonSize,\n boxShadow: '0 4px 16px rgba(22, 119, 255, 0.4)',\n }}\n />\n </div>\n )\n ) : (\n <div\n style={{\n ...getDialogStyle(),\n background: '#fff',\n boxShadow: '0 4px 20px rgba(0,0,0,0.15)',\n overflow: 'hidden',\n display: 'flex',\n flexDirection: 'row',\n marginBottom: isMaximized ? 0 : 12,\n transition: 'all 0.3s ease',\n }}\n >\n {/* 历史会话面板 - 左侧 */}\n {showSessionPanel && showSessions && (\n <div\n style={{\n width: 280,\n borderRight: '1px solid #f0f0f0',\n background: '#fafafa',\n display: 'flex',\n flexDirection: 'column',\n flexShrink: 0,\n }}\n >\n <div\n style={{\n padding: '12px 16px',\n borderBottom: '1px solid #f0f0f0',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n }}\n >\n <Title level={5} style={{ margin: 0, fontSize: 16 }}>\n 历史会话\n </Title>\n <Button type=\"text\" size=\"small\" icon={<CloseOutlined />} onClick={() => setShowSessions(false)} />\n </div>\n <div style={{ flex: 1, padding: 16, overflowY: 'auto' }}>\n <Button\n type=\"primary\"\n block\n icon={<PlusOutlined />}\n onClick={createNewSession}\n style={{ marginBottom: 12 }}\n >\n 新建会话\n </Button>\n <List dataSource={sessions} renderItem={renderSession} locale={{ emptyText: '暂无会话' }} />\n </div>\n </div>\n )}\n\n {/* 主对话区域 */}\n <div\n style={{\n flex: 1,\n display: 'flex',\n flexDirection: 'column',\n minWidth: 0,\n }}\n >\n {renderHeader || (\n <div\n className=\"ai-chat-header\"\n style={{\n padding: '12px 16px',\n background: '#fafafa',\n borderBottom: '1px solid #f0f0f0',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n flexShrink: 0,\n }}\n >\n <Space>\n {showSessionPanel && (\n <Button\n type={showSessions ? 'primary' : 'text'}\n icon={<MenuOutlined />}\n size=\"small\"\n onClick={() => setShowSessions(!showSessions)}\n >\n {showSessions ? '收起' : '会话'}\n </Button>\n )}\n <Button type=\"text\" icon={<PlusOutlined />} onClick={createNewSession} size=\"small\">\n 新会话\n </Button>\n </Space>\n <Space>\n <Tooltip title={isMaximized ? '还原' : '最大化'}>\n <Button\n type=\"text\"\n icon={isMaximized ? <CompressOutlined /> : <ExpandOutlined />}\n onClick={toggleMaximize}\n size=\"small\"\n />\n </Tooltip>\n <Tooltip title=\"关闭\">\n <Button type=\"text\" icon={<CloseOutlined />} onClick={() => toggleVisible(false)} size=\"small\" />\n </Tooltip>\n </Space>\n </div>\n )}\n\n <div\n style={{\n flex: 1,\n padding: 16,\n overflowY: 'auto',\n }}\n >\n {!activeSession || activeSession.messages.length === 0 ? (\n <div\n style={{\n textAlign: 'center',\n padding: '60px 0',\n color: '#999',\n }}\n >\n <MessageOutlined style={{ fontSize: 48, color: '#d9d9d9' }} />\n <div style={{ marginTop: 16, fontSize: 16 }}>开始你的 AI 对话</div>\n <div style={{ marginTop: 8, fontSize: 13, color: '#bfbfbf' }}>我可以帮你解答问题、创建任务</div>\n <Button type=\"primary\" onClick={createNewSession} style={{ marginTop: 20 }}>\n 新建会话\n </Button>\n </div>\n ) : (\n <List dataSource={activeSession.messages} renderItem={renderMsg} />\n )}\n <div ref={messagesEndRef} />\n </div>\n\n {renderFooter || (\n <div\n style={{\n padding: 12,\n borderTop: '1px solid #f0f0f0',\n background: '#fafafa',\n flexShrink: 0,\n }}\n >\n {/* 智能建议 */}\n {suggestions.length > 0 && (\n <div\n style={{\n marginBottom: 8,\n display: 'flex',\n flexWrap: 'wrap',\n gap: 4,\n }}\n >\n {suggestions.slice(0, 3).map((suggestion, index) => (\n <Button\n key={index}\n size=\"small\"\n type=\"dashed\"\n onClick={() => setInputValue(suggestion)}\n style={{ fontSize: 12 }}\n >\n {suggestion}\n </Button>\n ))}\n </div>\n )}\n\n {/* 文件预览 */}\n {fileList.length > 0 && (\n <div style={{ marginBottom: 8 }}>\n {fileList.map((file) => (\n <div\n key={file.uid}\n style={{\n display: 'inline-block',\n padding: '4px 8px',\n background: '#e6f7ff',\n borderRadius: 4,\n marginRight: 8,\n fontSize: 12,\n }}\n >\n {file.name}\n <CloseOutlined\n style={{\n marginLeft: 8,\n cursor: 'pointer',\n fontSize: 10,\n }}\n onClick={() => setFileList(fileList.filter((f) => f.uid !== file.uid))}\n />\n </div>\n ))}\n </div>\n )}\n\n <div style={{ position: 'relative' }}>\n <TextArea\n value={inputValue}\n onChange={handleInputChange}\n placeholder=\"输入问题或任务描述,如:帮我创建一个paimon调用任务\"\n autoSize={{ minRows: 2, maxRows: 4 }}\n onPressEnter={(e) => {\n if (!e.shiftKey) {\n e.preventDefault();\n handleSend();\n }\n }}\n style={{\n marginBottom: 8,\n borderRadius: 8,\n paddingRight: 80,\n }}\n disabled={loading}\n />\n {/* 上传文件和图片按钮 */}\n <div\n style={{\n position: 'absolute',\n right: 12,\n bottom: 16,\n display: 'flex',\n gap: 8,\n }}\n >\n <Upload\n fileList={fileList}\n onChange={handleFileChange}\n beforeUpload={() => false}\n showUploadList={false}\n accept=\"image/*\"\n >\n <Tooltip title=\"上传图片\">\n <Button type=\"text\" size=\"small\" icon={<FileImageOutlined />} />\n </Tooltip>\n </Upload>\n <Upload\n fileList={fileList}\n onChange={handleFileChange}\n beforeUpload={() => false}\n showUploadList={false}\n >\n <Tooltip title=\"上传文件\">\n <Button type=\"text\" size=\"small\" icon={<PaperClipOutlined />} />\n </Tooltip>\n </Upload>\n </div>\n </div>\n <div style={{ display: 'flex', justifyContent: 'flex-end' }}>\n <Space>\n {loading && (\n <Button danger icon={<StopOutlined />} onClick={cancelRequest} size=\"small\">\n 终止\n </Button>\n )}\n <Button\n type=\"primary\"\n icon={<SendOutlined />}\n onClick={handleSend}\n loading={loading}\n disabled={!inputValue.trim() && fileList.length === 0}\n size=\"small\"\n >\n 发送\n </Button>\n </Space>\n </div>\n </div>\n )}\n </div>\n </div>\n )}\n </div>\n );\n};\n\nexport default AiChat;\nexport { generateSuggestions, isTaskCreationIntent, parseTaskDescription, recognizeIntent } from './intentRecognizer';\nexport { delay, NavigationManager } from './navigationManager';\nexport type {\n AiChatConfig,\n ChatSession,\n EnhancedAiChatConfig,\n Intent,\n IntentType,\n Message,\n NavigationOptions,\n Task,\n TaskCategory,\n TaskCreationResult,\n TaskPriority,\n TaskStatus,\n UseAiChatOptions,\n UseAiChatReturn,\n} from './type';\nexport { useAiChat } from './useAiChat';\nexport { useTaskWorkflow } from './useTaskWorkflow';\n"],
4
+ "sourcesContent": ["import {\n CloseOutlined,\n CompressOutlined,\n CopyOutlined,\n DeleteOutlined,\n ExpandOutlined,\n FileImageOutlined,\n MenuOutlined,\n MessageOutlined,\n PaperClipOutlined,\n PlusOutlined,\n ReloadOutlined,\n SendOutlined,\n StopOutlined,\n} from '@ant-design/icons';\nimport type { UploadFile } from 'antd';\nimport { Button, Input, List, message, Popconfirm, Space, Tooltip, Typography, Upload } from 'antd';\nimport type { CSSProperties, FC, ReactNode } from 'react';\nimport React, { useCallback, useEffect, useRef, useState } from 'react';\nimport { generateSuggestions, isTaskCreationIntent, recognizeIntent } from './intentRecognizer';\nimport MarkdownRender from './MarkdownRender';\nimport type { AiChatConfig, ChatSession, EnhancedAiChatConfig, Message } from './type';\nimport { useAiChat } from './useAiChat';\nimport { useTaskWorkflow } from './useTaskWorkflow';\n\nconst { TextArea } = Input;\nconst { Title } = Typography;\n\n// 合并Props类型\nexport interface AiChatProps extends AiChatConfig, Partial<EnhancedAiChatConfig> {\n className?: string;\n style?: CSSProperties;\n renderTrigger?: ReactNode;\n renderHeader?: ReactNode;\n renderFooter?: ReactNode;\n}\n\nconst AiChat: FC<AiChatProps> = ({\n apiPath = '/api/ai/chat-stream',\n storageKey = 'ai_chat_sessions',\n position: initialPosition = { bottom: 24, right: 24 },\n buttonSize = 56,\n zIndex = 1080,\n showSessionPanel = true,\n onBeforeSend,\n onError,\n onSuccess,\n onFetchSessions,\n onFetchSessionDetail,\n onCreateSession,\n onSendMessage,\n onUploadFile,\n className,\n style,\n renderTrigger,\n renderHeader,\n renderFooter,\n // 增强配置\n taskWorkflow,\n intentRecognition: intentConfig,\n navigation: navConfig,\n}) => {\n const {\n sessions,\n activeSession,\n loading,\n visible,\n createNewSession,\n switchSession,\n deleteSession,\n sendQuestion,\n regenerateAnswer,\n deleteSingleMsg,\n cancelRequest,\n toggleVisible,\n } = useAiChat({\n apiPath,\n storageKey,\n onBeforeSend,\n onError,\n onSuccess,\n onFetchSessions,\n onFetchSessionDetail,\n onCreateSession,\n onSendMessage,\n onUploadFile,\n });\n\n // 任务工作流\n const {\n tasks,\n createTask,\n updateTaskStatus,\n deleteTask,\n createTaskFromNaturalLanguage,\n updateUrlWithTaskId,\n refreshCurrentPage,\n executeTaskCreationWorkflow,\n } = useTaskWorkflow({\n enabled: taskWorkflow?.enabled,\n createTaskApi: taskWorkflow?.createTaskApi,\n supportedTaskTypes: taskWorkflow?.supportedTaskTypes,\n autoNavigate: taskWorkflow?.autoNavigate,\n refreshDelay: navConfig?.refreshDelay,\n taskIdParamName: taskWorkflow?.taskIdParamName,\n taskTypeParamName: taskWorkflow?.taskTypeParamName,\n onTaskCreated: (taskId, taskData) => {\n // 添加系统消息到会话\n if (activeSession) {\n const systemMsg: Message = {\n id: `${Date.now()}_sys`,\n role: 'system',\n content: `任务创建成功,ID: ${taskId}`,\n taskId,\n };\n // 这里可以添加到消息列表的逻辑\n }\n },\n onError,\n });\n\n const [inputValue, setInputValue] = useState('');\n const [showSessions, setShowSessions] = useState(false);\n const [suggestions, setSuggestions] = useState<string[]>([]);\n const [fileList, setFileList] = useState<UploadFile<any>[]>([]);\n const [isMaximized, setIsMaximized] = useState(false);\n const [dialogPosition, setDialogPosition] = useState({ x: 0, y: 0 });\n const [isDragging, setIsDragging] = useState(false);\n const dragOffset = useRef({ x: 0, y: 0 });\n const messagesEndRef = useRef<HTMLDivElement>(null);\n const triggerRef = useRef<HTMLDivElement>(null);\n const fileInputRef = useRef<HTMLInputElement>(null);\n\n useEffect(() => {\n messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' });\n }, [activeSession?.messages]);\n\n // 拖拽处理 - 拖拽整个组件位置\n const handleTriggerMouseDown = useCallback(\n (e: React.MouseEvent) => {\n if (visible) return; // 对话框展开时不处理\n const target = e.target as HTMLElement;\n if (target.closest('.ant-upload, .ant-btn')) return; // 排除按钮和上传元素\n\n e.preventDefault();\n const triggerEl = triggerRef.current;\n if (!triggerEl) return;\n\n const rect = triggerEl.getBoundingClientRect();\n dragOffset.current = {\n x: e.clientX - rect.left,\n y: e.clientY - rect.top,\n };\n setIsDragging(true);\n },\n [visible],\n );\n\n const handleTriggerMouseMove = useCallback(\n (e: MouseEvent) => {\n if (!isDragging || visible) return;\n setDialogPosition({\n x: e.clientX - dragOffset.current.x,\n y: e.clientY - dragOffset.current.y,\n });\n },\n [isDragging, visible],\n );\n\n const handleTriggerMouseUp = useCallback(() => {\n setIsDragging(false);\n }, []);\n\n useEffect(() => {\n if (isDragging && !visible) {\n document.addEventListener('mousemove', handleTriggerMouseMove);\n document.addEventListener('mouseup', handleTriggerMouseUp);\n return () => {\n document.removeEventListener('mousemove', handleTriggerMouseMove);\n document.removeEventListener('mouseup', handleTriggerMouseUp);\n };\n }\n return undefined;\n }, [isDragging, visible, handleTriggerMouseMove, handleTriggerMouseUp]);\n\n // 实时分析输入内容生成建议\n const handleInputChange = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {\n const value = e.target.value;\n setInputValue(value);\n\n if (value.length > 3) {\n const intent = recognizeIntent(value);\n const newSuggestions = generateSuggestions(value, intent);\n setSuggestions(newSuggestions);\n } else {\n setSuggestions([]);\n }\n }, []);\n\n // 文件上传处理\n const handleFileChange = useCallback(\n async (info: { fileList: UploadFile<any>[]; file?: UploadFile<any> }) => {\n const uploadFile = info.file;\n if (!uploadFile || !uploadFile.originFileObj) {\n setFileList(info.fileList);\n return;\n }\n\n const file = uploadFile.originFileObj as File;\n const isImage = file.type.startsWith('image/');\n const fileType: 'image' | 'document' = isImage ? 'image' : 'document';\n\n try {\n // 调用 onUploadFile 回调\n if (onUploadFile && activeSession) {\n const result = await onUploadFile({\n sessionId: activeSession.id,\n file: file as any,\n type: fileType,\n });\n\n // 将上传结果添加到文件列表\n const uploadedFile: UploadFile = {\n uid: file.name,\n name: result.name || file.name,\n status: 'done',\n url: result.url,\n };\n setFileList([...fileList, uploadedFile]);\n message.success(`${fileType === 'image' ? '图片' : '文件'}上传成功`);\n } else {\n // 没有回调时,仅本地记录\n setFileList([...fileList, uploadFile]);\n }\n } catch (err) {\n console.error('文件上传失败:', err);\n message.error('文件上传失败');\n setFileList(fileList.filter((f) => f.uid !== file.name));\n }\n },\n [fileList, activeSession, onUploadFile],\n );\n\n // 判断是否为任务创建意图并执行工作流\n const handleSend = useCallback(async () => {\n if (!inputValue.trim() && fileList.length === 0) {\n message.warning('请输入问题或上传文件');\n return;\n }\n\n let content = inputValue.trim();\n\n // 如果有文件,附加文件信息\n if (fileList.length > 0) {\n const fileInfo = fileList.map((f) => f.name).join(', ');\n content += `\\n[附件: ${fileInfo}]`;\n }\n\n setInputValue('');\n setSuggestions([]);\n setFileList([]);\n\n // 检查是否为任务创建意图\n const intentEnabled = intentConfig?.enabled !== false;\n if (intentEnabled && isTaskCreationIntent(content)) {\n // 执行任务创建工作流\n await executeTaskCreationWorkflow(content);\n return;\n }\n\n // 普通问答\n sendQuestion(content);\n }, [inputValue, fileList, intentConfig?.enabled, executeTaskCreationWorkflow, sendQuestion]);\n\n const copyMsg = (c: string) => {\n navigator.clipboard.writeText(c);\n message.success('复制成功');\n };\n\n const toggleMaximize = useCallback(() => {\n setIsMaximized((prev) => !prev);\n }, []);\n\n const renderMsg = (item: Message) => (\n <List.Item\n key={item.id}\n style={{\n justifyContent: item.role === 'user' ? 'flex-end' : 'flex-start',\n marginBottom: 12,\n padding: '4px 0',\n }}\n >\n <div\n style={{\n maxWidth: '85%',\n padding: '10px 14px',\n borderRadius: 12,\n backgroundColor: item.role === 'user' ? '#1677ff' : item.role === 'system' ? '#f6ffed' : '#fafafa',\n color: item.role === 'user' ? '#fff' : '#333',\n border: item.role === 'ai' ? '1px solid #f0f0f0' : item.role === 'system' ? '1px solid #b7eb8f' : 'none',\n boxShadow:\n item.role === 'user'\n ? '0 2px 8px rgba(22, 119, 255, 0.3)'\n : item.role === 'system'\n ? '0 1px 4px rgba(82, 196, 26, 0.1)'\n : '0 1px 4px rgba(0,0,0,0.05)',\n }}\n >\n {item.role === 'user' ? (\n <div style={{ display: 'flex', alignItems: 'flex-start', gap: 8 }}>\n <span>{item.content}</span>\n <CopyOutlined\n onClick={() => copyMsg(item.content)}\n style={{\n fontSize: 12,\n color: 'rgba(255,255,255,0.7)',\n cursor: 'pointer',\n flexShrink: 0,\n }}\n />\n </div>\n ) : item.role === 'system' ? (\n <div\n style={{\n display: 'flex',\n alignItems: 'center',\n gap: 8,\n fontSize: 13,\n }}\n >\n <span style={{ color: '#52c41a' }}>✓</span>\n <span>{item.content}</span>\n {item.taskId && (\n <Button\n type='link'\n size='small'\n style={{ padding: 0, fontSize: 12 }}\n onClick={() => updateUrlWithTaskId(item.taskId!)}\n >\n 查看任务\n </Button>\n )}\n </div>\n ) : (\n <>\n <div style={{ minHeight: 20 }}>\n <MarkdownRender content={item.content} msgId={item.id} isAi={true} onDelete={deleteSingleMsg} />\n </div>\n {item.loading && <span style={{ color: '#1677ff', fontSize: 12 }}> 思考中...</span>}\n {!item.loading && (\n <Space size={8} style={{ fontSize: 12, marginTop: 6, display: 'flex' }}>\n <Tooltip title='复制'>\n <CopyOutlined onClick={() => copyMsg(item.content)} style={{ cursor: 'pointer', color: '#999' }} />\n </Tooltip>\n <Tooltip title='重新生成'>\n <ReloadOutlined onClick={() => regenerateAnswer(item)} style={{ cursor: 'pointer', color: '#999' }} />\n </Tooltip>\n </Space>\n )}\n </>\n )}\n </div>\n </List.Item>\n );\n\n const renderSession = (s: ChatSession) => (\n <List.Item\n key={s.id}\n onClick={(e) => {\n if ((e.target as HTMLElement).closest('.ant-popover, .ant-popconfirm')) {\n return;\n }\n switchSession(s.id);\n }}\n style={{\n padding: '10px 12px',\n borderRadius: 8,\n cursor: 'pointer',\n backgroundColor: activeSession?.id === s.id ? '#e6f7ff' : '#fff',\n transition: 'all 0.2s',\n marginBottom: 8,\n }}\n >\n <div\n style={{\n flex: 1,\n overflow: 'hidden',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n }}\n >\n <div style={{ flex: 1, overflow: 'hidden' }}>\n <div\n style={{\n fontSize: 14,\n fontWeight: 500,\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n }}\n >\n {s.title}\n </div>\n <div style={{ fontSize: 12, color: '#999', marginTop: 2 }}>\n {new Date(s.createTime).toLocaleString().slice(0, 16)}\n </div>\n </div>\n <Popconfirm\n title='确定删除?'\n onConfirm={(e: any) => {\n deleteSession(s.id, e as React.MouseEvent);\n }}\n >\n <DeleteOutlined\n style={{ color: '#ff4d4f', cursor: 'pointer', marginLeft: 8 }}\n onClick={(e) => e.stopPropagation()}\n />\n </Popconfirm>\n </div>\n </List.Item>\n );\n\n // 计算对话框位置和大小\n const getDialogStyle = (): CSSProperties => {\n if (isMaximized) {\n return {\n position: 'fixed',\n top: 0,\n left: 0,\n width: '100vw',\n height: '100vh',\n borderRadius: 0,\n };\n }\n\n const baseStyle: CSSProperties = {\n position: 'relative',\n width: showSessions ? 960 : 720,\n height: 680,\n borderRadius: 12,\n };\n\n if (dialogPosition.x !== 0 || dialogPosition.y !== 0) {\n return {\n ...baseStyle,\n transform: `translate(${dialogPosition.x}px, ${dialogPosition.y}px)`,\n };\n }\n\n return baseStyle;\n };\n\n return (\n <div\n className={className}\n style={{\n position: 'fixed',\n bottom: initialPosition.bottom,\n right: initialPosition.right,\n zIndex,\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'flex-end',\n transform:\n dialogPosition.x !== 0 || dialogPosition.y !== 0\n ? `translate(${dialogPosition.x}px, ${dialogPosition.y}px)`\n : undefined,\n ...style,\n }}\n >\n {!visible ? (\n renderTrigger ? (\n <div\n ref={triggerRef}\n onMouseDown={handleTriggerMouseDown}\n onClick={() => !isDragging && toggleVisible(true)}\n style={{ cursor: isDragging ? 'grabbing' : 'grab' }}\n >\n {renderTrigger}\n </div>\n ) : (\n <div\n ref={triggerRef}\n onMouseDown={handleTriggerMouseDown}\n onClick={() => !isDragging && toggleVisible(true)}\n style={{ cursor: isDragging ? 'grabbing' : 'grab' }}\n >\n <Button\n type='primary'\n shape='circle'\n size='large'\n icon={<MessageOutlined />}\n style={{\n width: buttonSize,\n height: buttonSize,\n boxShadow: '0 4px 16px rgba(22, 119, 255, 0.4)',\n }}\n />\n </div>\n )\n ) : (\n <div\n style={{\n ...getDialogStyle(),\n background: '#fff',\n boxShadow: '0 4px 20px rgba(0,0,0,0.15)',\n overflow: 'hidden',\n display: 'flex',\n flexDirection: 'row',\n marginBottom: isMaximized ? 0 : 12,\n transition: 'all 0.3s ease',\n }}\n >\n {/* 历史会话面板 - 左侧 */}\n {showSessionPanel && showSessions && (\n <div\n style={{\n width: 280,\n borderRight: '1px solid #f0f0f0',\n background: '#fafafa',\n display: 'flex',\n flexDirection: 'column',\n flexShrink: 0,\n }}\n >\n <div\n style={{\n padding: '12px 16px',\n borderBottom: '1px solid #f0f0f0',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n }}\n >\n <Title level={5} style={{ margin: 0, fontSize: 16 }}>\n 历史会话\n </Title>\n <Button type='text' size='small' icon={<CloseOutlined />} onClick={() => setShowSessions(false)} />\n </div>\n <div style={{ flex: 1, padding: 16, overflowY: 'auto' }}>\n <Button\n type='primary'\n block\n icon={<PlusOutlined />}\n onClick={createNewSession}\n style={{ marginBottom: 12 }}\n >\n 新建会话\n </Button>\n <List dataSource={sessions} renderItem={renderSession} locale={{ emptyText: '暂无会话' }} />\n </div>\n </div>\n )}\n\n {/* 主对话区域 */}\n <div\n style={{\n flex: 1,\n display: 'flex',\n flexDirection: 'column',\n minWidth: 0,\n }}\n >\n {renderHeader || (\n <div\n className='ai-chat-header'\n style={{\n padding: '12px 16px',\n background: '#fafafa',\n borderBottom: '1px solid #f0f0f0',\n display: 'flex',\n justifyContent: 'space-between',\n alignItems: 'center',\n flexShrink: 0,\n }}\n >\n <Space>\n {showSessionPanel && (\n <Button\n type={showSessions ? 'primary' : 'text'}\n icon={<MenuOutlined />}\n size='small'\n onClick={() => setShowSessions(!showSessions)}\n >\n {showSessions ? '收起' : '会话'}\n </Button>\n )}\n <Button type='text' icon={<PlusOutlined />} onClick={createNewSession} size='small'>\n 新会话\n </Button>\n </Space>\n <Space>\n <Tooltip title={isMaximized ? '还原' : '最大化'}>\n <Button\n type='text'\n icon={isMaximized ? <CompressOutlined /> : <ExpandOutlined />}\n onClick={toggleMaximize}\n size='small'\n />\n </Tooltip>\n <Tooltip title='关闭'>\n <Button type='text' icon={<CloseOutlined />} onClick={() => toggleVisible(false)} size='small' />\n </Tooltip>\n </Space>\n </div>\n )}\n\n <div\n style={{\n flex: 1,\n padding: 16,\n overflowY: 'auto',\n }}\n >\n {!activeSession || activeSession.messages.length === 0 ? (\n <div\n style={{\n textAlign: 'center',\n padding: '60px 0',\n color: '#999',\n }}\n >\n <MessageOutlined style={{ fontSize: 48, color: '#d9d9d9' }} />\n <div style={{ marginTop: 16, fontSize: 16 }}>开始你的 AI 对话</div>\n <div style={{ marginTop: 8, fontSize: 13, color: '#bfbfbf' }}>我可以帮你解答问题、创建任务</div>\n <Button type='primary' onClick={createNewSession} style={{ marginTop: 20 }}>\n 新建会话\n </Button>\n </div>\n ) : (\n <List dataSource={activeSession.messages} renderItem={renderMsg} />\n )}\n <div ref={messagesEndRef} />\n </div>\n\n {renderFooter || (\n <div\n style={{\n padding: 12,\n borderTop: '1px solid #f0f0f0',\n background: '#fafafa',\n flexShrink: 0,\n }}\n >\n {/* 智能建议 */}\n {suggestions.length > 0 && (\n <div\n style={{\n marginBottom: 8,\n display: 'flex',\n flexWrap: 'wrap',\n gap: 4,\n }}\n >\n {suggestions.slice(0, 3).map((suggestion, index) => (\n <Button\n key={index}\n size='small'\n type='dashed'\n onClick={() => setInputValue(suggestion)}\n style={{ fontSize: 12 }}\n >\n {suggestion}\n </Button>\n ))}\n </div>\n )}\n\n {/* 文件预览 */}\n {fileList.length > 0 && (\n <div style={{ marginBottom: 8 }}>\n {fileList.map((file) => (\n <div\n key={file.uid}\n style={{\n display: 'inline-block',\n padding: '4px 8px',\n background: '#e6f7ff',\n borderRadius: 4,\n marginRight: 8,\n fontSize: 12,\n }}\n >\n {file.name}\n <CloseOutlined\n style={{\n marginLeft: 8,\n cursor: 'pointer',\n fontSize: 10,\n }}\n onClick={() => setFileList(fileList.filter((f) => f.uid !== file.uid))}\n />\n </div>\n ))}\n </div>\n )}\n\n <div style={{ position: 'relative' }}>\n <TextArea\n value={inputValue}\n onChange={handleInputChange}\n placeholder='输入问题或任务描述,如:帮我创建一个paimon调用任务'\n autoSize={{ minRows: 2, maxRows: 4 }}\n onPressEnter={(e) => {\n if (!e.shiftKey) {\n e.preventDefault();\n handleSend();\n }\n }}\n style={{\n marginBottom: 8,\n borderRadius: 8,\n paddingRight: 80,\n }}\n disabled={loading}\n />\n {/* 上传文件和图片按钮 */}\n <div\n style={{\n position: 'absolute',\n right: 12,\n bottom: 16,\n display: 'flex',\n gap: 8,\n }}\n >\n <Upload\n fileList={fileList}\n onChange={handleFileChange}\n beforeUpload={() => false}\n showUploadList={false}\n accept='image/*'\n >\n <Tooltip title='上传图片'>\n <Button type='text' size='small' icon={<FileImageOutlined />} />\n </Tooltip>\n </Upload>\n <Upload\n fileList={fileList}\n onChange={handleFileChange}\n beforeUpload={() => false}\n showUploadList={false}\n >\n <Tooltip title='上传文件'>\n <Button type='text' size='small' icon={<PaperClipOutlined />} />\n </Tooltip>\n </Upload>\n </div>\n </div>\n <div style={{ display: 'flex', justifyContent: 'flex-end' }}>\n <Space>\n {loading && (\n <Button danger icon={<StopOutlined />} onClick={cancelRequest} size='small'>\n 终止\n </Button>\n )}\n <Button\n type='primary'\n icon={<SendOutlined />}\n onClick={handleSend}\n loading={loading}\n disabled={!inputValue.trim() && fileList.length === 0}\n size='small'\n >\n 发送\n </Button>\n </Space>\n </div>\n </div>\n )}\n </div>\n </div>\n )}\n </div>\n );\n};\n\nexport default AiChat;\nexport { generateSuggestions, isTaskCreationIntent, parseTaskDescription, recognizeIntent } from './intentRecognizer';\nexport { delay, NavigationManager } from './navigationManager';\nexport type {\n AiChatConfig,\n ChatSession,\n EnhancedAiChatConfig,\n Intent,\n IntentType,\n Message,\n NavigationOptions,\n Task,\n TaskCategory,\n TaskCreationResult,\n TaskPriority,\n TaskStatus,\n UseAiChatOptions,\n UseAiChatReturn,\n} from './type';\nexport { useAiChat } from './useAiChat';\nexport { useTaskWorkflow } from './useTaskWorkflow';\n"],
5
5
  "mappings": "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",
6
6
  "names": ["generateSuggestions", "isTaskCreationIntent", "recognizeIntent", "useAiChat", "useTaskWorkflow"]
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/business/AiChat/intentRecognizer.ts"],
4
- "sourcesContent": ["import type { Intent, IntentType, IntentPattern, TaskCategory } from './type';\n\n// 默认意图模式\nconst DEFAULT_PATTERNS: IntentPattern[] = [\n {\n type: 'create_task',\n patterns: [/创建|新建|启动|开始|生成|建立|发起/i, /帮我|请|帮忙|协助/i, /任务|作业|job|task/i],\n },\n {\n type: 'query_task',\n patterns: [/查询|查看|获取|显示|看看|找/i, /任务|作业|状态|进度|结果/i],\n },\n {\n type: 'update_task',\n patterns: [/更新|修改|编辑|更改|调整|优化/i, /任务|配置|参数|设置/i],\n },\n {\n type: 'delete_task',\n patterns: [/删除|取消|移除|清除|销毁/i, /任务|作业|job|task/i],\n },\n {\n type: 'call_api',\n patterns: [/调用|请求|发送|访问/i, /接口|API|服务|请求/i],\n },\n];\n\n// 任务类型关键词映射\nconst TASK_TYPE_KEYWORDS: Record<string, { patterns: RegExp[]; type: TaskCategory }> = {\n paimon: {\n patterns: [/paimon|帕蒙/i],\n type: 'paimon',\n },\n data_sync: {\n patterns: [/数据同步|同步|数据迁移|迁移/i],\n type: 'data_sync',\n },\n report: {\n patterns: [/报表|报告|report|数据报表|统计/i],\n type: 'report',\n },\n etl: {\n patterns: [/ETL|数据处理|数据清洗|数据转换/i],\n type: 'etl',\n },\n bug: {\n patterns: [/bug|缺陷|问题|错误|修复|fix/i],\n type: 'bug',\n },\n feature: {\n patterns: [/功能|新特性|feature|需求|新特性/i],\n type: 'feature',\n },\n analysis: {\n patterns: [/分析|解析|analyze|数据分析|统计分析/i],\n type: 'analysis',\n },\n};\n\n/**\n * 识别用户意图\n * @param text 用户输入文本\n * @param customPatterns 自定义意图模式\n * @returns 识别结果\n */\nexport function recognizeIntent(text: string, customPatterns?: IntentPattern[]): Intent {\n const patterns = customPatterns || DEFAULT_PATTERNS;\n\n let bestMatch: IntentType = 'general_question';\n let bestScore = 0;\n let matchedPattern = '';\n\n // 计算每个意图的匹配分数\n for (const pattern of patterns) {\n let score = 0;\n for (const regex of pattern.patterns) {\n if (regex.test(text)) {\n score += 1;\n matchedPattern = regex.source;\n }\n }\n\n // 如果有多个模式匹配,增加权重\n if (score === pattern.patterns.length && score > 0) {\n score += 0.5;\n }\n\n if (score > bestScore) {\n bestScore = score;\n bestMatch = pattern.type;\n }\n }\n\n // 提取任务类型参数\n const parameters: Record<string, any> = extractTaskParameters(text);\n\n // 计算置信度\n const confidence = Math.min(bestScore / 3, 1);\n\n return {\n type: bestMatch,\n confidence,\n parameters,\n rawText: text,\n matchedPattern: bestMatch !== 'general_question' ? matchedPattern : undefined,\n };\n}\n\n/**\n * 提取任务参数\n * @param text 用户输入文本\n * @returns 提取的参数\n */\nfunction extractTaskParameters(text: string): Record<string, any> {\n const params: Record<string, any> = {\n taskType: detectTaskType(text),\n originalText: text,\n };\n\n // 提取任务名称\n const nameMatch = text.match(/名为[\"']?([^\"']+)[\"']?/i);\n if (nameMatch) {\n params.taskName = nameMatch[1];\n }\n\n // 提取优先级\n if (/高优先级|紧急|urgent|critical/i.test(text)) {\n params.priority = 'high';\n } else if (/低优先级|optional|low/i.test(text)) {\n params.priority = 'low';\n } else {\n params.priority = 'medium';\n }\n\n // 提取描述信息\n const descMatch = text.match(/描述|说明|备注[::]?\\s*(.+)$/i);\n if (descMatch) {\n params.description = descMatch[1].trim();\n }\n\n // 提取时间信息\n const timeMatch = text.match(/(明天|下周|后天|今天|立即|马上|尽快)/i);\n if (timeMatch) {\n params.scheduledTime = timeMatch[1];\n }\n\n return params;\n}\n\n/**\n * 检测任务类型\n * @param text 用户输入文本\n * @returns 任务类型\n */\nfunction detectTaskType(text: string): TaskCategory {\n for (const [key, config] of Object.entries(TASK_TYPE_KEYWORDS)) {\n for (const pattern of config.patterns) {\n if (pattern.test(text)) {\n return config.type;\n }\n }\n }\n return 'other';\n}\n\n/**\n * 生成建议问题\n * @param text 用户输入文本\n * @param intent 识别的意图\n * @returns 建议问题列表\n */\nexport function generateSuggestions(text: string, intent: Intent): string[] {\n const suggestions: string[] = [];\n\n switch (intent.type) {\n case 'create_task':\n suggestions.push(\n '帮我创建一个数据同步任务',\n '创建一个新的报表生成任务',\n '启动Paimon表同步作业',\n '帮我创建一个ETL数据处理任务',\n );\n break;\n case 'query_task':\n suggestions.push('查询我的任务列表', '查看任务执行状态', '获取任务执行结果', '查看任务进度');\n break;\n case 'update_task':\n suggestions.push('更新任务配置', '修改任务参数', '调整任务执行时间', '优化任务设置');\n break;\n case 'delete_task':\n suggestions.push('取消当前任务', '删除失败任务', '清除已完成的任务');\n break;\n default:\n suggestions.push('帮我解答一个问题', '分析这段代码', '解释这个概念', '如何优化查询性能');\n }\n\n return suggestions;\n}\n\n/**\n * 判断是否为任务创建意图\n * @param text 用户输入文本\n * @param threshold 置信度阈值\n * @returns 是否为任务创建意图\n */\nexport function isTaskCreationIntent(text: string, threshold = 0.5): boolean {\n const intent = recognizeIntent(text);\n return intent.type === 'create_task' && intent.confidence >= threshold;\n}\n\n/**\n * 创建任务描述(自然语言转结构化数据)\n * @param text 用户输入文本\n * @returns 结构化的任务描述\n */\nexport function parseTaskDescription(text: string): {\n taskType: TaskCategory;\n taskName: string;\n description: string;\n priority: 'high' | 'medium' | 'low';\n} {\n const intent = recognizeIntent(text);\n const params = intent.parameters;\n\n return {\n taskType: params.taskType || detectTaskType(text),\n taskName: params.taskName || `${params.taskType || '新'}任务 - ${new Date().toLocaleTimeString().slice(0, 5)}`,\n description: params.description || text,\n priority: params.priority || 'medium',\n };\n}\n"],
4
+ "sourcesContent": ["import type { Intent, IntentPattern, IntentType, TaskCategory } from './type';\n\n// 默认意图模式\nconst DEFAULT_PATTERNS: IntentPattern[] = [\n {\n type: 'create_task',\n patterns: [/创建|新建|启动|开始|生成|建立|发起/i, /帮我|请|帮忙|协助/i, /任务|作业|job|task/i],\n },\n {\n type: 'query_task',\n patterns: [/查询|查看|获取|显示|看看|找/i, /任务|作业|状态|进度|结果/i],\n },\n {\n type: 'update_task',\n patterns: [/更新|修改|编辑|更改|调整|优化/i, /任务|配置|参数|设置/i],\n },\n {\n type: 'delete_task',\n patterns: [/删除|取消|移除|清除|销毁/i, /任务|作业|job|task/i],\n },\n {\n type: 'call_api',\n patterns: [/调用|请求|发送|访问/i, /接口|API|服务|请求/i],\n },\n];\n\n// 任务类型关键词映射\nconst TASK_TYPE_KEYWORDS: Record<string, { patterns: RegExp[]; type: TaskCategory }> = {\n paimon: {\n patterns: [/paimon|帕蒙/i],\n type: 'paimon',\n },\n data_sync: {\n patterns: [/数据同步|同步|数据迁移|迁移/i],\n type: 'data_sync',\n },\n report: {\n patterns: [/报表|报告|report|数据报表|统计/i],\n type: 'report',\n },\n etl: {\n patterns: [/ETL|数据处理|数据清洗|数据转换/i],\n type: 'etl',\n },\n bug: {\n patterns: [/bug|缺陷|问题|错误|修复|fix/i],\n type: 'bug',\n },\n feature: {\n patterns: [/功能|新特性|feature|需求|新特性/i],\n type: 'feature',\n },\n analysis: {\n patterns: [/分析|解析|analyze|数据分析|统计分析/i],\n type: 'analysis',\n },\n};\n\n/**\n * 识别用户意图\n * @param text 用户输入文本\n * @param customPatterns 自定义意图模式\n * @returns 识别结果\n */\nexport function recognizeIntent(text: string, customPatterns?: IntentPattern[]): Intent {\n const patterns = customPatterns || DEFAULT_PATTERNS;\n\n let bestMatch: IntentType = 'general_question';\n let bestScore = 0;\n let matchedPattern = '';\n\n // 计算每个意图的匹配分数\n for (const pattern of patterns) {\n let score = 0;\n for (const regex of pattern.patterns) {\n if (regex.test(text)) {\n score += 1;\n matchedPattern = regex.source;\n }\n }\n\n // 如果有多个模式匹配,增加权重\n if (score === pattern.patterns.length && score > 0) {\n score += 0.5;\n }\n\n if (score > bestScore) {\n bestScore = score;\n bestMatch = pattern.type;\n }\n }\n\n // 提取任务类型参数\n const parameters: Record<string, any> = extractTaskParameters(text);\n\n // 计算置信度\n const confidence = Math.min(bestScore / 3, 1);\n\n return {\n type: bestMatch,\n confidence,\n parameters,\n rawText: text,\n matchedPattern: bestMatch !== 'general_question' ? matchedPattern : undefined,\n };\n}\n\n/**\n * 提取任务参数\n * @param text 用户输入文本\n * @returns 提取的参数\n */\nfunction extractTaskParameters(text: string): Record<string, any> {\n const params: Record<string, any> = {\n taskType: detectTaskType(text),\n originalText: text,\n };\n\n // 提取任务名称\n const nameMatch = text.match(/名为[\"']?([^\"']+)[\"']?/i);\n if (nameMatch) {\n params.taskName = nameMatch[1];\n }\n\n // 提取优先级\n if (/高优先级|紧急|urgent|critical/i.test(text)) {\n params.priority = 'high';\n } else if (/低优先级|optional|low/i.test(text)) {\n params.priority = 'low';\n } else {\n params.priority = 'medium';\n }\n\n // 提取描述信息\n const descMatch = text.match(/描述|说明|备注[::]?\\s*(.+)$/i);\n if (descMatch) {\n params.description = descMatch[1].trim();\n }\n\n // 提取时间信息\n const timeMatch = text.match(/(明天|下周|后天|今天|立即|马上|尽快)/i);\n if (timeMatch) {\n params.scheduledTime = timeMatch[1];\n }\n\n return params;\n}\n\n/**\n * 检测任务类型\n * @param text 用户输入文本\n * @returns 任务类型\n */\nfunction detectTaskType(text: string): TaskCategory {\n for (const [key, config] of Object.entries(TASK_TYPE_KEYWORDS)) {\n for (const pattern of config.patterns) {\n if (pattern.test(text)) {\n return config.type;\n }\n }\n }\n return 'other';\n}\n\n/**\n * 生成建议问题\n * @param text 用户输入文本\n * @param intent 识别的意图\n * @returns 建议问题列表\n */\nexport function generateSuggestions(text: string, intent: Intent): string[] {\n const suggestions: string[] = [];\n\n switch (intent.type) {\n case 'create_task':\n suggestions.push(\n '帮我创建一个数据同步任务',\n '创建一个新的报表生成任务',\n '启动Paimon表同步作业',\n '帮我创建一个ETL数据处理任务',\n );\n break;\n case 'query_task':\n suggestions.push('查询我的任务列表', '查看任务执行状态', '获取任务执行结果', '查看任务进度');\n break;\n case 'update_task':\n suggestions.push('更新任务配置', '修改任务参数', '调整任务执行时间', '优化任务设置');\n break;\n case 'delete_task':\n suggestions.push('取消当前任务', '删除失败任务', '清除已完成的任务');\n break;\n default:\n suggestions.push('帮我解答一个问题', '分析这段代码', '解释这个概念', '如何优化查询性能');\n }\n\n return suggestions;\n}\n\n/**\n * 判断是否为任务创建意图\n * @param text 用户输入文本\n * @param threshold 置信度阈值\n * @returns 是否为任务创建意图\n */\nexport function isTaskCreationIntent(text: string, threshold = 0.5): boolean {\n const intent = recognizeIntent(text);\n return intent.type === 'create_task' && intent.confidence >= threshold;\n}\n\n/**\n * 创建任务描述(自然语言转结构化数据)\n * @param text 用户输入文本\n * @returns 结构化的任务描述\n */\nexport function parseTaskDescription(text: string): {\n taskType: TaskCategory;\n taskName: string;\n description: string;\n priority: 'high' | 'medium' | 'low';\n} {\n const intent = recognizeIntent(text);\n const params = intent.parameters;\n\n return {\n taskType: params.taskType || detectTaskType(text),\n taskName: params.taskName || `${params.taskType || '新'}任务 - ${new Date().toLocaleTimeString().slice(0, 5)}`,\n description: params.description || text,\n priority: params.priority || 'medium',\n };\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -100,7 +100,7 @@ var NavigationManager = class {
100
100
  * @param options 导航选项
101
101
  */
102
102
  static navigateToTask(taskId, options) {
103
- const url = this.buildTaskUrl(taskId, {
103
+ const url = NavigationManager.buildTaskUrl(taskId, {
104
104
  baseUrl: options == null ? void 0 : options.baseUrl,
105
105
  taskType: options == null ? void 0 : options.taskType,
106
106
  additionalParams: options == null ? void 0 : options.queryParams
@@ -117,7 +117,7 @@ var NavigationManager = class {
117
117
  */
118
118
  static refreshCurrentPage(taskId) {
119
119
  if (taskId) {
120
- this.updateQueryParams({ taskId }, { replace: false });
120
+ NavigationManager.updateQueryParams({ taskId }, { replace: false });
121
121
  }
122
122
  window.location.reload();
123
123
  }
@@ -126,7 +126,7 @@ var NavigationManager = class {
126
126
  * @param params 查询参数
127
127
  */
128
128
  static refreshWithParams(params) {
129
- this.updateQueryParams(params, { replace: false });
129
+ NavigationManager.updateQueryParams(params, { replace: false });
130
130
  window.location.reload();
131
131
  }
132
132
  /**
@@ -135,14 +135,14 @@ var NavigationManager = class {
135
135
  * @returns 任务ID
136
136
  */
137
137
  static extractTaskId(paramName = "taskId") {
138
- return this.getQueryParam(paramName);
138
+ return NavigationManager.getQueryParam(paramName);
139
139
  }
140
140
  /**
141
141
  * 从URL中提取所有任务相关信息
142
142
  * @returns 任务相关信息对象
143
143
  */
144
144
  static extractTaskInfo() {
145
- const params = this.getCurrentParams();
145
+ const params = NavigationManager.getCurrentParams();
146
146
  return {
147
147
  taskId: params.taskId,
148
148
  taskType: params.type,
@@ -158,7 +158,7 @@ var NavigationManager = class {
158
158
  paramsToClean.forEach((key) => {
159
159
  params[key] = "";
160
160
  });
161
- this.updateQueryParams(params, { merge: false });
161
+ NavigationManager.updateQueryParams(params, { merge: false });
162
162
  }
163
163
  /**
164
164
  * 监听URL变化(用于SPA路由变化检测)
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/business/AiChat/navigationManager.ts"],
4
- "sourcesContent": ["import type { NavigationOptions } from './type';\n\n/**\n * 导航管理器 - 用于处理URL更新和页面导航\n */\nexport class NavigationManager {\n /**\n * 更新URL查询参数\n * @param params 要更新的参数\n * @param options 导航选项\n */\n static updateQueryParams(params: Record<string, string>, options?: NavigationOptions): void {\n const { replace = false, merge = true } = options || {};\n\n const currentParams = new URLSearchParams(window.location.search);\n\n // 根据需要合并或替换参数\n if (merge) {\n Object.entries(params).forEach(([key, value]) => {\n if (value) {\n currentParams.set(key, value);\n } else {\n currentParams.delete(key);\n }\n });\n } else {\n // 替换所有参数\n Object.entries(params).forEach(([key, value]) => {\n if (value) {\n currentParams.set(key, value);\n }\n });\n }\n\n const newUrl = `${window.location.pathname}?${currentParams.toString()}${options?.fragment ? `#${options.fragment}` : ''}`;\n\n if (replace) {\n window.history.replaceState({ ...window.history.state, params }, '', newUrl);\n } else {\n window.history.pushState({ ...window.history.state, params }, '', newUrl);\n }\n }\n\n /**\n * 获取当前URL查询参数\n * @returns 查询参数对象\n */\n static getCurrentParams(): Record<string, string> {\n const params = new URLSearchParams(window.location.search);\n const result: Record<string, string> = {};\n\n params.forEach((value, key) => {\n result[key] = value;\n });\n\n return result;\n }\n\n /**\n * 获取单个查询参数值\n * @param key 参数名\n * @returns 参数值\n */\n static getQueryParam(key: string): string | undefined {\n const params = new URLSearchParams(window.location.search);\n return params.get(key) || undefined;\n }\n\n /**\n * 构建任务详情URL\n * @param taskId 任务ID\n * @param baseUrl 基础URL(可选,默认当前页面)\n * @param options 其他选项\n * @returns 完整的URL字符串\n */\n static buildTaskUrl(\n taskId: string,\n options?: {\n baseUrl?: string;\n taskType?: string;\n additionalParams?: Record<string, string>;\n },\n ): string {\n const baseUrl = options?.baseUrl || window.location.pathname;\n const taskTypeParamName = 'type';\n const taskIdParamName = 'taskId';\n\n const params = new URLSearchParams();\n params.set(taskIdParamName, taskId);\n\n if (options?.taskType) {\n params.set(taskTypeParamName, options.taskType);\n }\n\n if (options?.additionalParams) {\n Object.entries(options.additionalParams).forEach(([key, value]) => {\n params.set(key, value);\n });\n }\n\n return `${baseUrl}?${params.toString()}`;\n }\n\n /**\n * 导航到任务页面\n * @param taskId 任务ID\n * @param options 导航选项\n */\n static navigateToTask(\n taskId: string,\n options?: NavigationOptions & {\n baseUrl?: string;\n taskType?: string;\n },\n ): void {\n const url = this.buildTaskUrl(taskId, {\n baseUrl: options?.baseUrl,\n taskType: options?.taskType,\n additionalParams: options?.queryParams,\n });\n\n if (options?.replace) {\n window.history.replaceState({ taskId, taskType: options?.taskType }, '', url);\n } else {\n window.history.pushState({ taskId, taskType: options?.taskType }, '', url);\n }\n }\n\n /**\n * 刷新当前页面\n * @param taskId 可选的任务ID\n */\n static refreshCurrentPage(taskId?: string): void {\n if (taskId) {\n // 如果有任务ID,更新URL后再刷新\n this.updateQueryParams({ taskId }, { replace: false });\n }\n\n window.location.reload();\n }\n\n /**\n * 携带参数刷新页面\n * @param params 查询参数\n */\n static refreshWithParams(params: Record<string, string>): void {\n this.updateQueryParams(params, { replace: false });\n window.location.reload();\n }\n\n /**\n * 从URL中提取任务ID\n * @param paramName 参数名(默认taskId)\n * @returns 任务ID\n */\n static extractTaskId(paramName = 'taskId'): string | undefined {\n return this.getQueryParam(paramName);\n }\n\n /**\n * 从URL中提取所有任务相关信息\n * @returns 任务相关信息对象\n */\n static extractTaskInfo(): {\n taskId?: string;\n taskType?: string;\n queryParams: Record<string, string>;\n } {\n const params = this.getCurrentParams();\n return {\n taskId: params.taskId,\n taskType: params.type,\n queryParams: params,\n };\n }\n\n /**\n * 清理URL中的任务参数\n * @param paramsToClean 要清理的参数名列表\n */\n static clearTaskParams(paramsToClean: string[] = ['taskId', 'type']): void {\n const params: Record<string, string> = {};\n paramsToClean.forEach((key) => {\n params[key] = '';\n });\n this.updateQueryParams(params, { merge: false });\n }\n\n /**\n * 监听URL变化(用于SPA路由变化检测)\n * @param callback 回调函数\n * @returns 取消监听的函数\n */\n static onUrlChange(callback: () => void): () => void {\n let lastUrl = window.location.href;\n\n const checkUrl = () => {\n if (window.location.href !== lastUrl) {\n lastUrl = window.location.href;\n callback();\n }\n };\n\n // 监听popstate事件\n window.addEventListener('popstate', checkUrl);\n\n // 重写history方法\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n window.history.pushState = function (...args) {\n originalPushState.apply(this, args);\n checkUrl();\n };\n\n window.history.replaceState = function (...args) {\n originalReplaceState.apply(this, args);\n checkUrl();\n };\n\n // 返回取消监听的函数\n return () => {\n window.removeEventListener('popstate', checkUrl);\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n };\n }\n}\n\n/**\n * 延迟函数\n * @param ms 延迟毫秒数\n * @returns Promise\n */\nexport function delay(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * 创建带超时的Promise\n * @param promise 原始Promise\n * @param timeout 超时时间(毫秒)\n * @param timeoutMessage 超时错误消息\n * @returns Promise\n */\nexport function withTimeout<T>(\n promise: Promise<T>,\n timeout: number,\n timeoutMessage = 'Operation timed out',\n): Promise<T> {\n return Promise.race([\n promise,\n new Promise<T>((_, reject) => setTimeout(() => reject(new Error(timeoutMessage)), timeout)),\n ]);\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import type { NavigationOptions } from './type';\n\n/**\n * 导航管理器 - 用于处理URL更新和页面导航\n */\nexport class NavigationManager {\n /**\n * 更新URL查询参数\n * @param params 要更新的参数\n * @param options 导航选项\n */\n static updateQueryParams(params: Record<string, string>, options?: NavigationOptions): void {\n const { replace = false, merge = true } = options || {};\n\n const currentParams = new URLSearchParams(window.location.search);\n\n // 根据需要合并或替换参数\n if (merge) {\n Object.entries(params).forEach(([key, value]) => {\n if (value) {\n currentParams.set(key, value);\n } else {\n currentParams.delete(key);\n }\n });\n } else {\n // 替换所有参数\n Object.entries(params).forEach(([key, value]) => {\n if (value) {\n currentParams.set(key, value);\n }\n });\n }\n\n const newUrl = `${window.location.pathname}?${currentParams.toString()}${options?.fragment ? `#${options.fragment}` : ''}`;\n\n if (replace) {\n window.history.replaceState({ ...window.history.state, params }, '', newUrl);\n } else {\n window.history.pushState({ ...window.history.state, params }, '', newUrl);\n }\n }\n\n /**\n * 获取当前URL查询参数\n * @returns 查询参数对象\n */\n static getCurrentParams(): Record<string, string> {\n const params = new URLSearchParams(window.location.search);\n const result: Record<string, string> = {};\n\n params.forEach((value, key) => {\n result[key] = value;\n });\n\n return result;\n }\n\n /**\n * 获取单个查询参数值\n * @param key 参数名\n * @returns 参数值\n */\n static getQueryParam(key: string): string | undefined {\n const params = new URLSearchParams(window.location.search);\n return params.get(key) || undefined;\n }\n\n /**\n * 构建任务详情URL\n * @param taskId 任务ID\n * @param baseUrl 基础URL(可选,默认当前页面)\n * @param options 其他选项\n * @returns 完整的URL字符串\n */\n static buildTaskUrl(\n taskId: string,\n options?: {\n baseUrl?: string;\n taskType?: string;\n additionalParams?: Record<string, string>;\n },\n ): string {\n const baseUrl = options?.baseUrl || window.location.pathname;\n const taskTypeParamName = 'type';\n const taskIdParamName = 'taskId';\n\n const params = new URLSearchParams();\n params.set(taskIdParamName, taskId);\n\n if (options?.taskType) {\n params.set(taskTypeParamName, options.taskType);\n }\n\n if (options?.additionalParams) {\n Object.entries(options.additionalParams).forEach(([key, value]) => {\n params.set(key, value);\n });\n }\n\n return `${baseUrl}?${params.toString()}`;\n }\n\n /**\n * 导航到任务页面\n * @param taskId 任务ID\n * @param options 导航选项\n */\n static navigateToTask(\n taskId: string,\n options?: NavigationOptions & {\n baseUrl?: string;\n taskType?: string;\n },\n ): void {\n const url = NavigationManager.buildTaskUrl(taskId, {\n baseUrl: options?.baseUrl,\n taskType: options?.taskType,\n additionalParams: options?.queryParams,\n });\n\n if (options?.replace) {\n window.history.replaceState({ taskId, taskType: options?.taskType }, '', url);\n } else {\n window.history.pushState({ taskId, taskType: options?.taskType }, '', url);\n }\n }\n\n /**\n * 刷新当前页面\n * @param taskId 可选的任务ID\n */\n static refreshCurrentPage(taskId?: string): void {\n if (taskId) {\n // 如果有任务ID,更新URL后再刷新\n NavigationManager.updateQueryParams({ taskId }, { replace: false });\n }\n\n window.location.reload();\n }\n\n /**\n * 携带参数刷新页面\n * @param params 查询参数\n */\n static refreshWithParams(params: Record<string, string>): void {\n NavigationManager.updateQueryParams(params, { replace: false });\n window.location.reload();\n }\n\n /**\n * 从URL中提取任务ID\n * @param paramName 参数名(默认taskId)\n * @returns 任务ID\n */\n static extractTaskId(paramName = 'taskId'): string | undefined {\n return NavigationManager.getQueryParam(paramName);\n }\n\n /**\n * 从URL中提取所有任务相关信息\n * @returns 任务相关信息对象\n */\n static extractTaskInfo(): {\n taskId?: string;\n taskType?: string;\n queryParams: Record<string, string>;\n } {\n const params = NavigationManager.getCurrentParams();\n return {\n taskId: params.taskId,\n taskType: params.type,\n queryParams: params,\n };\n }\n\n /**\n * 清理URL中的任务参数\n * @param paramsToClean 要清理的参数名列表\n */\n static clearTaskParams(paramsToClean: string[] = ['taskId', 'type']): void {\n const params: Record<string, string> = {};\n paramsToClean.forEach((key) => {\n params[key] = '';\n });\n NavigationManager.updateQueryParams(params, { merge: false });\n }\n\n /**\n * 监听URL变化(用于SPA路由变化检测)\n * @param callback 回调函数\n * @returns 取消监听的函数\n */\n static onUrlChange(callback: () => void): () => void {\n let lastUrl = window.location.href;\n\n const checkUrl = () => {\n if (window.location.href !== lastUrl) {\n lastUrl = window.location.href;\n callback();\n }\n };\n\n // 监听popstate事件\n window.addEventListener('popstate', checkUrl);\n\n // 重写history方法\n const originalPushState = window.history.pushState;\n const originalReplaceState = window.history.replaceState;\n\n window.history.pushState = function (...args) {\n originalPushState.apply(this, args);\n checkUrl();\n };\n\n window.history.replaceState = function (...args) {\n originalReplaceState.apply(this, args);\n checkUrl();\n };\n\n // 返回取消监听的函数\n return () => {\n window.removeEventListener('popstate', checkUrl);\n window.history.pushState = originalPushState;\n window.history.replaceState = originalReplaceState;\n };\n }\n}\n\n/**\n * 延迟函数\n * @param ms 延迟毫秒数\n * @returns Promise\n */\nexport function delay(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * 创建带超时的Promise\n * @param promise 原始Promise\n * @param timeout 超时时间(毫秒)\n * @param timeoutMessage 超时错误消息\n * @returns Promise\n */\nexport function withTimeout<T>(\n promise: Promise<T>,\n timeout: number,\n timeoutMessage = 'Operation timed out',\n): Promise<T> {\n return Promise.race([\n promise,\n new Promise<T>((_, reject) => setTimeout(() => reject(new Error(timeoutMessage)), timeout)),\n ]);\n}\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;AAKO,IAAM,oBAAN,MAAwB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAM7B,OAAO,kBAAkB,QAAgC,SAAmC;AAC1F,UAAM,EAAE,UAAU,OAAO,QAAQ,KAAK,IAAI,WAAW,CAAC;AAEtD,UAAM,gBAAgB,IAAI,gBAAgB,OAAO,SAAS,MAAM;AAGhE,QAAI,OAAO;AACT,aAAO,QAAQ,MAAM,EAAE,QAAQ,CAAC,CAAC,KAAK,KAAK,MAAM;AAC/C,YAAI,OAAO;AACT,wBAAc,IAAI,KAAK,KAAK;AAAA,QAC9B,OAAO;AACL,wBAAc,OAAO,GAAG;AAAA,QAC1B;AAAA,MACF,CAAC;AAAA,IACH,OAAO;AAEL,aAAO,QAAQ,MAAM,EAAE,QAAQ,CAAC,CAAC,KAAK,KAAK,MAAM;AAC/C,YAAI,OAAO;AACT,wBAAc,IAAI,KAAK,KAAK;AAAA,QAC9B;AAAA,MACF,CAAC;AAAA,IACH;AAEA,UAAM,SAAS,GAAG,OAAO,SAAS,YAAY,cAAc,SAAS,KAAI,mCAAS,YAAW,IAAI,QAAQ,aAAa;AAEtH,QAAI,SAAS;AACX,aAAO,QAAQ,aAAa,iCAAK,OAAO,QAAQ,QAApB,EAA2B,OAAO,IAAG,IAAI,MAAM;AAAA,IAC7E,OAAO;AACL,aAAO,QAAQ,UAAU,iCAAK,OAAO,QAAQ,QAApB,EAA2B,OAAO,IAAG,IAAI,MAAM;AAAA,IAC1E;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,OAAO,mBAA2C;AAChD,UAAM,SAAS,IAAI,gBAAgB,OAAO,SAAS,MAAM;AACzD,UAAM,SAAiC,CAAC;AAExC,WAAO,QAAQ,CAAC,OAAO,QAAQ;AAC7B,aAAO,GAAG,IAAI;AAAA,IAChB,CAAC;AAED,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAO,cAAc,KAAiC;AACpD,UAAM,SAAS,IAAI,gBAAgB,OAAO,SAAS,MAAM;AACzD,WAAO,OAAO,IAAI,GAAG,KAAK;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,OAAO,aACL,QACA,SAKQ;AACR,UAAM,WAAU,mCAAS,YAAW,OAAO,SAAS;AACpD,UAAM,oBAAoB;AAC1B,UAAM,kBAAkB;AAExB,UAAM,SAAS,IAAI,gBAAgB;AACnC,WAAO,IAAI,iBAAiB,MAAM;AAElC,QAAI,mCAAS,UAAU;AACrB,aAAO,IAAI,mBAAmB,QAAQ,QAAQ;AAAA,IAChD;AAEA,QAAI,mCAAS,kBAAkB;AAC7B,aAAO,QAAQ,QAAQ,gBAAgB,EAAE,QAAQ,CAAC,CAAC,KAAK,KAAK,MAAM;AACjE,eAAO,IAAI,KAAK,KAAK;AAAA,MACvB,CAAC;AAAA,IACH;AAEA,WAAO,GAAG,WAAW,OAAO,SAAS;AAAA,EACvC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAO,eACL,QACA,SAIM;AACN,UAAM,MAAM,kBAAkB,aAAa,QAAQ;AAAA,MACjD,SAAS,mCAAS;AAAA,MAClB,UAAU,mCAAS;AAAA,MACnB,kBAAkB,mCAAS;AAAA,IAC7B,CAAC;AAED,QAAI,mCAAS,SAAS;AACpB,aAAO,QAAQ,aAAa,EAAE,QAAQ,UAAU,mCAAS,SAAS,GAAG,IAAI,GAAG;AAAA,IAC9E,OAAO;AACL,aAAO,QAAQ,UAAU,EAAE,QAAQ,UAAU,mCAAS,SAAS,GAAG,IAAI,GAAG;AAAA,IAC3E;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,OAAO,mBAAmB,QAAuB;AAC/C,QAAI,QAAQ;AAEV,wBAAkB,kBAAkB,EAAE,OAAO,GAAG,EAAE,SAAS,MAAM,CAAC;AAAA,IACpE;AAEA,WAAO,SAAS,OAAO;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,OAAO,kBAAkB,QAAsC;AAC7D,sBAAkB,kBAAkB,QAAQ,EAAE,SAAS,MAAM,CAAC;AAC9D,WAAO,SAAS,OAAO;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAO,cAAc,YAAY,UAA8B;AAC7D,WAAO,kBAAkB,cAAc,SAAS;AAAA,EAClD;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,OAAO,kBAIL;AACA,UAAM,SAAS,kBAAkB,iBAAiB;AAClD,WAAO;AAAA,MACL,QAAQ,OAAO;AAAA,MACf,UAAU,OAAO;AAAA,MACjB,aAAa;AAAA,IACf;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,OAAO,gBAAgB,gBAA0B,CAAC,UAAU,MAAM,GAAS;AACzE,UAAM,SAAiC,CAAC;AACxC,kBAAc,QAAQ,CAAC,QAAQ;AAC7B,aAAO,GAAG,IAAI;AAAA,IAChB,CAAC;AACD,sBAAkB,kBAAkB,QAAQ,EAAE,OAAO,MAAM,CAAC;AAAA,EAC9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAO,YAAY,UAAkC;AACnD,QAAI,UAAU,OAAO,SAAS;AAE9B,UAAM,WAAW,MAAM;AACrB,UAAI,OAAO,SAAS,SAAS,SAAS;AACpC,kBAAU,OAAO,SAAS;AAC1B,iBAAS;AAAA,MACX;AAAA,IACF;AAGA,WAAO,iBAAiB,YAAY,QAAQ;AAG5C,UAAM,oBAAoB,OAAO,QAAQ;AACzC,UAAM,uBAAuB,OAAO,QAAQ;AAE5C,WAAO,QAAQ,YAAY,YAAa,MAAM;AAC5C,wBAAkB,MAAM,MAAM,IAAI;AAClC,eAAS;AAAA,IACX;AAEA,WAAO,QAAQ,eAAe,YAAa,MAAM;AAC/C,2BAAqB,MAAM,MAAM,IAAI;AACrC,eAAS;AAAA,IACX;AAGA,WAAO,MAAM;AACX,aAAO,oBAAoB,YAAY,QAAQ;AAC/C,aAAO,QAAQ,YAAY;AAC3B,aAAO,QAAQ,eAAe;AAAA,IAChC;AAAA,EACF;AACF;AAOO,SAAS,MAAM,IAA2B;AAC/C,SAAO,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,EAAE,CAAC;AACzD;AASO,SAAS,YACd,SACA,SACA,iBAAiB,uBACL;AACZ,SAAO,QAAQ,KAAK;AAAA,IAClB;AAAA,IACA,IAAI,QAAW,CAAC,GAAG,WAAW,WAAW,MAAM,OAAO,IAAI,MAAM,cAAc,CAAC,GAAG,OAAO,CAAC;AAAA,EAC5F,CAAC;AACH;",
6
6
  "names": []
7
7
  }
@@ -1,5 +1,5 @@
1
1
  // src/business/DrawerPageInfo/index.tsx
2
- import { DislikeOutlined, LikeOutlined } from "@ant-design/icons";
2
+ import { CloseOutlined, DislikeOutlined, LikeOutlined } from "@ant-design/icons";
3
3
  import { Button, Divider, Drawer, Modal, Skeleton } from "antd";
4
4
  import React, { useCallback, useEffect, useRef, useState } from "react";
5
5
  import styles from "./index.module.less";
@@ -84,7 +84,7 @@ var DrawerPageInfo = ({ id, open, onOpenChange, loadData, onFeedback }) => {
84
84
  width: 600,
85
85
  open,
86
86
  onClose,
87
- extra: /* @__PURE__ */ React.createElement("div", { className: styles.pageInfoDrawerExtraClose, onClick: onClose, role: "button", tabIndex: 0 }, /* @__PURE__ */ React.createElement("i", { className: "iconfont icon-guanbi" })),
87
+ extra: /* @__PURE__ */ React.createElement("div", { className: styles.pageInfoDrawerExtraClose, onClick: onClose, role: "button", tabIndex: 0 }, /* @__PURE__ */ React.createElement(CloseOutlined, null)),
88
88
  maskClosable: false,
89
89
  className: styles.drawerTipInfo
90
90
  },