@opensumi/ide-ai-native 2.26.9-rc-1695091570.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (196) hide show
  1. package/LICENSE +21 -0
  2. package/lib/browser/ai-chat.contribution.d.ts +26 -0
  3. package/lib/browser/ai-chat.contribution.d.ts.map +1 -0
  4. package/lib/browser/ai-chat.contribution.js +227 -0
  5. package/lib/browser/ai-chat.contribution.js.map +1 -0
  6. package/lib/browser/ai-chat.module.less +174 -0
  7. package/lib/browser/ai-chat.service.d.ts +30 -0
  8. package/lib/browser/ai-chat.service.d.ts.map +1 -0
  9. package/lib/browser/ai-chat.service.js +134 -0
  10. package/lib/browser/ai-chat.service.js.map +1 -0
  11. package/lib/browser/ai-chat.view.d.ts +4 -0
  12. package/lib/browser/ai-chat.view.d.ts.map +1 -0
  13. package/lib/browser/ai-chat.view.js +305 -0
  14. package/lib/browser/ai-chat.view.js.map +1 -0
  15. package/lib/browser/ai-editor.contribution.d.ts +24 -0
  16. package/lib/browser/ai-editor.contribution.d.ts.map +1 -0
  17. package/lib/browser/ai-editor.contribution.js +419 -0
  18. package/lib/browser/ai-editor.contribution.js.map +1 -0
  19. package/lib/browser/ai-project/generate.service.d.ts +31 -0
  20. package/lib/browser/ai-project/generate.service.d.ts.map +1 -0
  21. package/lib/browser/ai-project/generate.service.js +453 -0
  22. package/lib/browser/ai-project/generate.service.js.map +1 -0
  23. package/lib/browser/ai-sumi/sumi.service.d.ts +11 -0
  24. package/lib/browser/ai-sumi/sumi.service.d.ts.map +1 -0
  25. package/lib/browser/ai-sumi/sumi.service.js +64 -0
  26. package/lib/browser/ai-sumi/sumi.service.js.map +1 -0
  27. package/lib/browser/code-widget/ai-code-document.provider.d.ts +10 -0
  28. package/lib/browser/code-widget/ai-code-document.provider.d.ts.map +1 -0
  29. package/lib/browser/code-widget/ai-code-document.provider.js +26 -0
  30. package/lib/browser/code-widget/ai-code-document.provider.js.map +1 -0
  31. package/lib/browser/code-widget/ai-code-widget.d.ts +20 -0
  32. package/lib/browser/code-widget/ai-code-widget.d.ts.map +1 -0
  33. package/lib/browser/code-widget/ai-code-widget.js +164 -0
  34. package/lib/browser/code-widget/ai-code-widget.js.map +1 -0
  35. package/lib/browser/components/AIImprove.d.ts +9 -0
  36. package/lib/browser/components/AIImprove.d.ts.map +1 -0
  37. package/lib/browser/components/AIImprove.js +34 -0
  38. package/lib/browser/components/AIImprove.js.map +1 -0
  39. package/lib/browser/components/AIInput.d.ts +5 -0
  40. package/lib/browser/components/AIInput.d.ts.map +1 -0
  41. package/lib/browser/components/AIInput.js +21 -0
  42. package/lib/browser/components/AIInput.js.map +1 -0
  43. package/lib/browser/components/ChatEditor.d.ts +5 -0
  44. package/lib/browser/components/ChatEditor.d.ts.map +1 -0
  45. package/lib/browser/components/ChatEditor.js +85 -0
  46. package/lib/browser/components/ChatEditor.js.map +1 -0
  47. package/lib/browser/components/ChatInput.d.ts +6 -0
  48. package/lib/browser/components/ChatInput.d.ts.map +1 -0
  49. package/lib/browser/components/ChatInput.js +127 -0
  50. package/lib/browser/components/ChatInput.js.map +1 -0
  51. package/lib/browser/components/ChatMoreActions.d.ts +5 -0
  52. package/lib/browser/components/ChatMoreActions.d.ts.map +1 -0
  53. package/lib/browser/components/ChatMoreActions.js +26 -0
  54. package/lib/browser/components/ChatMoreActions.js.map +1 -0
  55. package/lib/browser/components/Thinking.d.ts +3 -0
  56. package/lib/browser/components/Thinking.d.ts.map +1 -0
  57. package/lib/browser/components/Thinking.js +19 -0
  58. package/lib/browser/components/Thinking.js.map +1 -0
  59. package/lib/browser/components/components.module.less +259 -0
  60. package/lib/browser/components/lineVertical.d.ts +4 -0
  61. package/lib/browser/components/lineVertical.d.ts.map +1 -0
  62. package/lib/browser/components/lineVertical.js +22 -0
  63. package/lib/browser/components/lineVertical.js.map +1 -0
  64. package/lib/browser/content-widget/ai-content-widget.d.ts +38 -0
  65. package/lib/browser/content-widget/ai-content-widget.d.ts.map +1 -0
  66. package/lib/browser/content-widget/ai-content-widget.js +97 -0
  67. package/lib/browser/content-widget/ai-content-widget.js.map +1 -0
  68. package/lib/browser/content-widget/ai-inline-chat-panel.d.ts +6 -0
  69. package/lib/browser/content-widget/ai-inline-chat-panel.d.ts.map +1 -0
  70. package/lib/browser/content-widget/ai-inline-chat-panel.js +108 -0
  71. package/lib/browser/content-widget/ai-inline-chat-panel.js.map +1 -0
  72. package/lib/browser/content-widget/ai-inline-chat.module.less +74 -0
  73. package/lib/browser/content-widget/ai-inline-chat.service.d.ts +23 -0
  74. package/lib/browser/content-widget/ai-inline-chat.service.d.ts.map +1 -0
  75. package/lib/browser/content-widget/ai-inline-chat.service.js +47 -0
  76. package/lib/browser/content-widget/ai-inline-chat.service.js.map +1 -0
  77. package/lib/browser/diff-widget/ai-diff-document.provider.d.ts +10 -0
  78. package/lib/browser/diff-widget/ai-diff-document.provider.d.ts.map +1 -0
  79. package/lib/browser/diff-widget/ai-diff-document.provider.js +26 -0
  80. package/lib/browser/diff-widget/ai-diff-document.provider.js.map +1 -0
  81. package/lib/browser/diff-widget/ai-diff-widget.d.ts +20 -0
  82. package/lib/browser/diff-widget/ai-diff-widget.d.ts.map +1 -0
  83. package/lib/browser/diff-widget/ai-diff-widget.js +158 -0
  84. package/lib/browser/diff-widget/ai-diff-widget.js.map +1 -0
  85. package/lib/browser/index.d.ts +12 -0
  86. package/lib/browser/index.d.ts.map +1 -0
  87. package/lib/browser/index.js +84 -0
  88. package/lib/browser/index.js.map +1 -0
  89. package/lib/browser/inline-completions/constants.d.ts +62 -0
  90. package/lib/browser/inline-completions/constants.d.ts.map +1 -0
  91. package/lib/browser/inline-completions/constants.js +69 -0
  92. package/lib/browser/inline-completions/constants.js.map +1 -0
  93. package/lib/browser/inline-completions/provider.d.ts +27 -0
  94. package/lib/browser/inline-completions/provider.d.ts.map +1 -0
  95. package/lib/browser/inline-completions/provider.js +63 -0
  96. package/lib/browser/inline-completions/provider.js.map +1 -0
  97. package/lib/browser/override/ai-editor-tab.service.d.ts +6 -0
  98. package/lib/browser/override/ai-editor-tab.service.d.ts.map +1 -0
  99. package/lib/browser/override/ai-editor-tab.service.js +22 -0
  100. package/lib/browser/override/ai-editor-tab.service.js.map +1 -0
  101. package/lib/browser/override/ai-marker.service.d.ts +11 -0
  102. package/lib/browser/override/ai-marker.service.d.ts.map +1 -0
  103. package/lib/browser/override/ai-marker.service.js +55 -0
  104. package/lib/browser/override/ai-marker.service.js.map +1 -0
  105. package/lib/browser/override/global.styles.less +268 -0
  106. package/lib/browser/override/layout/layout-config.d.ts +6 -0
  107. package/lib/browser/override/layout/layout-config.d.ts.map +1 -0
  108. package/lib/browser/override/layout/layout-config.js +17 -0
  109. package/lib/browser/override/layout/layout-config.js.map +1 -0
  110. package/lib/browser/override/layout/layout.module.less +7 -0
  111. package/lib/browser/override/layout/main-slot-renderer.d.ts +4 -0
  112. package/lib/browser/override/layout/main-slot-renderer.d.ts.map +1 -0
  113. package/lib/browser/override/layout/main-slot-renderer.js +22 -0
  114. package/lib/browser/override/layout/main-slot-renderer.js.map +1 -0
  115. package/lib/browser/override/layout/menu-bar/menu-bar.contribution.d.ts +8 -0
  116. package/lib/browser/override/layout/menu-bar/menu-bar.contribution.d.ts.map +1 -0
  117. package/lib/browser/override/layout/menu-bar/menu-bar.contribution.js +32 -0
  118. package/lib/browser/override/layout/menu-bar/menu-bar.contribution.js.map +1 -0
  119. package/lib/browser/override/layout/menu-bar/menu-bar.module.less +41 -0
  120. package/lib/browser/override/layout/menu-bar/menu-bar.view.d.ts +3 -0
  121. package/lib/browser/override/layout/menu-bar/menu-bar.view.d.ts.map +1 -0
  122. package/lib/browser/override/layout/menu-bar/menu-bar.view.js +62 -0
  123. package/lib/browser/override/layout/menu-bar/menu-bar.view.js.map +1 -0
  124. package/lib/browser/override/layout/tabbar.view.d.ts +17 -0
  125. package/lib/browser/override/layout/tabbar.view.d.ts.map +1 -0
  126. package/lib/browser/override/layout/tabbar.view.js +38 -0
  127. package/lib/browser/override/layout/tabbar.view.js.map +1 -0
  128. package/lib/browser/override/override.module.less +32 -0
  129. package/lib/browser/override/theme/default-theme.d.ts +315 -0
  130. package/lib/browser/override/theme/default-theme.d.ts.map +1 -0
  131. package/lib/browser/override/theme/default-theme.js +821 -0
  132. package/lib/browser/override/theme/default-theme.js.map +1 -0
  133. package/lib/browser/run/run.service.d.ts +52 -0
  134. package/lib/browser/run/run.service.d.ts.map +1 -0
  135. package/lib/browser/run/run.service.js +176 -0
  136. package/lib/browser/run/run.service.js.map +1 -0
  137. package/lib/common/command.d.ts +17 -0
  138. package/lib/common/command.d.ts.map +1 -0
  139. package/lib/common/command.js +21 -0
  140. package/lib/common/command.js.map +1 -0
  141. package/lib/common/index.d.ts +37 -0
  142. package/lib/common/index.d.ts.map +1 -0
  143. package/lib/common/index.js +22 -0
  144. package/lib/common/index.js.map +1 -0
  145. package/lib/index.d.ts +2 -0
  146. package/lib/index.d.ts.map +1 -0
  147. package/lib/index.js +5 -0
  148. package/lib/index.js.map +1 -0
  149. package/lib/node/index.d.ts +10 -0
  150. package/lib/node/index.d.ts.map +1 -0
  151. package/lib/node/index.js +32 -0
  152. package/lib/node/index.js.map +1 -0
  153. package/package.json +38 -0
  154. package/src/browser/ai-chat.contribution.ts +294 -0
  155. package/src/browser/ai-chat.module.less +174 -0
  156. package/src/browser/ai-chat.service.ts +152 -0
  157. package/src/browser/ai-chat.view.tsx +421 -0
  158. package/src/browser/ai-editor.contribution.ts +521 -0
  159. package/src/browser/ai-project/generate.service.ts +505 -0
  160. package/src/browser/ai-sumi/sumi.service.ts +65 -0
  161. package/src/browser/code-widget/ai-code-document.provider.ts +22 -0
  162. package/src/browser/code-widget/ai-code-widget.tsx +202 -0
  163. package/src/browser/components/AIImprove.tsx +45 -0
  164. package/src/browser/components/AIInput.tsx +30 -0
  165. package/src/browser/components/ChatEditor.tsx +129 -0
  166. package/src/browser/components/ChatInput.tsx +197 -0
  167. package/src/browser/components/ChatMoreActions.tsx +33 -0
  168. package/src/browser/components/Thinking.tsx +23 -0
  169. package/src/browser/components/components.module.less +259 -0
  170. package/src/browser/components/lineVertical.tsx +19 -0
  171. package/src/browser/content-widget/ai-content-widget.tsx +132 -0
  172. package/src/browser/content-widget/ai-inline-chat-panel.tsx +154 -0
  173. package/src/browser/content-widget/ai-inline-chat.module.less +74 -0
  174. package/src/browser/content-widget/ai-inline-chat.service.ts +46 -0
  175. package/src/browser/diff-widget/ai-diff-document.provider.ts +22 -0
  176. package/src/browser/diff-widget/ai-diff-widget.tsx +196 -0
  177. package/src/browser/index.ts +91 -0
  178. package/src/browser/inline-completions/constants.ts +108 -0
  179. package/src/browser/inline-completions/provider.ts +55 -0
  180. package/src/browser/override/ai-editor-tab.service.tsx +22 -0
  181. package/src/browser/override/ai-marker.service.tsx +55 -0
  182. package/src/browser/override/global.styles.less +268 -0
  183. package/src/browser/override/layout/layout-config.ts +21 -0
  184. package/src/browser/override/layout/layout.module.less +7 -0
  185. package/src/browser/override/layout/main-slot-renderer.tsx +62 -0
  186. package/src/browser/override/layout/menu-bar/menu-bar.contribution.tsx +28 -0
  187. package/src/browser/override/layout/menu-bar/menu-bar.module.less +41 -0
  188. package/src/browser/override/layout/menu-bar/menu-bar.view.tsx +92 -0
  189. package/src/browser/override/layout/tabbar.view.tsx +99 -0
  190. package/src/browser/override/override.module.less +32 -0
  191. package/src/browser/override/theme/default-theme.ts +818 -0
  192. package/src/browser/run/run.service.ts +183 -0
  193. package/src/common/command.ts +21 -0
  194. package/src/common/index.ts +39 -0
  195. package/src/index.ts +1 -0
  196. package/src/node/index.ts +24 -0
@@ -0,0 +1,294 @@
1
+ import { Injectable, Autowired, INJECTOR_TOKEN, Injector } from '@opensumi/di';
2
+ import {
3
+ AppConfig,
4
+ CommandContribution,
5
+ CommandRegistry,
6
+ ComponentContribution,
7
+ ComponentRegistry,
8
+ Domain,
9
+ Position,
10
+ URI,
11
+ getIcon,
12
+ IRange,
13
+ SlotRendererContribution,
14
+ SlotRendererRegistry,
15
+ SlotLocation,
16
+ } from '@opensumi/ide-core-browser';
17
+ import { IMenuRegistry, MenuContribution, MenuId } from '@opensumi/ide-core-browser/lib/menu/next';
18
+ import { DebugConsoleNode } from '@opensumi/ide-debug/lib/browser/tree';
19
+ import { IEditor } from '@opensumi/ide-editor';
20
+ import { WorkbenchEditorService } from '@opensumi/ide-editor';
21
+ import { ResourceService } from '@opensumi/ide-editor';
22
+ import { IResource } from '@opensumi/ide-editor';
23
+ import {
24
+ BrowserEditorContribution,
25
+ IEditorDocumentModelContentRegistry,
26
+ IEditorFeatureRegistry,
27
+ } from '@opensumi/ide-editor/lib/browser';
28
+ import { WorkbenchEditorServiceImpl } from '@opensumi/ide-editor/lib/browser/workbench-editor.service';
29
+ import { IFileTreeAPI } from '@opensumi/ide-file-tree-next';
30
+ import { ITerminalController, ITerminalGroupViewService } from '@opensumi/ide-terminal-next';
31
+
32
+ import { Ai_CHAT_CONTAINER_VIEW_ID } from '../common';
33
+ import { AI_EXPLAIN_DEBUG_COMMANDS, AI_EXPLAIN_TERMINAL_COMMANDS, AI_RUN_DEBUG_COMMANDS } from '../common/command';
34
+
35
+ import { AiChatService } from './ai-chat.service';
36
+ import { AiChatView } from './ai-chat.view';
37
+ import { AiEditorContribution } from './ai-editor.contribution';
38
+ import { AiDiffDocumentProvider } from './diff-widget/ai-diff-document.provider';
39
+ import { AiChatTabRenderer, AiLeftTabRenderer, AiRightTabRenderer } from './override/layout/tabbar.view';
40
+ import { AiRunService } from './run/run.service';
41
+
42
+ @Injectable()
43
+ @Domain(
44
+ ComponentContribution,
45
+ BrowserEditorContribution,
46
+ MenuContribution,
47
+ CommandContribution,
48
+ SlotRendererContribution,
49
+ )
50
+ export class AiChatContribution
51
+ implements
52
+ ComponentContribution,
53
+ BrowserEditorContribution,
54
+ MenuContribution,
55
+ CommandContribution,
56
+ SlotRendererContribution
57
+ {
58
+ @Autowired()
59
+ private readonly aiDiffDocumentProvider: AiDiffDocumentProvider;
60
+
61
+ @Autowired(INJECTOR_TOKEN)
62
+ private readonly injector: Injector;
63
+
64
+ @Autowired(AppConfig)
65
+ private readonly appConfig: AppConfig;
66
+
67
+ @Autowired(IFileTreeAPI)
68
+ private readonly fileTreeAPI: IFileTreeAPI;
69
+
70
+ @Autowired(WorkbenchEditorService)
71
+ private readonly editorService: WorkbenchEditorServiceImpl;
72
+
73
+ @Autowired(AiChatService)
74
+ protected readonly aiChatService: AiChatService;
75
+
76
+ @Autowired(ITerminalGroupViewService)
77
+ protected readonly view: ITerminalGroupViewService;
78
+
79
+ @Autowired(ITerminalController)
80
+ protected readonly terminalController: ITerminalController;
81
+
82
+ @Autowired(AiRunService)
83
+ protected readonly aiRunService: AiRunService;
84
+
85
+ registerComponent(registry: ComponentRegistry): void {
86
+ registry.register(
87
+ Ai_CHAT_CONTAINER_VIEW_ID,
88
+ {
89
+ component: AiChatView,
90
+ id: Ai_CHAT_CONTAINER_VIEW_ID,
91
+ },
92
+ {
93
+ containerId: Ai_CHAT_CONTAINER_VIEW_ID,
94
+ },
95
+ );
96
+ }
97
+
98
+ registerEditorFeature(registry: IEditorFeatureRegistry): void {
99
+ registry.registerEditorFeatureContribution({
100
+ contribute: (editor: IEditor) => {
101
+ const aiEditorContribution = this.injector.get(AiEditorContribution, [editor]);
102
+ return aiEditorContribution.contribute(editor);
103
+ },
104
+ });
105
+ }
106
+
107
+ registerCommands(commands: CommandRegistry): void {
108
+ commands.registerCommand(
109
+ {
110
+ id: 'ai.suggest.documentation',
111
+ },
112
+ {
113
+ execute: async (range: IRange) => {
114
+ const currentEditor = this.editorService.currentEditor;
115
+ if (!currentEditor || !range) {
116
+ return;
117
+ }
118
+
119
+ const getContent = currentEditor.monacoEditor.getModel()!.getValueInRange(range);
120
+
121
+ if (getContent) {
122
+ const messageWithPrompt = `基于提供的代码,需要按照以下这种格式为这段代码添加 javadoc。
123
+ 格式要求是:
124
+ /**
125
+ * $\{关于这段代码的注释\}
126
+ *
127
+ * 如果有参数,则添加 @param $\{参数信息\}。如果没有参数,则不需要添加 @param
128
+ * @return $\{返回类型\}
129
+ */
130
+
131
+ 代码内容是:
132
+ ${getContent}
133
+ `;
134
+
135
+ const aiResult = await this.aiChatService.aiBackService.aiMFTCompletion(messageWithPrompt);
136
+ const resultContent = aiResult.data;
137
+
138
+ // 提取 markdown 里的代码
139
+ const regex = /```java\s*([\s\S]+?)\s*```/;
140
+ let code = regex.exec(resultContent)![1];
141
+
142
+ if (!code) {
143
+ return;
144
+ }
145
+
146
+ const monacoEditor = currentEditor.monacoEditor;
147
+
148
+ if (monacoEditor) {
149
+ const model = monacoEditor.getModel()!;
150
+
151
+ const indents = ' '.repeat(4);
152
+
153
+ const spcode = code.split('\n');
154
+ code = spcode.map((s, i) => (i === 0 ? s : indents + s)).join('\n');
155
+
156
+ model.pushStackElement();
157
+ model.pushEditOperations(
158
+ null,
159
+ [
160
+ {
161
+ range,
162
+ text: code,
163
+ },
164
+ ],
165
+ () => null,
166
+ );
167
+ model.pushStackElement();
168
+
169
+ monacoEditor.focus();
170
+ }
171
+ }
172
+ },
173
+ },
174
+ );
175
+
176
+ commands.registerCommand(
177
+ {
178
+ id: 'ai.chat.createNewFile',
179
+ },
180
+ {
181
+ execute: async (fileName) => {
182
+ const workspaceDir = this.appConfig.workspaceDir;
183
+ return await this.fileTreeAPI.createFile(URI.parse(`${workspaceDir}/${fileName}`));
184
+ },
185
+ },
186
+ );
187
+
188
+ commands.registerCommand(
189
+ {
190
+ id: 'ai.chat.focusLine',
191
+ },
192
+ {
193
+ execute: async (line: number) => {
194
+ line = Number(line);
195
+ let currentEditor = this.editorService.currentEditor;
196
+ if (!currentEditor) {
197
+ this.editorService.setCurrentGroup(this.editorService.editorGroups[0]);
198
+ }
199
+
200
+ currentEditor = this.editorService.currentEditor;
201
+
202
+ currentEditor?.monacoEditor.focus();
203
+ setTimeout(() => {
204
+ currentEditor?.monacoEditor.revealLineInCenter(Number(line));
205
+ currentEditor?.monacoEditor.setPosition(new Position(line, 0));
206
+ }, 0);
207
+ },
208
+ },
209
+ );
210
+
211
+ commands.registerCommand(
212
+ {
213
+ id: 'cloudide.command.workspace.getRuntimeConfig',
214
+ },
215
+ {
216
+ execute: async (key: string) => {
217
+ const obj = {
218
+ workspaceIP: '127.0.0.1',
219
+ workspaceDir: '/Users/mushi/Documents/workcode/opensumi/core/tools/workspace',
220
+ proxyHost: 'https://ide.cloudbaseapp-sanbox.cn',
221
+ };
222
+ return obj[key];
223
+ },
224
+ },
225
+ );
226
+
227
+ commands.registerCommand(AI_EXPLAIN_TERMINAL_COMMANDS, {
228
+ execute: () => {
229
+ const current = this.view.currentWidgetId;
230
+ const client = this.terminalController.findClientFromWidgetId(current);
231
+ if (client) {
232
+ const name = client.name;
233
+ const selectionContent = client.getSelection();
234
+
235
+ this.aiChatService.launchChatMessage({
236
+ message: '/explain @terminalSelection',
237
+ prompt: `请解释一下这一段 ${name} 终端面板里的这部分内容: \`\`\`\n${selectionContent}\n\`\`\` `,
238
+ });
239
+ }
240
+ },
241
+ });
242
+
243
+ commands.registerCommand(AI_EXPLAIN_DEBUG_COMMANDS, {
244
+ execute: (node: DebugConsoleNode) => {
245
+ const description = node.description;
246
+ if (description) {
247
+ this.aiChatService.launchChatMessage({
248
+ message: '/explain @debugSelection',
249
+ prompt: `我在运行并调试我的项目代码,请解释调试运行过程当中的这段日志: \`\`\`\n${description}\n\`\`\` `,
250
+ });
251
+ }
252
+ },
253
+ });
254
+
255
+ commands.registerCommand(AI_RUN_DEBUG_COMMANDS, {
256
+ execute: () => {
257
+ this.aiRunService.run();
258
+ },
259
+ });
260
+ }
261
+ // TerminalClient
262
+ registerMenus(menus: IMenuRegistry): void {
263
+ menus.registerMenuItem(MenuId.TerminalInstanceContext, {
264
+ command: AI_EXPLAIN_TERMINAL_COMMANDS,
265
+ group: '0_ai',
266
+ });
267
+
268
+ menus.registerMenuItem(MenuId.DebugConsoleContext, {
269
+ command: AI_EXPLAIN_DEBUG_COMMANDS,
270
+ group: '0_ai',
271
+ });
272
+ }
273
+
274
+ registerEditorDocumentModelContentProvider(registry: IEditorDocumentModelContentRegistry) {
275
+ registry.registerEditorDocumentModelContentProvider(this.aiDiffDocumentProvider);
276
+ }
277
+
278
+ registerResource(resourceService: ResourceService): void {
279
+ resourceService.registerResourceProvider({
280
+ scheme: 'AI',
281
+ provideResource: async (uri: URI): Promise<IResource<Partial<{ [prop: string]: any }>>> => ({
282
+ uri,
283
+ icon: getIcon('file-text'),
284
+ name: `AI Diff ${uri.displayName}`,
285
+ }),
286
+ });
287
+ }
288
+
289
+ registerRenderer(registry: SlotRendererRegistry): void {
290
+ registry.registerSlotRenderer(SlotLocation.left, AiLeftTabRenderer);
291
+ registry.registerSlotRenderer(SlotLocation.right, AiRightTabRenderer);
292
+ registry.registerSlotRenderer(Ai_CHAT_CONTAINER_VIEW_ID, AiChatTabRenderer);
293
+ }
294
+ }
@@ -0,0 +1,174 @@
1
+ .ai_chat_view {
2
+ display: flex;
3
+ flex-direction: column;
4
+ height: 100%;
5
+ padding-top: 4px;
6
+
7
+ background-color: var(--panel-background);
8
+ border: 1px solid var(--sideBar-border);
9
+ border-radius: 12px;
10
+ overflow: hidden;
11
+
12
+ user-select: text;
13
+
14
+ // ai 助手头像
15
+ .ai_chat_avatar_icon {
16
+ width: 24px;
17
+ height: 24px;
18
+ background-image: radial-gradient(circle at 21% 21%, #00f6ff 0%, #9c03ff 95%);
19
+ border-radius: 12px;
20
+
21
+ img {
22
+ width: 12px;
23
+ height: 12px;
24
+ }
25
+ }
26
+
27
+ .header_container {
28
+ height: 48px;
29
+ display: flex;
30
+ justify-content: space-between;
31
+ align-items: center;
32
+ padding: 12px 16px;
33
+ border-bottom: 1px solid var(--sideBar-border);
34
+
35
+ .left {
36
+ font-size: 14px;
37
+ display: flex;
38
+ align-items: center;
39
+ height: 16px;
40
+
41
+ & > * {
42
+ margin-right: 8px;
43
+ }
44
+ }
45
+
46
+ .right {
47
+ display: flex;
48
+ align-items: center;
49
+
50
+ span {
51
+ color: #666;
52
+ }
53
+
54
+ div:first-child {
55
+ margin-right: 10px;
56
+ }
57
+ }
58
+ }
59
+
60
+ .body_container {
61
+ display: flex;
62
+ height: calc(100% - 48px);
63
+
64
+ .left_bar {
65
+ display: flex;
66
+ flex-direction: column;
67
+ width: calc(100% - 40px);
68
+
69
+ // chat loading
70
+ .chat_loading_msg_box {
71
+ display: flex;
72
+ align-items: flex-start;
73
+ color: #ffffff;
74
+ font-size: 12px;
75
+ font-weight: 300;
76
+ margin-left: 16px;
77
+
78
+ .chat_loading_mgs_avatar {
79
+ height: 32px;
80
+ width: 32px;
81
+ }
82
+ }
83
+
84
+ // 聊天框
85
+ .chat_container {
86
+ flex: 1;
87
+ height: 100%;
88
+ overflow: auto;
89
+ padding-top: 10px;
90
+
91
+ .message_list {
92
+ .rce-mbox-left-notch,
93
+ .rce-mbox-right-notch {
94
+ display: none;
95
+ }
96
+ }
97
+ }
98
+
99
+ .quick_way {
100
+ display: flex;
101
+ align-items: center;
102
+ padding: 4px 8px;
103
+
104
+ .quick_way_item {
105
+ margin-right: 8px;
106
+ cursor: pointer;
107
+ }
108
+ }
109
+
110
+ .chat_input {
111
+ padding: 16px;
112
+ display: flex;
113
+ flex-direction: column;
114
+
115
+ .input_wrapper {
116
+ height: 40px;
117
+ border-radius: 8px;
118
+ background-color: #ffffff14;
119
+
120
+ textarea {
121
+ resize: none;
122
+ }
123
+ }
124
+
125
+ .send_chat_btn {
126
+ border-radius: 2px;
127
+ background-color: rgba(95, 101, 107, 0.25);
128
+ width: 24px;
129
+ height: 24px;
130
+ display: flex;
131
+ align-items: center;
132
+ justify-content: center;
133
+ cursor: pointer;
134
+
135
+ span {
136
+ color: rgba(95, 101, 107, 1);
137
+ }
138
+ }
139
+ }
140
+ }
141
+
142
+ .right_bar {
143
+ width: 40px;
144
+ padding-top: 10px;
145
+ border-left: 1px solid #2a3344e6;
146
+
147
+ .chat_list {
148
+ margin: 0;
149
+ padding: 0;
150
+ display: flex;
151
+ flex-direction: column;
152
+ align-items: center;
153
+ font-size: 24px;
154
+
155
+ li {
156
+ width: 40px;
157
+ height: 40px;
158
+ line-height: 40px;
159
+ text-align: center;
160
+ cursor: pointer;
161
+ display: flex;
162
+ align-items: center;
163
+ justify-content: center;
164
+
165
+ &.active_chat_bar {
166
+ background-color: #ffffff14;
167
+ border-left: 2px solid #1677ff73;
168
+ margin-left: -2px;
169
+ }
170
+ }
171
+ }
172
+ }
173
+ }
174
+ }
@@ -0,0 +1,152 @@
1
+ import { Injectable, Autowired } from '@opensumi/di';
2
+ import { PreferenceService } from '@opensumi/ide-core-browser';
3
+ import { Emitter, Event } from '@opensumi/ide-core-common';
4
+ import { WorkbenchEditorService } from '@opensumi/ide-editor';
5
+ import { WorkbenchEditorServiceImpl } from '@opensumi/ide-editor/lib/browser/workbench-editor.service';
6
+
7
+ import { AISerivceType, AiGPTBackSerivcePath } from '../common';
8
+
9
+ const aiSearchKey = '/search ';
10
+ const aiSearchCodeKey = '/searchCode ';
11
+ const aiSumiKey = '/sumi ';
12
+ const aiExplainKey = '/explain ';
13
+ const aiRunKey = '/run ';
14
+
15
+ export interface IChatMessageStructure {
16
+ /**
17
+ * 用于 chat 面板展示
18
+ */
19
+ message: string | React.ReactNode;
20
+ /**
21
+ * 实际调用的 prompt
22
+ */
23
+ prompt?: string;
24
+ }
25
+
26
+ @Injectable()
27
+ export class AiChatService {
28
+ @Autowired(AiGPTBackSerivcePath)
29
+ aiBackService: any;
30
+
31
+ @Autowired(PreferenceService)
32
+ protected preferenceService: PreferenceService;
33
+
34
+ @Autowired(WorkbenchEditorService)
35
+ private readonly editorService: WorkbenchEditorServiceImpl;
36
+
37
+ private readonly _onChatMessageLaunch = new Emitter<IChatMessageStructure>();
38
+ public readonly onChatMessageLaunch: Event<IChatMessageStructure> = this._onChatMessageLaunch.event;
39
+
40
+ private get currentEditor() {
41
+ return this.editorService.currentEditor;
42
+ }
43
+
44
+ public launchChatMessage(data: IChatMessageStructure) {
45
+ this._onChatMessageLaunch.fire(data);
46
+ }
47
+
48
+ public async switchAIService(input: string, prompt = '') {
49
+ let type: AISerivceType | undefined;
50
+ let message: string | undefined;
51
+
52
+ if (!this.currentEditor) {
53
+ return;
54
+ }
55
+
56
+ const currentUri = this.currentEditor.currentUri;
57
+ if (!currentUri) {
58
+ return;
59
+ }
60
+
61
+ if (input.startsWith(aiSumiKey)) {
62
+ type = AISerivceType.Sumi;
63
+ message = input.split(aiSumiKey)[1];
64
+
65
+ return { type: AISerivceType.Sumi, message };
66
+ }
67
+
68
+ if (input.startsWith(aiExplainKey)) {
69
+ message = input.split(aiExplainKey)[1];
70
+
71
+ if (!prompt) {
72
+ prompt = this.explainCodePrompt(message);
73
+ }
74
+
75
+ return { type: AISerivceType.Explain, message: prompt };
76
+ }
77
+
78
+ if (input.startsWith(aiRunKey)) {
79
+ return { type: AISerivceType.Run, message: prompt };
80
+ }
81
+
82
+ if (input.startsWith(aiSearchKey)) {
83
+ type = AISerivceType.Search;
84
+ message = input.split(aiSearchKey)[1];
85
+ } else if (input.startsWith(aiSearchCodeKey)) {
86
+ type = AISerivceType.SearchCode;
87
+ message = input.split(aiSearchCodeKey)[1];
88
+ } else {
89
+ type = AISerivceType.GPT;
90
+ message = input;
91
+ }
92
+
93
+ return { type, message };
94
+ }
95
+
96
+ // 解释当前文件的代码或者选中的某个代码片段的 prompt,也可以用于对选中的代码加上用户的描述进行解释
97
+ public explainCodePrompt(message = ''): string {
98
+ if (!this.currentEditor) {
99
+ return '';
100
+ }
101
+
102
+ const currentUri = this.currentEditor.currentUri;
103
+ if (!currentUri) {
104
+ return '';
105
+ }
106
+
107
+ const displayName = currentUri.displayName;
108
+ const fsPath = currentUri.codeUri.fsPath;
109
+ const content = this.currentEditor.monacoEditor.getValue();
110
+ const selectionContent =
111
+ this.currentEditor.monacoEditor.getModel()?.getValueInRange(this.currentEditor.monacoEditor.getSelection()!) ||
112
+ '';
113
+ let messageWithPrompt = '';
114
+
115
+ /**
116
+ * 分三种情况
117
+ * 1. 没打开任意一个文件,则提供当前文件目录树给出当前项目的解释。如果用户有 prompt,则在最后带上
118
+ * 2. 没选中任意代码,则解释当前打开的代码文件(文件路径、代码)
119
+ * 3. 选中任意代码,则带上当前文件信息(文件路径、代码)和选中片段
120
+ * 4. 打开当前文件,用户如果有 prompt,则在最后带上。此时如果有选中代码片段,则带上,没有则带上文件代码
121
+ */
122
+ if (!this.currentEditor || !this.currentEditor.currentUri) {
123
+ //
124
+ }
125
+
126
+ if (!selectionContent) {
127
+ messageWithPrompt = `这是 ${displayName} 文件, 位置是在 ${fsPath}, 代码内容是 \`\`\`\n${content}\n\`\`\`。向我解释这个代码内容的意思`;
128
+ }
129
+
130
+ if (selectionContent) {
131
+ messageWithPrompt = `这是 ${displayName} 文件, 位置是在 ${fsPath}, 代码内容是 \`\`\`\n${content}\n\`\`\`。我会给你一段代码片段,你需要给我解释这段代码片段的意思。我的代码片段是: \`\`\`\n${selectionContent}\n\`\`\` `;
132
+ }
133
+
134
+ if (message.trim()) {
135
+ if (selectionContent) {
136
+ messageWithPrompt = `这是 ${displayName} 文件,代码内容是 \`\`\`\n${content}\n\`\`\`。我会提供给你其中的某个代码片段以及我的问题, 你需要根据我给的代码片段来解释我的问题。我提供的代码片段是: \`\`\`\n${selectionContent}\n\`\`\`,我的问题是: "${message}" `;
137
+ } else {
138
+ messageWithPrompt = `这是 ${displayName} 文件,代码内容是 \`\`\`\n${content}\n\`\`\`。根据我提供的代码内容来回答我的问题,我的问题是: "${message}" `;
139
+ }
140
+ }
141
+ return messageWithPrompt;
142
+ }
143
+
144
+ public async messageWithGPT(input: string) {
145
+ const res = await this.aiBackService.aiGPTcompletionRequest(input);
146
+ if (res.errorCode !== 0) {
147
+ return res.errorMsg || '';
148
+ } else {
149
+ return res.data || '';
150
+ }
151
+ }
152
+ }