From 237e913e53f0a77abe2ffb2dc069d15a9bdf658f Mon Sep 17 00:00:00 2001 From: Dogus Atasoy Date: Thu, 23 May 2024 18:37:15 +0200 Subject: [PATCH 1/3] updated rendered card maps to add normal cards to the list too --- example/src/main.ts | 12 +- example/src/samples/sample-data.ts | 739 ++++++++++----------- package-lock.json | 4 +- package.json | 2 +- src/components/chat-item/chat-item-card.ts | 13 +- src/components/chat-item/chat-wrapper.ts | 56 +- src/helper/chat-item.ts | 8 + 7 files changed, 400 insertions(+), 434 deletions(-) create mode 100644 src/helper/chat-item.ts diff --git a/example/src/main.ts b/example/src/main.ts index 23e1be90..67293c30 100644 --- a/example/src/main.ts +++ b/example/src/main.ts @@ -192,19 +192,19 @@ export const createMynahUI = (initialData?: MynahUIDataModel): MynahUI => { Log(`File clicked: ${filePath}`); }, onFileActionClick: (tabId, messageId, filePath, actionName) => { + console.log("YARAK!"); Log(`File action clicked: ${filePath} -> ${actionName}`); switch (actionName) { - case 'update-comment': - case 'comment-to-change': - showCustomForm(tabId); - break; case 'reject-change': - mynahUI.updateChatAnswerWithMessageId(tabId, 'file-list-message', exampleFileListChatItemForUpdate); + mynahUI.updateChatAnswerWithMessageId(tabId, messageId, exampleFileListChatItemForUpdate); + break; + case 'revert-rejection': + mynahUI.updateChatAnswerWithMessageId(tabId, messageId, {fileList: exampleFileListChatItem.fileList}); break; default: break; } - mynahUI.updateChatAnswerWithMessageId(tabId, 'file-list-message', exampleFileListChatItemForUpdate); + // mynahUI.updateChatAnswerWithMessageId(tabId, messageId, exampleFileListChatItemForUpdate); }, onCustomFormAction: (tabId, action) => { Log(`Custom form action clicked for tab ${tabId}:
diff --git a/example/src/samples/sample-data.ts b/example/src/samples/sample-data.ts index 18aeeb43..2bab82ca 100644 --- a/example/src/samples/sample-data.ts +++ b/example/src/samples/sample-data.ts @@ -19,29 +19,29 @@ import SampleCode from './sample-code.md'; import { Commands } from '../commands'; export const mynahUIQRImageBase64 = - ''; + ''; export const mynahUIQRMarkdown = `![Mynah UI](${mynahUIQRImageBase64})`; // react stateless function component example export const exampleSources = [ - { - url: 'https://github.com/aws/mynah-ui', - title: 'MynahUI', - body: '#### A Data & Event Drivent Chat Interface Library for Browsers and Webviews', - }, - { - url: 'https://github.com/aws/mynah-ui/blob/main/docs/STARTUP.md', - title: 'MynahUI initial setup', - body: `Simply install it from npm with your favorite package manager. + { + url: 'https://github.com/aws/mynah-ui', + title: 'MynahUI', + body: '#### A Data & Event Drivent Chat Interface Library for Browsers and Webviews', + }, + { + url: 'https://github.com/aws/mynah-ui/blob/main/docs/STARTUP.md', + title: 'MynahUI initial setup', + body: `Simply install it from npm with your favorite package manager. \`\`\` npm install @aws/mynah-ui \`\`\` `, - }, - { - url: 'https://github.com/aws/mynah-ui/blob/main/docs/USAGE.md', - title: 'How to use MynahUI', - body: `To see how to configure statics for MynahUI please refer to **[Configuration](./CONFIG.md)** document. + }, + { + url: 'https://github.com/aws/mynah-ui/blob/main/docs/USAGE.md', + title: 'How to use MynahUI', + body: `To see how to configure statics for MynahUI please refer to **[Configuration](./CONFIG.md)** document. Lastly before you start reading here, you can find more details on the **[Data Model](./DATAMODEL.md)** document. That document also contains visuals related with each type of the chat message in detail. @@ -56,344 +56,297 @@ export const exampleSources = [ mynahUI.updateStore(...); \`\`\` `, - }, + }, ] as SourceLink[]; export const sampleMarkdownList: Partial[] = [ - { body: `${sampleList0 as string}`}, - { body: `${sampleList1 as string}`}, - { body: `${sampleList2 as string}`}, - { body: `${sampleList3 as string}`}, - { body: `${sampleList4 as string}`}, + { body: `${sampleList0 as string}` }, + { body: `${sampleList1 as string}` }, + { body: `${sampleList2 as string}` }, + { body: `${sampleList3 as string}` }, + { body: `${sampleList4 as string}` }, ]; export const exampleStreamParts: Partial[] = [ - { body: `${md0 as string}` }, - { body: `${md1 as string}` }, - { body: `${md2 as string}` }, - { body: `${md3 as string}` }, - { body: `${md4 as string}` }, - { body: `${md5 as string}` }, - { body: `${md6 as string}` }, - { body: `${md7 as string}` }, - { body: `${md8 as string}` }, - { body: `${md9 as string}` }, - { body: `${md10 as string}` }, - { - relatedContent: { - content: exampleSources, - title: 'Sources', - }, - codeReference: [ - { - recommendationContentSpan: { - start: 952, - end: 967, - }, - information: 'Say Hello to **`MynahUI`**.', - }, - { - recommendationContentSpan: { - start: 1034, - end: 1409, + { body: `${md0 as string}` }, + { body: `${md1 as string}` }, + { body: `${md2 as string}` }, + { body: `${md3 as string}` }, + { body: `${md4 as string}` }, + { body: `${md5 as string}` }, + { body: `${md6 as string}` }, + { body: `${md7 as string}` }, + { body: `${md8 as string}` }, + { body: `${md9 as string}` }, + { body: `${md10 as string}` }, + { + relatedContent: { + content: exampleSources, + title: 'Sources', }, - information: 'Reference code *under the Apache License 2.0 license* from repository **`@aws/mynah-ui`**.', - }, - ], - }, + codeReference: [ + { + recommendationContentSpan: { + start: 952, + end: 967, + }, + information: 'Say Hello to **`MynahUI`**.', + }, + { + recommendationContentSpan: { + start: 1034, + end: 1409, + }, + information: 'Reference code *under the Apache License 2.0 license* from repository **`@aws/mynah-ui`**.', + }, + ], + }, ]; export const exampleCodeBlockToInsert = SampleCode; export const exampleRichFollowups: ChatItem = { - type: ChatItemType.SYSTEM_PROMPT, - messageId: new Date().getTime().toString(), - followUp: { - text: 'Rich followups', - options: [ - { - pillText: 'Accept', - icon: MynahIcons.OK, - description: 'You can accept by clicking this.', - status: 'success', - }, - { - pillText: 'Reject', - icon: MynahIcons.CANCEL, - status: 'error', - }, - { - pillText: 'Retry', - icon: MynahIcons.REFRESH, - status: 'warning', - }, - { - pillText: 'Do nothing', - icon: MynahIcons.BLOCK, - status: 'info', - }, - ], - }, + type: ChatItemType.SYSTEM_PROMPT, + messageId: new Date().getTime().toString(), + followUp: { + text: 'Rich followups', + options: [ + { + pillText: 'Accept', + icon: MynahIcons.OK, + description: 'You can accept by clicking this.', + status: 'success', + }, + { + pillText: 'Reject', + icon: MynahIcons.CANCEL, + status: 'error', + }, + { + pillText: 'Retry', + icon: MynahIcons.REFRESH, + status: 'warning', + }, + { + pillText: 'Do nothing', + icon: MynahIcons.BLOCK, + status: 'info', + }, + ], + }, }; export const defaultFollowUps: ChatItem = { - type: ChatItemType.ANSWER, - messageId: new Date().getTime().toString(), - followUp: { - text: 'Example card types', - options: [ - { - command: Commands.STATUS_CARDS, - pillText: 'Cards with status', - }, - { - command: Commands.FORM_CARD, - pillText: 'Form items', - }, - { - command: Commands.CARD_WITH_MARKDOWN_LIST, - pillText: 'Markdown list', - }, - { - command: Commands.CARD_SNAPS_TO_TOP, - pillText: 'Snaps to top', - }, - { - command: Commands.FILE_LIST_CARD, - pillText: 'File list', - }, - { - command: Commands.PROGRESSIVE_CARD, - pillText: 'Progressive', - }, - { - command: Commands.IMAGE_IN_CARD, - pillText: 'Image inside', - }, - { - command: Commands.CUSTOM_RENDERER_CARDS, - pillText: 'Custom renderers', - }, - { - pillText: 'Followups on right', - command: Commands.FOLLOWUPS_AT_RIGHT, - }, - { - pillText: 'Some auto reply', - prompt: 'Some random auto reply here.', - }, - ], - }, + type: ChatItemType.ANSWER, + messageId: new Date().getTime().toString(), + followUp: { + text: 'Example card types', + options: [ + { + command: Commands.STATUS_CARDS, + pillText: 'Cards with status', + }, + { + command: Commands.FORM_CARD, + pillText: 'Form items', + }, + { + command: Commands.CARD_WITH_MARKDOWN_LIST, + pillText: 'Markdown list', + }, + { + command: Commands.CARD_SNAPS_TO_TOP, + pillText: 'Snaps to top', + }, + { + command: Commands.FILE_LIST_CARD, + pillText: 'File list', + }, + { + command: Commands.PROGRESSIVE_CARD, + pillText: 'Progressive', + }, + { + command: Commands.IMAGE_IN_CARD, + pillText: 'Image inside', + }, + { + command: Commands.CUSTOM_RENDERER_CARDS, + pillText: 'Custom renderers', + }, + { + pillText: 'Followups on right', + command: Commands.FOLLOWUPS_AT_RIGHT, + }, + { + pillText: 'Some auto reply', + prompt: 'Some random auto reply here.', + }, + ], + }, }; export const exampleFileListChatItem: ChatItem = { - type: ChatItemType.CODE_RESULT, - body: '#### Here are the changed files:', - buttons: [ - { - id: 'open-diff-viewer', - text: 'Open Diff Viewer', - icon: MynahIcons.EXTERNAL, - status: 'info', - disabled: false, - }, - ], - fileList: { - filePaths: ['dummy.ts', 'src/App.tsx', 'devfile.yaml', 'src/App.test.tsx'], - deletedFiles: ['src/devfile.yaml'], - actions: { - 'src/App.tsx': [ - { - icon: MynahIcons.CANCEL_CIRCLE, - status: 'info', - name: 'reject-change', - description: 'Reject change', - }, - { - icon: MynahIcons.COMMENT, - name: 'comment-to-change', - description: 'Comment', - }, - ], - 'devfile.yaml': [ - { - icon: MynahIcons.CANCEL_CIRCLE, - status: 'info', - name: 'reject-change', - description: 'Reject change', + type: ChatItemType.ANSWER, + body: '', + messageId: `FILE_LIST_${new Date().getTime().toString()}`, + fileList: { + rootFolderTitle: 'Changes', + filePaths: ['src/index.ts'], + deletedFiles: [], + actions: { + 'src/index.ts': [ + { + icon: MynahIcons.CANCEL_CIRCLE, + name: 'reject-change', + description: 'Reject Change', + }, + ], }, - ], - }, - details: { - 'src/devfile.yaml': { - status: 'error', - label: 'Change rejected', - icon: MynahIcons.REVERT, - }, - }, - }, - codeReference: [ - { - information: 'Reference code *under the MIT license* from repository `amazon`.', - }, - { - information: 'Reference code *under the MIT license* from repository `aws`.', }, - ], - canBeVoted: true, - messageId: 'file-list-message', }; export const exampleFileListChatItemForUpdate: Partial = { - fileList: { - filePaths: ['src/App.tsx', 'src/App.test.tsx'], - details: { - 'src/App.tsx': { - status: 'error', - label: 'File rejected', - icon: MynahIcons.CANCEL_CIRCLE, - }, - 'src/App.test.tsx': { - status: 'warning', - label: 'Comment added', - icon: MynahIcons.COMMENT, - }, - }, - actions: { - 'src/App.tsx': [ - { - icon: MynahIcons.REVERT, - name: 'revert-rejection', - description: 'Revert rejection', + type: ChatItemType.ANSWER, + fileList: { + rootFolderTitle: 'Changes', + filePaths: ['src/index.ts'], + deletedFiles: [], + details: { + 'src/index.ts': { + status: 'error', + label: 'File rejected', + icon: MynahIcons.CANCEL_CIRCLE, + }, }, - ], - 'src/App.test.tsx': [ - { - icon: MynahIcons.PENCIL, - name: 'update-comment', - description: 'Update comment', + actions: { + 'src/index.ts': [ + { + icon: MynahIcons.REVERT, + name: 'revert-rejection', + description: 'Revert rejection', + }, + ], }, - ], }, - }, }; export const exampleFormChatItem: ChatItem = { - type: ChatItemType.ANSWER, - messageId: new Date().getTime().toString(), - body: `Can you help us to improve our AI Assistant? Please fill the form below and hit **Submit** to send your feedback. + type: ChatItemType.ANSWER, + messageId: new Date().getTime().toString(), + body: `Can you help us to improve our AI Assistant? Please fill the form below and hit **Submit** to send your feedback. _To send the form, mandatory items should be filled._`, - formItems: [ - { - id: 'expertise-area', - type: 'select', - title: `Area of expertise`, - options: [ + formItems: [ + { + id: 'expertise-area', + type: 'select', + title: `Area of expertise`, + options: [ + { + label: 'Frontend', + value: 'frontend', + }, + { + label: 'Backend', + value: 'backend', + }, + { + label: 'Data Science', + value: 'datascience', + }, + { + label: 'Other', + value: 'other', + }, + ], + }, { - label: 'Frontend', - value: 'frontend', + id: 'preferred-ide', + type: 'radiogroup', + title: `Preferred IDE`, + options: [ + { + label: 'VSCode', + value: 'vscode', + }, + { + label: 'JetBrains IntelliJ', + value: 'intellij', + }, + { + label: 'Visual Studio', + value: 'visualstudio', + }, + ], }, { - label: 'Backend', - value: 'backend', + id: 'working-hours', + type: 'numericinput', + title: `How many hours are you using an IDE weekly?`, + placeholder: 'IDE working hours', }, { - label: 'Data Science', - value: 'datascience', + id: 'email', + type: 'textinput', + mandatory: true, + title: `Email`, + placeholder: 'email', }, { - label: 'Other', - value: 'other', + id: 'name', + type: 'textinput', + mandatory: true, + title: `Name`, + placeholder: 'Name and Surname', }, - ], - }, - { - id: 'preferred-ide', - type: 'radiogroup', - title: `Preferred IDE`, - options: [ { - label: 'VSCode', - value: 'vscode', + id: 'ease-of-usage-rating', + type: 'stars', + mandatory: true, + title: `How easy is it to use our AI assistant?`, }, { - label: 'JetBrains IntelliJ', - value: 'intellij', + id: 'accuracy-rating', + type: 'stars', + mandatory: true, + title: `How accurate are the answers you get from our AI assistant?`, }, { - label: 'Visual Studio', - value: 'visualstudio', + id: 'general-rating', + type: 'stars', + title: `How do feel about our AI assistant in general?`, }, - ], - }, - { - id: 'working-hours', - type: 'numericinput', - title: `How many hours are you using an IDE weekly?`, - placeholder: 'IDE working hours', - }, - { - id: 'email', - type: 'textinput', - mandatory: true, - title: `Email`, - placeholder: 'email', - }, - { - id: 'name', - type: 'textinput', - mandatory: true, - title: `Name`, - placeholder: 'Name and Surname', - }, - { - id: 'ease-of-usage-rating', - type: 'stars', - mandatory: true, - title: `How easy is it to use our AI assistant?`, - }, - { - id: 'accuracy-rating', - type: 'stars', - mandatory: true, - title: `How accurate are the answers you get from our AI assistant?`, - }, - { - id: 'general-rating', - type: 'stars', - title: `How do feel about our AI assistant in general?`, - }, - { - id: 'description', - type: 'textarea', - title: `Any other things you would like to share?`, - placeholder: 'Write your feelings about our tool', - }, - ], - buttons: [ - { - id: 'submit', - text: 'Submit', - status: 'info', - }, - { - id: 'cancel-feedback', - text: 'Cancel', - keepCardAfterClick: false, - waitMandatoryFormItems: false, - }, - ], + { + id: 'description', + type: 'textarea', + title: `Any other things you would like to share?`, + placeholder: 'Write your feelings about our tool', + }, + ], + buttons: [ + { + id: 'submit', + text: 'Submit', + status: 'info', + }, + { + id: 'cancel-feedback', + text: 'Cancel', + keepCardAfterClick: false, + waitMandatoryFormItems: false, + }, + ], }; const checkIcons = { - wait: '☐', - current: '☐', - done: '☑', + wait: '☐', + current: '☐', + done: '☑', }; export const exampleProgressCards: Partial[] = [ - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.wait} Reading your files in the project, @@ -408,9 +361,9 @@ ${checkIcons.wait} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.current} Reading your files in the project, @@ -425,9 +378,9 @@ ${checkIcons.wait} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -442,9 +395,9 @@ ${checkIcons.wait} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -459,9 +412,9 @@ ${checkIcons.wait} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -476,9 +429,9 @@ ${checkIcons.wait} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -493,9 +446,9 @@ ${checkIcons.current} Creating a refactor plan ${checkIcons.wait} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -510,9 +463,9 @@ ${checkIcons.done} Creating a refactor plan ${checkIcons.current} Showing the plan details Once it is done, you'll be notified.`, - }, - { - body: `Hi there, we're currently working on your task. You can follow the steps below: + }, + { + body: `Hi there, we're currently working on your task. You can follow the steps below: ${checkIcons.done} Reading your files in the project, @@ -531,32 +484,32 @@ Your Refactor analysis is ready! You can review it by opening the Markdown file: You can also ask me any follow-up questions that you have or adjust any part by generating a revised analysis. `, - fileList: { - fileTreeTitle: '', - filePaths: ['Refactor_analysis_[id].pdf'], + fileList: { + fileTreeTitle: '', + filePaths: ['Refactor_analysis_[id].pdf'], + }, }, - }, ]; export const exampleImageCard = (): ChatItem => { - return { - messageId: new Date().getTime().toString(), - type: ChatItemType.ANSWER, - body: ` + return { + messageId: new Date().getTime().toString(), + type: ChatItemType.ANSWER, + body: ` ### Image! Here's a QR code for mynah-ui github link: ${mynahUIQRMarkdown} `, - }; + }; }; export const exampleCustomRendererWithHTMLMarkup = (): ChatItem => { - return { - messageId: new Date().getTime().toString(), - type: ChatItemType.ANSWER, - canBeVoted: true, - customRenderer: ` + return { + messageId: new Date().getTime().toString(), + type: ChatItemType.ANSWER, + canBeVoted: true, + customRenderer: `

Custom renderer's with HTML markup string

Here you will find some custom html rendering examples which may not be available with markdown or pretty hard to generate. @@ -640,7 +593,7 @@ You can find more information and references documentation for details and limitations.

`, - }; + }; }; const attachmentIcon = ` @@ -648,68 +601,68 @@ const attachmentIcon = ` `; export const exampleCustomRendererWithDomBuilderJson: ChatItem = { - messageId: new Date().getTime().toString(), - type: ChatItemType.ANSWER, - canBeVoted: true, - body: `Your Refactor analysis is ready! You can review it by opening the Markdown file: [file_name](#hello-pdf) + messageId: new Date().getTime().toString(), + type: ChatItemType.ANSWER, + canBeVoted: true, + body: `Your Refactor analysis is ready! You can review it by opening the Markdown file: [file_name](#hello-pdf) You can also ask me any follow-up questions that you have or adjust any part by generating a revised analysis.`, - customRenderer: [ - { - type: 'blockquote', - events: { - click: (e: Event) => { - console.log('Hello!', e); - }, - }, - - children: [ + customRenderer: [ { - type: 'table', - children: [ - { - type: 'tr', - children: [ - { - type: 'td', - attributes: { - style: 'min-width: 30px; width: 30px;', - }, - children: [ - { - type: 'img', - attributes: { - src: `data:image/svg+xml;base64,${window.btoa(attachmentIcon)}`, - }, - }, - ], + type: 'blockquote', + events: { + click: (e: Event) => { + console.log('Hello!', e); }, + }, + + children: [ { - type: 'td', - children: [ - { - type: 'strong', - children: ['Refactor_analysis_[id] .pdf'], - }, - ], + type: 'table', + children: [ + { + type: 'tr', + children: [ + { + type: 'td', + attributes: { + style: 'min-width: 30px; width: 30px;', + }, + children: [ + { + type: 'img', + attributes: { + src: `data:image/svg+xml;base64,${window.btoa(attachmentIcon)}`, + }, + }, + ], + }, + { + type: 'td', + children: [ + { + type: 'strong', + children: ['Refactor_analysis_[id] .pdf'], + }, + ], + }, + ], + }, + ], }, - ], - }, - ], + ], }, - ], - }, - ], + ], }; export const exampleDownloadFile: ChatItem = { - messageId: new Date().getTime().toString(), - type: ChatItemType.ANSWER, - canBeVoted: true, - body: `Your Refactor analysis is ready! You can review it by opening the Markdown file: [file_name](#hello-pdf) + messageId: new Date().getTime().toString(), + type: ChatItemType.ANSWER, + canBeVoted: true, + body: `Your Refactor analysis is ready! You can review it by opening the Markdown file: [file_name](#hello-pdf) You can also ask me any follow-up questions that you have or adjust any part by generating a revised analysis.`, - fileList: { - fileTreeTitle: 'Report', - rootFolderTitle: '', - filePaths: ['Refactor_analysis_[id] .pdf'], - }, + fileList: { + fileTreeTitle: 'Report', + rootFolderTitle: '', + filePaths: ['Refactor_analysis_[id] .pdf'], + }, }; diff --git a/package-lock.json b/package-lock.json index 1f9ed289..8f1bc2bf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "@aws/mynah-ui", - "version": "4.9.0", + "version": "4.9.1", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@aws/mynah-ui", - "version": "4.9.0", + "version": "4.9.1", "hasInstallScript": true, "license": "Apache License 2.0", "dependencies": { diff --git a/package.json b/package.json index af8a69b2..c663e44f 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@aws/mynah-ui", "displayName": "AWS Mynah UI", - "version": "4.9.0", + "version": "4.9.1", "description": "AWS Toolkit VSCode and Intellij IDE Extension Mynah UI", "publisher": "Amazon Web Services", "license": "Apache License 2.0", diff --git a/src/components/chat-item/chat-item-card.ts b/src/components/chat-item/chat-item-card.ts index 06e419ff..75dcf35c 100644 --- a/src/components/chat-item/chat-item-card.ts +++ b/src/components/chat-item/chat-item-card.ts @@ -20,6 +20,7 @@ import { ChatItemFormItemsWrapper } from './chat-item-form-items'; import { ChatItemButtonsWrapper } from './chat-item-buttons'; import { cleanHtml } from '../../helper/sanitize'; import { CONTAINER_GAP } from './chat-wrapper'; +import { chatItemHasContent } from '../../helper/chat-item'; const TYPEWRITER_STACK_TIME = 500; export interface ChatItemCardProps { @@ -98,15 +99,9 @@ export class ChatItemCard { return generatedCard; }; - private readonly cardHasContent = (): boolean => ((this.props.chatItem.body != null && this.props.chatItem.body !== '') || - this.props.chatItem.fileList != null || - this.props.chatItem.formItems != null || - this.props.chatItem.customRenderer != null || - this.props.chatItem.buttons != null); - private readonly getCardClasses = (): string[] => { const isNoContent = - !this.cardHasContent() && + !chatItemHasContent(this.props.chatItem) && this.props.chatItem.followUp == null && this.props.chatItem.relatedContent == null && this.props.chatItem.type === ChatItemType.ANSWER; @@ -115,7 +110,7 @@ export class ChatItemCard { `mynah-chat-item-card-status-${this.props.chatItem.status ?? 'default'}`, 'mynah-chat-item-card', `mynah-chat-item-${this.props.chatItem.type ?? ChatItemType.ANSWER}`, - ...(!this.cardHasContent() ? [ 'mynah-chat-item-empty' ] : []), + ...(!chatItemHasContent(this.props.chatItem) ? [ 'mynah-chat-item-empty' ] : []), ...(isNoContent ? [ 'mynah-chat-item-no-content' ] : []), ]; }; @@ -293,7 +288,7 @@ export class ChatItemCard { return [ ...(MynahUITabsStore.getInstance().getTabDataStore(this.props.tabId).getValue('showChatAvatars') === true ? [ this.chatAvatar ] : []), - ...(this.cardHasContent() + ...(chatItemHasContent(this.props.chatItem) ? [ new Card({ onCardEngaged: engagement => { diff --git a/src/components/chat-item/chat-wrapper.ts b/src/components/chat-item/chat-wrapper.ts index 00cd99da..03ee54b0 100644 --- a/src/components/chat-item/chat-wrapper.ts +++ b/src/components/chat-item/chat-wrapper.ts @@ -3,6 +3,7 @@ * SPDX-License-Identifier: Apache-2.0 */ +import { chatItemHasContent } from '../../helper/chat-item'; import { Config } from '../../helper/config'; import { DomBuilder, ExtendedHTMLElement } from '../../helper/dom'; import { generateUID } from '../../helper/guid'; @@ -28,8 +29,8 @@ export class ChatWrapper { private readonly promptInput: ChatPromptInput; private readonly promptInfo: ExtendedHTMLElement; private readonly promptStickyCard: ExtendedHTMLElement; - private lastChatItemCard: ChatItemCard | null; - private lastChatItemMessageId: string | null; + private lastStreamingChatItemCard: ChatItemCard | null; + private lastStreamingChatItemMessageId: string | null; private allRenderedChatItems: Record = {}; render: ExtendedHTMLElement; constructor (props: ChatWrapperProps) { @@ -141,31 +142,40 @@ export class ChatWrapper { } private readonly insertChatItem = (chatItem: ChatItem): void => { - this.lastChatItemMessageId = (chatItem.messageId != null && chatItem.messageId !== '') ? chatItem.messageId : `TEMP_${generateUID()}`; + const currentMessageId: string = (chatItem.messageId != null && chatItem.messageId !== '') ? chatItem.messageId : `TEMP_${generateUID()}`; const chatItemCard = new ChatItemCard({ tabId: this.props.tabId, chatItem: { ...chatItem, - messageId: this.lastChatItemMessageId + messageId: currentMessageId } }); + if (chatItem.type === ChatItemType.ANSWER_STREAM) { - this.lastChatItemCard?.render.addClass('stream-ended'); - this.lastChatItemCard = chatItemCard; + // End previous streaming card if there is + this.lastStreamingChatItemCard?.render.addClass('stream-ended'); + + // Update the lastStreaming variables with the new one + this.lastStreamingChatItemMessageId = currentMessageId; + this.lastStreamingChatItemCard = chatItemCard; } else if ( - (chatItem.type === ChatItemType.ANSWER || - chatItem.type === ChatItemType.PROMPT || - chatItem.type === ChatItemType.AI_PROMPT || - chatItem.type === ChatItemType.SYSTEM_PROMPT) && chatItem.body !== undefined) { - this.lastChatItemCard?.render.addClass('stream-ended'); - this.lastChatItemCard = null; - this.lastChatItemMessageId = null; + chatItem.type !== ChatItemType.ANSWER && + chatItem.type !== ChatItemType.ANSWER_PART && + chatItemHasContent(chatItem)) { + // If the new card is not a streaming one and it has any kind of content, + // it means that the last card is not a streaming card anymore. + // So end the previous stream and reset the lastStreaming variables + this.lastStreamingChatItemCard?.render.addClass('stream-ended'); + this.lastStreamingChatItemCard = null; + this.lastStreamingChatItemMessageId = null; } + + // Add to render this.chatItemsContainer.insertChild('afterbegin', chatItemCard.render); - if (this.lastChatItemMessageId != null) { - this.allRenderedChatItems[this.lastChatItemMessageId] = chatItemCard; - } + // Add to all rendered chat items map + this.allRenderedChatItems[currentMessageId] = chatItemCard; + if (chatItem.type === ChatItemType.PROMPT || chatItem.type === ChatItemType.SYSTEM_PROMPT) { // Make sure we scroll the chat window to the bottom // Only if it is a PROMPT @@ -174,20 +184,20 @@ export class ChatWrapper { }; public updateLastChatAnswer = (updateWith: Partial): void => { - if (this.lastChatItemCard !== null) { - this.lastChatItemCard.updateCardStack(updateWith); + if (this.lastStreamingChatItemCard !== null) { + this.lastStreamingChatItemCard.updateCardStack(updateWith); if (updateWith.messageId != null && updateWith.messageId !== '') { - if (this.lastChatItemMessageId != null && this.lastChatItemMessageId !== updateWith.messageId) { - const renderChatItemInMap = this.allRenderedChatItems[this.lastChatItemMessageId]; + if (this.lastStreamingChatItemMessageId != null && this.lastStreamingChatItemMessageId !== updateWith.messageId) { + const renderChatItemInMap = this.allRenderedChatItems[this.lastStreamingChatItemMessageId]; if (renderChatItemInMap != null) { this.allRenderedChatItems[updateWith.messageId] = renderChatItemInMap; - if (this.lastChatItemMessageId != null) { + if (this.lastStreamingChatItemMessageId != null) { // eslint-disable-next-line @typescript-eslint/no-dynamic-delete - delete this.allRenderedChatItems[this.lastChatItemMessageId]; + delete this.allRenderedChatItems[this.lastStreamingChatItemMessageId]; } } } - this.lastChatItemMessageId = updateWith.messageId; + this.lastStreamingChatItemMessageId = updateWith.messageId; } } }; diff --git a/src/helper/chat-item.ts b/src/helper/chat-item.ts new file mode 100644 index 00000000..58eeeb8d --- /dev/null +++ b/src/helper/chat-item.ts @@ -0,0 +1,8 @@ +import { ChatItem } from '../static'; + +export const chatItemHasContent = (chatItem: Partial): boolean => ( + (chatItem.body != null && chatItem.body !== '') || +chatItem.fileList != null || +chatItem.formItems != null || +chatItem.customRenderer != null || +chatItem.buttons != null); From b5aced36882c159bf8c808416836f79457b62fae Mon Sep 17 00:00:00 2001 From: Dogus Atasoy Date: Thu, 23 May 2024 18:38:09 +0200 Subject: [PATCH 2/3] removed unnecessary comment --- example/src/main.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/example/src/main.ts b/example/src/main.ts index 67293c30..96fa8978 100644 --- a/example/src/main.ts +++ b/example/src/main.ts @@ -204,7 +204,6 @@ export const createMynahUI = (initialData?: MynahUIDataModel): MynahUI => { default: break; } - // mynahUI.updateChatAnswerWithMessageId(tabId, messageId, exampleFileListChatItemForUpdate); }, onCustomFormAction: (tabId, action) => { Log(`Custom form action clicked for tab ${tabId}:
From 02ad7fdf2f521a44c597e7d8e0e5eafce4080fdd Mon Sep 17 00:00:00 2001 From: Dogus Atasoy Date: Fri, 24 May 2024 11:39:58 +0200 Subject: [PATCH 3/3] removed unnecesaary comments --- example/src/main.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/example/src/main.ts b/example/src/main.ts index 96fa8978..836c527a 100644 --- a/example/src/main.ts +++ b/example/src/main.ts @@ -192,7 +192,6 @@ export const createMynahUI = (initialData?: MynahUIDataModel): MynahUI => { Log(`File clicked: ${filePath}`); }, onFileActionClick: (tabId, messageId, filePath, actionName) => { - console.log("YARAK!"); Log(`File action clicked: ${filePath} -> ${actionName}`); switch (actionName) { case 'reject-change':