* feat: desktop revamp * feat: refactor system monitor * fix linter CI * remove unused import component * added responsive and resizeable component * responsive and resizeable local server page * finalize responsive and resizeable component * fix scroll custom ui * remove react scroll to bottom from modal troubleshoot * fix modal troubleshoot ui * fix setting gpu list * text area custom scroll bar * fix padding message input * cleanup classname * update inference engine model dropdown * update loader style * update quick ask ui * prepare theme provider * update dark theme * remove update hotkey list model and navigation * fix: cleanup hardcode classname * fix: update feedback * Set native theme electron * update destop ui revamp from feedback * update button icon component insider icon chat input message * update model dropdown ui * update tranaparent baclground * update logo model provider * fix: set background material acrylic support to blur background windows * fix: update tranparent left and right panel * fix: linter CI * update app using frameless window * styling custom style minimize, maximize and close app * temporary hidden maximize window * fix: responsive left and right panel * fix: enable click outside when leftpanel responsive * fix: remove unused import * update transparent variable css windows * fix: ui import model * feat: Support Theme system (#2946) * feat: update support theme system * update select component * feat: add theme folder in root project * fix: padding left and right center panel * fix: update padding left and right * chore: migrate themes * fix: rmdirsync error * chore: update gitignore * fix: cp recursive * fix: files electron package json * fix: migration * fix: update fgit ignore --------- Co-authored-by: Louis <louis@jan.ai> * fix: update feedback missing state when refrash app * fix: error test CI * chore: refactor useLoadThemes * chore: cleanup unused vars * fix: revert back menubar windows * fix minor ui * fix: minor ui --------- Co-authored-by: Louis <louis@jan.ai>
167 lines
4.8 KiB
TypeScript
167 lines
4.8 KiB
TypeScript
import { useCallback } from 'react'
|
|
|
|
import {
|
|
Assistant,
|
|
ConversationalExtension,
|
|
ExtensionTypeEnum,
|
|
Thread,
|
|
ThreadAssistantInfo,
|
|
ThreadState,
|
|
Model,
|
|
AssistantTool,
|
|
} from '@janhq/core'
|
|
import { atom, useAtomValue, useSetAtom } from 'jotai'
|
|
|
|
import { fileUploadAtom } from '@/containers/Providers/Jotai'
|
|
|
|
import { generateThreadId } from '@/utils/thread'
|
|
|
|
import { useActiveModel } from './useActiveModel'
|
|
import useRecommendedModel from './useRecommendedModel'
|
|
|
|
import useSetActiveThread from './useSetActiveThread'
|
|
|
|
import { extensionManager } from '@/extension'
|
|
|
|
import { experimentalFeatureEnabledAtom } from '@/helpers/atoms/AppConfig.atom'
|
|
import { selectedModelAtom } from '@/helpers/atoms/Model.atom'
|
|
import {
|
|
threadsAtom,
|
|
threadStatesAtom,
|
|
updateThreadAtom,
|
|
setThreadModelParamsAtom,
|
|
isGeneratingResponseAtom,
|
|
} from '@/helpers/atoms/Thread.atom'
|
|
|
|
const createNewThreadAtom = atom(null, (get, set, newThread: Thread) => {
|
|
// create thread state for this new thread
|
|
const currentState = { ...get(threadStatesAtom) }
|
|
|
|
const threadState: ThreadState = {
|
|
hasMore: false,
|
|
waitingForResponse: false,
|
|
lastMessage: undefined,
|
|
}
|
|
currentState[newThread.id] = threadState
|
|
set(threadStatesAtom, currentState)
|
|
|
|
// add the new thread on top of the thread list to the state
|
|
const threads = get(threadsAtom)
|
|
set(threadsAtom, [newThread, ...threads])
|
|
})
|
|
|
|
export const useCreateNewThread = () => {
|
|
const createNewThread = useSetAtom(createNewThreadAtom)
|
|
const { setActiveThread } = useSetActiveThread()
|
|
const updateThread = useSetAtom(updateThreadAtom)
|
|
const setFileUpload = useSetAtom(fileUploadAtom)
|
|
const setSelectedModel = useSetAtom(selectedModelAtom)
|
|
const setThreadModelParams = useSetAtom(setThreadModelParamsAtom)
|
|
|
|
const experimentalEnabled = useAtomValue(experimentalFeatureEnabledAtom)
|
|
const setIsGeneratingResponse = useSetAtom(isGeneratingResponseAtom)
|
|
|
|
const { recommendedModel, downloadedModels } = useRecommendedModel()
|
|
|
|
const threads = useAtomValue(threadsAtom)
|
|
const { stopInference } = useActiveModel()
|
|
|
|
const requestCreateNewThread = async (
|
|
assistant: Assistant,
|
|
model?: Model | undefined
|
|
) => {
|
|
// Stop generating if any
|
|
setIsGeneratingResponse(false)
|
|
stopInference()
|
|
|
|
const defaultModel = model ?? recommendedModel ?? downloadedModels[0]
|
|
|
|
if (!model) {
|
|
// if we have model, which means user wants to create new thread from Model hub. Allow them.
|
|
|
|
// check last thread message, if there empty last message use can not create thread
|
|
const lastMessage = threads[0]?.metadata?.lastMessage
|
|
|
|
if (!lastMessage && threads.length) {
|
|
return null
|
|
}
|
|
}
|
|
|
|
// modify assistant tools when experimental on, retieval toggle enabled in default
|
|
const assistantTools: AssistantTool = {
|
|
type: 'retrieval',
|
|
enabled: true,
|
|
settings: assistant.tools && assistant.tools[0].settings,
|
|
}
|
|
|
|
const overriddenSettings =
|
|
defaultModel?.settings.ctx_len && defaultModel.settings.ctx_len > 2048
|
|
? { ctx_len: 2048 }
|
|
: {}
|
|
|
|
const overriddenParameters =
|
|
defaultModel?.parameters.max_tokens && defaultModel.parameters.max_tokens
|
|
? { max_tokens: 2048 }
|
|
: {}
|
|
|
|
const createdAt = Date.now()
|
|
const assistantInfo: ThreadAssistantInfo = {
|
|
assistant_id: assistant.id,
|
|
assistant_name: assistant.name,
|
|
tools: experimentalEnabled ? [assistantTools] : assistant.tools,
|
|
model: {
|
|
id: defaultModel?.id ?? '*',
|
|
settings: { ...defaultModel?.settings, ...overriddenSettings } ?? {},
|
|
parameters:
|
|
{ ...defaultModel?.parameters, ...overriddenParameters } ?? {},
|
|
engine: defaultModel?.engine,
|
|
},
|
|
instructions: assistant.instructions,
|
|
}
|
|
|
|
const threadId = generateThreadId(assistant.id)
|
|
const thread: Thread = {
|
|
id: threadId,
|
|
object: 'thread',
|
|
title: 'New Thread',
|
|
assistants: [assistantInfo],
|
|
created: createdAt,
|
|
updated: createdAt,
|
|
}
|
|
|
|
// add the new thread on top of the thread list to the state
|
|
//TODO: Why do we have thread list then thread states? Should combine them
|
|
createNewThread(thread)
|
|
|
|
setSelectedModel(defaultModel)
|
|
setThreadModelParams(thread.id, {
|
|
...defaultModel?.settings,
|
|
...defaultModel?.parameters,
|
|
...overriddenSettings,
|
|
})
|
|
|
|
// Delete the file upload state
|
|
setFileUpload([])
|
|
// Update thread metadata
|
|
await updateThreadMetadata(thread)
|
|
|
|
setActiveThread(thread)
|
|
}
|
|
|
|
const updateThreadMetadata = useCallback(
|
|
async (thread: Thread) => {
|
|
updateThread(thread)
|
|
|
|
await extensionManager
|
|
.get<ConversationalExtension>(ExtensionTypeEnum.Conversational)
|
|
?.saveThread(thread)
|
|
},
|
|
[updateThread]
|
|
)
|
|
|
|
return {
|
|
requestCreateNewThread,
|
|
updateThreadMetadata,
|
|
}
|
|
}
|