Phase 9: PWA & Workbox Caching (merged)

This commit is contained in:
Khalid A
2026-04-21 01:52:43 -05:00
3 changed files with 74 additions and 9 deletions

View File

@@ -79,6 +79,7 @@ function App() {
}; };
const handleAddTemplate = (template) => { const handleAddTemplate = (template) => {
// Apply template elements to canvas
if (template && template.elements) { if (template && template.elements) {
template.elements.forEach((el, index) => { template.elements.forEach((el, index) => {
setTimeout(() => addElement({ ...el }), index * 50); setTimeout(() => addElement({ ...el }), index * 50);

View File

@@ -164,6 +164,12 @@ input, textarea, select {
max-width: 400px; max-width: 400px;
} }
.canvas-actions {
display: flex;
gap: 0.75rem;
align-items: center;
}
.undo-redo-buttons { .undo-redo-buttons {
display: flex; display: flex;
gap: 0.5rem; gap: 0.5rem;
@@ -196,6 +202,64 @@ input, textarea, select {
cursor: not-allowed; cursor: not-allowed;
} }
.export-btn {
padding: 0.5rem 1rem;
background: linear-gradient(135deg, #22c55e, #16a34a);
color: white;
border: none;
border-radius: var(--radius-md);
font-size: 0.875rem;
font-weight: 500;
cursor: pointer;
transition: all 0.2s;
display: flex;
align-items: center;
gap: 0.5rem;
}
.export-btn:hover:not(:disabled) {
transform: translateY(-1px);
box-shadow: var(--shadow-md);
}
.export-btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.export-error {
display: flex;
align-items: center;
justify-content: space-between;
padding: 0.75rem 1rem;
background: #fef2f2;
border: 1px solid #fecaca;
border-radius: var(--radius-md);
color: #dc2626;
font-size: 0.875rem;
margin-bottom: 1rem;
width: 100%;
max-width: 400px;
}
.export-error p {
margin: 0;
}
.close-error {
background: transparent;
border: none;
color: #dc2626;
cursor: pointer;
font-size: 1.25rem;
padding: 0;
line-height: 1;
}
.close-error:hover {
color: #991b1b;
}
.canvas-wrapper { .canvas-wrapper {
margin-bottom: 1rem; margin-bottom: 1rem;
} }

View File

@@ -17,7 +17,7 @@ const PORT = process.env.PORT || 3001;
// Ensure upload and export directories exist // Ensure upload and export directories exist
const uploadsDir = join(__dirname, 'uploads'); const uploadsDir = join(__dirname, 'uploads');
const exportsDir = join(__dirname, 'exports'); const exportsDir = join(__dirname, 'exports');
[uploadsDir, exportsDir].forEach(dir => { [uploadsDir, exportsDir].forEach((dir) => {
if (!existsSync(dir)) mkdirSync(dir, { recursive: true }); if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
}); });
@@ -39,7 +39,7 @@ const storage = multer.diskStorage({
const ext = file.originalname.split('.').pop(); const ext = file.originalname.split('.').pop();
const filename = `${uuidv4()}.${ext}`; const filename = `${uuidv4()}.${ext}`;
cb(null, filename); cb(null, filename);
} },
}); });
const fileFilter = (req, file, cb) => { const fileFilter = (req, file, cb) => {
@@ -55,8 +55,8 @@ const upload = multer({
storage, storage,
fileFilter, fileFilter,
limits: { limits: {
fileSize: 20 * 1024 * 1024 // 20MB fileSize: 20 * 1024 * 1024, // 20MB
} },
}); });
// Health check endpoint // Health check endpoint
@@ -96,13 +96,13 @@ app.post('/api/upload', upload.single('image'), async (req, res) => {
url: originalUrl, url: originalUrl,
filename: req.file.filename, filename: req.file.filename,
size: req.file.size, size: req.file.size,
mimetype: req.file.mimetype mimetype: req.file.mimetype,
}, },
preview: { preview: {
path: previewPath, path: previewPath,
url: previewUrl, url: previewUrl,
filename: previewFilename filename: previewFilename,
} },
}); });
} catch (error) { } catch (error) {
console.error('Upload error:', error); console.error('Upload error:', error);
@@ -192,8 +192,8 @@ app.post('/api/export', async (req, res) => {
width: EXPORT_SIZE, width: EXPORT_SIZE,
height: EXPORT_SIZE, height: EXPORT_SIZE,
dpi: 300, dpi: 300,
sizeInches: '15x15' sizeInches: '15x15',
} },
}); });
} catch (error) { } catch (error) {
console.error('Export error:', error); console.error('Export error:', error);