From 8d0e0407b032b3dfca2179279f377c7faf9664a2 Mon Sep 17 00:00:00 2001 From: Coen Warmer Date: Mon, 8 Jan 2024 16:59:07 +0100 Subject: [PATCH] Tweaks to prompt editor (#174030) ## Summary This fixes the following edge case when using the chat interface: * When the main editor has a value in the text area, and the user edits an existing message, and presses `` on the keyboard, the value that was in the main editor is appended as a new message, instead of editing the existing message. It also does a bit of cleanup (moving of ChatPromptEditor components to `/components/prompt_editor`, and renaming to `PromptEditor`.) ### Additional fixes * [Don't stick to bottom when changing to edit mode, re-stick to bottom when done editing](https://github.com/elastic/kibana/pull/174030/commits/e8a01c1d4ddd449fdf85f0fef11de3d7cc9b2637) * [Autofocus function popover list search box upon opening](https://github.com/elastic/kibana/pull/174030/commits/2329d1c0a791716bf192b5e567c49336853edbd4) * [Remove focus trap as it wasn't doing anything anymore](https://github.com/elastic/kibana/pull/174030/commits/7fcb4e0b775a768c07b79c9c362f030c8f6036cb) * [Move constants used when creating monaco model inside function scope to avoid sharing of model between multiple editor instances](https://github.com/elastic/kibana/pull/174030/commits/c9cab2c15566a01f21342dcef66964c13574939d) * [Disable submitting function editor when json is not valid](https://github.com/elastic/kibana/pull/174030/commits/2a6f1e1cfb2ee5f917ac4862b68aff02813341be) (cherry picked from commit f4265ca731be471481518a24794a3664a46774ff) # Conflicts: # x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_function.tsx --- .../public/components/chat/chat_body.tsx | 9 +- .../public/components/chat/chat_item.tsx | 1 + ...chat_item_content_inline_prompt_editor.tsx | 9 +- .../public/components/chat/chat_timeline.tsx | 2 +- .../components/chat/function_list_popover.tsx | 94 +++++------- .../prompt_editor.stories.tsx} | 12 +- .../prompt_editor.tsx} | 143 ++++++++++-------- .../prompt_editor_function.tsx} | 46 ++++-- .../prompt_editor_natural_language.tsx} | 14 +- .../public/hooks/use_json_editor_model.ts | 11 +- 10 files changed, 186 insertions(+), 155 deletions(-) rename x-pack/plugins/observability_ai_assistant/public/components/{chat/chat_prompt_editor.stories.tsx => prompt_editor/prompt_editor.stories.tsx} (66%) rename x-pack/plugins/observability_ai_assistant/public/components/{chat/chat_prompt_editor.tsx => prompt_editor/prompt_editor.tsx} (51%) rename x-pack/plugins/observability_ai_assistant/public/components/{chat/chat_prompt_editor_function.tsx => prompt_editor/prompt_editor_function.tsx} (81%) rename x-pack/plugins/observability_ai_assistant/public/components/{chat/chat_prompt_editor_prompt.tsx => prompt_editor/prompt_editor_natural_language.tsx} (88%) diff --git a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_body.tsx b/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_body.tsx index d8eda3fb27418..d0aa6303005c8 100644 --- a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_body.tsx +++ b/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_body.tsx @@ -27,7 +27,7 @@ import type { UseGenAIConnectorsResult } from '../../hooks/use_genai_connectors' import type { UseKnowledgeBaseResult } from '../../hooks/use_knowledge_base'; import { type Conversation, type Message, MessageRole } from '../../../common/types'; import { ChatHeader } from './chat_header'; -import { ChatPromptEditor } from './chat_prompt_editor'; +import { PromptEditor } from '../prompt_editor/prompt_editor'; import { ChatTimeline } from './chat_timeline'; import { Feedback } from '../feedback_buttons'; import { IncorrectLicensePanel } from './incorrect_license_panel'; @@ -210,7 +210,7 @@ export function ChatBody({ - + ); } diff --git a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_function.tsx b/x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_function.tsx similarity index 81% rename from x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_function.tsx rename to x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_function.tsx index 33f467be26cb2..0a06037dd204d 100644 --- a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_function.tsx +++ b/x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_function.tsx @@ -4,12 +4,13 @@ * 2.0; you may not use this file except in compliance with the Elastic License * 2.0. */ -import React, { useCallback, useEffect, useState } from 'react'; -import { CodeEditor } from '@kbn/kibana-react-plugin/public'; +import React, { useCallback, useEffect, useRef, useState } from 'react'; import { i18n } from '@kbn/i18n'; import usePrevious from 'react-use/lib/usePrevious'; -import { EuiCode, EuiPanel } from '@elastic/eui'; import { css } from '@emotion/css'; +import { CodeEditor } from '@kbn/code-editor'; +import { monaco } from '@kbn/monaco'; +import { EuiCode, EuiPanel } from '@elastic/eui'; import { useJsonEditorModel } from '../../hooks/use_json_editor_model'; import { type Message, MessageRole } from '../../../common'; @@ -17,13 +18,22 @@ export interface Props { functionName: string; functionPayload?: string; onChange: (message: Message['message']) => void; + onFocus: () => void; + onBlur: () => void; } const functionNameClassName = css` display: inline-block; `; -export function ChatPromptEditorFunction({ functionName, functionPayload, onChange }: Props) { +export function PromptEditorFunction({ + functionName, + functionPayload, + onChange, + onFocus, + onBlur, +}: Props) { + const editorRef = useRef(null); const [functionEditorLineCount, setFunctionEditorLineCount] = useState(0); const previousPayload = usePrevious(functionPayload); @@ -33,8 +43,8 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan initialJson: functionPayload, }); - const handleChangeFunctionPayload = (params: string) => { - recalculateFunctionEditorLineCount(); + const handleChangePayload = (args: string) => { + recalculateLineCount(); onChange({ role: MessageRole.Assistant, @@ -42,12 +52,12 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan function_call: { name: functionName, trigger: MessageRole.User, - arguments: params, + arguments: args, }, }); }; - const recalculateFunctionEditorLineCount = useCallback(() => { + const recalculateLineCount = useCallback(() => { const newLineCount = model?.getLineCount() || 0; if (newLineCount !== functionEditorLineCount) { setFunctionEditorLineCount(newLineCount + 1); @@ -55,8 +65,8 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan }, [functionEditorLineCount, model]); useEffect(() => { - recalculateFunctionEditorLineCount(); - }, [model, recalculateFunctionEditorLineCount]); + recalculateLineCount(); + }, [model, recalculateLineCount]); useEffect(() => { if (previousPayload === undefined && initialJsonString) { @@ -72,6 +82,10 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan } }, [functionName, functionPayload, initialJsonString, onChange, previousPayload]); + editorRef.current?.onDidBlurEditorWidget(() => { + onBlur(); + }); + return ( {functionName} @@ -81,10 +95,15 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan { defaultMessage: 'payloadEditor' } )} data-test-subj="observabilityAiAssistantChatPromptEditorCodeEditor" + editorDidMount={(editor) => { + editorRef.current = editor; + editor.focus(); + onFocus(); + }} fullWidth height={'180px'} - languageId="json" isCopyable + languageId="json" languageConfiguration={{ autoClosingPairs: [ { @@ -93,9 +112,6 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan }, ], }} - editorDidMount={(editor) => { - editor.focus(); - }} options={{ accessibilitySupport: 'off', acceptSuggestionOnEnter: 'on', @@ -121,7 +137,7 @@ export function ChatPromptEditorFunction({ functionName, functionPayload, onChan }} transparentBackground value={functionPayload || ''} - onChange={handleChangeFunctionPayload} + onChange={handleChangePayload} /> ); diff --git a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_prompt.tsx b/x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_natural_language.tsx similarity index 88% rename from x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_prompt.tsx rename to x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_natural_language.tsx index 57c083d9704fd..e7bd833aee356 100644 --- a/x-pack/plugins/observability_ai_assistant/public/components/chat/chat_prompt_editor_prompt.tsx +++ b/x-pack/plugins/observability_ai_assistant/public/components/prompt_editor/prompt_editor_natural_language.tsx @@ -14,9 +14,18 @@ interface Props { prompt: string | undefined; onChange: (message: Message['message']) => void; onChangeHeight: (height: number) => void; + onFocus: () => void; + onBlur: () => void; } -export function ChatPromptEditorPrompt({ disabled, prompt, onChange, onChangeHeight }: Props) { +export function PromptEditorNaturalLanguage({ + disabled, + prompt, + onChange, + onChangeHeight, + onFocus, + onBlur, +}: Props) { const textAreaRef = useRef(null); const handleChange = (event: React.ChangeEvent) => { @@ -44,6 +53,7 @@ export function ChatPromptEditorPrompt({ disabled, prompt, onChange, onChangeHei if (textarea) { textarea.focus(); + textarea.select(); } }, [handleResizeTextArea]); @@ -65,6 +75,8 @@ export function ChatPromptEditorPrompt({ disabled, prompt, onChange, onChangeHei rows={1} value={prompt || ''} onChange={handleChange} + onFocus={onFocus} + onBlur={onBlur} /> ); } diff --git a/x-pack/plugins/observability_ai_assistant/public/hooks/use_json_editor_model.ts b/x-pack/plugins/observability_ai_assistant/public/hooks/use_json_editor_model.ts index 230d0065e29c3..6f4535d84acef 100644 --- a/x-pack/plugins/observability_ai_assistant/public/hooks/use_json_editor_model.ts +++ b/x-pack/plugins/observability_ai_assistant/public/hooks/use_json_editor_model.ts @@ -12,9 +12,6 @@ import { safeJsonParse } from '../utils/safe_json_parse'; const { editor, languages, Uri } = monaco; -const SCHEMA_URI = 'http://elastic.co/foo.json'; -const modelUri = Uri.parse(SCHEMA_URI); - export const useJsonEditorModel = ({ functionName, initialJson, @@ -28,13 +25,17 @@ export const useJsonEditorModel = ({ const [initialJsonValue, setInitialJsonValue] = useState(initialJson); + const SCHEMA_URI = `http://elastic.co/${functionName}.json`; + + const modelUri = useMemo(() => Uri.parse(SCHEMA_URI), [SCHEMA_URI]); + useEffect(() => { setInitialJsonValue(initialJson); // eslint-disable-next-line react-hooks/exhaustive-deps }, [functionName]); return useMemo(() => { - if (!functionDefinition) { + if (!functionDefinition || !modelUri) { return {}; } @@ -66,5 +67,5 @@ export const useJsonEditorModel = ({ } return { model, initialJsonString }; - }, [functionDefinition, initialJsonValue]); + }, [SCHEMA_URI, functionDefinition, initialJsonValue, modelUri]); };