* Make thread screen as default screen * Blank state when user have not any model * Cleanup topbar thread screen * Improve style right panel * Add instructions right panel * Styling thread list history * Resolve conflict * Default title new thread * Fix trigger panel sidebar * Make default right panel false when no activethread * Fix CI test * chore: assistant instruction with system prompt * Fix title and blank state explore the hub * Claenup style thread screen and add buble message for assitant * Remove unused import * Styling more menus on thread list and right panel, and make max height textarea 400 pixel * Finished revamp ui thread * Finished system monitor UI * Style box running models * Make animate right panel more smooth * Add status arround textarea for starting model info * Temporary disable hide left panel * chore: system resource monitoring update * copy nits * chore: typo * Reverse icon chevron accordion * Move my models into setting page --------- Co-authored-by: Louis <louis@jan.ai> Co-authored-by: 0xSage <n@pragmatic.vc>
111 lines
3.6 KiB
TypeScript
111 lines
3.6 KiB
TypeScript
import {
|
|
ChatCompletionMessage,
|
|
EventName,
|
|
MessageRequest,
|
|
MessageStatus,
|
|
ExtensionType,
|
|
ThreadMessage,
|
|
events,
|
|
} from '@janhq/core'
|
|
import { ConversationalExtension, InferenceExtension } from '@janhq/core'
|
|
import { useAtomValue, useSetAtom } from 'jotai'
|
|
import { RefreshCcw, Copy, Trash2Icon, StopCircle } from 'lucide-react'
|
|
|
|
import { twMerge } from 'tailwind-merge'
|
|
|
|
import { toaster } from '@/containers/Toast'
|
|
|
|
import { extensionManager } from '@/extension'
|
|
import {
|
|
deleteMessageAtom,
|
|
getCurrentChatMessagesAtom,
|
|
} from '@/helpers/atoms/ChatMessage.atom'
|
|
import { activeThreadAtom } from '@/helpers/atoms/Conversation.atom'
|
|
|
|
const MessageToolbar = ({ message }: { message: ThreadMessage }) => {
|
|
const deleteMessage = useSetAtom(deleteMessageAtom)
|
|
const thread = useAtomValue(activeThreadAtom)
|
|
const messages = useAtomValue(getCurrentChatMessagesAtom)
|
|
// const threadStateAtom = useMemo(
|
|
// () => atom((get) => get(threadStatesAtom)[thread?.id ?? '']),
|
|
// [thread?.id]
|
|
// )
|
|
// const threadState = useAtomValue(threadStateAtom)
|
|
const stopInference = async () => {
|
|
await extensionManager
|
|
.get<InferenceExtension>(ExtensionType.Inference)
|
|
?.stopInference()
|
|
setTimeout(() => {
|
|
events.emit(EventName.OnMessageUpdate, {
|
|
...message,
|
|
status: MessageStatus.Ready,
|
|
})
|
|
}, 300)
|
|
}
|
|
|
|
return (
|
|
<div className={twMerge('flex flex-row items-center')}>
|
|
<div className="flex overflow-hidden rounded-md border border-border bg-background/20">
|
|
{message.status === MessageStatus.Pending && (
|
|
<div
|
|
className="cursor-pointer border-r border-border px-2 py-2 hover:bg-background/80"
|
|
onClick={() => stopInference()}
|
|
>
|
|
<StopCircle size={14} />
|
|
</div>
|
|
)}
|
|
{message.status !== MessageStatus.Pending &&
|
|
message.id === messages[messages.length - 1]?.id && (
|
|
<div
|
|
className="cursor-pointer border-r border-border px-2 py-2 hover:bg-background/80"
|
|
onClick={() => {
|
|
const messageRequest: MessageRequest = {
|
|
id: message.id ?? '',
|
|
messages: messages.slice(0, -1).map((e) => {
|
|
const msg: ChatCompletionMessage = {
|
|
role: e.role,
|
|
content: e.content[0].text.value,
|
|
}
|
|
return msg
|
|
}),
|
|
threadId: message.thread_id ?? '',
|
|
}
|
|
events.emit(EventName.OnMessageSent, messageRequest)
|
|
}}
|
|
>
|
|
<RefreshCcw size={14} />
|
|
</div>
|
|
)}
|
|
<div
|
|
className="cursor-pointer border-r border-border px-2 py-2 hover:bg-background/80"
|
|
onClick={() => {
|
|
navigator.clipboard.writeText(message.content[0]?.text?.value ?? '')
|
|
toaster({
|
|
title: 'Copied to clipboard',
|
|
})
|
|
}}
|
|
>
|
|
<Copy size={14} />
|
|
</div>
|
|
<div
|
|
className="cursor-pointer px-2 py-2 hover:bg-background/80"
|
|
onClick={async () => {
|
|
deleteMessage(message.id ?? '')
|
|
if (thread)
|
|
await extensionManager
|
|
.get<ConversationalExtension>(ExtensionType.Conversational)
|
|
?.writeMessages(
|
|
thread.id,
|
|
messages.filter((msg) => msg.id !== message.id)
|
|
)
|
|
}}
|
|
>
|
|
<Trash2Icon size={14} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default MessageToolbar
|