Files
budget-tracker/penpot-poster.mjs
T

276 lines
7.8 KiB
JavaScript

/**
* Création d'une affiche "SOLDES -50% Chaussures" dans Penpot
*/
import http from 'http';
const MCP_HOST = '192.168.1.150';
const MCP_PORT = 9002;
let sessionId = null;
let msgId = 1;
function postMCP(body) {
return new Promise((resolve, reject) => {
const payload = JSON.stringify(body);
const headers = {
'Content-Type': 'application/json',
'Content-Length': Buffer.byteLength(payload),
'Accept': 'application/json, text/event-stream',
};
if (sessionId) headers['mcp-session-id'] = sessionId;
const req = http.request({
hostname: MCP_HOST, port: MCP_PORT, path: '/mcp', method: 'POST', headers
}, res => {
if (res.headers['mcp-session-id']) sessionId = res.headers['mcp-session-id'];
let data = '';
res.setEncoding('utf8');
res.on('data', c => data += c);
res.on('end', () => {
const ct = res.headers['content-type'] || '';
if (ct.includes('text/event-stream')) {
let result = null, error = null;
for (const line of data.split('\n')) {
if (line.startsWith('data:')) {
const raw = line.slice(5).trim();
if (!raw || raw === '[DONE]') continue;
try {
const j = JSON.parse(raw);
if (j.result !== undefined) result = j.result;
if (j.error) error = j.error;
} catch {}
}
}
if (error) reject(new Error(JSON.stringify(error)));
else resolve(result);
} else {
try {
const j = JSON.parse(data);
if (j.error) reject(new Error(JSON.stringify(j.error)));
else resolve(j.result ?? j);
} catch { resolve(data || null); }
}
});
});
req.on('error', reject);
req.write(payload);
req.end();
});
}
async function init() {
await postMCP({
jsonrpc: '2.0', id: msgId++, method: 'initialize',
params: { protocolVersion: '2025-03-26', capabilities: {}, clientInfo: { name: 'nox', version: '1.0' } }
});
await postMCP({ jsonrpc: '2.0', method: 'notifications/initialized', params: {} }).catch(() => {});
console.log('✅ Session:', sessionId);
}
async function callTool(name, args = {}) {
const result = await postMCP({ jsonrpc: '2.0', id: msgId++, method: 'tools/call', params: { name, arguments: args } });
if (!result) return null;
if (result.content && result.content[0]) {
const text = result.content[0].text;
try { return JSON.parse(text); } catch { return text; }
}
return result;
}
// ─── MAIN ───────────────────────────────────────────────────────────────────
await init();
// 1. Profil pour récupérer defaultTeamId et defaultProjectId
const profile = await callTool('get_profile', {});
const teamId = profile?.defaultTeamId;
const projectId = profile?.defaultProjectId;
console.log('TeamId:', teamId);
console.log('ProjectId:', projectId);
if (!projectId || !teamId) {
console.error('❌ Pas de profil/projet', JSON.stringify(profile));
process.exit(1);
}
// 2. Créer le fichier
console.log('\n📄 Création du fichier...');
const file = await callTool('create_file', { project_id: projectId, name: 'Affiche SOLDES -50%' });
console.log('File:', JSON.stringify(file).slice(0, 300));
const fileId = file?.id || file?.file_id;
if (!fileId) { console.error('❌ Pas de fileId', file); process.exit(1); }
console.log('FileId:', fileId);
// 3. Récupérer la première page
const pages = await callTool('list_pages', { file_id: fileId });
console.log('Pages:', JSON.stringify(pages).slice(0, 200));
let pageId = null;
if (Array.isArray(pages)) pageId = pages[0]?.id;
else if (pages?.pages) pageId = pages.pages[0]?.id;
console.log('PageId:', pageId);
// ─── DESIGN ─────────────────────────────────────────────────────────────────
// Format affiche : 600 x 900px
// Palette : fond rouge foncé, texte blanc/jaune, accent orange
// 4. Frame principale (l'affiche)
console.log('\n🎨 Création du frame principal...');
const mainFrame = await callTool('create_frame', {
file_id: fileId,
page_id: pageId,
name: 'Affiche SOLDES',
x: 50,
y: 50,
width: 600,
height: 900,
fill_color: '#C0392B',
});
console.log('MainFrame:', JSON.stringify(mainFrame).slice(0, 200));
const frameId = mainFrame?.id;
// 5. Bande décorative haut (jaune)
await callTool('create_rectangle', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Bande haut',
x: 0, y: 0, width: 600, height: 12,
fill_color: '#F1C40F',
});
// 6. Bande décorative bas (jaune)
await callTool('create_rectangle', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Bande bas',
x: 0, y: 888, width: 600, height: 12,
fill_color: '#F1C40F',
});
// 7. Bloc accroche en haut
await callTool('create_rectangle', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Bloc top',
x: 0, y: 12, width: 600, height: 80,
fill_color: '#922B21',
});
// 8. Texte "MEGA SOLDES" (haut)
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Label MEGA',
content: 'MEGA SOLDES',
x: 0, y: 25,
width: 600, height: 60,
font_size: 32,
font_weight: '700',
fill_color: '#F1C40F',
align: 'center',
});
// 9. Gros texte "-50%"
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: '-50%',
content: '-50%',
x: 0, y: 120,
width: 600, height: 280,
font_size: 200,
font_weight: '900',
fill_color: '#FFFFFF',
align: 'center',
});
// 10. Texte "SUR TOUTES NOS CHAUSSURES"
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Sous-titre',
content: 'SUR TOUTES NOS CHAUSSURES',
x: 0, y: 415,
width: 600, height: 60,
font_size: 26,
font_weight: '600',
fill_color: '#F1C40F',
align: 'center',
});
// 11. Séparateur
await callTool('create_rectangle', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Separateur',
x: 80, y: 490, width: 440, height: 4,
fill_color: '#F1C40F',
});
// 12. Bloc central jaune pour "OFFRE LIMITÉE"
await callTool('create_rectangle', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Badge offre',
x: 130, y: 520, width: 340, height: 70,
fill_color: '#F1C40F',
corner_radius: 8,
});
// 13. Texte "OFFRE LIMITÉE"
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Offre limitee',
content: 'OFFRE LIMITÉE',
x: 130, y: 534,
width: 340, height: 44,
font_size: 24,
font_weight: '800',
fill_color: '#C0392B',
align: 'center',
});
// 14. Texte "du 1er au 31 janvier"
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Dates',
content: 'du 1er au 31 mars 2026',
x: 0, y: 620,
width: 600, height: 40,
font_size: 20,
font_weight: '400',
fill_color: '#FADBD8',
align: 'center',
});
// 15. Grande icône chaussure (emoji via texte)
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Icone chaussure',
content: '👟',
x: 0, y: 660,
width: 600, height: 160,
font_size: 130,
align: 'center',
});
// 16. Texte bas "boutique-example.fr"
await callTool('create_text', {
file_id: fileId, page_id: pageId,
parent_id: frameId,
name: 'Site web',
content: 'www.votre-boutique.fr',
x: 0, y: 845,
width: 600, height: 35,
font_size: 16,
font_weight: '400',
fill_color: '#F8C471',
align: 'center',
});
console.log('\n✅ Affiche créée dans Penpot !');
console.log(`🔗 Ouvrir : http://192.168.1.150:9001 → projet → fichier "Affiche SOLDES -50%"`);