Files
vscode/src/vs/workbench/contrib/inlineChat/browser/inlineChatCurrentLine.ts
Benjamin Pasero 6b924c5152 ESM merge to main (#227184)
Co-authored-by: Johannes Rieken <jrieken@microsoft.com>
Co-authored-by: Alexandru Dima <alexdima@microsoft.com>
2024-08-30 10:31:46 +02:00

167 lines
5.8 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import { DisposableStore, MutableDisposable } from '../../../../base/common/lifecycle.js';
import { IActiveCodeEditor, ICodeEditor } from '../../../../editor/browser/editorBrowser.js';
import { CursorChangeReason } from '../../../../editor/common/cursorEvents.js';
import { IEditorContribution } from '../../../../editor/common/editorCommon.js';
import { localize, localize2 } from '../../../../nls.js';
import { ContextKeyExpr, IContextKey, IContextKeyService, RawContextKey } from '../../../../platform/contextkey/common/contextkey.js';
import { IChatAgentService } from '../../chat/common/chatAgents.js';
import { InlineChatController, State } from './inlineChatController.js';
import { CTX_INLINE_CHAT_HAS_AGENT, CTX_INLINE_CHAT_VISIBLE } from '../common/inlineChat.js';
import { EditorAction2, ServicesAccessor } from '../../../../editor/browser/editorExtensions.js';
import { EditOperation } from '../../../../editor/common/core/editOperation.js';
import { Range } from '../../../../editor/common/core/range.js';
import { Position } from '../../../../editor/common/core/position.js';
import { AbstractInlineChatAction } from './inlineChatActions.js';
import { EditorContextKeys } from '../../../../editor/common/editorContextKeys.js';
import { IValidEditOperation } from '../../../../editor/common/model.js';
export const CTX_INLINE_CHAT_EXPANSION = new RawContextKey<boolean>('inlineChatExpansion', false, localize('inlineChatExpansion', "Whether the inline chat expansion is enabled when at the end of a just-typed line"));
export class InlineChatExansionContextKey implements IEditorContribution {
static Id = 'editor.inlineChatExpansion';
private readonly _store = new DisposableStore();
private readonly _editorListener = this._store.add(new MutableDisposable());
private readonly _ctxInlineChatExpansion: IContextKey<boolean>;
constructor(
editor: ICodeEditor,
@IContextKeyService contextKeyService: IContextKeyService,
@IChatAgentService chatAgentService: IChatAgentService
) {
this._ctxInlineChatExpansion = CTX_INLINE_CHAT_EXPANSION.bindTo(contextKeyService);
const update = () => {
if (editor.hasModel() && chatAgentService.getAgents().length > 0) {
this._install(editor);
} else {
this._uninstall();
}
};
this._store.add(chatAgentService.onDidChangeAgents(update));
this._store.add(editor.onDidChangeModel(update));
update();
}
dispose(): void {
this._ctxInlineChatExpansion.reset();
this._store.dispose();
}
private _install(editor: IActiveCodeEditor): void {
const store = new DisposableStore();
this._editorListener.value = store;
const model = editor.getModel();
const lastChangeEnds: number[] = [];
store.add(editor.onDidChangeCursorPosition(e => {
let enabled = false;
if (e.reason === CursorChangeReason.NotSet) {
const position = editor.getPosition();
const positionOffset = model.getOffsetAt(position);
const lineLength = model.getLineLength(position.lineNumber);
const firstNonWhitespace = model.getLineFirstNonWhitespaceColumn(position.lineNumber);
if (firstNonWhitespace !== 0 && position.column > lineLength && lastChangeEnds.includes(positionOffset)) {
enabled = true;
}
}
lastChangeEnds.length = 0;
this._ctxInlineChatExpansion.set(enabled);
}));
store.add(editor.onDidChangeModelContent(e => {
lastChangeEnds.length = 0;
for (const change of e.changes) {
const changeEnd = change.rangeOffset + change.text.length;
lastChangeEnds.push(changeEnd);
}
queueMicrotask(() => {
if (lastChangeEnds.length > 0) {
// this is a signal that onDidChangeCursorPosition didn't run which means some outside change
// which means we should disable the context key
this._ctxInlineChatExpansion.set(false);
}
});
}));
}
private _uninstall(): void {
this._editorListener.clear();
}
}
export class InlineChatExpandLineAction extends EditorAction2 {
constructor() {
super({
id: 'inlineChat.startWithCurrentLine',
category: AbstractInlineChatAction.category,
title: localize2('startWithCurrentLine', "Start in Editor with Current Line"),
f1: true,
precondition: ContextKeyExpr.and(CTX_INLINE_CHAT_VISIBLE.negate(), CTX_INLINE_CHAT_HAS_AGENT, EditorContextKeys.writable),
// keybinding: {
// when: CTX_INLINE_CHAT_EXPANSION,
// weight: KeybindingWeight.EditorContrib,
// primary: KeyCode.Tab
// }
});
}
override async runEditorCommand(_accessor: ServicesAccessor, editor: ICodeEditor) {
const ctrl = InlineChatController.get(editor);
if (!ctrl || !editor.hasModel()) {
return;
}
const model = editor.getModel();
const lineNumber = editor.getSelection().positionLineNumber;
const lineContent = model.getLineContent(lineNumber);
const startColumn = model.getLineFirstNonWhitespaceColumn(lineNumber);
const endColumn = model.getLineMaxColumn(lineNumber);
// clear the line
let undoEdits: IValidEditOperation[] = [];
model.pushEditOperations(null, [EditOperation.replace(new Range(lineNumber, startColumn, lineNumber, endColumn), '')], (edits) => {
undoEdits = edits;
return null;
});
let lastState: State | undefined;
const d = ctrl.onDidEnterState(e => lastState = e);
try {
// trigger chat
await ctrl.run({
autoSend: true,
message: lineContent.trim(),
position: new Position(lineNumber, startColumn)
});
} finally {
d.dispose();
}
if (lastState === State.CANCEL) {
model.pushEditOperations(null, undoEdits, () => null);
}
}
}