mirror of
https://github.com/stackblitz/bolt.new
synced 2025-02-06 04:48:04 +00:00
feat: added sync files to selected local folder function is created. Yarn package manager fixes, styling fixes. Sass module fix. Added Claude model for open router.
This commit is contained in:
parent
50a501ecb1
commit
49217f2461
@ -1,7 +1,7 @@
|
|||||||
import { useStore } from '@nanostores/react';
|
import { useStore } from '@nanostores/react';
|
||||||
import { motion, type HTMLMotionProps, type Variants } from 'framer-motion';
|
import { motion, type HTMLMotionProps, type Variants } from 'framer-motion';
|
||||||
import { computed } from 'nanostores';
|
import { computed } from 'nanostores';
|
||||||
import { memo, useCallback, useEffect } from 'react';
|
import { memo, useCallback, useEffect, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
import {
|
import {
|
||||||
type OnChangeCallback as OnEditorChange,
|
type OnChangeCallback as OnEditorChange,
|
||||||
@ -55,6 +55,8 @@ const workbenchVariants = {
|
|||||||
export const Workbench = memo(({ chatStarted, isStreaming }: WorkspaceProps) => {
|
export const Workbench = memo(({ chatStarted, isStreaming }: WorkspaceProps) => {
|
||||||
renderLogger.trace('Workbench');
|
renderLogger.trace('Workbench');
|
||||||
|
|
||||||
|
const [isSyncing, setIsSyncing] = useState(false);
|
||||||
|
|
||||||
const hasPreview = useStore(computed(workbenchStore.previews, (previews) => previews.length > 0));
|
const hasPreview = useStore(computed(workbenchStore.previews, (previews) => previews.length > 0));
|
||||||
const showWorkbench = useStore(workbenchStore.showWorkbench);
|
const showWorkbench = useStore(workbenchStore.showWorkbench);
|
||||||
const selectedFile = useStore(workbenchStore.selectedFile);
|
const selectedFile = useStore(workbenchStore.selectedFile);
|
||||||
@ -99,6 +101,21 @@ export const Workbench = memo(({ chatStarted, isStreaming }: WorkspaceProps) =>
|
|||||||
workbenchStore.resetCurrentDocument();
|
workbenchStore.resetCurrentDocument();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const handleSyncFiles = useCallback(async () => {
|
||||||
|
setIsSyncing(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const directoryHandle = await window.showDirectoryPicker();
|
||||||
|
await workbenchStore.syncFiles(directoryHandle);
|
||||||
|
toast.success('Files synced successfully');
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error syncing files:', error);
|
||||||
|
toast.error('Failed to sync files');
|
||||||
|
} finally {
|
||||||
|
setIsSyncing(false);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
chatStarted && (
|
chatStarted && (
|
||||||
<motion.div
|
<motion.div
|
||||||
@ -132,6 +149,10 @@ export const Workbench = memo(({ chatStarted, isStreaming }: WorkspaceProps) =>
|
|||||||
<div className="i-ph:code" />
|
<div className="i-ph:code" />
|
||||||
Download Code
|
Download Code
|
||||||
</PanelHeaderButton>
|
</PanelHeaderButton>
|
||||||
|
<PanelHeaderButton className="mr-1 text-sm" onClick={handleSyncFiles} disabled={isSyncing}>
|
||||||
|
{isSyncing ? <div className="i-ph:spinner" /> : <div className="i-ph:cloud-arrow-down" />}
|
||||||
|
{isSyncing ? 'Syncing...' : 'Sync Files'}
|
||||||
|
</PanelHeaderButton>
|
||||||
<PanelHeaderButton
|
<PanelHeaderButton
|
||||||
className="mr-1 text-sm"
|
className="mr-1 text-sm"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@ -184,7 +205,6 @@ export const Workbench = memo(({ chatStarted, isStreaming }: WorkspaceProps) =>
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
interface ViewProps extends HTMLMotionProps<'div'> {
|
interface ViewProps extends HTMLMotionProps<'div'> {
|
||||||
children: JSX.Element;
|
children: JSX.Element;
|
||||||
}
|
}
|
||||||
|
@ -280,21 +280,22 @@ export class WorkbenchStore {
|
|||||||
|
|
||||||
for (const [filePath, dirent] of Object.entries(files)) {
|
for (const [filePath, dirent] of Object.entries(files)) {
|
||||||
if (dirent?.type === 'file' && !dirent.isBinary) {
|
if (dirent?.type === 'file' && !dirent.isBinary) {
|
||||||
// Remove '/home/project/' from the beginning of the path
|
// remove '/home/project/' from the beginning of the path
|
||||||
const relativePath = filePath.replace(/^\/home\/project\//, '');
|
const relativePath = filePath.replace(/^\/home\/project\//, '');
|
||||||
|
|
||||||
// Split the path into segments
|
// split the path into segments
|
||||||
const pathSegments = relativePath.split('/');
|
const pathSegments = relativePath.split('/');
|
||||||
|
|
||||||
// If there's more than one segment, we need to create folders
|
// if there's more than one segment, we need to create folders
|
||||||
if (pathSegments.length > 1) {
|
if (pathSegments.length > 1) {
|
||||||
let currentFolder = zip;
|
let currentFolder = zip;
|
||||||
|
|
||||||
for (let i = 0; i < pathSegments.length - 1; i++) {
|
for (let i = 0; i < pathSegments.length - 1; i++) {
|
||||||
currentFolder = currentFolder.folder(pathSegments[i])!;
|
currentFolder = currentFolder.folder(pathSegments[i])!;
|
||||||
}
|
}
|
||||||
currentFolder.file(pathSegments[pathSegments.length - 1], dirent.content);
|
currentFolder.file(pathSegments[pathSegments.length - 1], dirent.content);
|
||||||
} else {
|
} else {
|
||||||
// If there's only one segment, it's a file in the root
|
// if there's only one segment, it's a file in the root
|
||||||
zip.file(relativePath, dirent.content);
|
zip.file(relativePath, dirent.content);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -303,6 +304,35 @@ export class WorkbenchStore {
|
|||||||
const content = await zip.generateAsync({ type: 'blob' });
|
const content = await zip.generateAsync({ type: 'blob' });
|
||||||
saveAs(content, 'project.zip');
|
saveAs(content, 'project.zip');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async syncFiles(targetHandle: FileSystemDirectoryHandle) {
|
||||||
|
const files = this.files.get();
|
||||||
|
const syncedFiles = [];
|
||||||
|
|
||||||
|
for (const [filePath, dirent] of Object.entries(files)) {
|
||||||
|
if (dirent?.type === 'file' && !dirent.isBinary) {
|
||||||
|
const relativePath = filePath.replace(/^\/home\/project\//, '');
|
||||||
|
const pathSegments = relativePath.split('/');
|
||||||
|
let currentHandle = targetHandle;
|
||||||
|
|
||||||
|
for (let i = 0; i < pathSegments.length - 1; i++) {
|
||||||
|
currentHandle = await currentHandle.getDirectoryHandle(pathSegments[i], { create: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
// create or get the file
|
||||||
|
const fileHandle = await currentHandle.getFileHandle(pathSegments[pathSegments.length - 1], { create: true });
|
||||||
|
|
||||||
|
// write the file content
|
||||||
|
const writable = await fileHandle.createWritable();
|
||||||
|
await writable.write(dirent.content);
|
||||||
|
await writable.close();
|
||||||
|
|
||||||
|
syncedFiles.push(relativePath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return syncedFiles;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const workbenchStore = new WorkbenchStore();
|
export const workbenchStore = new WorkbenchStore();
|
||||||
|
3
app/types/global.d.ts
vendored
Normal file
3
app/types/global.d.ts
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
interface Window {
|
||||||
|
showDirectoryPicker(): Promise<FileSystemDirectoryHandle>;
|
||||||
|
}
|
@ -10,6 +10,8 @@ export const DEFAULT_PROVIDER = 'Anthropic';
|
|||||||
const staticModels: ModelInfo[] = [
|
const staticModels: ModelInfo[] = [
|
||||||
{ name: 'claude-3-5-sonnet-20240620', label: 'Claude 3.5 Sonnet', provider: 'Anthropic' },
|
{ name: 'claude-3-5-sonnet-20240620', label: 'Claude 3.5 Sonnet', provider: 'Anthropic' },
|
||||||
{ name: 'gpt-4o', label: 'GPT-4o', provider: 'OpenAI' },
|
{ name: 'gpt-4o', label: 'GPT-4o', provider: 'OpenAI' },
|
||||||
|
{ name: 'anthropic/claude-3.5-sonnet', label: 'Anthropic: Claude 3.5 Sonnet (OpenRouter)', provider: 'OpenRouter' },
|
||||||
|
{ name: 'anthropic/claude-3-haiku', label: 'Anthropic: Claude 3 Haiku (OpenRouter)', provider: 'OpenRouter' },
|
||||||
{ name: 'deepseek/deepseek-coder', label: 'Deepseek-Coder V2 236B (OpenRouter)', provider: 'OpenRouter' },
|
{ name: 'deepseek/deepseek-coder', label: 'Deepseek-Coder V2 236B (OpenRouter)', provider: 'OpenRouter' },
|
||||||
{ name: 'google/gemini-flash-1.5', label: 'Google Gemini Flash 1.5 (OpenRouter)', provider: 'OpenRouter' },
|
{ name: 'google/gemini-flash-1.5', label: 'Google Gemini Flash 1.5 (OpenRouter)', provider: 'OpenRouter' },
|
||||||
{ name: 'google/gemini-pro-1.5', label: 'Google Gemini Pro 1.5 (OpenRouter)', provider: 'OpenRouter' },
|
{ name: 'google/gemini-pro-1.5', label: 'Google Gemini Pro 1.5 (OpenRouter)', provider: 'OpenRouter' },
|
||||||
|
@ -3,7 +3,6 @@
|
|||||||
"description": "StackBlitz AI Agent",
|
"description": "StackBlitz AI Agent",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"packageManager": "pnpm@9.4.0",
|
|
||||||
"sideEffects": false,
|
"sideEffects": false,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -94,6 +93,7 @@
|
|||||||
"is-ci": "^3.0.1",
|
"is-ci": "^3.0.1",
|
||||||
"node-fetch": "^3.3.2",
|
"node-fetch": "^3.3.2",
|
||||||
"prettier": "^3.3.2",
|
"prettier": "^3.3.2",
|
||||||
|
"sass-embedded": "^1.80.3",
|
||||||
"typescript": "^5.5.2",
|
"typescript": "^5.5.2",
|
||||||
"unified": "^11.0.5",
|
"unified": "^11.0.5",
|
||||||
"unocss": "^0.61.3",
|
"unocss": "^0.61.3",
|
||||||
|
@ -27,6 +27,13 @@ export default defineConfig((config) => {
|
|||||||
chrome129IssuePlugin(),
|
chrome129IssuePlugin(),
|
||||||
config.mode === 'production' && optimizeCssModules({ apply: 'build' }),
|
config.mode === 'production' && optimizeCssModules({ apply: 'build' }),
|
||||||
],
|
],
|
||||||
|
css: {
|
||||||
|
preprocessorOptions: {
|
||||||
|
scss: {
|
||||||
|
api: 'modern-compiler',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user