add sidePanel for Chat page
This commit is contained in:
parent
66e43c9d9b
commit
a9819139b8
@ -269,5 +269,6 @@
|
|||||||
"Server is working on deployment mode, please close the terminal window manually": "サーバーはデプロイモードで動作しています、ターミナルウィンドウを手動で閉じてください",
|
"Server is working on deployment mode, please close the terminal window manually": "サーバーはデプロイモードで動作しています、ターミナルウィンドウを手動で閉じてください",
|
||||||
"Server is working on deployment mode, please exit the program manually to stop the server": "サーバーはデプロイモードで動作しています、サーバーを停止するにはプログラムを手動で終了してください",
|
"Server is working on deployment mode, please exit the program manually to stop the server": "サーバーはデプロイモードで動作しています、サーバーを停止するにはプログラムを手動で終了してください",
|
||||||
"You can increase the number of stored layers in Configs page to improve performance": "パフォーマンスを向上させるために、保存されるレイヤーの数を設定ページで増やすことができます",
|
"You can increase the number of stored layers in Configs page to improve performance": "パフォーマンスを向上させるために、保存されるレイヤーの数を設定ページで増やすことができます",
|
||||||
"Failed to load model, try to increase the virtual memory (Swap of WSL) or use a smaller base model.": "モデルの読み込みに失敗しました、仮想メモリ (WSL Swap) を増やすか小さなベースモデルを使用してみてください。"
|
"Failed to load model, try to increase the virtual memory (Swap of WSL) or use a smaller base model.": "モデルの読み込みに失敗しました、仮想メモリ (WSL Swap) を増やすか小さなベースモデルを使用してみてください。",
|
||||||
|
"Save Conversation": "会話を保存"
|
||||||
}
|
}
|
@ -269,5 +269,6 @@
|
|||||||
"Server is working on deployment mode, please close the terminal window manually": "服务器正在部署模式下运行,请手动关闭终端窗口",
|
"Server is working on deployment mode, please close the terminal window manually": "服务器正在部署模式下运行,请手动关闭终端窗口",
|
||||||
"Server is working on deployment mode, please exit the program manually to stop the server": "服务器正在部署模式下运行,请手动退出程序以停止服务器",
|
"Server is working on deployment mode, please exit the program manually to stop the server": "服务器正在部署模式下运行,请手动退出程序以停止服务器",
|
||||||
"You can increase the number of stored layers in Configs page to improve performance": "你可以在配置页面增加载入显存层数以提升性能",
|
"You can increase the number of stored layers in Configs page to improve performance": "你可以在配置页面增加载入显存层数以提升性能",
|
||||||
"Failed to load model, try to increase the virtual memory (Swap of WSL) or use a smaller base model.": "模型载入失败,尝试增加虚拟内存(WSL Swap),或使用一个更小规模的基底模型"
|
"Failed to load model, try to increase the virtual memory (Swap of WSL) or use a smaller base model.": "模型载入失败,尝试增加虚拟内存(WSL Swap),或使用一个更小规模的基底模型",
|
||||||
|
"Save Conversation": "保存对话"
|
||||||
}
|
}
|
@ -16,8 +16,11 @@ import {
|
|||||||
Attach16Regular,
|
Attach16Regular,
|
||||||
Delete28Regular,
|
Delete28Regular,
|
||||||
Dismiss16Regular,
|
Dismiss16Regular,
|
||||||
|
Dismiss24Regular,
|
||||||
RecordStop28Regular,
|
RecordStop28Regular,
|
||||||
Save28Regular
|
SaveRegular,
|
||||||
|
TextAlignJustify24Regular,
|
||||||
|
TextAlignJustifyRotate9024Regular
|
||||||
} from '@fluentui/react-icons';
|
} from '@fluentui/react-icons';
|
||||||
import { CopyButton } from '../components/CopyButton';
|
import { CopyButton } from '../components/CopyButton';
|
||||||
import { ReadButton } from '../components/ReadButton';
|
import { ReadButton } from '../components/ReadButton';
|
||||||
@ -26,9 +29,11 @@ import { WorkHeader } from '../components/WorkHeader';
|
|||||||
import { DialogButton } from '../components/DialogButton';
|
import { DialogButton } from '../components/DialogButton';
|
||||||
import { OpenFileFolder, OpenOpenFileDialog, OpenSaveFileDialog } from '../../wailsjs/go/backend_golang/App';
|
import { OpenFileFolder, OpenOpenFileDialog, OpenSaveFileDialog } from '../../wailsjs/go/backend_golang/App';
|
||||||
import { absPathAsset, bytesToReadable, getServerRoot, toastWithButton } from '../utils';
|
import { absPathAsset, bytesToReadable, getServerRoot, toastWithButton } from '../utils';
|
||||||
import { PresetsButton } from './PresetsManager/PresetsButton';
|
|
||||||
import { useMediaQuery } from 'usehooks-ts';
|
import { useMediaQuery } from 'usehooks-ts';
|
||||||
import { botName, ConversationMessage, MessageType, userName, welcomeUuid } from '../types/chat';
|
import { botName, ConversationMessage, MessageType, userName, welcomeUuid } from '../types/chat';
|
||||||
|
import { Labeled } from '../components/Labeled';
|
||||||
|
import { ValuedSlider } from '../components/ValuedSlider';
|
||||||
|
import { PresetsButton } from './PresetsManager/PresetsButton';
|
||||||
|
|
||||||
let chatSseControllers: {
|
let chatSseControllers: {
|
||||||
[id: string]: AbortController
|
[id: string]: AbortController
|
||||||
@ -188,11 +193,125 @@ const ChatMessageItem: FC<{
|
|||||||
</div>;
|
</div>;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const SidePanel: FC = observer(() => {
|
||||||
|
const [t] = useTranslation();
|
||||||
|
const mq = useMediaQuery('(min-width: 640px)');
|
||||||
|
const params = commonStore.chatParams;
|
||||||
|
|
||||||
|
return <div
|
||||||
|
className={classnames(
|
||||||
|
'flex flex-col gap-1 h-full flex-shrink-0 transition-width duration-300 ease-in-out',
|
||||||
|
commonStore.sidePanelCollapsed ? 'w-0' : (mq ? 'w-64' : 'w-full'),
|
||||||
|
!commonStore.sidePanelCollapsed && 'ml-1')
|
||||||
|
}>
|
||||||
|
<div className="flex m-1">
|
||||||
|
<div className="grow" />
|
||||||
|
<PresetsButton tab="Chat" size="medium" shape="circular" appearance="subtle" />
|
||||||
|
<Button size="medium" shape="circular" appearance="subtle" icon={<Dismiss24Regular />}
|
||||||
|
onClick={() => commonStore.setSidePanelCollapsed(true)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1 overflow-x-hidden overflow-y-auto p-1">
|
||||||
|
<Labeled flex breakline label={t('Max Response Token')}
|
||||||
|
desc={t('By default, the maximum number of tokens that can be answered in a single response, it can be changed by the user by specifying API parameters.')}
|
||||||
|
content={
|
||||||
|
<ValuedSlider value={params.maxResponseToken} min={100} max={8100}
|
||||||
|
step={100}
|
||||||
|
input
|
||||||
|
onChange={(e, data) => {
|
||||||
|
commonStore.setChatParams({
|
||||||
|
maxResponseToken: data.value
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
} />
|
||||||
|
<Labeled flex breakline label={t('Temperature')}
|
||||||
|
desc={t('Sampling temperature, it\'s like giving alcohol to a model, the higher the stronger the randomness and creativity, while the lower, the more focused and deterministic it will be.')}
|
||||||
|
content={
|
||||||
|
<ValuedSlider value={params.temperature} min={0} max={2} step={0.1}
|
||||||
|
input
|
||||||
|
onChange={(e, data) => {
|
||||||
|
commonStore.setChatParams({
|
||||||
|
temperature: data.value
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
} />
|
||||||
|
<Labeled flex breakline label={t('Top_P')}
|
||||||
|
desc={t('Just like feeding sedatives to the model. Consider the results of the top n% probability mass, 0.1 considers the top 10%, with higher quality but more conservative, 1 considers all results, with lower quality but more diverse.')}
|
||||||
|
content={
|
||||||
|
<ValuedSlider value={params.topP} min={0} max={1} step={0.1} input
|
||||||
|
onChange={(e, data) => {
|
||||||
|
commonStore.setChatParams({
|
||||||
|
topP: data.value
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
} />
|
||||||
|
<Labeled flex breakline label={t('Presence Penalty')}
|
||||||
|
desc={t('Positive values penalize new tokens based on whether they appear in the text so far, increasing the model\'s likelihood to talk about new topics.')}
|
||||||
|
content={
|
||||||
|
<ValuedSlider value={params.presencePenalty} min={0} max={2}
|
||||||
|
step={0.1} input
|
||||||
|
onChange={(e, data) => {
|
||||||
|
commonStore.setChatParams({
|
||||||
|
presencePenalty: data.value
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
} />
|
||||||
|
<Labeled flex breakline label={t('Frequency Penalty')}
|
||||||
|
desc={t('Positive values penalize new tokens based on their existing frequency in the text so far, decreasing the model\'s likelihood to repeat the same line verbatim.')}
|
||||||
|
content={
|
||||||
|
<ValuedSlider value={params.frequencyPenalty} min={0} max={2}
|
||||||
|
step={0.1} input
|
||||||
|
onChange={(e, data) => {
|
||||||
|
commonStore.setChatParams({
|
||||||
|
frequencyPenalty: data.value
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
} />
|
||||||
|
</div>
|
||||||
|
<div className="grow" />
|
||||||
|
{/*<Button*/}
|
||||||
|
{/* icon={<FolderOpenVerticalRegular />}*/}
|
||||||
|
{/* onClick={() => {*/}
|
||||||
|
{/* }}>*/}
|
||||||
|
{/* {t('Load Conversation')}*/}
|
||||||
|
{/*</Button>*/}
|
||||||
|
<Button
|
||||||
|
icon={<SaveRegular />}
|
||||||
|
onClick={() => {
|
||||||
|
let savedContent: string = '';
|
||||||
|
const isWorldModel = commonStore.getCurrentModelConfig().modelParameters.modelName.toLowerCase().includes('world');
|
||||||
|
const user = isWorldModel ? 'User' : 'Bob';
|
||||||
|
const bot = isWorldModel ? 'Assistant' : 'Alice';
|
||||||
|
commonStore.conversationOrder.forEach((uuid) => {
|
||||||
|
if (uuid === welcomeUuid)
|
||||||
|
return;
|
||||||
|
const messageItem = commonStore.conversation[uuid];
|
||||||
|
if (messageItem.type !== MessageType.Error) {
|
||||||
|
savedContent += `${messageItem.sender === userName ? user : bot}: ${messageItem.content}\n\n`;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
OpenSaveFileDialog('*.txt', 'conversation.txt', savedContent).then((path) => {
|
||||||
|
if (path)
|
||||||
|
toastWithButton(t('Conversation Saved'), t('Open'), () => {
|
||||||
|
OpenFileFolder(path, false);
|
||||||
|
});
|
||||||
|
}).catch(e => {
|
||||||
|
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
||||||
|
});
|
||||||
|
}}>
|
||||||
|
{t('Save Conversation')}
|
||||||
|
</Button>
|
||||||
|
</div>;
|
||||||
|
});
|
||||||
|
|
||||||
const ChatPanel: FC = observer(() => {
|
const ChatPanel: FC = observer(() => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const bodyRef = useRef<HTMLDivElement>(null);
|
const bodyRef = useRef<HTMLDivElement>(null);
|
||||||
const inputRef = useRef<HTMLTextAreaElement>(null);
|
const inputRef = useRef<HTMLTextAreaElement>(null);
|
||||||
const mq = useMediaQuery('(min-width: 640px)');
|
const mq = useMediaQuery('(min-width: 640px)');
|
||||||
|
if (commonStore.sidePanelCollapsed === 'auto')
|
||||||
|
commonStore.setSidePanelCollapsed(!mq);
|
||||||
const currentConfig = commonStore.getCurrentModelConfig();
|
const currentConfig = commonStore.getCurrentModelConfig();
|
||||||
const apiParams = currentConfig.apiParameters;
|
const apiParams = currentConfig.apiParameters;
|
||||||
const port = apiParams.apiPort;
|
const port = apiParams.apiPort;
|
||||||
@ -327,8 +446,10 @@ const ChatPanel: FC = observer(() => {
|
|||||||
messages,
|
messages,
|
||||||
stream: true,
|
stream: true,
|
||||||
model: commonStore.settings.apiChatModelName, // 'gpt-3.5-turbo'
|
model: commonStore.settings.apiChatModelName, // 'gpt-3.5-turbo'
|
||||||
temperature: apiParams.temperature,
|
temperature: commonStore.chatParams.temperature,
|
||||||
top_p: apiParams.topP,
|
top_p: commonStore.chatParams.topP,
|
||||||
|
presence_penalty: commonStore.chatParams.presencePenalty,
|
||||||
|
frequency_penalty: commonStore.chatParams.frequencyPenalty,
|
||||||
user_name: commonStore.activePreset?.userName || undefined,
|
user_name: commonStore.activePreset?.userName || undefined,
|
||||||
assistant_name: commonStore.activePreset?.assistantName || undefined,
|
assistant_name: commonStore.activePreset?.assistantName || undefined,
|
||||||
presystem: commonStore.activePreset?.presystem && undefined
|
presystem: commonStore.activePreset?.presystem && undefined
|
||||||
@ -391,185 +512,164 @@ const ChatPanel: FC = observer(() => {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col w-full grow gap-4 pt-4 overflow-hidden">
|
<div className="flex h-full grow pt-4 overflow-hidden">
|
||||||
<div ref={bodyRef} className="grow overflow-y-scroll overflow-x-hidden pr-2">
|
<div className="relative flex flex-col w-full grow gap-4 overflow-hidden">
|
||||||
{commonStore.conversationOrder.map(uuid =>
|
<Button className="absolute top-1 right-1" size="medium" shape="circular" appearance="subtle"
|
||||||
<ChatMessageItem key={uuid} uuid={uuid} onSubmit={onSubmit} />
|
icon={commonStore.sidePanelCollapsed ? <TextAlignJustify24Regular /> : <TextAlignJustifyRotate9024Regular />}
|
||||||
)}
|
onClick={() => commonStore.setSidePanelCollapsed(!commonStore.sidePanelCollapsed)} />
|
||||||
</div>
|
<div ref={bodyRef} className="grow overflow-y-scroll overflow-x-hidden pr-2">
|
||||||
<div className={classnames('flex items-end', mq ? 'gap-2' : '')}>
|
{commonStore.conversationOrder.map(uuid =>
|
||||||
<PresetsButton tab="Chat" size={mq ? 'large' : 'small'} shape="circular" appearance="subtle" />
|
<ChatMessageItem key={uuid} uuid={uuid} onSubmit={onSubmit} />
|
||||||
<DialogButton tooltip={t('Clear')}
|
)}
|
||||||
icon={<Delete28Regular />}
|
</div>
|
||||||
size={mq ? 'large' : 'small'} shape="circular" appearance="subtle" title={t('Clear')}
|
<div className={classnames('flex items-end', mq ? 'gap-2' : '')}>
|
||||||
contentText={t('Are you sure you want to clear the conversation? It cannot be undone.')}
|
<DialogButton tooltip={t('Clear')}
|
||||||
onConfirm={() => {
|
icon={<Delete28Regular />}
|
||||||
if (generating) {
|
size={mq ? 'large' : 'small'} shape="circular" appearance="subtle" title={t('Clear')}
|
||||||
for (const id in chatSseControllers) {
|
contentText={t('Are you sure you want to clear the conversation? It cannot be undone.')}
|
||||||
chatSseControllers[id].abort();
|
onConfirm={() => {
|
||||||
|
if (generating) {
|
||||||
|
for (const id in chatSseControllers) {
|
||||||
|
chatSseControllers[id].abort();
|
||||||
|
}
|
||||||
|
chatSseControllers = {};
|
||||||
}
|
}
|
||||||
chatSseControllers = {};
|
commonStore.setConversation({});
|
||||||
}
|
commonStore.setConversationOrder([]);
|
||||||
commonStore.setConversation({});
|
}} />
|
||||||
commonStore.setConversationOrder([]);
|
<div className="relative flex grow">
|
||||||
}} />
|
<Textarea
|
||||||
<div className="relative flex grow">
|
ref={inputRef}
|
||||||
<Textarea
|
style={{ minWidth: 0 }}
|
||||||
ref={inputRef}
|
className="grow"
|
||||||
style={{ minWidth: 0 }}
|
resize="vertical"
|
||||||
className="grow"
|
placeholder={t('Type your message here')!}
|
||||||
resize="vertical"
|
value={commonStore.currentInput}
|
||||||
placeholder={t('Type your message here')!}
|
onChange={(e) => commonStore.setCurrentInput(e.target.value)}
|
||||||
value={commonStore.currentInput}
|
onKeyDown={handleKeyDownOrClick}
|
||||||
onChange={(e) => commonStore.setCurrentInput(e.target.value)}
|
/>
|
||||||
onKeyDown={handleKeyDownOrClick}
|
<div className="absolute right-2 bottom-2">
|
||||||
/>
|
{!commonStore.currentTempAttachment ?
|
||||||
<div className="absolute right-2 bottom-2">
|
<ToolTipButton
|
||||||
{!commonStore.currentTempAttachment ?
|
desc={commonStore.attachmentUploading ?
|
||||||
<ToolTipButton
|
t('Uploading Attachment') :
|
||||||
desc={commonStore.attachmentUploading ?
|
t('Add An Attachment (Accepts pdf, txt)')}
|
||||||
t('Uploading Attachment') :
|
icon={commonStore.attachmentUploading ?
|
||||||
t('Add An Attachment (Accepts pdf, txt)')}
|
<ArrowClockwise16Regular className="animate-spin" />
|
||||||
icon={commonStore.attachmentUploading ?
|
: <Attach16Regular />}
|
||||||
<ArrowClockwise16Regular className="animate-spin" />
|
size="small" shape="circular" appearance="secondary"
|
||||||
: <Attach16Regular />}
|
onClick={() => {
|
||||||
size="small" shape="circular" appearance="secondary"
|
if (commonStore.status.status === ModelStatus.Offline && !commonStore.settings.apiUrl && commonStore.platform !== 'web') {
|
||||||
onClick={() => {
|
toast(t('Please click the button in the top right corner to start the model'), { type: 'warning' });
|
||||||
if (commonStore.status.status === ModelStatus.Offline && !commonStore.settings.apiUrl && commonStore.platform !== 'web') {
|
return;
|
||||||
toast(t('Please click the button in the top right corner to start the model'), { type: 'warning' });
|
}
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (commonStore.attachmentUploading)
|
if (commonStore.attachmentUploading)
|
||||||
return;
|
|
||||||
|
|
||||||
OpenOpenFileDialog('*.txt;*.pdf').then(async filePath => {
|
|
||||||
if (!filePath)
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
commonStore.setAttachmentUploading(true);
|
OpenOpenFileDialog('*.txt;*.pdf').then(async filePath => {
|
||||||
|
if (!filePath)
|
||||||
|
return;
|
||||||
|
|
||||||
let blob: Blob;
|
commonStore.setAttachmentUploading(true);
|
||||||
let attachmentName: string | undefined;
|
|
||||||
let attachmentContent: string | undefined;
|
let blob: Blob;
|
||||||
if (commonStore.platform === 'web') {
|
let attachmentName: string | undefined;
|
||||||
const webReturn = filePath as any;
|
let attachmentContent: string | undefined;
|
||||||
blob = webReturn.blob;
|
if (commonStore.platform === 'web') {
|
||||||
attachmentName = blob.name;
|
const webReturn = filePath as any;
|
||||||
attachmentContent = webReturn.content;
|
blob = webReturn.blob;
|
||||||
} else {
|
attachmentName = blob.name;
|
||||||
// Both are slow. Communication between frontend and backend is slow. Use AssetServer Handler to read the file.
|
attachmentContent = webReturn.content;
|
||||||
// const blob = new Blob([atob(info.content as unknown as string)]); // await fetch(`data:application/octet-stream;base64,${info.content}`).then(r => r.blob());
|
} else {
|
||||||
blob = await fetch(absPathAsset(filePath)).then(r => r.blob());
|
// Both are slow. Communication between frontend and backend is slow. Use AssetServer Handler to read the file.
|
||||||
attachmentName = filePath.split(/[\\/]/).pop();
|
// const blob = new Blob([atob(info.content as unknown as string)]); // await fetch(`data:application/octet-stream;base64,${info.content}`).then(r => r.blob());
|
||||||
}
|
blob = await fetch(absPathAsset(filePath)).then(r => r.blob());
|
||||||
if (attachmentContent) {
|
attachmentName = filePath.split(/[\\/]/).pop();
|
||||||
commonStore.setCurrentTempAttachment(
|
}
|
||||||
{
|
if (attachmentContent) {
|
||||||
name: attachmentName!,
|
commonStore.setCurrentTempAttachment(
|
||||||
size: blob.size,
|
{
|
||||||
content: attachmentContent
|
name: attachmentName!,
|
||||||
});
|
size: blob.size,
|
||||||
commonStore.setAttachmentUploading(false);
|
content: attachmentContent
|
||||||
} else {
|
});
|
||||||
const urlPath = `/file-to-text?file_name=${attachmentName}`;
|
|
||||||
const bodyForm = new FormData();
|
|
||||||
bodyForm.append('file_data', blob, attachmentName);
|
|
||||||
fetch(getServerRoot(port) + urlPath, {
|
|
||||||
method: 'POST',
|
|
||||||
body: bodyForm
|
|
||||||
}).then(async r => {
|
|
||||||
if (r.status === 200) {
|
|
||||||
const pages = (await r.json()).pages as any[];
|
|
||||||
if (pages.length === 1)
|
|
||||||
attachmentContent = pages[0].page_content;
|
|
||||||
else
|
|
||||||
attachmentContent = pages.map((p, i) => `Page ${i + 1}:\n${p.page_content}`).join('\n\n');
|
|
||||||
commonStore.setCurrentTempAttachment(
|
|
||||||
{
|
|
||||||
name: attachmentName!,
|
|
||||||
size: blob.size,
|
|
||||||
content: attachmentContent!
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
toast(r.statusText + '\n' + (await r.text()), {
|
|
||||||
type: 'error'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
commonStore.setAttachmentUploading(false);
|
|
||||||
}
|
|
||||||
).catch(e => {
|
|
||||||
commonStore.setAttachmentUploading(false);
|
commonStore.setAttachmentUploading(false);
|
||||||
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
} else {
|
||||||
});
|
const urlPath = `/file-to-text?file_name=${attachmentName}`;
|
||||||
|
const bodyForm = new FormData();
|
||||||
|
bodyForm.append('file_data', blob, attachmentName);
|
||||||
|
fetch(getServerRoot(port) + urlPath, {
|
||||||
|
method: 'POST',
|
||||||
|
body: bodyForm
|
||||||
|
}).then(async r => {
|
||||||
|
if (r.status === 200) {
|
||||||
|
const pages = (await r.json()).pages as any[];
|
||||||
|
if (pages.length === 1)
|
||||||
|
attachmentContent = pages[0].page_content;
|
||||||
|
else
|
||||||
|
attachmentContent = pages.map((p, i) => `Page ${i + 1}:\n${p.page_content}`).join('\n\n');
|
||||||
|
commonStore.setCurrentTempAttachment(
|
||||||
|
{
|
||||||
|
name: attachmentName!,
|
||||||
|
size: blob.size,
|
||||||
|
content: attachmentContent!
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
toast(r.statusText + '\n' + (await r.text()), {
|
||||||
|
type: 'error'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
commonStore.setAttachmentUploading(false);
|
||||||
|
}
|
||||||
|
).catch(e => {
|
||||||
|
commonStore.setAttachmentUploading(false);
|
||||||
|
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}).catch(e => {
|
||||||
|
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/> :
|
||||||
|
<div>
|
||||||
|
<ToolTipButton
|
||||||
|
text={
|
||||||
|
commonStore.currentTempAttachment.name.replace(
|
||||||
|
new RegExp('(^[^\\.]{5})[^\\.]+'), '$1...')
|
||||||
}
|
}
|
||||||
}).catch(e => {
|
desc={`${commonStore.currentTempAttachment.name} (${bytesToReadable(commonStore.currentTempAttachment.size)})`}
|
||||||
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
size="small" shape="circular" appearance="secondary" />
|
||||||
});
|
<ToolTipButton desc={t('Remove Attachment')}
|
||||||
}}
|
icon={<Dismiss16Regular />}
|
||||||
/> :
|
size="small" shape="circular" appearance="subtle"
|
||||||
<div>
|
onClick={() => {
|
||||||
<ToolTipButton
|
commonStore.setCurrentTempAttachment(null);
|
||||||
text={
|
}} />
|
||||||
commonStore.currentTempAttachment.name.replace(
|
</div>
|
||||||
new RegExp('(^[^\\.]{5})[^\\.]+'), '$1...')
|
}
|
||||||
}
|
</div>
|
||||||
desc={`${commonStore.currentTempAttachment.name} (${bytesToReadable(commonStore.currentTempAttachment.size)})`}
|
|
||||||
size="small" shape="circular" appearance="secondary" />
|
|
||||||
<ToolTipButton desc={t('Remove Attachment')}
|
|
||||||
icon={<Dismiss16Regular />}
|
|
||||||
size="small" shape="circular" appearance="subtle"
|
|
||||||
onClick={() => {
|
|
||||||
commonStore.setCurrentTempAttachment(null);
|
|
||||||
}} />
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
|
<ToolTipButton desc={generating ? t('Stop') : t('Send')}
|
||||||
|
icon={generating ? <RecordStop28Regular /> : <ArrowCircleUp28Regular />}
|
||||||
|
size={mq ? 'large' : 'small'} shape="circular" appearance="subtle"
|
||||||
|
onClick={(e) => {
|
||||||
|
if (generating) {
|
||||||
|
for (const id in chatSseControllers) {
|
||||||
|
chatSseControllers[id].abort();
|
||||||
|
commonStore.conversation[id].type = MessageType.Error;
|
||||||
|
commonStore.conversation[id].done = true;
|
||||||
|
}
|
||||||
|
chatSseControllers = {};
|
||||||
|
commonStore.setConversation(commonStore.conversation);
|
||||||
|
commonStore.setConversationOrder([...commonStore.conversationOrder]);
|
||||||
|
} else {
|
||||||
|
handleKeyDownOrClick(e);
|
||||||
|
}
|
||||||
|
}} />
|
||||||
</div>
|
</div>
|
||||||
<ToolTipButton desc={generating ? t('Stop') : t('Send')}
|
|
||||||
icon={generating ? <RecordStop28Regular /> : <ArrowCircleUp28Regular />}
|
|
||||||
size={mq ? 'large' : 'small'} shape="circular" appearance="subtle"
|
|
||||||
onClick={(e) => {
|
|
||||||
if (generating) {
|
|
||||||
for (const id in chatSseControllers) {
|
|
||||||
chatSseControllers[id].abort();
|
|
||||||
commonStore.conversation[id].type = MessageType.Error;
|
|
||||||
commonStore.conversation[id].done = true;
|
|
||||||
}
|
|
||||||
chatSseControllers = {};
|
|
||||||
commonStore.setConversation(commonStore.conversation);
|
|
||||||
commonStore.setConversationOrder([...commonStore.conversationOrder]);
|
|
||||||
} else {
|
|
||||||
handleKeyDownOrClick(e);
|
|
||||||
}
|
|
||||||
}} />
|
|
||||||
<ToolTipButton desc={t('Save')}
|
|
||||||
icon={<Save28Regular />}
|
|
||||||
size={mq ? 'large' : 'small'} shape="circular" appearance="subtle"
|
|
||||||
onClick={() => {
|
|
||||||
let savedContent: string = '';
|
|
||||||
const isWorldModel = commonStore.getCurrentModelConfig().modelParameters.modelName.toLowerCase().includes('world');
|
|
||||||
const user = isWorldModel ? 'User' : 'Bob';
|
|
||||||
const bot = isWorldModel ? 'Assistant' : 'Alice';
|
|
||||||
commonStore.conversationOrder.forEach((uuid) => {
|
|
||||||
if (uuid === welcomeUuid)
|
|
||||||
return;
|
|
||||||
const messageItem = commonStore.conversation[uuid];
|
|
||||||
if (messageItem.type !== MessageType.Error) {
|
|
||||||
savedContent += `${messageItem.sender === userName ? user : bot}: ${messageItem.content}\n\n`;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
OpenSaveFileDialog('*.txt', 'conversation.txt', savedContent).then((path) => {
|
|
||||||
if (path)
|
|
||||||
toastWithButton(t('Conversation Saved'), t('Open'), () => {
|
|
||||||
OpenFileFolder(path, false);
|
|
||||||
});
|
|
||||||
}).catch(e => {
|
|
||||||
toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 });
|
|
||||||
});
|
|
||||||
}} />
|
|
||||||
</div>
|
</div>
|
||||||
|
<SidePanel />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@ -7,7 +7,7 @@ import { defaultCompositionPrompt, defaultModelConfigs, defaultModelConfigsMac }
|
|||||||
import { ChartData } from 'chart.js';
|
import { ChartData } from 'chart.js';
|
||||||
import { Preset } from '../types/presets';
|
import { Preset } from '../types/presets';
|
||||||
import { AboutContent } from '../types/about';
|
import { AboutContent } from '../types/about';
|
||||||
import { Attachment, Conversation } from '../types/chat';
|
import { Attachment, ChatParams, Conversation } from '../types/chat';
|
||||||
import { CompletionPreset } from '../types/completion';
|
import { CompletionPreset } from '../types/completion';
|
||||||
import { CompositionParams } from '../types/composition';
|
import { CompositionParams } from '../types/composition';
|
||||||
import { ModelConfig } from '../types/configs';
|
import { ModelConfig } from '../types/configs';
|
||||||
@ -65,6 +65,14 @@ class CommonStore {
|
|||||||
attachmentUploading: boolean = false;
|
attachmentUploading: boolean = false;
|
||||||
attachments: { [uuid: string]: Attachment[] } = {};
|
attachments: { [uuid: string]: Attachment[] } = {};
|
||||||
currentTempAttachment: Attachment | null = null;
|
currentTempAttachment: Attachment | null = null;
|
||||||
|
chatParams: ChatParams = {
|
||||||
|
maxResponseToken: 1000,
|
||||||
|
temperature: 1,
|
||||||
|
topP: 0.3,
|
||||||
|
presencePenalty: 0,
|
||||||
|
frequencyPenalty: 1
|
||||||
|
};
|
||||||
|
sidePanelCollapsed: boolean | 'auto' = 'auto';
|
||||||
// completion
|
// completion
|
||||||
completionPreset: CompletionPreset | null = null;
|
completionPreset: CompletionPreset | null = null;
|
||||||
completionGenerating: boolean = false;
|
completionGenerating: boolean = false;
|
||||||
@ -363,6 +371,14 @@ class CommonStore {
|
|||||||
setCurrentTempAttachment(value: Attachment | null) {
|
setCurrentTempAttachment(value: Attachment | null) {
|
||||||
this.currentTempAttachment = value;
|
this.currentTempAttachment = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setChatParams(value: Partial<ChatParams>) {
|
||||||
|
this.chatParams = { ...this.chatParams, ...value };
|
||||||
|
}
|
||||||
|
|
||||||
|
setSidePanelCollapsed(value: boolean | 'auto') {
|
||||||
|
this.sidePanelCollapsed = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default new CommonStore();
|
export default new CommonStore();
|
@ -1,3 +1,5 @@
|
|||||||
|
import { ApiParameters } from './configs';
|
||||||
|
|
||||||
export const userName = 'M E';
|
export const userName = 'M E';
|
||||||
export const botName = 'A I';
|
export const botName = 'A I';
|
||||||
export const welcomeUuid = 'welcome';
|
export const welcomeUuid = 'welcome';
|
||||||
@ -32,3 +34,4 @@ export type Attachment = {
|
|||||||
size: number;
|
size: number;
|
||||||
content: string;
|
content: string;
|
||||||
}
|
}
|
||||||
|
export type ChatParams = Omit<ApiParameters, 'apiPort'>
|
Loading…
x
Reference in New Issue
Block a user