mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-01-20 17:26:12 +01: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
189 lines
5.8 KiB
TypeScript
189 lines
5.8 KiB
TypeScript
import React, { useEffect, useRef, useState } from 'react';
|
|
import mermaid from 'mermaid';
|
|
import { TransformWrapper, TransformComponent, ReactZoomPanPinchRef } from 'react-zoom-pan-pinch';
|
|
// import { Button } from '/components/ui/Button'; // Live component
|
|
import { Button } from '~/components/ui/Button';
|
|
import { ZoomIn, ZoomOut, RefreshCw } from 'lucide-react';
|
|
|
|
interface MermaidDiagramProps {
|
|
content: string;
|
|
}
|
|
|
|
/** Note: this is just for testing purposes, don't actually use this component */
|
|
const MermaidDiagram: React.FC<MermaidDiagramProps> = ({ content }) => {
|
|
const mermaidRef = useRef<HTMLDivElement>(null);
|
|
const transformRef = useRef<ReactZoomPanPinchRef>(null);
|
|
const [isRendered, setIsRendered] = useState(false);
|
|
|
|
useEffect(() => {
|
|
mermaid.initialize({
|
|
startOnLoad: false,
|
|
theme: 'base',
|
|
securityLevel: 'sandbox',
|
|
themeVariables: {
|
|
background: '#282C34',
|
|
primaryColor: '#333842',
|
|
secondaryColor: '#333842',
|
|
tertiaryColor: '#333842',
|
|
primaryTextColor: '#ABB2BF',
|
|
secondaryTextColor: '#ABB2BF',
|
|
lineColor: '#636D83',
|
|
fontSize: '16px',
|
|
nodeBorder: '#636D83',
|
|
mainBkg: '#282C34',
|
|
altBackground: '#282C34',
|
|
textColor: '#ABB2BF',
|
|
edgeLabelBackground: '#282C34',
|
|
clusterBkg: '#282C34',
|
|
clusterBorder: '#636D83',
|
|
labelBoxBkgColor: '#333842',
|
|
labelBoxBorderColor: '#636D83',
|
|
labelTextColor: '#ABB2BF',
|
|
},
|
|
flowchart: {
|
|
curve: 'basis',
|
|
nodeSpacing: 50,
|
|
rankSpacing: 50,
|
|
diagramPadding: 8,
|
|
htmlLabels: true,
|
|
useMaxWidth: true,
|
|
defaultRenderer: 'dagre-d3',
|
|
padding: 15,
|
|
wrappingWidth: 200,
|
|
},
|
|
});
|
|
|
|
const renderDiagram = async () => {
|
|
if (mermaidRef.current) {
|
|
try {
|
|
const { svg } = await mermaid.render('mermaid-diagram', content);
|
|
mermaidRef.current.innerHTML = svg;
|
|
|
|
const svgElement = mermaidRef.current.querySelector('svg');
|
|
if (svgElement) {
|
|
svgElement.style.width = '100%';
|
|
svgElement.style.height = '100%';
|
|
|
|
const pathElements = svgElement.querySelectorAll('path');
|
|
pathElements.forEach((path) => {
|
|
path.style.strokeWidth = '1.5px';
|
|
});
|
|
|
|
const rectElements = svgElement.querySelectorAll('rect');
|
|
rectElements.forEach((rect) => {
|
|
const parent = rect.parentElement;
|
|
if (parent && parent.classList.contains('node')) {
|
|
rect.style.stroke = '#636D83';
|
|
rect.style.strokeWidth = '1px';
|
|
} else {
|
|
rect.style.stroke = 'none';
|
|
}
|
|
});
|
|
}
|
|
setIsRendered(true);
|
|
} catch (error) {
|
|
console.error('Mermaid rendering error:', error);
|
|
mermaidRef.current.innerHTML = 'Error rendering diagram';
|
|
}
|
|
}
|
|
};
|
|
|
|
renderDiagram();
|
|
}, [content]);
|
|
|
|
const centerAndFitDiagram = () => {
|
|
if (transformRef.current && mermaidRef.current) {
|
|
const { centerView, zoomToElement } = transformRef.current;
|
|
zoomToElement(mermaidRef.current as HTMLElement);
|
|
centerView(1, 0);
|
|
}
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (isRendered) {
|
|
centerAndFitDiagram();
|
|
}
|
|
}, [isRendered]);
|
|
|
|
const handlePanning = () => {
|
|
if (transformRef.current) {
|
|
const { state, instance } = (transformRef.current as ReactZoomPanPinchRef | undefined) ?? {};
|
|
if (!state || !instance) {
|
|
return;
|
|
}
|
|
const { scale, positionX, positionY } = state;
|
|
const { wrapperComponent, contentComponent } = instance;
|
|
|
|
if (wrapperComponent && contentComponent) {
|
|
const wrapperRect = wrapperComponent.getBoundingClientRect();
|
|
const contentRect = contentComponent.getBoundingClientRect();
|
|
const maxX = wrapperRect.width - contentRect.width * scale;
|
|
const maxY = wrapperRect.height - contentRect.height * scale;
|
|
|
|
let newX = positionX;
|
|
let newY = positionY;
|
|
|
|
if (newX > 0) {
|
|
newX = 0;
|
|
}
|
|
if (newY > 0) {
|
|
newY = 0;
|
|
}
|
|
if (newX < maxX) {
|
|
newX = maxX;
|
|
}
|
|
if (newY < maxY) {
|
|
newY = maxY;
|
|
}
|
|
|
|
if (newX !== positionX || newY !== positionY) {
|
|
instance.setTransformState(scale, newX, newY);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="relative h-screen w-screen cursor-move bg-[#282C34] p-5">
|
|
<TransformWrapper
|
|
ref={transformRef}
|
|
initialScale={1}
|
|
minScale={0.1}
|
|
maxScale={4}
|
|
limitToBounds={false}
|
|
centerOnInit={true}
|
|
initialPositionY={0}
|
|
wheel={{ step: 0.1 }}
|
|
panning={{ velocityDisabled: true }}
|
|
alignmentAnimation={{ disabled: true }}
|
|
onPanning={handlePanning}
|
|
>
|
|
{({ zoomIn, zoomOut }) => (
|
|
<>
|
|
<TransformComponent
|
|
wrapperStyle={{ width: '100%', height: '100%', overflow: 'hidden' }}
|
|
>
|
|
<div
|
|
ref={mermaidRef}
|
|
style={{ width: 'auto', height: 'auto', minWidth: '100%', minHeight: '100%' }}
|
|
/>
|
|
</TransformComponent>
|
|
<div className="absolute bottom-2 right-2 flex space-x-2">
|
|
<Button onClick={() => zoomIn(0.1)} variant="outline" size="icon">
|
|
<ZoomIn className="h-4 w-4" />
|
|
</Button>
|
|
<Button onClick={() => zoomOut(0.1)} variant="outline" size="icon">
|
|
<ZoomOut className="h-4 w-4" />
|
|
</Button>
|
|
<Button onClick={centerAndFitDiagram} variant="outline" size="icon">
|
|
<RefreshCw className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
</>
|
|
)}
|
|
</TransformWrapper>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default MermaidDiagram;
|