Phase 9: PWA & Workbox Caching (merged)
This commit is contained in:
@@ -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);
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
Reference in New Issue
Block a user