* chore: initial new FE setup * chore: update namespace text-left-panel foreground variable * chore: enable dynamic mainview color * chore: remove greetings new chat * chore: fix chat input style * chore: simplify hook useAppearance * chore: enable internationalization * chore: prepare vn locale * chore: keyboardshortcut layout * chore: update keyboard shortcut exclude pathname * chore: update state active setting route * chore: fix update theme by system * chore: handle dynamic primary color * chore: fix left panel navigation active state and styled item privacy analytic * chore: reorder general setting being a first * chore: add function reset appearance * chore: update scrollbar * chore: update delete thread with dialog confirmation * chore: update state dialog inside dropdown menu * chore: wip thread detail or chat page * chore: wip model dropdown * chore: prepare model dropdown select * chore: update model providers setting * chore: show provider on model dropdown based isActive toogle * chore: update layout model provider * chore: update state active on storage * chore: update gap of item dropdown model * chore: update select model base on id * chore: update edit model capabilities * chore: add dialog to add model * chore: update sheet for model setting * chore: add sheet setting each model * chore: make dynamic syntax highlight * chore: fix menu setting appearance theme * chore: markdown render support emoji * chore: markdown support latex * chore: change codeblock default theme * chore: update ui codeblock * chore: custom render link taget new window * chore: fix copy button codeblock * chore: update accent and desctructive color * chore: setup user chat message * chore: prepare some page settings * chore: simple list extension and prepare mcp, local api, and hardware * chore: mcp-serve * chore: MCP server UI * chore: update local api server config * chore: adjust chat input * chore: update local api server log * chore: prepare hub page * chore: remove help page * chore: update mock * chore: prepare http proxy setting UI * chore: adjust local api server and title every action * fix: chore FE package (#4962) * fix: update command which referred to non-existent web app * fix: added commented out macos platform for now * fix: remove the platform name as macos * fix: remove unnecessary line for platform name in HeaderPage component * fix: update dev script to specify port 3000 for Vite * feat: model providers and chat completion * enhancement: threads performance * fix: thread content update * chore: clean up threads * fix: performance issue with streaming and state loop * fix: streaming * fix: react markdow * feat: extension manager * chore: add nodePolyfills include path * chore: improve performance avoid unhandle rejection * chore: update pre margin bottom * chore: swith thread should be deafult scroll to bottom * chore: wip scroll to bottom * chore: add model loader * chore: add platform utils * feat: threads functionality * chore: setup toaster * chore: persist threads deletion * fix: create thread with new message * chore: create new thread should change route path * chore: navigate after delet dialog thread * chore: thread favorites and orders * chore: dismiss deleting modal on delete * chore: remove undefined properties * chore: remove deprecated run step * chore: fix delete thread * chore: create empty thread content on started streaming * chore: correct messages store key * chore: stuck at generating state * chore: preapre chat toolbar * chore: introduce in-memory app state * chore: update extensions migration logic * chore: remove redundant extensions migration gate * chore: message toolbar user and assistant * chore: add logo gemini * feat: remote providers with model capabilities * chore: maintain provider settings * chore: move speed token into chat input * chore: temp harcoded model loader * chore: make chat text selectable and truncate model list * chore: update shortcut UI * Feat/implement threads (#4977) * chore: add fuse.js library for enhanced search functionality * feat: implement thread filtering with Fuse.js for improved search capabilities * fix: update the fuseOptions * feat: add search functionality to LeftPanel and refactor thread retrieval logic * refactor: optimize thread filtering and improve search functionality in LeftPanel * fix: more edits * refactor: remove duplicate import of useAppState in StreamingContent component * chore: update navigate after delete all thread * chore: pass prop speedToken from new chat input * chore: persist provider general settings * chore: styling search left panel * chore: cleanup margin * chore: update size icon * chore: improve chat input * chore: imprve list markdown * chore: animate border * feat: local model provider work * chore: persist manually added model * chore: prepare download management ui and show version on general setting * chore: improve pre tag * chore: remove buton install extension and improve light theme download * chore: add missing hardware information handler * chore: cleanup small ui * chore: update default provider settings * fix: missing fs commands * chore: correct provider models * chore: prepare delete model * chore: handle thinking block * chore: fix conditional message toolbar * chore: pophover download select none * enhancement: add prune mode * chore: model settings * chore: bump engine version tauri * chore: update style thinking * chore: add indicator and toogle mcp server * chore: wip hub * chore: update model settings * chore: mvp hub * chore: add function rename title * chore: update function delete message * chore: update rename title * chore: update model settings * chore: persist MCP configs * refactor: clean up utils * chore: add tools to completion request * chore: clean up * chore: ignore assets --------- Co-authored-by: Ivan Leo <ivanleomk@gmail.com> Co-authored-by: Louis <louis@jan.ai>
176 lines
5.3 KiB
TypeScript
176 lines
5.3 KiB
TypeScript
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
import { useContext, useEffect, useRef, useState } from 'react'
|
|
|
|
import {
|
|
ConversationalExtension,
|
|
ExtensionTypeEnum,
|
|
MessageStatus,
|
|
ThreadMessage,
|
|
} from '@janhq/core'
|
|
|
|
import { TextArea, Button, Modal, ModalClose } from '@janhq/joi'
|
|
import { useAtom, useAtomValue, useSetAtom } from 'jotai'
|
|
|
|
import { twMerge } from 'tailwind-merge'
|
|
|
|
import { editPromptAtom } from '@/containers/Providers/Jotai'
|
|
|
|
import { useActiveModel } from '@/hooks/useActiveModel'
|
|
|
|
import useSendChatMessage from '@/hooks/useSendChatMessage'
|
|
|
|
import { ChatContext } from '../../ThreadCenterPanel'
|
|
|
|
import { ChatContext } from '../../ThreadCenterPanel'
|
|
|
|
import { extensionManager } from '@/extension'
|
|
|
|
import {
|
|
editMessageAtom,
|
|
getCurrentChatMessagesAtom,
|
|
setConvoMessagesAtom,
|
|
} from '@/helpers/atoms/ChatMessage.atom'
|
|
import { spellCheckAtom } from '@/helpers/atoms/Setting.atom'
|
|
import {
|
|
activeThreadAtom,
|
|
getActiveThreadIdAtom,
|
|
} from '@/helpers/atoms/Thread.atom'
|
|
|
|
type Props = {
|
|
message: ThreadMessage
|
|
}
|
|
|
|
const EditChatInput: React.FC<Props> = ({ message }) => {
|
|
const activeThread = useAtomValue(activeThreadAtom)
|
|
const { stateModel, stopInference } = useActiveModel()
|
|
const messages = useAtomValue(getCurrentChatMessagesAtom)
|
|
|
|
const [editPrompt, setEditPrompt] = useAtom(editPromptAtom)
|
|
const { showApprovalModal } = useContext(ChatContext)
|
|
const { sendChatMessage } = useSendChatMessage(showApprovalModal)
|
|
const setMessages = useSetAtom(setConvoMessagesAtom)
|
|
const activeThreadId = useAtomValue(getActiveThreadIdAtom)
|
|
const spellCheck = useAtomValue(spellCheckAtom)
|
|
const textareaRef = useRef<HTMLTextAreaElement>(null)
|
|
const setEditMessage = useSetAtom(editMessageAtom)
|
|
const [showDialog, setshowDialog] = useState(false)
|
|
|
|
const onPromptChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
|
|
setEditPrompt(e.target.value)
|
|
}
|
|
|
|
useEffect(() => {
|
|
if (textareaRef.current) {
|
|
textareaRef.current.focus()
|
|
const length = textareaRef.current.value.length
|
|
textareaRef.current.setSelectionRange(length, length)
|
|
}
|
|
}, [activeThreadId])
|
|
|
|
useEffect(() => {
|
|
if (textareaRef.current) {
|
|
textareaRef.current.style.height = '40px'
|
|
textareaRef.current.style.height = textareaRef.current.scrollHeight + 'px'
|
|
textareaRef.current.style.overflow =
|
|
textareaRef.current.clientHeight >= 390 ? 'auto' : 'hidden'
|
|
}
|
|
}, [editPrompt])
|
|
|
|
useEffect(() => {
|
|
if (message.content?.[0]?.text?.value)
|
|
setEditPrompt(message.content[0].text.value)
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [])
|
|
|
|
const sendEditMessage = async () => {
|
|
setEditMessage('')
|
|
const messageIdx = messages.findIndex((msg) => msg.id === message.id)
|
|
const newMessages = messages.slice(0, messageIdx)
|
|
const toDeleteMessages = messages.slice(messageIdx)
|
|
const threadId = messages[0].thread_id
|
|
await Promise.all(
|
|
toDeleteMessages.map(async (message) =>
|
|
extensionManager
|
|
.get<ConversationalExtension>(ExtensionTypeEnum.Conversational)
|
|
?.deleteMessage(message.thread_id, message.id)
|
|
)
|
|
).catch(console.error)
|
|
setMessages(threadId, newMessages)
|
|
sendChatMessage(editPrompt, false, newMessages)
|
|
}
|
|
|
|
const onKeyDown = async (e: React.KeyboardEvent<HTMLTextAreaElement>) => {
|
|
if (e.key === 'Enter' && !e.shiftKey) {
|
|
e.preventDefault()
|
|
if (messages[messages.length - 1]?.status !== MessageStatus.Pending)
|
|
sendEditMessage()
|
|
else onStopInferenceClick()
|
|
}
|
|
}
|
|
|
|
const onStopInferenceClick = async () => {
|
|
stopInference()
|
|
}
|
|
|
|
return (
|
|
<div className="mx-auto flex w-full flex-shrink-0 flex-col items-start justify-center space-y-4 pb-0 pt-1">
|
|
<div className="relative flex w-full flex-col">
|
|
<TextArea
|
|
className={twMerge('max-h-[400px] resize-none pr-20')}
|
|
style={{ height: '40px' }}
|
|
ref={textareaRef}
|
|
spellCheck={spellCheck}
|
|
onKeyDown={onKeyDown}
|
|
placeholder="Enter your message..."
|
|
disabled={stateModel.loading || !activeThread}
|
|
value={editPrompt}
|
|
onChange={onPromptChange}
|
|
/>
|
|
</div>
|
|
|
|
<div className="flex items-center space-x-2">
|
|
<Button
|
|
disabled={
|
|
stateModel.loading ||
|
|
!activeThread ||
|
|
editPrompt.trim().length === 0
|
|
}
|
|
onClick={sendEditMessage}
|
|
>
|
|
Submit
|
|
</Button>
|
|
<Button
|
|
theme="ghost"
|
|
variant="outline"
|
|
onClick={() => setEditMessage('')}
|
|
>
|
|
Cancel
|
|
</Button>
|
|
</div>
|
|
|
|
<Modal
|
|
open={showDialog}
|
|
onOpenChange={() => setshowDialog(false)}
|
|
title="Edit Message"
|
|
content={
|
|
<div>
|
|
<p className="text-[hsla(var(--text-secondary)]">
|
|
Do you want to discard the change
|
|
</p>
|
|
<div className="mt-4 flex justify-end gap-x-2">
|
|
<ModalClose asChild onClick={() => setshowDialog(false)}>
|
|
<Button theme="ghost">Cancel</Button>
|
|
</ModalClose>
|
|
<ModalClose asChild onClick={() => setEditMessage('')}>
|
|
<Button autoFocus>Yes</Button>
|
|
</ModalClose>
|
|
</div>
|
|
</div>
|
|
}
|
|
/>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default EditChatInput
|