mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-14 21:27:20 +00:00
75 lines
2.3 KiB
TypeScript
75 lines
2.3 KiB
TypeScript
import { IconButton } from '@affine/component';
|
|
import { SendIcon } from '@blocksuite/icons';
|
|
import { contentLayoutAtom } from '@toeverything/plugin-infra/atom';
|
|
import { useAtomValue, useSetAtom } from 'jotai';
|
|
import type { ReactElement } from 'react';
|
|
import { Suspense, useCallback, useState } from 'react';
|
|
|
|
import { ConversationList } from '../core/components/conversation-list';
|
|
import { FollowingUp } from '../core/components/following-up';
|
|
import { openAIApiKeyAtom, useChatAtoms } from '../core/hooks';
|
|
import {
|
|
detailContentActionsStyle,
|
|
detailContentStyle,
|
|
sendButtonStyle,
|
|
textareaStyle,
|
|
} from './index.css';
|
|
|
|
const Actions = () => {
|
|
const { conversationAtom, followingUpAtoms } = useChatAtoms();
|
|
const call = useSetAtom(conversationAtom);
|
|
const questions = useAtomValue(followingUpAtoms.questionsAtom);
|
|
const generateFollowingUp = useSetAtom(followingUpAtoms.generateChatAtom);
|
|
const [input, setInput] = useState('');
|
|
return (
|
|
<>
|
|
<FollowingUp questions={questions} />
|
|
<div className={detailContentActionsStyle}>
|
|
<textarea
|
|
className={textareaStyle}
|
|
value={input}
|
|
placeholder="Type here ask Copilot some thing..."
|
|
onChange={e => {
|
|
setInput(e.target.value);
|
|
}}
|
|
/>
|
|
<IconButton
|
|
className={sendButtonStyle}
|
|
onClick={useCallback(async () => {
|
|
await call(input);
|
|
await generateFollowingUp();
|
|
}, [call, generateFollowingUp, input])}
|
|
>
|
|
<SendIcon />
|
|
</IconButton>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|
|
|
|
const DetailContentImpl = () => {
|
|
const { conversationAtom } = useChatAtoms();
|
|
const conversations = useAtomValue(conversationAtom);
|
|
|
|
return (
|
|
<div className={detailContentStyle}>
|
|
<ConversationList conversations={conversations} />
|
|
<Suspense fallback="generating follow-up question">
|
|
<Actions />
|
|
</Suspense>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export const DetailContent = (): ReactElement => {
|
|
const layout = useAtomValue(contentLayoutAtom);
|
|
const key = useAtomValue(openAIApiKeyAtom);
|
|
if (layout === 'editor' || layout.second !== 'copilot') {
|
|
return <></>;
|
|
}
|
|
if (!key) {
|
|
return <span>Please set OpenAI API Key in the debug panel.</span>;
|
|
}
|
|
return <DetailContentImpl />;
|
|
};
|