mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-21 21:50:49 +02:00

* feat: Add CodeArtifacts component to Beta settings tab * chore: Update npm dependency to @codesandbox/sandpack-react@2.18.2 * WIP: artifacts first pass * WIP first pass remark-directive * chore: revert markdown to original component + new artifacts rendering * refactor: first pass rewrite * refactor: add throttling * first pass styling * style: Add Radix Tabs, more styling changes * feat: second pass * style: code styling * fix: package markdown fixes * feat: Add useEffect hook to Artifacts component for visibility control, slide in animation * fix: only set artifact if there is content * refactor: typing and make latest artifact active if the number of artifacts changed * feat: artifacts + shadcnui * feat: Add Copy Code button to Artifacts component * feat: first pass streaming updates * refactor: optimize ordering of artifacts in Artifacts component * refactor: optimize ordering of artifacts and add latest artifact activation in Artifacts component * refactor: add order prop to Artifact * feat: update to latest, use update time for ordering * refactor: optimize ordering of artifacts and activate latest artifact in Artifacts component * wip: remove thinking text and artifact formatting if empty * refactor: optimize Markdown rendering and add support for code artifacts * feat: global state for current artifact Id and set on artifact preview click * refactor: Rename CodePreview component to ArtifactButton * refactor: apply growth to artifact frame so artifact preview can take full space * refactor: remove artifactIdsState * refactor: nullify artifact state and reset on empty conversation * feat: reset artifact state on conversation change * feat: artifacts system prompt in backend * refactor: update UI artifact toggle label to match localization key * style: remove ArtifactButton inline-block styling * feat: memoize ArtifactPreview, add html support * refactor: abstract out components * chore: bump react-resizable-panel * refactor: resizable panel order props * fix: side panel resizing crashes * style: temporarily remove scrolling, add better styling * chore: remove thinking for now * chore: preprocess artifacts for now * feat: Add auto scrolling to CodeMarkdown (artifacts) * feat: autoswitch to preview * feat: auto switch to code, adjust prompt, remove unused code * feat: refresh button * feat: open/close artifacts * wip: mermaid * refactor: w-fit Artifact button * chore: organize code * feat: first pass mermaid * refactor: improve panning logic in MermaidDiagram component * feat: center/zoom on first render * refactor: add centering with reset button * style: mermaid styling * refactor: add back MermaidDiagram * fix: static/html template * fix: mermaid * add examples to artifacts prompt * refactor: fix CodeBar plugin prop logic * refactor: remove unnecessary mention of artifacts when not requested * fix: remove preprocessCodeArtifacts function and fix imports * feat: improve artifacts guidelines and remove unnecessary mentions * refactor: improve artifacts guidelines and remove unnecessary mentions * chore: uninstall unused packages * chore: bump vite * chore: update three dependency to version 0.167.1 * refactor: move beta settings, add additional artifacts toggles * feat: artifacts mode toggles * refactor: adjust prompt * feat: shadcnui instructions * feat: code artifacts custom prompt mode * chore: Update artifacts UI labels and instructions localizations * refactor: Remove unused code in Markdown component
123 lines
4.1 KiB
TypeScript
123 lines
4.1 KiB
TypeScript
import { useRecoilValue } from 'recoil';
|
|
import { useEffect, useMemo } from 'react';
|
|
import { useGetStartupConfig } from 'librechat-data-provider/react-query';
|
|
import { FileSources, LocalStorageKeys, getConfigDefaults } from 'librechat-data-provider';
|
|
import type { ExtendedFile } from '~/common';
|
|
import { useDragHelpers, useSetFilesToDelete } from '~/hooks';
|
|
import DragDropOverlay from './Input/Files/DragDropOverlay';
|
|
import { useDeleteFilesMutation } from '~/data-provider';
|
|
import Artifacts from '~/components/Artifacts/Artifacts';
|
|
import { SidePanel } from '~/components/SidePanel';
|
|
import store from '~/store';
|
|
|
|
const defaultInterface = getConfigDefaults().interface;
|
|
|
|
export default function Presentation({
|
|
children,
|
|
useSidePanel = false,
|
|
panel,
|
|
}: {
|
|
children: React.ReactNode;
|
|
panel?: React.ReactNode;
|
|
useSidePanel?: boolean;
|
|
}) {
|
|
const { data: startupConfig } = useGetStartupConfig();
|
|
const artifacts = useRecoilValue(store.artifactsState);
|
|
const codeArtifacts = useRecoilValue(store.codeArtifacts);
|
|
const hideSidePanel = useRecoilValue(store.hideSidePanel);
|
|
const artifactsVisible = useRecoilValue(store.artifactsVisible);
|
|
|
|
const interfaceConfig = useMemo(
|
|
() => startupConfig?.interface ?? defaultInterface,
|
|
[startupConfig],
|
|
);
|
|
|
|
const setFilesToDelete = useSetFilesToDelete();
|
|
const { isOver, canDrop, drop } = useDragHelpers();
|
|
|
|
const { mutateAsync } = useDeleteFilesMutation({
|
|
onSuccess: () => {
|
|
console.log('Temporary Files deleted');
|
|
setFilesToDelete({});
|
|
},
|
|
onError: (error) => {
|
|
console.log('Error deleting temporary files:', error);
|
|
},
|
|
});
|
|
|
|
useEffect(() => {
|
|
const filesToDelete = localStorage.getItem(LocalStorageKeys.FILES_TO_DELETE);
|
|
const map = JSON.parse(filesToDelete ?? '{}') as Record<string, ExtendedFile>;
|
|
const files = Object.values(map)
|
|
.filter(
|
|
(file) =>
|
|
file.filepath != null && file.source && !(file.embedded ?? false) && file.temp_file_id,
|
|
)
|
|
.map((file) => ({
|
|
file_id: file.file_id,
|
|
filepath: file.filepath as string,
|
|
source: file.source as FileSources,
|
|
embedded: !!(file.embedded ?? false),
|
|
}));
|
|
|
|
if (files.length === 0) {
|
|
return;
|
|
}
|
|
mutateAsync({ files });
|
|
}, [mutateAsync]);
|
|
|
|
const isActive = canDrop && isOver;
|
|
|
|
const defaultLayout = useMemo(() => {
|
|
const resizableLayout = localStorage.getItem('react-resizable-panels:layout');
|
|
return typeof resizableLayout === 'string' ? JSON.parse(resizableLayout) : undefined;
|
|
}, []);
|
|
const defaultCollapsed = useMemo(() => {
|
|
const collapsedPanels = localStorage.getItem('react-resizable-panels:collapsed');
|
|
return typeof collapsedPanels === 'string' ? JSON.parse(collapsedPanels) : true;
|
|
}, []);
|
|
const fullCollapse = useMemo(() => localStorage.getItem('fullPanelCollapse') === 'true', []);
|
|
|
|
const layout = () => (
|
|
<div className="transition-width relative flex h-full w-full flex-1 flex-col items-stretch overflow-hidden bg-white pt-0 dark:bg-gray-800">
|
|
<div className="flex h-full flex-col" role="presentation">
|
|
{children}
|
|
{isActive && <DragDropOverlay />}
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
if (useSidePanel && !hideSidePanel && interfaceConfig.sidePanel === true) {
|
|
return (
|
|
<div
|
|
ref={drop}
|
|
className="relative flex w-full grow overflow-hidden bg-white dark:bg-gray-800"
|
|
>
|
|
<SidePanel
|
|
defaultLayout={defaultLayout}
|
|
defaultCollapsed={defaultCollapsed}
|
|
fullPanelCollapse={fullCollapse}
|
|
artifacts={
|
|
artifactsVisible === true &&
|
|
codeArtifacts === true &&
|
|
Object.keys(artifacts ?? {}).length > 0 ? (
|
|
<Artifacts />
|
|
) : null
|
|
}
|
|
>
|
|
<main className="flex h-full flex-col" role="main">
|
|
{children}
|
|
{isActive && <DragDropOverlay />}
|
|
</main>
|
|
</SidePanel>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div ref={drop} className="relative flex w-full grow overflow-hidden bg-white dark:bg-gray-800">
|
|
{layout()}
|
|
{panel != null && panel}
|
|
</div>
|
|
);
|
|
}
|