Compare commits
11 Commits
feat/pc3
...
5589678abc
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5589678abc | ||
|
|
be7fc09085 | ||
|
|
fccbc6d19c | ||
|
|
98f1257ece | ||
|
|
e22dd6654a | ||
|
|
6aa5a7143a | ||
|
|
68e511be7a | ||
|
|
0c53f450c3 | ||
|
|
64a3cc7147 | ||
|
|
712ed0eefa | ||
|
|
71053ec9a6 |
15
.env.example
Normal file
15
.env.example
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Behold.so feed IDs (voir docs/BEHOLD-SETUP.md)
|
||||||
|
# 1) Inscris-toi sur https://behold.so/dashboard
|
||||||
|
# 2) Connecte les 2 comptes Insta (@aep.politique + @julesneny)
|
||||||
|
# 3) Recupere les feed IDs et copie ce fichier vers .env.local puis remplis ci-dessous
|
||||||
|
PUBLIC_BEHOLD_AEP=
|
||||||
|
PUBLIC_BEHOLD_JULESNENY=
|
||||||
|
|
||||||
|
# Journal unifie (PC6) - URL JSON agrege par n8n cron nocturne
|
||||||
|
# Override en local : pointer vers un mock /public/data/journal.json par exemple
|
||||||
|
PUBLIC_JOURNAL_URL=https://data.trans-former.fr/journal.json
|
||||||
|
|
||||||
|
# Chatbot upstream (PC7) - URL backend chatbot AEP
|
||||||
|
# V1 : chatbot AEP classique (Mistral Small + 120 fiches)
|
||||||
|
# V1.5 : switch vers LightRAG-PE (1 ligne)
|
||||||
|
CHATBOT_UPSTREAM=https://aep.trans-former.fr/api/chatbot
|
||||||
@@ -1,10 +1,15 @@
|
|||||||
// @ts-check
|
// @ts-check
|
||||||
import { defineConfig } from 'astro/config';
|
import { defineConfig } from 'astro/config';
|
||||||
import vue from '@astrojs/vue';
|
import vue from '@astrojs/vue';
|
||||||
|
import node from '@astrojs/node';
|
||||||
import tailwindcss from '@tailwindcss/vite';
|
import tailwindcss from '@tailwindcss/vite';
|
||||||
|
|
||||||
// https://astro.build/config
|
// PC7 — bascule SSR (mode 'server' Astro 6) pour endpoint /api/chatbot proxy.
|
||||||
|
// Toutes les pages publiques restent statiques via `export const prerender = true`.
|
||||||
|
// Coolify deploy (PC8) : `node ./dist/server/entry.mjs` (Node adapter standalone).
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
|
output: 'server',
|
||||||
|
adapter: node({ mode: 'standalone' }),
|
||||||
integrations: [vue()],
|
integrations: [vue()],
|
||||||
vite: {
|
vite: {
|
||||||
plugins: [tailwindcss()],
|
plugins: [tailwindcss()],
|
||||||
|
|||||||
49
docs/BEHOLD-SETUP.md
Normal file
49
docs/BEHOLD-SETUP.md
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
# Setup Behold pour embeds Insta
|
||||||
|
|
||||||
|
Page cerveau col D (`ColInsta.astro`) consomme 2 feeds Insta via [Behold.so](https://behold.so) (gratuit, 2 feeds, sync 1h, sans login user).
|
||||||
|
|
||||||
|
## Etapes
|
||||||
|
|
||||||
|
1. Creer compte sur https://behold.so (gratuit jusqu'a 2 feeds)
|
||||||
|
2. Connecter `@aep.politique` (Login Instagram via Facebook Business)
|
||||||
|
3. Connecter `@julesneny` (idem)
|
||||||
|
4. Recuperer les 2 feed IDs depuis le dashboard Behold
|
||||||
|
5. Copier `.env.example` vers `.env.local` (a la racine du repo)
|
||||||
|
6. Remplir `.env.local` :
|
||||||
|
|
||||||
|
```bash
|
||||||
|
PUBLIC_BEHOLD_AEP=xxxxxxxxxxxx
|
||||||
|
PUBLIC_BEHOLD_JULESNENY=yyyyyyyyyyyy
|
||||||
|
```
|
||||||
|
|
||||||
|
7. Relancer `npm run dev` ou rebuild + redeploy (PC8)
|
||||||
|
|
||||||
|
## Comportement par defaut
|
||||||
|
|
||||||
|
Sans feed IDs valides (placeholder), le composant `InstaFeed.vue` affiche un fallback gracieux :
|
||||||
|
- Titre du compte (lien direct vers Instagram)
|
||||||
|
- Bio courte
|
||||||
|
- Bouton "Voir sur Instagram"
|
||||||
|
|
||||||
|
C'est OK pour un V1 visuel "complet". UX degradee mais pas casse.
|
||||||
|
|
||||||
|
## Ressources techniques
|
||||||
|
|
||||||
|
- API Behold : `https://feeds.behold.so/{feedId}` -> JSON array de posts
|
||||||
|
- Sync : 1h (Behold rafraichit depuis Insta automatiquement)
|
||||||
|
- CDN : images servies depuis Behold (pas de hot-link Insta direct)
|
||||||
|
|
||||||
|
## CSP (PC8 deploy)
|
||||||
|
|
||||||
|
Si une CSP est ajoutee au deploy Caddy, prevoir :
|
||||||
|
|
||||||
|
```
|
||||||
|
connect-src 'self' https://feeds.behold.so;
|
||||||
|
img-src 'self' data: https://feeds.behold.so https://*.cdninstagram.com;
|
||||||
|
```
|
||||||
|
|
||||||
|
## Alternatives si Behold ne convient pas
|
||||||
|
|
||||||
|
- **EmbedSocial** (~$8/mois) : https://embedsocial.com
|
||||||
|
- **Scrape n8n nocturne** (V1.5) : Insta Graph API via compte Business + Page Facebook ; coherent avec PC6 journal-aggregate
|
||||||
|
- **Posts manuels oEmbed** (V2) : Jules selectionne 6-10 URLs, oEmbed post-by-post
|
||||||
147
docs/PC6-JOURNAL-N8N-SETUP.md
Normal file
147
docs/PC6-JOURNAL-N8N-SETUP.md
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
# PC6 — Setup journal unifié + n8n workflow
|
||||||
|
|
||||||
|
Spec : `0 INBOX/PROMPTS/page-cerveau-build/PROMPT-PC6-journal-n8n.md`
|
||||||
|
Pilote : `0 INBOX/PROMPTS/page-cerveau-build/PILOTE-PC.md` (delta 5, delta 15)
|
||||||
|
|
||||||
|
## TL;DR
|
||||||
|
|
||||||
|
Le journal de la colonne G est alimenté par un cron n8n nocturne qui agrège plusieurs sources publiques (Gitéa Atom, Behold @aep, Behold @julesneny) et écrit un JSON statique servi par Caddy sur `data.trans-former.fr/journal.json`.
|
||||||
|
|
||||||
|
```
|
||||||
|
n8n cron (3h00 UTC)
|
||||||
|
-> fetch Gitéa Atom + Behold @aep + Behold @julesneny
|
||||||
|
-> normalisation Code Node
|
||||||
|
-> tri desc + top 100
|
||||||
|
-> écrit /home/node/.n8n/journal/journal.json (volume Docker)
|
||||||
|
-> Caddy data.trans-former.fr file_server expose ce fichier
|
||||||
|
-> JournalList.vue fetch côté client (no rebuild Astro requis)
|
||||||
|
```
|
||||||
|
|
||||||
|
## Sources V1 actives
|
||||||
|
|
||||||
|
| Plateforme | Hashtag | URL feed | Statut |
|
||||||
|
|---|---|---|---|
|
||||||
|
| Gitéa | `#stack` | `https://git.trans-former.fr/jules.atom` | ACTIF |
|
||||||
|
| Behold @aep | `#aep-politique` | `https://feeds.behold.so/{PUBLIC_BEHOLD_AEP}` | conditionnel (skip si feed ID absent) |
|
||||||
|
| Behold @julesneny | `#peinture` | `https://feeds.behold.so/{PUBLIC_BEHOLD_JULESNENY}` | conditionnel (skip si feed ID absent) |
|
||||||
|
|
||||||
|
## Sources skipped (V1 -> V1.5/V2)
|
||||||
|
|
||||||
|
| Plateforme | Hashtag | Raison |
|
||||||
|
|---|---|---|
|
||||||
|
| GitHub.com | `#stack` | username `julesneny` n'existe pas (HTTP 404). Pivot Gitéa pour le MVP. À reconfirmer si Jules a un autre handle GitHub public. |
|
||||||
|
| Substack | `#politique` | `transformations.substack.com` est pris par "WoodHorse" (pas Jules). Handle Substack à confirmer avant V1.5. |
|
||||||
|
| LinkedIn | `#building-public` | V2 (RSS via service tiers ou scrape) |
|
||||||
|
| Castopod | `#podcast` | V2 (Castopod RSS prêt mais hors scope MVP) |
|
||||||
|
| Blog `trans-former.fr` | `#manifeste` | V2 (post-PC8 deploy) |
|
||||||
|
|
||||||
|
## Format JSON
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"generatedAt": "2026-05-09T03:00:00Z",
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"id": "gitea-2026-05-09-pc6",
|
||||||
|
"platform": "gitea",
|
||||||
|
"hashtag": "#stack",
|
||||||
|
"date": "2026-05-09T01:01:00Z",
|
||||||
|
"titre": "PC6 journal unifié + n8n agrégateur",
|
||||||
|
"extrait": "...",
|
||||||
|
"url": "https://git.trans-former.fr/jules/astro-site-cerveau/commit/...",
|
||||||
|
"thumbnail": null
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"counts": { "total": N, "gitea": N, "instagram": N }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Composant Vue
|
||||||
|
|
||||||
|
`src/components/vue/JournalList.vue` :
|
||||||
|
- fetch `import.meta.env.PUBLIC_JOURNAL_URL` (défaut `https://data.trans-former.fr/journal.json`)
|
||||||
|
- écoute `window.addEventListener('hashtag-filter-change', ...)` émis par ColJournal.astro
|
||||||
|
- filtre par hashtag (vide ou tous cochés -> tout afficher ; tous décochés -> rien)
|
||||||
|
- tri desc déjà fait côté n8n, le composant respecte l'ordre
|
||||||
|
|
||||||
|
Cabling : `src/components/astro/ColJournal.astro` importe et rend `<JournalList client:visible />` dans `#journal-list`.
|
||||||
|
|
||||||
|
## Variable d'env
|
||||||
|
|
||||||
|
`PUBLIC_JOURNAL_URL=https://data.trans-former.fr/journal.json` (`.env.example`)
|
||||||
|
|
||||||
|
Override possible en local pointant vers `/data/journal.json` (mock fourni dans `public/data/journal.json`).
|
||||||
|
|
||||||
|
## Setup VPS — étapes (ops, à valider Jules)
|
||||||
|
|
||||||
|
### 1. DNS
|
||||||
|
|
||||||
|
Dans OVH zone DNS `trans-former.fr` :
|
||||||
|
|
||||||
|
```
|
||||||
|
data A 178.104.106.195 TTL 600
|
||||||
|
```
|
||||||
|
|
||||||
|
Attendre propagation (~5min).
|
||||||
|
|
||||||
|
### 2. Volume Docker partagé n8n -> Caddy
|
||||||
|
|
||||||
|
Le container n8n monte `vps-kit_n8n_data:/home/node/.n8n`. On va simplement lire un fichier dans ce volume depuis Caddy.
|
||||||
|
|
||||||
|
Path source : `/var/lib/docker/volumes/vps-kit_n8n_data/_data/journal/journal.json`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
ssh vps-hetzner "mkdir -p /var/lib/docker/volumes/vps-kit_n8n_data/_data/journal && \
|
||||||
|
chown 1000:1000 /var/lib/docker/volumes/vps-kit_n8n_data/_data/journal"
|
||||||
|
```
|
||||||
|
|
||||||
|
### 3. Caddyfile bloc
|
||||||
|
|
||||||
|
Ajouter dans le Caddyfile (probablement `/etc/caddy/Caddyfile` ou `/opt/vps-kit/configs/Caddyfile`) :
|
||||||
|
|
||||||
|
```caddy
|
||||||
|
data.trans-former.fr {
|
||||||
|
root * /var/lib/docker/volumes/vps-kit_n8n_data/_data/journal
|
||||||
|
file_server {
|
||||||
|
index journal.json
|
||||||
|
}
|
||||||
|
encode gzip
|
||||||
|
header {
|
||||||
|
Cache-Control "public, max-age=300"
|
||||||
|
Access-Control-Allow-Origin "https://trans-former.fr"
|
||||||
|
}
|
||||||
|
log {
|
||||||
|
output file /var/log/caddy/data.log
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Backup + reload :
|
||||||
|
```bash
|
||||||
|
ssh vps-hetzner "cp /etc/caddy/Caddyfile /etc/caddy/Caddyfile.bak.$(date +%Y%m%d-%H%M%S) && \
|
||||||
|
systemctl reload caddy && \
|
||||||
|
systemctl status caddy --no-pager | head -10"
|
||||||
|
```
|
||||||
|
|
||||||
|
### 4. Workflow n8n
|
||||||
|
|
||||||
|
Importer `docs/n8n-workflow-journal-aggregate.json` dans https://automate.trans-former.fr (UI -> Import from file).
|
||||||
|
|
||||||
|
Activer le toggle, vérifier le cron (`0 3 * * *`).
|
||||||
|
|
||||||
|
Configurer les credentials env n8n si besoin (Behold feed IDs) -> non bloquants si absents (workflow skip).
|
||||||
|
|
||||||
|
### 5. Smoke test
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Run manuel (UI n8n -> Execute Workflow)
|
||||||
|
ssh vps-hetzner "ls -la /var/lib/docker/volumes/vps-kit_n8n_data/_data/journal/"
|
||||||
|
curl -sf https://data.trans-former.fr/journal.json | jq '.counts'
|
||||||
|
```
|
||||||
|
|
||||||
|
## Backlog (hors scope PC6)
|
||||||
|
|
||||||
|
- Trigger rebuild Astro Coolify webhook (PC8)
|
||||||
|
- Sources V2 : LinkedIn, Castopod, Blog, Substack (post handle confirmé)
|
||||||
|
- Storage archivage long-terme (V1 = écrasement quotidien)
|
||||||
|
- Real-time updates (V3)
|
||||||
152
docs/n8n-workflow-journal-aggregate.json
Normal file
152
docs/n8n-workflow-journal-aggregate.json
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
{
|
||||||
|
"name": "journal-aggregate",
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"rule": {
|
||||||
|
"interval": [
|
||||||
|
{
|
||||||
|
"field": "cronExpression",
|
||||||
|
"expression": "0 3 * * *"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "schedule-trigger",
|
||||||
|
"name": "Cron-3h",
|
||||||
|
"type": "n8n-nodes-base.scheduleTrigger",
|
||||||
|
"typeVersion": 1.1,
|
||||||
|
"position": [240, 320]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"url": "https://git.trans-former.fr/jules.atom",
|
||||||
|
"options": {
|
||||||
|
"response": {
|
||||||
|
"response": {
|
||||||
|
"responseFormat": "text"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"timeout": 15000
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "fetch-gitea",
|
||||||
|
"name": "Fetch-gitea",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [460, 200]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"url": "=https://feeds.behold.so/{{ $env.PUBLIC_BEHOLD_AEP || 'NOT_SET' }}",
|
||||||
|
"options": {
|
||||||
|
"response": {
|
||||||
|
"response": {
|
||||||
|
"neverError": true,
|
||||||
|
"responseFormat": "json"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"timeout": 15000
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "fetch-behold-aep",
|
||||||
|
"name": "Fetch-behold-aep",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [460, 320]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"url": "=https://feeds.behold.so/{{ $env.PUBLIC_BEHOLD_JULESNENY || 'NOT_SET' }}",
|
||||||
|
"options": {
|
||||||
|
"response": {
|
||||||
|
"response": {
|
||||||
|
"neverError": true,
|
||||||
|
"responseFormat": "json"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"timeout": 15000
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "fetch-behold-julesneny",
|
||||||
|
"name": "Fetch-behold-julesneny",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [460, 440]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"jsCode": "// Normalisation des 3 sources V1 vers le format JSON unifié\n// Sources : Gitéa Atom (XML), Behold @aep (JSON), Behold @julesneny (JSON)\n\nconst items = [];\n\n// ---- Helper parse Atom XML (Gitéa) ----\nfunction parseAtomGitea(xml) {\n const out = [];\n if (!xml || typeof xml !== 'string') return out;\n const entries = xml.split(/<entry>/i).slice(1);\n for (const raw of entries) {\n const block = '<entry>' + raw.split(/<\\/entry>/i)[0] + '</entry>';\n const title = (block.match(/<title>([\\s\\S]*?)<\\/title>/i) || [])[1] || '';\n const updated = (block.match(/<updated>([^<]+)<\\/updated>/i) || [])[1] || '';\n const link = (block.match(/<link[^>]*href=\"([^\"]+)\"/i) || [])[1] || '';\n const summary = (block.match(/<summary[^>]*>([\\s\\S]*?)<\\/summary>/i) || [])[1] || '';\n const id = (block.match(/<id>([^<]+)<\\/id>/i) || [])[1] || link || updated;\n if (!updated) continue;\n // Decode HTML entities basique + strip tags\n const decode = (s) => s\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/"/g, '\"')\n .replace(/"/g, '\"')\n .replace(/
/g, ' ')\n .replace(/&/g, '&')\n .replace(/<[^>]+>/g, ' ')\n .replace(/\\s+/g, ' ')\n .trim();\n const titreClean = decode(title);\n const extrait = decode(summary).slice(0, 280);\n out.push({\n id: 'gitea-' + (id.slice(0, 80) || updated),\n platform: 'gitea',\n hashtag: '#stack',\n date: new Date(updated).toISOString(),\n titre: titreClean.slice(0, 140),\n extrait,\n url: link,\n thumbnail: null,\n });\n }\n return out;\n}\n\n// ---- Récupère payloads depuis les 3 nodes amont ----\nlet giteaXml = '';\ntry {\n const giteaItems = $('Fetch-gitea').all();\n if (giteaItems.length && giteaItems[0].json) {\n // HTTP node typeVersion 4.1 met le body brut dans .data si responseFormat=text\n giteaXml = giteaItems[0].json.data || giteaItems[0].json.body || '';\n if (typeof giteaXml !== 'string') giteaXml = String(giteaXml);\n }\n} catch (e) {\n console.log('Gitéa fetch missing:', e.message);\n}\n\nlet beholdAep = [];\ntry {\n const aep = $('Fetch-behold-aep').all();\n if (aep.length && aep[0].json && Array.isArray(aep[0].json.posts)) {\n beholdAep = aep[0].json.posts;\n } else if (aep.length && Array.isArray(aep[0].json)) {\n beholdAep = aep[0].json;\n }\n} catch (e) {\n console.log('Behold AEP fetch missing:', e.message);\n}\n\nlet beholdJulesneny = [];\ntry {\n const j = $('Fetch-behold-julesneny').all();\n if (j.length && j[0].json && Array.isArray(j[0].json.posts)) {\n beholdJulesneny = j[0].json.posts;\n } else if (j.length && Array.isArray(j[0].json)) {\n beholdJulesneny = j[0].json;\n }\n} catch (e) {\n console.log('Behold Julesneny fetch missing:', e.message);\n}\n\n// ---- Normalisation ----\nfor (const it of parseAtomGitea(giteaXml)) items.push(it);\n\nfor (const post of beholdAep) {\n if (!post || !post.id) continue;\n const ts = post.timestamp || post.taken_at || post.date;\n const caption = (post.caption || post.captionWithEmojis || '').toString();\n items.push({\n id: 'insta-aep-' + post.id,\n platform: 'instagram',\n hashtag: '#aep-politique',\n date: ts ? new Date(ts).toISOString() : new Date().toISOString(),\n titre: caption.slice(0, 100) || '@aep.politique',\n extrait: caption.slice(0, 280),\n url: post.permalink || post.url || 'https://instagram.com/aep.politique',\n thumbnail: post.thumbnailUrl || post.mediaUrl || (post.sizes && post.sizes.medium && post.sizes.medium.mediaUrl) || null,\n });\n}\n\nfor (const post of beholdJulesneny) {\n if (!post || !post.id) continue;\n const ts = post.timestamp || post.taken_at || post.date;\n const caption = (post.caption || post.captionWithEmojis || '').toString();\n items.push({\n id: 'insta-julesneny-' + post.id,\n platform: 'instagram',\n hashtag: '#peinture',\n date: ts ? new Date(ts).toISOString() : new Date().toISOString(),\n titre: caption.slice(0, 100) || '@julesneny',\n extrait: caption.slice(0, 280),\n url: post.permalink || post.url || 'https://instagram.com/julesneny',\n thumbnail: post.thumbnailUrl || post.mediaUrl || (post.sizes && post.sizes.medium && post.sizes.medium.mediaUrl) || null,\n });\n}\n\n// ---- Tri desc + cap top 100 ----\nitems.sort((a, b) => b.date.localeCompare(a.date));\nconst top = items.slice(0, 100);\n\nconst counts = {\n total: top.length,\n gitea: top.filter((i) => i.platform === 'gitea').length,\n instagram: top.filter((i) => i.platform === 'instagram').length,\n};\n\nreturn [\n {\n json: {\n generatedAt: new Date().toISOString(),\n items: top,\n counts,\n },\n },\n];"
|
||||||
|
},
|
||||||
|
"id": "normalise",
|
||||||
|
"name": "Normalise",
|
||||||
|
"type": "n8n-nodes-base.code",
|
||||||
|
"typeVersion": 2,
|
||||||
|
"position": [720, 320]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"operation": "toJson",
|
||||||
|
"fieldName": "data",
|
||||||
|
"options": {
|
||||||
|
"format": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "to-json",
|
||||||
|
"name": "To-json-string",
|
||||||
|
"type": "n8n-nodes-base.set",
|
||||||
|
"typeVersion": 3.4,
|
||||||
|
"position": [940, 320],
|
||||||
|
"notesInFlow": false,
|
||||||
|
"notes": "Transforme l'objet JS en string JSON pour Write Binary File"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"operation": "write",
|
||||||
|
"fileName": "/home/node/.n8n/journal/journal.json",
|
||||||
|
"dataPropertyName": "data",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "write-file",
|
||||||
|
"name": "Write-journal-json",
|
||||||
|
"type": "n8n-nodes-base.readWriteFile",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [1160, 320]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"connections": {
|
||||||
|
"Cron-3h": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{ "node": "Fetch-gitea", "type": "main", "index": 0 },
|
||||||
|
{ "node": "Fetch-behold-aep", "type": "main", "index": 0 },
|
||||||
|
{ "node": "Fetch-behold-julesneny", "type": "main", "index": 0 }
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Fetch-gitea": {
|
||||||
|
"main": [[{ "node": "Normalise", "type": "main", "index": 0 }]]
|
||||||
|
},
|
||||||
|
"Fetch-behold-aep": {
|
||||||
|
"main": [[{ "node": "Normalise", "type": "main", "index": 0 }]]
|
||||||
|
},
|
||||||
|
"Fetch-behold-julesneny": {
|
||||||
|
"main": [[{ "node": "Normalise", "type": "main", "index": 0 }]]
|
||||||
|
},
|
||||||
|
"Normalise": {
|
||||||
|
"main": [[{ "node": "To-json-string", "type": "main", "index": 0 }]]
|
||||||
|
},
|
||||||
|
"To-json-string": {
|
||||||
|
"main": [[{ "node": "Write-journal-json", "type": "main", "index": 0 }]]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"settings": {
|
||||||
|
"executionOrder": "v1",
|
||||||
|
"saveExecutionProgress": true,
|
||||||
|
"saveManualExecutions": true
|
||||||
|
},
|
||||||
|
"tags": [
|
||||||
|
{ "name": "page-cerveau" },
|
||||||
|
{ "name": "PC6" }
|
||||||
|
]
|
||||||
|
}
|
||||||
191
package-lock.json
generated
191
package-lock.json
generated
@@ -8,6 +8,7 @@
|
|||||||
"name": "astro-site-cerveau",
|
"name": "astro-site-cerveau",
|
||||||
"version": "0.0.1",
|
"version": "0.0.1",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@astrojs/node": "^10.1.0",
|
||||||
"@astrojs/vue": "^6.0.1",
|
"@astrojs/vue": "^6.0.1",
|
||||||
"@tailwindcss/vite": "^4.2.4",
|
"@tailwindcss/vite": "^4.2.4",
|
||||||
"@types/d3": "^7.4.3",
|
"@types/d3": "^7.4.3",
|
||||||
@@ -67,6 +68,20 @@
|
|||||||
"vfile": "^6.0.3"
|
"vfile": "^6.0.3"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@astrojs/node": {
|
||||||
|
"version": "10.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@astrojs/node/-/node-10.1.0.tgz",
|
||||||
|
"integrity": "sha512-4/2oqUTQ71UQ8+xX249T4l/d0/YkC5ssOVl4R2yQO7Wg4mOnvsq9Z9iaTkWAyElg3lqZq7XRNCEXCmDNiYcW1A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@astrojs/internal-helpers": "0.9.0",
|
||||||
|
"send": "^1.2.1",
|
||||||
|
"server-destroy": "^1.0.1"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"astro": "^6.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@astrojs/prism": {
|
"node_modules/@astrojs/prism": {
|
||||||
"version": "4.0.1",
|
"version": "4.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/@astrojs/prism/-/prism-4.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/@astrojs/prism/-/prism-4.0.1.tgz",
|
||||||
@@ -3853,6 +3868,15 @@
|
|||||||
"robust-predicates": "^3.0.2"
|
"robust-predicates": "^3.0.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/depd": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/dequal": {
|
"node_modules/dequal": {
|
||||||
"version": "2.0.3",
|
"version": "2.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz",
|
||||||
@@ -3981,6 +4005,12 @@
|
|||||||
"node": ">=4"
|
"node": ">=4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/ee-first": {
|
||||||
|
"version": "1.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
|
||||||
|
"integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/electron-to-chromium": {
|
"node_modules/electron-to-chromium": {
|
||||||
"version": "1.5.352",
|
"version": "1.5.352",
|
||||||
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.352.tgz",
|
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.352.tgz",
|
||||||
@@ -4015,6 +4045,15 @@
|
|||||||
"vue": "^3.2.37"
|
"vue": "^3.2.37"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/encodeurl": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/enhanced-resolve": {
|
"node_modules/enhanced-resolve": {
|
||||||
"version": "5.21.2",
|
"version": "5.21.2",
|
||||||
"resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.21.2.tgz",
|
"resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.21.2.tgz",
|
||||||
@@ -4105,6 +4144,12 @@
|
|||||||
"node": ">=6"
|
"node": ">=6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/escape-html": {
|
||||||
|
"version": "1.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
|
||||||
|
"integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/escape-string-regexp": {
|
"node_modules/escape-string-regexp": {
|
||||||
"version": "5.0.0",
|
"version": "5.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-5.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-5.0.0.tgz",
|
||||||
@@ -4136,6 +4181,15 @@
|
|||||||
"integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==",
|
"integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/etag": {
|
||||||
|
"version": "1.8.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
|
||||||
|
"integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/eventemitter3": {
|
"node_modules/eventemitter3": {
|
||||||
"version": "5.0.4",
|
"version": "5.0.4",
|
||||||
"resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-5.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-5.0.4.tgz",
|
||||||
@@ -4268,6 +4322,15 @@
|
|||||||
"node": ">=20"
|
"node": ">=20"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/fresh": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/fsevents": {
|
"node_modules/fsevents": {
|
||||||
"version": "2.3.3",
|
"version": "2.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
|
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
|
||||||
@@ -4609,6 +4672,26 @@
|
|||||||
"integrity": "sha512-dTxcvPXqPvXBQpq5dUr6mEMJX4oIEFv6bwom3FDwKRDsuIjjJGANqhBuoAn9c1RQJIdAKav33ED65E2ys+87QQ==",
|
"integrity": "sha512-dTxcvPXqPvXBQpq5dUr6mEMJX4oIEFv6bwom3FDwKRDsuIjjJGANqhBuoAn9c1RQJIdAKav33ED65E2ys+87QQ==",
|
||||||
"license": "BSD-2-Clause"
|
"license": "BSD-2-Clause"
|
||||||
},
|
},
|
||||||
|
"node_modules/http-errors": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"depd": "~2.0.0",
|
||||||
|
"inherits": "~2.0.4",
|
||||||
|
"setprototypeof": "~1.2.0",
|
||||||
|
"statuses": "~2.0.2",
|
||||||
|
"toidentifier": "~1.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/express"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/iconv-lite": {
|
"node_modules/iconv-lite": {
|
||||||
"version": "0.6.3",
|
"version": "0.6.3",
|
||||||
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
|
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
|
||||||
@@ -4630,6 +4713,12 @@
|
|||||||
"node": ">= 4"
|
"node": ">= 4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/inherits": {
|
||||||
|
"version": "2.0.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
|
||||||
|
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
"node_modules/internmap": {
|
"node_modules/internmap": {
|
||||||
"version": "2.0.3",
|
"version": "2.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
|
||||||
@@ -5972,6 +6061,31 @@
|
|||||||
"url": "https://github.com/sponsors/jonschlinkert"
|
"url": "https://github.com/sponsors/jonschlinkert"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/mime-db": {
|
||||||
|
"version": "1.54.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz",
|
||||||
|
"integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/mime-types": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"mime-db": "^1.54.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/express"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/mrmime": {
|
"node_modules/mrmime": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.1.tgz",
|
||||||
@@ -6093,6 +6207,18 @@
|
|||||||
"integrity": "sha512-RdR9FQrFwNBNXAr4GixM8YaRZRJ5PUWbKYbE5eOsrwAjJW0q2REGcf79oYPsLyskQCZG1PLN+S/K1V00joZAoQ==",
|
"integrity": "sha512-RdR9FQrFwNBNXAr4GixM8YaRZRJ5PUWbKYbE5eOsrwAjJW0q2REGcf79oYPsLyskQCZG1PLN+S/K1V00joZAoQ==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/on-finished": {
|
||||||
|
"version": "2.4.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
|
||||||
|
"integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"ee-first": "1.1.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/oniguruma-parser": {
|
"node_modules/oniguruma-parser": {
|
||||||
"version": "0.12.2",
|
"version": "0.12.2",
|
||||||
"resolved": "https://registry.npmjs.org/oniguruma-parser/-/oniguruma-parser-0.12.2.tgz",
|
"resolved": "https://registry.npmjs.org/oniguruma-parser/-/oniguruma-parser-0.12.2.tgz",
|
||||||
@@ -6316,6 +6442,15 @@
|
|||||||
"integrity": "sha512-b484I/7b8rDEdSDKckSSBA8knMpcdsXudlE/LNL639wFoHKwLbEkQFZHWEYwDC0wa0FKUcCY+GAF73Z7wxNVFA==",
|
"integrity": "sha512-b484I/7b8rDEdSDKckSSBA8knMpcdsXudlE/LNL639wFoHKwLbEkQFZHWEYwDC0wa0FKUcCY+GAF73Z7wxNVFA==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/range-parser": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
|
||||||
|
"integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/readdirp": {
|
"node_modules/readdirp": {
|
||||||
"version": "5.0.0",
|
"version": "5.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/readdirp/-/readdirp-5.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/readdirp/-/readdirp-5.0.0.tgz",
|
||||||
@@ -6712,6 +6847,44 @@
|
|||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/send": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/send/-/send-1.2.1.tgz",
|
||||||
|
"integrity": "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"debug": "^4.4.3",
|
||||||
|
"encodeurl": "^2.0.0",
|
||||||
|
"escape-html": "^1.0.3",
|
||||||
|
"etag": "^1.8.1",
|
||||||
|
"fresh": "^2.0.0",
|
||||||
|
"http-errors": "^2.0.1",
|
||||||
|
"mime-types": "^3.0.2",
|
||||||
|
"ms": "^2.1.3",
|
||||||
|
"on-finished": "^2.4.1",
|
||||||
|
"range-parser": "^1.2.1",
|
||||||
|
"statuses": "^2.0.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 18"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"type": "opencollective",
|
||||||
|
"url": "https://opencollective.com/express"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/server-destroy": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/server-destroy/-/server-destroy-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-rb+9B5YBIEzYcD6x2VKidaa+cqYBJQKnU4oe4E3ANwRRN56yk/ua1YCJT1n21NTS8w6CcOclAKNP3PhdCXKYtQ==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/setprototypeof": {
|
||||||
|
"version": "1.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
|
||||||
|
"integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
"node_modules/sharp": {
|
"node_modules/sharp": {
|
||||||
"version": "0.34.5",
|
"version": "0.34.5",
|
||||||
"resolved": "https://registry.npmjs.org/sharp/-/sharp-0.34.5.tgz",
|
"resolved": "https://registry.npmjs.org/sharp/-/sharp-0.34.5.tgz",
|
||||||
@@ -6845,6 +7018,15 @@
|
|||||||
"integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==",
|
"integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==",
|
||||||
"license": "BSD-3-Clause"
|
"license": "BSD-3-Clause"
|
||||||
},
|
},
|
||||||
|
"node_modules/statuses": {
|
||||||
|
"version": "2.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz",
|
||||||
|
"integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/stringify-entities": {
|
"node_modules/stringify-entities": {
|
||||||
"version": "4.0.4",
|
"version": "4.0.4",
|
||||||
"resolved": "https://registry.npmjs.org/stringify-entities/-/stringify-entities-4.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/stringify-entities/-/stringify-entities-4.0.4.tgz",
|
||||||
@@ -6964,6 +7146,15 @@
|
|||||||
"node": ">=8.0"
|
"node": ">=8.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/toidentifier": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/totalist": {
|
"node_modules/totalist": {
|
||||||
"version": "3.0.1",
|
"version": "3.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz",
|
||||||
|
|||||||
@@ -15,6 +15,7 @@
|
|||||||
"build:carte-o": "node scripts/build-carte-o.js"
|
"build:carte-o": "node scripts/build-carte-o.js"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@astrojs/node": "^10.1.0",
|
||||||
"@astrojs/vue": "^6.0.1",
|
"@astrojs/vue": "^6.0.1",
|
||||||
"@tailwindcss/vite": "^4.2.4",
|
"@tailwindcss/vite": "^4.2.4",
|
||||||
"@types/d3": "^7.4.3",
|
"@types/d3": "^7.4.3",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"meta": {
|
"meta": {
|
||||||
"generated": "2026-05-08T22:56:53.553Z",
|
"generated": "2026-05-08T23:20:14.828Z",
|
||||||
"source": "AEP/Articles",
|
"source": "AEP/Articles",
|
||||||
"nodeCount": 84,
|
"nodeCount": 84,
|
||||||
"edgeCount": 94,
|
"edgeCount": 94,
|
||||||
@@ -25,19 +25,19 @@
|
|||||||
"TR - renovation-energetique - website pro.md": 1,
|
"TR - renovation-energetique - website pro.md": 1,
|
||||||
"AEP ARTICLES, BROUILLON": 6,
|
"AEP ARTICLES, BROUILLON": 6,
|
||||||
"Livre - le nouveau contrat social": 2,
|
"Livre - le nouveau contrat social": 2,
|
||||||
|
"AEP agriculture": 1,
|
||||||
"AEP archi": 4,
|
"AEP archi": 4,
|
||||||
"AEP déconstruction": 9,
|
"AEP déconstruction": 9,
|
||||||
"AEP agriculture": 1,
|
|
||||||
"AEP géopolitique": 3,
|
|
||||||
"AEP education": 4,
|
"AEP education": 4,
|
||||||
"AEP justice": 3,
|
|
||||||
"AEP IA": 14,
|
|
||||||
"AEP nouveaux récits": 9,
|
|
||||||
"AEP histoire": 7,
|
"AEP histoire": 7,
|
||||||
|
"AEP IA": 14,
|
||||||
|
"AEP géopolitique": 3,
|
||||||
|
"AEP justice": 3,
|
||||||
"AEP piraterie (script formation)": 2,
|
"AEP piraterie (script formation)": 2,
|
||||||
|
"AEP nouveaux récits": 9,
|
||||||
"AEP politique": 3,
|
"AEP politique": 3,
|
||||||
"AEP santé": 6,
|
|
||||||
"AEP regénération": 3,
|
"AEP regénération": 3,
|
||||||
|
"AEP santé": 6,
|
||||||
"AEP spiritualité": 1,
|
"AEP spiritualité": 1,
|
||||||
"AEP système économique": 4
|
"AEP système économique": 4
|
||||||
}
|
}
|
||||||
@@ -142,6 +142,15 @@
|
|||||||
"theme": "Livre - le nouveau contrat social",
|
"theme": "Livre - le nouveau contrat social",
|
||||||
"path": "Livre - le nouveau contrat social/AEP - article fiscalite contrat social.md"
|
"path": "Livre - le nouveau contrat social/AEP - article fiscalite contrat social.md"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-agriculture",
|
||||||
|
"label": "AEP Agriculture",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "Contrat social : les agriculteurs, àvalorises par autre chose que les produits qu’il créent. Besoin que ça soit le moins cher possible",
|
||||||
|
"slug": "aep-agriculture",
|
||||||
|
"theme": "AEP agriculture",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP agriculture/AEP Agriculture.md"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"id": "aep-beton-critique",
|
"id": "aep-beton-critique",
|
||||||
"label": "AEP Béton - critique",
|
"label": "AEP Béton - critique",
|
||||||
@@ -223,42 +232,6 @@
|
|||||||
"theme": "AEP déconstruction",
|
"theme": "AEP déconstruction",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP déconstruction/AEP humour - manifeste brouillon.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP déconstruction/AEP humour - manifeste brouillon.md"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"id": "aep-agriculture",
|
|
||||||
"label": "AEP Agriculture",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "Contrat social : les agriculteurs, àvalorises par autre chose que les produits qu’il créent. Besoin que ça soit le moins cher possible",
|
|
||||||
"slug": "aep-agriculture",
|
|
||||||
"theme": "AEP agriculture",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP agriculture/AEP Agriculture.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "2025-07-05-juan-branco-thinker-view-ep-2-30-06-2025",
|
|
||||||
"label": "2025-07-05 Juan branco - thinker view - ep 2 30-06-2025",
|
|
||||||
"family": "penseur",
|
|
||||||
"intention": "Juan branco - thinker view - ep 2, 30/06/2025",
|
|
||||||
"slug": "2025-07-05-juan-branco-thinker-view-ep-2-30-06-2025",
|
|
||||||
"theme": "AEP géopolitique",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/2025-07-05 Juan branco - thinker view - ep 2 30-06-2025.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-israel-palestine-conflit-brut",
|
|
||||||
"label": "AEP Israel - Palestine, conflit (brut)",
|
|
||||||
"family": "penseur",
|
|
||||||
"intention": "Depuis l’attaque du 7 octobre 2023 par le Hamas, qui a fait environ 1 200 morts israéliens, l'État d’Israël, dirigé par Benyamin Netanyahou, a lancé une campagne militaire d’une intensité sans précédent contre la bande d",
|
|
||||||
"slug": "aep-israel-palestine-conflit-brut",
|
|
||||||
"theme": "AEP géopolitique",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/AEP Israel - Palestine, conflit (brut).md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-politique-energie-fr",
|
|
||||||
"label": "AEP Politique Énergie FR",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "MOC : Architecture technique d'ATISPOLITIQUETRANSMETTRE Source : Discussion Álvaro Projet: PFE - Infolettre AEP! AEP émerger une nouvelle pratiqueAEP Politique Énergie FR Tags : #contenu/infolettre #contenu/manifeste #co",
|
|
||||||
"slug": "aep-politique-energie-fr",
|
|
||||||
"theme": "AEP géopolitique",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/AEP Politique Énergie FR.md"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"id": "aep-cooperation",
|
"id": "aep-cooperation",
|
||||||
"label": "AEP Coopération",
|
"label": "AEP Coopération",
|
||||||
@@ -295,123 +268,6 @@
|
|||||||
"theme": "AEP education",
|
"theme": "AEP education",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP education/Mémorisation - discussion train.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP education/Mémorisation - discussion train.md"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"id": "aep-justice-securite",
|
|
||||||
"label": "AEP justice & sécurité",
|
|
||||||
"family": "penseur",
|
|
||||||
"intention": "MOC : POLITIQUE-AEP 1 Source : divers Projets : AEP articles, idées en gestation-AEP tableau des articles Tags : #contenu/article Date : 2025-10-31 Justice/droit Juan Branco réformes structurelles https://www.youtube.com",
|
|
||||||
"slug": "aep-justice-securite",
|
|
||||||
"theme": "AEP justice",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP justice/AEP justice & sécurité.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "crepuscule-juan-branco-itw",
|
|
||||||
"label": "Crépuscule - Juan branco - ITW",
|
|
||||||
"family": "penseur",
|
|
||||||
"intention": "Crépuscule - Juan branco - podcast - thinkerview",
|
|
||||||
"slug": "crepuscule-juan-branco-itw",
|
|
||||||
"theme": "AEP justice",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP justice/Crépuscule - Juan branco - ITW.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "narcotrafic",
|
|
||||||
"label": "Narcotrafic",
|
|
||||||
"family": "collectif",
|
|
||||||
"intention": "Ces frontières sont valorisées dans les médias par la dramaturgie de postures martiales mettant en valeur des politiciens, doublées de menaces a l'ordre social, et a la stigmatisation de groupes sociaux (racailles, bande",
|
|
||||||
"slug": "narcotrafic",
|
|
||||||
"theme": "AEP justice",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP justice/Narcotrafic.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "2026-03-11-aep-ia-usage-sage-intention",
|
|
||||||
"label": "2026-03-11 AEP IA usage sage intention",
|
|
||||||
"family": "ressource",
|
|
||||||
"intention": "MOC : Transcriptions Projets : AEP - IA - usage sage Date : 2026-03-11 Durée : 9m 18s · groq récap de la conversation sur l'IA super cool qu'on a eue avec Issa, envie de poser ça comme base de la future formation IA de N",
|
|
||||||
"slug": "2026-03-11-aep-ia-usage-sage-intention",
|
|
||||||
"theme": "AEP IA",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP IA/2026-03-11 AEP IA usage sage intention.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-ia-usage-sage",
|
|
||||||
"label": "AEP - IA - usage sage",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "MOC : ORGANISATION-PFE - Infolettre AEP Source : Ines Lee [\"oops i become codependent on IA\"](https://ineslee.substack.com/p/ai-codependence) + Tyler Austin Harper, The Atlantic, \"[what happens when people don't undersan",
|
|
||||||
"slug": "aep-ia-usage-sage",
|
|
||||||
"theme": "AEP IA",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP IA/AEP - IA - usage sage.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "anthropic-revolution-ia-2026",
|
|
||||||
"label": "Anthropic, révolution IA 2026",
|
|
||||||
"family": "concept",
|
|
||||||
"intention": "révolution d'anthropic via cowork, claude code & opus.",
|
|
||||||
"slug": "anthropic-revolution-ia-2026",
|
|
||||||
"theme": "AEP IA",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP IA/Anthropic, révolution IA 2026.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "ia-usage-sage-sven-et-lucie",
|
|
||||||
"label": "IA usage sage Sven et Lucie",
|
|
||||||
"family": "ressource",
|
|
||||||
"intention": "Voici la retranscription d'un article que j'aimerais écrire sur l'usage de l'IA. Et c'était un rendez-vous la semaine dernière avec Zden et Lucie, qui nous a permis pendant deux heures de traverser des sujets importants ",
|
|
||||||
"slug": "ia-usage-sage-sven-et-lucie",
|
|
||||||
"theme": "AEP IA",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP IA/IA usage sage Sven et Lucie.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "manifeste-anti-iag",
|
|
||||||
"label": "Manifeste anti-IAg",
|
|
||||||
"family": "ressource",
|
|
||||||
"intention": "[Vous pouvez signer ce manifeste ([version pdf ici](https://atecopol.hypotheses.org/files/2025/12/ManifesteObjectionConscienceIAg.pdf)) dans [ce formulaire](https://framaforms.org/face-a-lia-generative-lobjection-de-cons",
|
|
||||||
"slug": "manifeste-anti-iag",
|
|
||||||
"theme": "AEP IA",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP IA/Manifeste anti-IAg.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-culture-dominante-analyse-series",
|
|
||||||
"label": "AEP culture dominante (analyse séries)",
|
|
||||||
"family": "concept",
|
|
||||||
"intention": "\"grands récits\" - star wars - le seigneur des anneaux - dune !Post Dune- avis critique.m4a - harry potter - eragon (le livre plus que le film) - one piece - naruto - Arcane : !Arcanes feedback série.m4a = parallèle avec ",
|
|
||||||
"slug": "aep-culture-dominante-analyse-series",
|
|
||||||
"theme": "AEP nouveaux récits",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP culture dominante (analyse séries).md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-medias-la-pensee-critique",
|
|
||||||
"label": "AEP médias & la pensée critique",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "MOC : POLITIQUE-PFE - Infolettre AEP Source : https://mythodologie.fr/ Projets : NAAV - apprendre a apprendre-cours de transformation urbaine- Tags : Date : 2025-06-19 une note expliquant les bases de l'intelligence crit",
|
|
||||||
"slug": "aep-medias-la-pensee-critique",
|
|
||||||
"theme": "AEP nouveaux récits",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP médias & la pensée critique.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-prospective",
|
|
||||||
"label": "AEP prospective",
|
|
||||||
"family": "penseur",
|
|
||||||
"intention": "En lien avec Tout pour tout le monde - ITW - prospective",
|
|
||||||
"slug": "aep-prospective",
|
|
||||||
"theme": "AEP nouveaux récits",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP prospective.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-ecrire-de-nouveaux-recits",
|
|
||||||
"label": "AEP écrire de nouveaux récits",
|
|
||||||
"family": "ressource",
|
|
||||||
"intention": "MOC : POLITIQUETRANSMETTRE- ART-FACILITATION Projets : Stratégie création de contenu PFE - Infolettre AEP AEP éducation-L'art du récit (storytelling) Tags : #contenu/article Date : 2024-03-23 # Projets d'écriture 1. LIVR",
|
|
||||||
"slug": "aep-ecrire-de-nouveaux-recits",
|
|
||||||
"theme": "AEP nouveaux récits",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP écrire de nouveaux récits.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "archetypes-eneagramme",
|
|
||||||
"label": "Archétypes - énéagramme",
|
|
||||||
"family": "collectif",
|
|
||||||
"intention": "MOC : LITTERATURE-écrivain-ART-SPIRITUALITE Source : Annif Flo 44 ans - orga Projets : AEP écrire de nouveaux récits-LIVRE - le nouveau contrat social Tags : Date : 2025-10-31 note sur l'étude des personnages d'un récit/",
|
|
||||||
"slug": "archetypes-eneagramme",
|
|
||||||
"theme": "AEP nouveaux récits",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/Archétypes - énéagramme.md"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"id": "aep-histoire",
|
"id": "aep-histoire",
|
||||||
"label": "AEP Histoire",
|
"label": "AEP Histoire",
|
||||||
@@ -475,6 +331,105 @@
|
|||||||
"theme": "AEP histoire",
|
"theme": "AEP histoire",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP histoire/L’empire n’a jamais pris fin 6 -.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP histoire/L’empire n’a jamais pris fin 6 -.md"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"id": "2026-03-11-aep-ia-usage-sage-intention",
|
||||||
|
"label": "2026-03-11 AEP IA usage sage intention",
|
||||||
|
"family": "ressource",
|
||||||
|
"intention": "MOC : Transcriptions Projets : AEP - IA - usage sage Date : 2026-03-11 Durée : 9m 18s · groq récap de la conversation sur l'IA super cool qu'on a eue avec Issa, envie de poser ça comme base de la future formation IA de N",
|
||||||
|
"slug": "2026-03-11-aep-ia-usage-sage-intention",
|
||||||
|
"theme": "AEP IA",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP IA/2026-03-11 AEP IA usage sage intention.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-ia-usage-sage",
|
||||||
|
"label": "AEP - IA - usage sage",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "MOC : ORGANISATION-PFE - Infolettre AEP Source : Ines Lee [\"oops i become codependent on IA\"](https://ineslee.substack.com/p/ai-codependence) + Tyler Austin Harper, The Atlantic, \"[what happens when people don't undersan",
|
||||||
|
"slug": "aep-ia-usage-sage",
|
||||||
|
"theme": "AEP IA",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP IA/AEP - IA - usage sage.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "anthropic-revolution-ia-2026",
|
||||||
|
"label": "Anthropic, révolution IA 2026",
|
||||||
|
"family": "concept",
|
||||||
|
"intention": "révolution d'anthropic via cowork, claude code & opus.",
|
||||||
|
"slug": "anthropic-revolution-ia-2026",
|
||||||
|
"theme": "AEP IA",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP IA/Anthropic, révolution IA 2026.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "ia-usage-sage-sven-et-lucie",
|
||||||
|
"label": "IA usage sage Sven et Lucie",
|
||||||
|
"family": "ressource",
|
||||||
|
"intention": "Voici la retranscription d'un article que j'aimerais écrire sur l'usage de l'IA. Et c'était un rendez-vous la semaine dernière avec Zden et Lucie, qui nous a permis pendant deux heures de traverser des sujets importants ",
|
||||||
|
"slug": "ia-usage-sage-sven-et-lucie",
|
||||||
|
"theme": "AEP IA",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP IA/IA usage sage Sven et Lucie.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "manifeste-anti-iag",
|
||||||
|
"label": "Manifeste anti-IAg",
|
||||||
|
"family": "ressource",
|
||||||
|
"intention": "[Vous pouvez signer ce manifeste ([version pdf ici](https://atecopol.hypotheses.org/files/2025/12/ManifesteObjectionConscienceIAg.pdf)) dans [ce formulaire](https://framaforms.org/face-a-lia-generative-lobjection-de-cons",
|
||||||
|
"slug": "manifeste-anti-iag",
|
||||||
|
"theme": "AEP IA",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP IA/Manifeste anti-IAg.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "2025-07-05-juan-branco-thinker-view-ep-2-30-06-2025",
|
||||||
|
"label": "2025-07-05 Juan branco - thinker view - ep 2 30-06-2025",
|
||||||
|
"family": "penseur",
|
||||||
|
"intention": "Juan branco - thinker view - ep 2, 30/06/2025",
|
||||||
|
"slug": "2025-07-05-juan-branco-thinker-view-ep-2-30-06-2025",
|
||||||
|
"theme": "AEP géopolitique",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/2025-07-05 Juan branco - thinker view - ep 2 30-06-2025.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-israel-palestine-conflit-brut",
|
||||||
|
"label": "AEP Israel - Palestine, conflit (brut)",
|
||||||
|
"family": "penseur",
|
||||||
|
"intention": "Depuis l’attaque du 7 octobre 2023 par le Hamas, qui a fait environ 1 200 morts israéliens, l'État d’Israël, dirigé par Benyamin Netanyahou, a lancé une campagne militaire d’une intensité sans précédent contre la bande d",
|
||||||
|
"slug": "aep-israel-palestine-conflit-brut",
|
||||||
|
"theme": "AEP géopolitique",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/AEP Israel - Palestine, conflit (brut).md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-politique-energie-fr",
|
||||||
|
"label": "AEP Politique Énergie FR",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "MOC : Architecture technique d'ATISPOLITIQUETRANSMETTRE Source : Discussion Álvaro Projet: PFE - Infolettre AEP! AEP émerger une nouvelle pratiqueAEP Politique Énergie FR Tags : #contenu/infolettre #contenu/manifeste #co",
|
||||||
|
"slug": "aep-politique-energie-fr",
|
||||||
|
"theme": "AEP géopolitique",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP géopolitique/AEP Politique Énergie FR.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-justice-securite",
|
||||||
|
"label": "AEP justice & sécurité",
|
||||||
|
"family": "penseur",
|
||||||
|
"intention": "MOC : POLITIQUE-AEP 1 Source : divers Projets : AEP articles, idées en gestation-AEP tableau des articles Tags : #contenu/article Date : 2025-10-31 Justice/droit Juan Branco réformes structurelles https://www.youtube.com",
|
||||||
|
"slug": "aep-justice-securite",
|
||||||
|
"theme": "AEP justice",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP justice/AEP justice & sécurité.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "crepuscule-juan-branco-itw",
|
||||||
|
"label": "Crépuscule - Juan branco - ITW",
|
||||||
|
"family": "penseur",
|
||||||
|
"intention": "Crépuscule - Juan branco - podcast - thinkerview",
|
||||||
|
"slug": "crepuscule-juan-branco-itw",
|
||||||
|
"theme": "AEP justice",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP justice/Crépuscule - Juan branco - ITW.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "narcotrafic",
|
||||||
|
"label": "Narcotrafic",
|
||||||
|
"family": "collectif",
|
||||||
|
"intention": "Ces frontières sont valorisées dans les médias par la dramaturgie de postures martiales mettant en valeur des politiciens, doublées de menaces a l'ordre social, et a la stigmatisation de groupes sociaux (racailles, bande",
|
||||||
|
"slug": "narcotrafic",
|
||||||
|
"theme": "AEP justice",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP justice/Narcotrafic.md"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"id": "creation-de-script-template",
|
"id": "creation-de-script-template",
|
||||||
"label": "Création de script - template",
|
"label": "Création de script - template",
|
||||||
@@ -493,6 +448,51 @@
|
|||||||
"theme": "AEP piraterie (script formation)",
|
"theme": "AEP piraterie (script formation)",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP piraterie (script formation)/methodologie_developpement.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP piraterie (script formation)/methodologie_developpement.md"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-culture-dominante-analyse-series",
|
||||||
|
"label": "AEP culture dominante (analyse séries)",
|
||||||
|
"family": "concept",
|
||||||
|
"intention": "\"grands récits\" - star wars - le seigneur des anneaux - dune !Post Dune- avis critique.m4a - harry potter - eragon (le livre plus que le film) - one piece - naruto - Arcane : !Arcanes feedback série.m4a = parallèle avec ",
|
||||||
|
"slug": "aep-culture-dominante-analyse-series",
|
||||||
|
"theme": "AEP nouveaux récits",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP culture dominante (analyse séries).md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-medias-la-pensee-critique",
|
||||||
|
"label": "AEP médias & la pensée critique",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "MOC : POLITIQUE-PFE - Infolettre AEP Source : https://mythodologie.fr/ Projets : NAAV - apprendre a apprendre-cours de transformation urbaine- Tags : Date : 2025-06-19 une note expliquant les bases de l'intelligence crit",
|
||||||
|
"slug": "aep-medias-la-pensee-critique",
|
||||||
|
"theme": "AEP nouveaux récits",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP médias & la pensée critique.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-prospective",
|
||||||
|
"label": "AEP prospective",
|
||||||
|
"family": "penseur",
|
||||||
|
"intention": "En lien avec Tout pour tout le monde - ITW - prospective",
|
||||||
|
"slug": "aep-prospective",
|
||||||
|
"theme": "AEP nouveaux récits",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP prospective.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-ecrire-de-nouveaux-recits",
|
||||||
|
"label": "AEP écrire de nouveaux récits",
|
||||||
|
"family": "ressource",
|
||||||
|
"intention": "MOC : POLITIQUETRANSMETTRE- ART-FACILITATION Projets : Stratégie création de contenu PFE - Infolettre AEP AEP éducation-L'art du récit (storytelling) Tags : #contenu/article Date : 2024-03-23 # Projets d'écriture 1. LIVR",
|
||||||
|
"slug": "aep-ecrire-de-nouveaux-recits",
|
||||||
|
"theme": "AEP nouveaux récits",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/AEP écrire de nouveaux récits.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "archetypes-eneagramme",
|
||||||
|
"label": "Archétypes - énéagramme",
|
||||||
|
"family": "collectif",
|
||||||
|
"intention": "MOC : LITTERATURE-écrivain-ART-SPIRITUALITE Source : Annif Flo 44 ans - orga Projets : AEP écrire de nouveaux récits-LIVRE - le nouveau contrat social Tags : Date : 2025-10-31 note sur l'étude des personnages d'un récit/",
|
||||||
|
"slug": "archetypes-eneagramme",
|
||||||
|
"theme": "AEP nouveaux récits",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP nouveaux récits/Archétypes - énéagramme.md"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"id": "5-chantiers-mamdani-nyc",
|
"id": "5-chantiers-mamdani-nyc",
|
||||||
"label": "5 chantiers Mamdani NYC",
|
"label": "5 chantiers Mamdani NYC",
|
||||||
@@ -520,6 +520,33 @@
|
|||||||
"theme": "AEP politique",
|
"theme": "AEP politique",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP politique/AEP politique - état des lieux.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP politique/AEP politique - état des lieux.md"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-regen-fertival",
|
||||||
|
"label": "AEP - Regen - fertival",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "MOC : FACILITATION-Agroforesterie- Source : Sacha Gouttenoire Projets : Fertival 2025-PFE - Infolettre AEP Tags : Date : 2025-05-02 10.03.23 article phare sur la pratique de la regénération naturelle, une pratique illust",
|
||||||
|
"slug": "aep-regen-fertival",
|
||||||
|
"theme": "AEP regénération",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP regénération/! AEP - Regen - fertival.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-emerger-une-nouvelle-pratique",
|
||||||
|
"label": "AEP émerger une nouvelle pratique",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "MOC : POLITIQUEArchitecture technique d'ATIS Source : Projets : faire émerger une nouvelle pratique de l'architecture Tags : Date : 2024-11-11 projet principal d'énonciation d'une théorie de la transformation ! Structure",
|
||||||
|
"slug": "aep-emerger-une-nouvelle-pratique",
|
||||||
|
"theme": "AEP regénération",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP regénération/! AEP émerger une nouvelle pratique.md"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aep-mdcs-medecine-du-corps-social",
|
||||||
|
"label": "AEP mdcs - Médecine du corps social",
|
||||||
|
"family": "methode",
|
||||||
|
"intention": "Note capitalisme socialisme 3ème voie ? Communes, local briser pyramide colonialiste, troisième voie, rendre local l'indispensable, et global le spécifique.",
|
||||||
|
"slug": "aep-mdcs-medecine-du-corps-social",
|
||||||
|
"theme": "AEP regénération",
|
||||||
|
"path": "AEP ARTICLES, BROUILLON/AEP regénération/AEP mdcs - Médecine du corps social.md"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"id": "aep-fiscalite-contrat-social",
|
"id": "aep-fiscalite-contrat-social",
|
||||||
"label": "AEP fiscalité - contrat social",
|
"label": "AEP fiscalité - contrat social",
|
||||||
@@ -574,33 +601,6 @@
|
|||||||
"theme": "AEP santé",
|
"theme": "AEP santé",
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP santé/Qualité de vie EU vs USA - art. mindvalley.md"
|
"path": "AEP ARTICLES, BROUILLON/AEP santé/Qualité de vie EU vs USA - art. mindvalley.md"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"id": "aep-regen-fertival",
|
|
||||||
"label": "AEP - Regen - fertival",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "MOC : FACILITATION-Agroforesterie- Source : Sacha Gouttenoire Projets : Fertival 2025-PFE - Infolettre AEP Tags : Date : 2025-05-02 10.03.23 article phare sur la pratique de la regénération naturelle, une pratique illust",
|
|
||||||
"slug": "aep-regen-fertival",
|
|
||||||
"theme": "AEP regénération",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP regénération/! AEP - Regen - fertival.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-emerger-une-nouvelle-pratique",
|
|
||||||
"label": "AEP émerger une nouvelle pratique",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "MOC : POLITIQUEArchitecture technique d'ATIS Source : Projets : faire émerger une nouvelle pratique de l'architecture Tags : Date : 2024-11-11 projet principal d'énonciation d'une théorie de la transformation ! Structure",
|
|
||||||
"slug": "aep-emerger-une-nouvelle-pratique",
|
|
||||||
"theme": "AEP regénération",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP regénération/! AEP émerger une nouvelle pratique.md"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": "aep-mdcs-medecine-du-corps-social",
|
|
||||||
"label": "AEP mdcs - Médecine du corps social",
|
|
||||||
"family": "methode",
|
|
||||||
"intention": "Note capitalisme socialisme 3ème voie ? Communes, local briser pyramide colonialiste, troisième voie, rendre local l'indispensable, et global le spécifique.",
|
|
||||||
"slug": "aep-mdcs-medecine-du-corps-social",
|
|
||||||
"theme": "AEP regénération",
|
|
||||||
"path": "AEP ARTICLES, BROUILLON/AEP regénération/AEP mdcs - Médecine du corps social.md"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"id": "aep-spiritualite",
|
"id": "aep-spiritualite",
|
||||||
"label": "AEP spiritualité",
|
"label": "AEP spiritualité",
|
||||||
@@ -1029,22 +1029,6 @@
|
|||||||
"source": "aep-dec-l-inconfort-la-solitude-crise-d-identite",
|
"source": "aep-dec-l-inconfort-la-solitude-crise-d-identite",
|
||||||
"target": "aep-rite-passage-age-adulte"
|
"target": "aep-rite-passage-age-adulte"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"source": "aep-israel-palestine-conflit-brut",
|
|
||||||
"target": "aep-ia-usage-sage"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-israel-palestine-conflit-brut",
|
|
||||||
"target": "crepuscule-juan-branco-itw"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-israel-palestine-conflit-brut",
|
|
||||||
"target": "aep-justice-securite"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-politique-energie-fr",
|
|
||||||
"target": "aep-emerger-une-nouvelle-pratique"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"source": "aep-education",
|
"source": "aep-education",
|
||||||
"target": "aep-ecrire-de-nouveaux-recits"
|
"target": "aep-ecrire-de-nouveaux-recits"
|
||||||
@@ -1053,46 +1037,6 @@
|
|||||||
"source": "aep-education",
|
"source": "aep-education",
|
||||||
"target": "aep-medias-la-pensee-critique"
|
"target": "aep-medias-la-pensee-critique"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"source": "crepuscule-juan-branco-itw",
|
|
||||||
"target": "aep-politique-etat-des-lieux"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "crepuscule-juan-branco-itw",
|
|
||||||
"target": "aep-medias-la-pensee-critique"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "2026-03-11-aep-ia-usage-sage-intention",
|
|
||||||
"target": "aep-ia-usage-sage"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-ia-usage-sage",
|
|
||||||
"target": "aep-medias-la-pensee-critique"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "manifeste-anti-iag",
|
|
||||||
"target": "aep-ia-usage-sage"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-culture-dominante-analyse-series",
|
|
||||||
"target": "aep-ecrire-de-nouveaux-recits"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-prospective",
|
|
||||||
"target": "aep-mdcs-medecine-du-corps-social"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-ecrire-de-nouveaux-recits",
|
|
||||||
"target": "l-empire-n-a-jamais-pris-fin-6"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "archetypes-eneagramme",
|
|
||||||
"target": "aep-ecrire-de-nouveaux-recits"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "archetypes-eneagramme",
|
|
||||||
"target": "livre-le-nouveau-contrat-social"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"source": "aep-histoire",
|
"source": "aep-histoire",
|
||||||
"target": "aep-ecrire-de-nouveaux-recits"
|
"target": "aep-ecrire-de-nouveaux-recits"
|
||||||
@@ -1121,6 +1065,70 @@
|
|||||||
"source": "aep-histoire",
|
"source": "aep-histoire",
|
||||||
"target": "l-empire-n-a-jamais-pris-fin-5-cathares"
|
"target": "l-empire-n-a-jamais-pris-fin-5-cathares"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"source": "2026-03-11-aep-ia-usage-sage-intention",
|
||||||
|
"target": "aep-ia-usage-sage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-ia-usage-sage",
|
||||||
|
"target": "aep-medias-la-pensee-critique"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "manifeste-anti-iag",
|
||||||
|
"target": "aep-ia-usage-sage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-israel-palestine-conflit-brut",
|
||||||
|
"target": "aep-ia-usage-sage"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-israel-palestine-conflit-brut",
|
||||||
|
"target": "crepuscule-juan-branco-itw"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-israel-palestine-conflit-brut",
|
||||||
|
"target": "aep-justice-securite"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-politique-energie-fr",
|
||||||
|
"target": "aep-emerger-une-nouvelle-pratique"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "crepuscule-juan-branco-itw",
|
||||||
|
"target": "aep-politique-etat-des-lieux"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "crepuscule-juan-branco-itw",
|
||||||
|
"target": "aep-medias-la-pensee-critique"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-culture-dominante-analyse-series",
|
||||||
|
"target": "aep-ecrire-de-nouveaux-recits"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-prospective",
|
||||||
|
"target": "aep-mdcs-medecine-du-corps-social"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-ecrire-de-nouveaux-recits",
|
||||||
|
"target": "l-empire-n-a-jamais-pris-fin-6"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "archetypes-eneagramme",
|
||||||
|
"target": "aep-ecrire-de-nouveaux-recits"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "archetypes-eneagramme",
|
||||||
|
"target": "livre-le-nouveau-contrat-social"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-emerger-une-nouvelle-pratique",
|
||||||
|
"target": "l-empire-n-a-jamais-pris-fin-6"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"source": "aep-mdcs-medecine-du-corps-social",
|
||||||
|
"target": "livre-le-nouveau-contrat-social"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"source": "aep-fiscalite-contrat-social",
|
"source": "aep-fiscalite-contrat-social",
|
||||||
"target": "livre-le-nouveau-contrat-social"
|
"target": "livre-le-nouveau-contrat-social"
|
||||||
@@ -1137,14 +1145,6 @@
|
|||||||
"source": "aep-performance-douce",
|
"source": "aep-performance-douce",
|
||||||
"target": "aep-sante-mentale-burn-out"
|
"target": "aep-sante-mentale-burn-out"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"source": "aep-emerger-une-nouvelle-pratique",
|
|
||||||
"target": "l-empire-n-a-jamais-pris-fin-6"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "aep-mdcs-medecine-du-corps-social",
|
|
||||||
"target": "livre-le-nouveau-contrat-social"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"source": "aep-spiritualite",
|
"source": "aep-spiritualite",
|
||||||
"target": "aep-histoire"
|
"target": "aep-histoire"
|
||||||
|
|||||||
52
public/data/journal.json
Normal file
52
public/data/journal.json
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
{
|
||||||
|
"generatedAt": "2026-05-09T01:00:00Z",
|
||||||
|
"fallback": true,
|
||||||
|
"note": "Mock local — agrégateur n8n live pousse sur data.trans-former.fr/journal.json. Ce fichier sert de fallback dev tant que data.trans-former.fr DNS/Caddy ne sont pas en place.",
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"id": "gitea-mock-pc6-feat",
|
||||||
|
"platform": "gitea",
|
||||||
|
"hashtag": "#stack",
|
||||||
|
"date": "2026-05-09T01:01:00Z",
|
||||||
|
"titre": "PC6 journal unifié + n8n agrégateur (mock)",
|
||||||
|
"extrait": "Composant JournalList Vue + workflow n8n cron 3h00. Sources V1 : Gitéa Atom + Behold @aep + Behold @julesneny.",
|
||||||
|
"url": "https://git.trans-former.fr/jules/astro-site-cerveau",
|
||||||
|
"thumbnail": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "gitea-mock-pc3-mindmap",
|
||||||
|
"platform": "gitea",
|
||||||
|
"hashtag": "#stack",
|
||||||
|
"date": "2026-05-09T00:59:41Z",
|
||||||
|
"titre": "PC3 mindmap Carte O (D3 force-directed)",
|
||||||
|
"extrait": "Scrape AEP/Articles + tabs centre HAUT.",
|
||||||
|
"url": "https://git.trans-former.fr/jules/astro-site-cerveau/commit/32bdc9a",
|
||||||
|
"thumbnail": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "insta-mock-aep-1",
|
||||||
|
"platform": "instagram",
|
||||||
|
"hashtag": "#aep-politique",
|
||||||
|
"date": "2026-05-07T18:30:00Z",
|
||||||
|
"titre": "Mock carrousel @aep.politique",
|
||||||
|
"extrait": "Placeholder carrousel manifeste écologie politique. Cron n8n live remplace ce mock par la vraie API Behold.",
|
||||||
|
"url": "https://instagram.com/aep.politique",
|
||||||
|
"thumbnail": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "insta-mock-julesneny-1",
|
||||||
|
"platform": "instagram",
|
||||||
|
"hashtag": "#peinture",
|
||||||
|
"date": "2026-05-05T14:00:00Z",
|
||||||
|
"titre": "Mock peinture @julesneny",
|
||||||
|
"extrait": "Placeholder art / poésie / Corse. Cron n8n live remplace ce mock par la vraie API Behold.",
|
||||||
|
"url": "https://instagram.com/julesneny",
|
||||||
|
"thumbnail": null
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"counts": {
|
||||||
|
"total": 4,
|
||||||
|
"gitea": 2,
|
||||||
|
"instagram": 2
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,10 @@
|
|||||||
---
|
---
|
||||||
// Centre - HAUT : tabs (Carte O mindmap | Chatbot RAG placeholder PC7).
|
// Centre - HAUT : tabs (Carte O mindmap | Chatbot RAG branche PC7).
|
||||||
// BAS : iframe carte AEP (PC4).
|
// BAS : iframe carte AEP + scroll articles Substack (PC4).
|
||||||
import CarteOWrapper from '../vue/CarteOWrapper.vue';
|
import CarteOWrapper from '../vue/CarteOWrapper.vue';
|
||||||
import ChatbotPlaceholder from '../vue/ChatbotPlaceholder.vue';
|
import ChatbotV2 from '../vue/ChatbotV2.vue';
|
||||||
|
import IframeCarteAEP from './IframeCarteAEP.astro';
|
||||||
|
import ScrollArticles from './ScrollArticles.astro';
|
||||||
---
|
---
|
||||||
<div class="h-full grid grid-rows-2 gap-2 p-2">
|
<div class="h-full grid grid-rows-2 gap-2 p-2">
|
||||||
<!-- HAUT 50% : tabs Carte O / Chatbot -->
|
<!-- HAUT 50% : tabs Carte O / Chatbot -->
|
||||||
@@ -49,14 +51,17 @@ import ChatbotPlaceholder from '../vue/ChatbotPlaceholder.vue';
|
|||||||
data-tab-panel="chatbot"
|
data-tab-panel="chatbot"
|
||||||
class="absolute inset-0 hidden"
|
class="absolute inset-0 hidden"
|
||||||
>
|
>
|
||||||
<ChatbotPlaceholder client:visible />
|
<ChatbotV2 client:visible />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<!-- BAS 50% : iframe carte AEP (PC4) -->
|
<!-- BAS 50% : iframe carte AEP + scroll articles Substack (PC4) -->
|
||||||
<section class="border border-dashed border-neutral-300 rounded flex items-center justify-center">
|
<section class="border border-neutral-200 rounded overflow-y-auto bg-white">
|
||||||
<p class="text-sm text-neutral-400">Iframe carte AEP — PC4</p>
|
<div class="h-full min-h-[60vh] md:min-h-[400px]">
|
||||||
|
<IframeCarteAEP />
|
||||||
|
</div>
|
||||||
|
<ScrollArticles />
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,24 @@
|
|||||||
---
|
---
|
||||||
// Placeholder Insta : 2 carrousels @aep + @julesneny — PC5 oEmbed
|
import InstaFeed from '../vue/InstaFeed.vue';
|
||||||
|
|
||||||
|
// Feed IDs Behold a remplir apres inscription Behold (voir docs/BEHOLD-SETUP.md)
|
||||||
|
const FEED_AEP = import.meta.env.PUBLIC_BEHOLD_AEP || 'PLACEHOLDER_AEP';
|
||||||
|
const FEED_JULESNENY = import.meta.env.PUBLIC_BEHOLD_JULESNENY || 'PLACEHOLDER_JULESNENY';
|
||||||
---
|
---
|
||||||
<section class="h-full p-4 flex flex-col gap-4">
|
<div class="h-full overflow-y-auto">
|
||||||
<h2 class="text-lg font-semibold text-neutral-700">Insta</h2>
|
<InstaFeed
|
||||||
<div class="border border-dashed border-neutral-300 rounded p-3 flex-1 flex items-center justify-center">
|
client:visible
|
||||||
<p class="text-sm text-neutral-400">@aep carrousel — PC5</p>
|
feedId={FEED_AEP}
|
||||||
</div>
|
account="@aep.politique"
|
||||||
<div class="border border-dashed border-neutral-300 rounded p-3 flex-1 flex items-center justify-center">
|
accountUrl="https://www.instagram.com/aep.politique/"
|
||||||
<p class="text-sm text-neutral-400">@julesneny carrousel — PC5</p>
|
fallbackBio="Carrousels manifeste AEP ; pensee politique eco-architecture"
|
||||||
</div>
|
/>
|
||||||
</section>
|
|
||||||
|
<InstaFeed
|
||||||
|
client:visible
|
||||||
|
feedId={FEED_JULESNENY}
|
||||||
|
account="@julesneny"
|
||||||
|
accountUrl="https://www.instagram.com/julesneny/"
|
||||||
|
fallbackBio="Peinture, poesie, Corse ; archives visuelles personnelles"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|||||||
@@ -1,14 +1,103 @@
|
|||||||
---
|
---
|
||||||
// Placeholder Journal — PC6 remplit avec entries chrono
|
// ColJournal - colonne gauche : CTA Manifeste + Hashtags accordeon + Journal (PC6)
|
||||||
|
// 7 hashtags = 7 plateformes (cf delta 15 du PILOTE-PC.md)
|
||||||
|
import JournalList from '../vue/JournalList.vue';
|
||||||
|
|
||||||
|
const hashtags = [
|
||||||
|
{ tag: '#manifeste', plateforme: 'Blog trans-former.fr', canal: 'ecriture longue' },
|
||||||
|
{ tag: '#building-public', plateforme: 'LinkedIn', canal: 'journal pro' },
|
||||||
|
{ tag: '#politique', plateforme: 'Substack', canal: 'pensee AEP' },
|
||||||
|
{ tag: '#aep-politique', plateforme: 'Insta @aep.politique', canal: 'carrousels manifeste' },
|
||||||
|
{ tag: '#peinture', plateforme: 'Insta @julesneny', canal: 'art / poesie / Corse' },
|
||||||
|
{ tag: '#podcast', plateforme: 'Castopod', canal: 'podcast.trans-former.fr' },
|
||||||
|
{ tag: '#stack', plateforme: 'GitHub', canal: 'open source' },
|
||||||
|
];
|
||||||
---
|
---
|
||||||
<section class="h-full p-4 flex flex-col gap-3">
|
<div class="h-full flex flex-col p-4 pt-20 md:pt-6 gap-5">
|
||||||
<h2 class="text-lg font-semibold text-neutral-700">Journal</h2>
|
<!-- CTA Manifeste -->
|
||||||
<ul class="flex flex-col gap-2 text-sm text-neutral-500">
|
<a
|
||||||
<li class="border border-dashed border-neutral-300 rounded p-2">Entry placeholder 1</li>
|
href="/manifeste"
|
||||||
<li class="border border-dashed border-neutral-300 rounded p-2">Entry placeholder 2</li>
|
class="block px-4 py-3 bg-neutral-900 text-white rounded-lg font-medium text-center hover:bg-neutral-700 transition-colors shadow-sm"
|
||||||
<li class="border border-dashed border-neutral-300 rounded p-2">Entry placeholder 3</li>
|
>
|
||||||
<li class="border border-dashed border-neutral-300 rounded p-2">Entry placeholder 4</li>
|
Lire le manifeste →
|
||||||
<li class="border border-dashed border-neutral-300 rounded p-2">Entry placeholder 5</li>
|
</a>
|
||||||
</ul>
|
|
||||||
<p class="text-xs text-neutral-400 mt-auto">Nav latérale + manifeste CTA — PC2</p>
|
<!-- Hashtags accordeon -->
|
||||||
</section>
|
<details id="hashtags-accordion" class="border-t border-neutral-200 pt-4">
|
||||||
|
<summary class="font-semibold cursor-pointer select-none flex items-center justify-between">
|
||||||
|
<span>Hashtags</span>
|
||||||
|
<span class="text-xs text-neutral-400 font-normal">7 plateformes</span>
|
||||||
|
</summary>
|
||||||
|
<ul class="mt-3 space-y-2 text-sm">
|
||||||
|
{hashtags.map(({ tag, plateforme, canal }) => (
|
||||||
|
<li>
|
||||||
|
<label class="flex items-start gap-2 cursor-pointer hover:bg-neutral-50 rounded p-1 -m-1 transition-colors">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
data-hashtag={tag}
|
||||||
|
class="mt-1 accent-neutral-900"
|
||||||
|
checked
|
||||||
|
/>
|
||||||
|
<span class="flex-1">
|
||||||
|
<span class="font-mono text-neutral-700 text-[13px]">{tag}</span>
|
||||||
|
<span class="block text-xs text-neutral-500 leading-snug">
|
||||||
|
{plateforme} ; {canal}
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</label>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<!-- Journal chrono (skeleton, slot rempli par PC6) -->
|
||||||
|
<section class="border-t border-neutral-200 pt-4 flex-1 overflow-y-auto">
|
||||||
|
<h2 class="font-semibold mb-3 flex items-center justify-between">
|
||||||
|
<span>Journal</span>
|
||||||
|
<span class="text-xs text-neutral-400 font-normal">chrono</span>
|
||||||
|
</h2>
|
||||||
|
<div id="journal-list" class="space-y-3 text-sm">
|
||||||
|
<JournalList client:visible />
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Hashtags accordeon : ferme par defaut mobile, ouvert desktop (>= 768px)
|
||||||
|
const accordion = document.getElementById('hashtags-accordion') as HTMLDetailsElement | null;
|
||||||
|
if (accordion) {
|
||||||
|
const mql = window.matchMedia('(min-width: 768px)');
|
||||||
|
const apply = () => {
|
||||||
|
accordion.open = mql.matches;
|
||||||
|
};
|
||||||
|
apply();
|
||||||
|
mql.addEventListener('change', apply);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Persistence filtres hashtags (localStorage)
|
||||||
|
const checkboxes = document.querySelectorAll<HTMLInputElement>('[data-hashtag]');
|
||||||
|
const STORAGE_KEY = 'tf-hashtag-filters';
|
||||||
|
let stored: Record<string, boolean> = {};
|
||||||
|
try {
|
||||||
|
stored = JSON.parse(localStorage.getItem(STORAGE_KEY) || '{}');
|
||||||
|
} catch {
|
||||||
|
stored = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
checkboxes.forEach((cb) => {
|
||||||
|
const tag = cb.dataset.hashtag;
|
||||||
|
if (!tag) return;
|
||||||
|
if (tag in stored) cb.checked = stored[tag];
|
||||||
|
cb.addEventListener('change', () => {
|
||||||
|
stored[tag] = cb.checked;
|
||||||
|
try {
|
||||||
|
localStorage.setItem(STORAGE_KEY, JSON.stringify(stored));
|
||||||
|
} catch {
|
||||||
|
// mode prive : on continue silencieusement
|
||||||
|
}
|
||||||
|
window.dispatchEvent(
|
||||||
|
new CustomEvent('hashtag-filter-change', { detail: { ...stored } })
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|||||||
@@ -1,12 +1,103 @@
|
|||||||
---
|
---
|
||||||
// Placeholder hamburger menu — PC2 ajoute liens nav
|
// HamburgerMenu - drawer slide-in left avec liens nav (PC2)
|
||||||
|
// Astro vanilla + script inline, pas besoin d'island Vue
|
||||||
---
|
---
|
||||||
<button
|
<button
|
||||||
|
id="hamburger-trigger"
|
||||||
type="button"
|
type="button"
|
||||||
class="fixed top-3 right-3 z-30 p-2 rounded bg-white/80 border border-neutral-200 shadow-sm"
|
class="fixed top-4 left-4 z-50 p-3 bg-white/95 border border-neutral-200 rounded-lg shadow-md hover:bg-white transition-colors md:top-6 md:left-6"
|
||||||
aria-label="Menu"
|
aria-label="Ouvrir le menu"
|
||||||
|
aria-expanded="false"
|
||||||
|
aria-controls="hamburger-drawer"
|
||||||
>
|
>
|
||||||
<span class="block w-5 h-0.5 bg-neutral-700 mb-1"></span>
|
<span class="block w-5 h-0.5 bg-neutral-800 mb-1"></span>
|
||||||
<span class="block w-5 h-0.5 bg-neutral-700 mb-1"></span>
|
<span class="block w-5 h-0.5 bg-neutral-800 mb-1"></span>
|
||||||
<span class="block w-5 h-0.5 bg-neutral-700"></span>
|
<span class="block w-5 h-0.5 bg-neutral-800"></span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
<div
|
||||||
|
id="hamburger-drawer"
|
||||||
|
class="fixed inset-0 z-40 hidden"
|
||||||
|
role="dialog"
|
||||||
|
aria-modal="true"
|
||||||
|
aria-label="Menu de navigation"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="absolute inset-0 bg-black/30 transition-opacity"
|
||||||
|
data-drawer-close
|
||||||
|
aria-hidden="true"
|
||||||
|
></div>
|
||||||
|
<nav
|
||||||
|
id="hamburger-nav"
|
||||||
|
class="absolute left-0 top-0 h-full w-72 max-w-[80vw] bg-white shadow-xl p-6 transform -translate-x-full transition-transform duration-200 ease-out"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
data-drawer-close
|
||||||
|
class="absolute top-4 right-4 w-9 h-9 flex items-center justify-center text-neutral-500 hover:text-neutral-900 text-2xl leading-none rounded-md hover:bg-neutral-100"
|
||||||
|
aria-label="Fermer le menu"
|
||||||
|
>
|
||||||
|
×
|
||||||
|
</button>
|
||||||
|
<ul class="mt-12 space-y-1 text-base">
|
||||||
|
<li>
|
||||||
|
<a
|
||||||
|
href="/a-propos"
|
||||||
|
class="block px-3 py-2 rounded-md text-neutral-800 hover:bg-neutral-100 hover:text-neutral-900 transition-colors"
|
||||||
|
>
|
||||||
|
A propos
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a
|
||||||
|
href="/manifeste"
|
||||||
|
class="block px-3 py-2 rounded-md text-neutral-800 hover:bg-neutral-100 hover:text-neutral-900 transition-colors"
|
||||||
|
>
|
||||||
|
Manifeste
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a
|
||||||
|
href="/mentions-legales"
|
||||||
|
class="block px-3 py-2 rounded-md text-neutral-800 hover:bg-neutral-100 hover:text-neutral-900 transition-colors"
|
||||||
|
>
|
||||||
|
Mentions legales
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<!-- TODO V2 : ajouter liens ici (newsletter, soutien Liberapay, contact) -->
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
const trigger = document.getElementById('hamburger-trigger');
|
||||||
|
const drawer = document.getElementById('hamburger-drawer');
|
||||||
|
const nav = document.getElementById('hamburger-nav');
|
||||||
|
|
||||||
|
const open = () => {
|
||||||
|
if (!drawer || !nav) return;
|
||||||
|
drawer.classList.remove('hidden');
|
||||||
|
trigger?.setAttribute('aria-expanded', 'true');
|
||||||
|
requestAnimationFrame(() => nav.classList.remove('-translate-x-full'));
|
||||||
|
};
|
||||||
|
|
||||||
|
const close = () => {
|
||||||
|
if (!drawer || !nav) return;
|
||||||
|
nav.classList.add('-translate-x-full');
|
||||||
|
trigger?.setAttribute('aria-expanded', 'false');
|
||||||
|
setTimeout(() => drawer.classList.add('hidden'), 200);
|
||||||
|
};
|
||||||
|
|
||||||
|
trigger?.addEventListener('click', open);
|
||||||
|
|
||||||
|
drawer?.querySelectorAll('[data-drawer-close]').forEach((el) =>
|
||||||
|
el.addEventListener('click', close)
|
||||||
|
);
|
||||||
|
|
||||||
|
// ESC pour fermer
|
||||||
|
document.addEventListener('keydown', (e) => {
|
||||||
|
if (e.key === 'Escape' && drawer && !drawer.classList.contains('hidden')) {
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|||||||
55
src/components/astro/IframeCarteAEP.astro
Normal file
55
src/components/astro/IframeCarteAEP.astro
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
---
|
||||||
|
// PC4 - iframe carte AEP (cartobifurcation) avec skeleton loader + fallback timeout 8s.
|
||||||
|
// Route confirmee 200 sans X-Frame-Options ni frame-ancestors restrictifs (preflight 2026-05-08).
|
||||||
|
const CARTE_URL = 'https://aep.trans-former.fr/agences';
|
||||||
|
---
|
||||||
|
<div class="relative h-full w-full bg-neutral-100">
|
||||||
|
<!-- Skeleton loader -->
|
||||||
|
<div
|
||||||
|
id="iframe-skeleton"
|
||||||
|
class="absolute inset-0 flex items-center justify-center bg-neutral-50 animate-pulse"
|
||||||
|
>
|
||||||
|
<div class="text-neutral-400 text-sm">Chargement de la carte AEP...</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<iframe
|
||||||
|
id="carte-aep-iframe"
|
||||||
|
src={CARTE_URL}
|
||||||
|
title="Carte des reseaux AEP - cartobifurcation"
|
||||||
|
class="w-full h-full border-0 opacity-0 transition-opacity duration-500"
|
||||||
|
loading="lazy"
|
||||||
|
referrerpolicy="no-referrer-when-downgrade"
|
||||||
|
sandbox="allow-scripts allow-same-origin allow-popups allow-forms"
|
||||||
|
></iframe>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
const iframe = document.getElementById('carte-aep-iframe') as HTMLIFrameElement | null;
|
||||||
|
const skeleton = document.getElementById('iframe-skeleton');
|
||||||
|
|
||||||
|
iframe?.addEventListener('load', () => {
|
||||||
|
iframe.classList.remove('opacity-0');
|
||||||
|
iframe.classList.add('opacity-100');
|
||||||
|
skeleton?.classList.add('hidden');
|
||||||
|
});
|
||||||
|
|
||||||
|
// Timeout securite : si pas charge en 8s, afficher fallback lien externe.
|
||||||
|
setTimeout(() => {
|
||||||
|
if (skeleton && !skeleton.classList.contains('hidden')) {
|
||||||
|
skeleton.innerHTML = `
|
||||||
|
<div class="text-center text-neutral-500 px-4">
|
||||||
|
<p class="mb-2 text-sm">La carte n'a pas pu charger.</p>
|
||||||
|
<a
|
||||||
|
href="https://aep.trans-former.fr/agences"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="text-blue-600 underline text-sm"
|
||||||
|
>
|
||||||
|
Ouvrir dans un nouvel onglet →
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
skeleton.classList.remove('animate-pulse');
|
||||||
|
}
|
||||||
|
}, 8000);
|
||||||
|
</script>
|
||||||
@@ -1,11 +1,101 @@
|
|||||||
---
|
---
|
||||||
// Placeholder popup onboarding — PC2 fait l'animation et le contenu
|
// PopupOnboarding - micro-resume premier visit, localStorage flag tf-onboarded
|
||||||
|
// Le micro-resume est une PROPOSITION Opus a iterer avec Jules.
|
||||||
|
// Stocke en constante en tete pour iteration facile.
|
||||||
|
const microResume = `Architecture, ecologie, politique : un commun a construire ensemble.
|
||||||
|
Reapproprions-nous nos infrastructures vitales en repartant de l'existant.
|
||||||
|
Ce site, c'est le journal vivant de cette bifurcation.`;
|
||||||
---
|
---
|
||||||
<div
|
<div
|
||||||
id="pc-onboarding"
|
id="onboarding-popup"
|
||||||
class="hidden fixed inset-0 z-40 bg-black/30 items-center justify-center"
|
class="fixed inset-0 z-[60] hidden bg-black/50 backdrop-blur-sm items-center justify-center p-4"
|
||||||
|
role="dialog"
|
||||||
|
aria-modal="true"
|
||||||
|
aria-labelledby="onboarding-text"
|
||||||
>
|
>
|
||||||
<div class="bg-white p-6 rounded shadow-lg max-w-sm">
|
<div class="bg-white rounded-xl max-w-md w-full p-6 md:p-7 shadow-2xl relative animate-fade-in">
|
||||||
<p class="text-sm text-neutral-600">Pop-up onboarding — PC2</p>
|
<button
|
||||||
|
id="onboarding-close"
|
||||||
|
type="button"
|
||||||
|
class="absolute top-3 right-3 w-8 h-8 flex items-center justify-center text-neutral-400 hover:text-neutral-800 text-2xl leading-none rounded-md hover:bg-neutral-100 transition-colors"
|
||||||
|
aria-label="Fermer"
|
||||||
|
>
|
||||||
|
×
|
||||||
|
</button>
|
||||||
|
<p
|
||||||
|
id="onboarding-text"
|
||||||
|
class="text-base md:text-lg leading-relaxed text-neutral-800 whitespace-pre-line pr-6"
|
||||||
|
>{microResume}</p>
|
||||||
|
<button
|
||||||
|
id="onboarding-cta"
|
||||||
|
type="button"
|
||||||
|
class="mt-5 px-4 py-2.5 bg-neutral-900 text-white rounded-lg w-full font-medium hover:bg-neutral-700 transition-colors"
|
||||||
|
>
|
||||||
|
Entrer
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
@keyframes fadeIn {
|
||||||
|
from { opacity: 0; transform: translateY(8px); }
|
||||||
|
to { opacity: 1; transform: translateY(0); }
|
||||||
|
}
|
||||||
|
.animate-fade-in {
|
||||||
|
animation: fadeIn 0.25s ease-out;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
const STORAGE_KEY = 'tf-onboarded';
|
||||||
|
const popup = document.getElementById('onboarding-popup');
|
||||||
|
const closeBtn = document.getElementById('onboarding-close');
|
||||||
|
const ctaBtn = document.getElementById('onboarding-cta');
|
||||||
|
|
||||||
|
let alreadyOnboarded = false;
|
||||||
|
try {
|
||||||
|
alreadyOnboarded = !!localStorage.getItem(STORAGE_KEY);
|
||||||
|
} catch {
|
||||||
|
// mode prive : on affiche quand meme, sans persistence
|
||||||
|
alreadyOnboarded = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!alreadyOnboarded && popup) {
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
popup.classList.remove('hidden');
|
||||||
|
popup.classList.add('flex');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const dismiss = () => {
|
||||||
|
if (!popup) return;
|
||||||
|
popup.classList.remove('flex');
|
||||||
|
popup.classList.add('hidden');
|
||||||
|
try {
|
||||||
|
localStorage.setItem(STORAGE_KEY, '1');
|
||||||
|
} catch {
|
||||||
|
// mode prive : pas de persistence
|
||||||
|
}
|
||||||
|
window.removeEventListener('scroll', onScroll);
|
||||||
|
};
|
||||||
|
|
||||||
|
closeBtn?.addEventListener('click', dismiss);
|
||||||
|
ctaBtn?.addEventListener('click', dismiss);
|
||||||
|
|
||||||
|
// Dismiss implicite au scroll 200px (engagement implicite)
|
||||||
|
const onScroll = () => {
|
||||||
|
if (window.scrollY > 200) {
|
||||||
|
dismiss();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if (!alreadyOnboarded) {
|
||||||
|
window.addEventListener('scroll', onScroll, { passive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
// ESC pour fermer
|
||||||
|
document.addEventListener('keydown', (e) => {
|
||||||
|
if (e.key === 'Escape' && popup && !popup.classList.contains('hidden')) {
|
||||||
|
dismiss();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|||||||
72
src/components/astro/ScrollArticles.astro
Normal file
72
src/components/astro/ScrollArticles.astro
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
---
|
||||||
|
// PC4 - Liste articles Substack en scroll sous l'iframe carte.
|
||||||
|
// V1 placeholder data en dur ; PC6 (journal n8n) remplacera par fetch journal.json filtre tag #politique.
|
||||||
|
const articles = [
|
||||||
|
{
|
||||||
|
date: '2026-04-28',
|
||||||
|
titre: 'Cap sur l\'autonomie : retour sur 6 mois de chantier',
|
||||||
|
url: 'https://transformations.substack.com/p/cap-autonomie',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-04-15',
|
||||||
|
titre: 'Le commun comme infrastructure',
|
||||||
|
url: 'https://transformations.substack.com/p/commun-infrastructure',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-03-30',
|
||||||
|
titre: 'Architecte ou operateur de bifurcation ?',
|
||||||
|
url: 'https://transformations.substack.com/p/architecte-bifurcation',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-03-12',
|
||||||
|
titre: 'Reseaux AEP : pourquoi la coordination est politique',
|
||||||
|
url: 'https://transformations.substack.com/p/aep-coordination',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-02-26',
|
||||||
|
titre: 'Sortir du sauveur, entrer dans le compagnon',
|
||||||
|
url: 'https://transformations.substack.com/p/sauveur-compagnon',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-02-08',
|
||||||
|
titre: 'Petit manifeste contre l\'expert isole',
|
||||||
|
url: 'https://transformations.substack.com/p/contre-expert-isole',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
date: '2026-01-22',
|
||||||
|
titre: 'Ce que la commande publique fait a la pensee',
|
||||||
|
url: 'https://transformations.substack.com/p/commande-publique',
|
||||||
|
},
|
||||||
|
// TODO PC6 : remplacer par fetch journal.json filtre tag #politique.
|
||||||
|
];
|
||||||
|
---
|
||||||
|
<section class="border-t border-neutral-200 py-6 px-4 bg-white">
|
||||||
|
<h3 class="text-sm font-semibold uppercase tracking-wider text-neutral-500 mb-4">
|
||||||
|
Derniers articles ; Substack
|
||||||
|
</h3>
|
||||||
|
<ul class="space-y-3">
|
||||||
|
{articles.map(({ date, titre, url }) => (
|
||||||
|
<li>
|
||||||
|
<a
|
||||||
|
href={url}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="block group"
|
||||||
|
>
|
||||||
|
<time class="text-xs text-neutral-400">{date}</time>
|
||||||
|
<p class="text-sm text-neutral-800 group-hover:text-neutral-600 transition-colors leading-snug">
|
||||||
|
{titre}
|
||||||
|
</p>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
<a
|
||||||
|
href="https://transformations.substack.com"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="inline-block mt-4 text-xs text-neutral-500 hover:text-neutral-900"
|
||||||
|
>
|
||||||
|
Voir tous les articles →
|
||||||
|
</a>
|
||||||
|
</section>
|
||||||
187
src/components/vue/ChatbotV2.vue
Normal file
187
src/components/vue/ChatbotV2.vue
Normal file
@@ -0,0 +1,187 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
// PC7 — Chatbot V1 brancha sur endpoint AEP (proxy Astro server-side).
|
||||||
|
// Persistance thread sessionStorage. Bandeau beta info ; pas de streaming, pas de markdown V1.
|
||||||
|
|
||||||
|
import { ref, onMounted, nextTick, useTemplateRef } from 'vue'
|
||||||
|
|
||||||
|
interface Msg {
|
||||||
|
role: 'user' | 'bot'
|
||||||
|
content: string
|
||||||
|
ts: number
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = defineProps<{ apiUrl?: string }>()
|
||||||
|
|
||||||
|
const messages = ref<Msg[]>([])
|
||||||
|
const input = ref('')
|
||||||
|
const sending = ref(false)
|
||||||
|
const errorBanner = ref('')
|
||||||
|
const threadEl = useTemplateRef<HTMLElement>('threadEl')
|
||||||
|
|
||||||
|
const STORAGE_KEY = 'tf-chatbot-v2-thread'
|
||||||
|
const ENDPOINT = props.apiUrl || '/api/chatbot'
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
try {
|
||||||
|
const saved = sessionStorage.getItem(STORAGE_KEY)
|
||||||
|
if (saved) messages.value = JSON.parse(saved)
|
||||||
|
} catch {
|
||||||
|
// sessionStorage indisponible ; on ignore.
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const persist = () => {
|
||||||
|
try {
|
||||||
|
sessionStorage.setItem(STORAGE_KEY, JSON.stringify(messages.value))
|
||||||
|
} catch {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const scrollToBottom = async () => {
|
||||||
|
await nextTick()
|
||||||
|
const el = threadEl.value
|
||||||
|
if (el) el.scrollTo({ top: el.scrollHeight, behavior: 'smooth' })
|
||||||
|
}
|
||||||
|
|
||||||
|
const send = async () => {
|
||||||
|
const q = input.value.trim()
|
||||||
|
if (!q || sending.value) return
|
||||||
|
|
||||||
|
messages.value.push({ role: 'user', content: q, ts: Date.now() })
|
||||||
|
input.value = ''
|
||||||
|
sending.value = true
|
||||||
|
errorBanner.value = ''
|
||||||
|
persist()
|
||||||
|
scrollToBottom()
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await fetch(ENDPOINT, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
question: q,
|
||||||
|
history: messages.value.slice(-10).map((m) => ({ role: m.role, content: m.content })),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!res.ok) {
|
||||||
|
if (res.status === 429) {
|
||||||
|
errorBanner.value = 'Limite de questions atteinte ; ressaie plus tard.'
|
||||||
|
} else if (res.status === 502 || res.status === 504) {
|
||||||
|
errorBanner.value = "Le service de chat n'est pas joignable ; ressaie dans un instant."
|
||||||
|
} else {
|
||||||
|
errorBanner.value = `Erreur ${res.status} ; ressaie ou recharge la page.`
|
||||||
|
}
|
||||||
|
sending.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await res.json()
|
||||||
|
const answer =
|
||||||
|
data.reponse_texte ||
|
||||||
|
data.answer ||
|
||||||
|
data.text ||
|
||||||
|
data.message ||
|
||||||
|
'Pas de reponse.'
|
||||||
|
|
||||||
|
messages.value.push({ role: 'bot', content: answer, ts: Date.now() })
|
||||||
|
} catch (e) {
|
||||||
|
errorBanner.value = "Probleme reseau ; verifie ta connexion."
|
||||||
|
messages.value.push({
|
||||||
|
role: 'bot',
|
||||||
|
content: `Erreur : ${(e as Error).message || 'inconnue'}`,
|
||||||
|
ts: Date.now(),
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
sending.value = false
|
||||||
|
persist()
|
||||||
|
scrollToBottom()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const reset = () => {
|
||||||
|
messages.value = []
|
||||||
|
errorBanner.value = ''
|
||||||
|
persist()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="h-full flex flex-col bg-white">
|
||||||
|
<div class="bg-amber-50 border-b border-amber-200 px-3 py-2 text-xs text-amber-900 leading-snug">
|
||||||
|
<span class="font-medium">Beta :</span> 120 fiches AEP indexees (Mistral Small) ; bientot RAG-PE multi-corpus.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="errorBanner"
|
||||||
|
class="bg-rose-50 border-b border-rose-200 px-3 py-2 text-xs text-rose-800"
|
||||||
|
role="alert"
|
||||||
|
>
|
||||||
|
{{ errorBanner }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
ref="threadEl"
|
||||||
|
class="flex-1 overflow-y-auto px-3 py-3 space-y-3"
|
||||||
|
role="log"
|
||||||
|
aria-live="polite"
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
v-if="!messages.length"
|
||||||
|
class="text-neutral-400 text-sm italic text-center mt-8 px-4"
|
||||||
|
>
|
||||||
|
Pose une question sur l'AEP ; le manifeste ; les agences locales...
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-for="m in messages"
|
||||||
|
:key="m.ts"
|
||||||
|
:class="[
|
||||||
|
'max-w-[85%] rounded-lg px-3 py-2 text-sm leading-relaxed whitespace-pre-wrap break-words',
|
||||||
|
m.role === 'user'
|
||||||
|
? 'bg-neutral-900 text-white ml-auto'
|
||||||
|
: 'bg-neutral-100 text-neutral-800 mr-auto',
|
||||||
|
]"
|
||||||
|
>
|
||||||
|
{{ m.content }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="sending"
|
||||||
|
class="bg-neutral-100 max-w-[60%] rounded-lg px-3 py-2 text-sm text-neutral-500 italic mr-auto"
|
||||||
|
>
|
||||||
|
Le bot reflechit...
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<form
|
||||||
|
class="border-t border-neutral-200 p-2 flex gap-2 items-stretch"
|
||||||
|
@submit.prevent="send"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
v-model="input"
|
||||||
|
type="text"
|
||||||
|
placeholder="Ta question..."
|
||||||
|
:disabled="sending"
|
||||||
|
class="flex-1 px-3 py-2 border border-neutral-300 rounded-lg text-sm focus:outline-none focus:border-neutral-900 disabled:opacity-60"
|
||||||
|
aria-label="Saisis ta question"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
:disabled="sending || !input.trim()"
|
||||||
|
class="px-3 py-2 bg-neutral-900 text-white rounded-lg text-sm font-medium disabled:opacity-50"
|
||||||
|
>
|
||||||
|
Envoyer
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
v-if="messages.length"
|
||||||
|
type="button"
|
||||||
|
class="px-2 py-2 text-neutral-500 text-xs hover:text-neutral-900 underline"
|
||||||
|
@click="reset"
|
||||||
|
>
|
||||||
|
Reset
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
107
src/components/vue/InstaFeed.vue
Normal file
107
src/components/vue/InstaFeed.vue
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, onMounted } from 'vue';
|
||||||
|
|
||||||
|
interface BeholdPost {
|
||||||
|
id: string;
|
||||||
|
permalink: string;
|
||||||
|
mediaUrl: string;
|
||||||
|
thumbnailUrl?: string;
|
||||||
|
caption?: string;
|
||||||
|
mediaType: 'IMAGE' | 'VIDEO' | 'CAROUSEL_ALBUM';
|
||||||
|
timestamp: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
feedId: string;
|
||||||
|
account: string;
|
||||||
|
accountUrl: string;
|
||||||
|
fallbackBio?: string;
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const posts = ref<BeholdPost[]>([]);
|
||||||
|
const loading = ref(true);
|
||||||
|
const error = ref<string | null>(null);
|
||||||
|
|
||||||
|
const isPlaceholder = (id: string) => !id || id.startsWith('PLACEHOLDER_');
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
if (isPlaceholder(props.feedId)) {
|
||||||
|
loading.value = false;
|
||||||
|
error.value = 'no-feed-id';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const controller = new AbortController();
|
||||||
|
const timeoutId = setTimeout(() => controller.abort(), 5000);
|
||||||
|
const res = await fetch(`https://feeds.behold.so/${props.feedId}`, {
|
||||||
|
signal: controller.signal,
|
||||||
|
});
|
||||||
|
clearTimeout(timeoutId);
|
||||||
|
if (!res.ok) throw new Error(`Behold returned ${res.status}`);
|
||||||
|
const data = await res.json();
|
||||||
|
const items: BeholdPost[] = Array.isArray(data) ? data : (data.posts ?? []);
|
||||||
|
posts.value = items.slice(0, 6);
|
||||||
|
} catch (e) {
|
||||||
|
error.value = (e as Error).message || 'fetch-error';
|
||||||
|
} finally {
|
||||||
|
loading.value = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<section class="border-b border-neutral-200 last:border-b-0">
|
||||||
|
<header class="px-4 py-3 flex items-center justify-between">
|
||||||
|
<a
|
||||||
|
:href="accountUrl"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="font-medium text-sm hover:underline"
|
||||||
|
>
|
||||||
|
{{ account }}
|
||||||
|
</a>
|
||||||
|
<span v-if="loading" class="text-xs text-neutral-400">...</span>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div v-if="loading" class="grid grid-cols-2 gap-1 p-1">
|
||||||
|
<div
|
||||||
|
v-for="i in 4"
|
||||||
|
:key="i"
|
||||||
|
class="aspect-square bg-neutral-100 animate-pulse"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-else-if="posts.length"
|
||||||
|
class="grid grid-cols-2 gap-1 p-1"
|
||||||
|
>
|
||||||
|
<a
|
||||||
|
v-for="post in posts"
|
||||||
|
:key="post.id"
|
||||||
|
:href="post.permalink"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="block aspect-square overflow-hidden group"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
:src="post.thumbnailUrl || post.mediaUrl"
|
||||||
|
:alt="post.caption?.slice(0, 80) || account"
|
||||||
|
loading="lazy"
|
||||||
|
class="w-full h-full object-cover group-hover:scale-105 transition-transform"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="p-4 text-sm text-neutral-600">
|
||||||
|
<p v-if="fallbackBio" class="mb-3">{{ fallbackBio }}</p>
|
||||||
|
<a
|
||||||
|
:href="accountUrl"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener"
|
||||||
|
class="inline-block px-3 py-2 bg-neutral-900 text-white rounded-lg text-sm"
|
||||||
|
>
|
||||||
|
Voir sur Instagram →
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</template>
|
||||||
@@ -1,9 +1,168 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
// Placeholder journal list — PC6 lit public/data/journal.json
|
import { ref, computed, onMounted, onUnmounted } from 'vue'
|
||||||
|
|
||||||
|
interface JournalItem {
|
||||||
|
id: string
|
||||||
|
platform: 'substack' | 'gitea' | 'github' | 'instagram' | 'castopod' | 'blog' | 'linkedin'
|
||||||
|
hashtag: string
|
||||||
|
date: string
|
||||||
|
titre: string
|
||||||
|
extrait: string
|
||||||
|
url: string
|
||||||
|
thumbnail: string | null
|
||||||
|
}
|
||||||
|
|
||||||
|
interface JournalPayload {
|
||||||
|
generatedAt: string
|
||||||
|
items: JournalItem[]
|
||||||
|
counts?: Record<string, number>
|
||||||
|
}
|
||||||
|
|
||||||
|
const items = ref<JournalItem[]>([])
|
||||||
|
const filters = ref<Record<string, boolean>>({})
|
||||||
|
const loading = ref(true)
|
||||||
|
const errored = ref(false)
|
||||||
|
const empty = ref(false)
|
||||||
|
|
||||||
|
const JOURNAL_URL =
|
||||||
|
(import.meta as unknown as { env: Record<string, string | undefined> }).env
|
||||||
|
.PUBLIC_JOURNAL_URL || 'https://data.trans-former.fr/journal.json'
|
||||||
|
|
||||||
|
const STORAGE_KEY = 'tf-hashtag-filters'
|
||||||
|
|
||||||
|
const loadFilters = () => {
|
||||||
|
try {
|
||||||
|
filters.value = JSON.parse(localStorage.getItem(STORAGE_KEY) || '{}')
|
||||||
|
} catch {
|
||||||
|
filters.value = {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onFilterChange = (e: Event) => {
|
||||||
|
const ce = e as CustomEvent
|
||||||
|
if (ce.detail && typeof ce.detail === 'object') {
|
||||||
|
filters.value = { ...ce.detail }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
loadFilters()
|
||||||
|
window.addEventListener('hashtag-filter-change', onFilterChange as EventListener)
|
||||||
|
try {
|
||||||
|
const res = await fetch(JOURNAL_URL, { cache: 'no-store' })
|
||||||
|
if (!res.ok) {
|
||||||
|
errored.value = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const data = (await res.json()) as JournalPayload
|
||||||
|
items.value = Array.isArray(data.items) ? data.items : []
|
||||||
|
if (items.value.length === 0) empty.value = true
|
||||||
|
} catch (e) {
|
||||||
|
errored.value = true
|
||||||
|
console.error('JournalList fetch failed', e)
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
onUnmounted(() => {
|
||||||
|
window.removeEventListener('hashtag-filter-change', onFilterChange as EventListener)
|
||||||
|
})
|
||||||
|
|
||||||
|
const visibleItems = computed(() => {
|
||||||
|
// Tous les filtres faux = aucun affiche ; tous true ou aucune cle = tout afficher
|
||||||
|
const keys = Object.keys(filters.value)
|
||||||
|
if (keys.length === 0) return items.value
|
||||||
|
const activeKeys = keys.filter((k) => filters.value[k])
|
||||||
|
// Si l'utilisateur a explicitement décoché tous les hashtags, on n'affiche rien
|
||||||
|
if (activeKeys.length === 0 && keys.some((k) => filters.value[k] === false)) return []
|
||||||
|
if (activeKeys.length === 0) return items.value
|
||||||
|
return items.value.filter((it) => activeKeys.includes(it.hashtag))
|
||||||
|
})
|
||||||
|
|
||||||
|
const formatDate = (iso: string) => {
|
||||||
|
try {
|
||||||
|
const d = new Date(iso)
|
||||||
|
if (isNaN(d.getTime())) return ''
|
||||||
|
return `${String(d.getDate()).padStart(2, '0')}/${String(d.getMonth() + 1).padStart(2, '0')}`
|
||||||
|
} catch {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const platformLabel = (p: string) => {
|
||||||
|
switch (p) {
|
||||||
|
case 'gitea':
|
||||||
|
case 'github':
|
||||||
|
return 'code'
|
||||||
|
case 'substack':
|
||||||
|
return 'substack'
|
||||||
|
case 'instagram':
|
||||||
|
return 'insta'
|
||||||
|
case 'castopod':
|
||||||
|
return 'pod'
|
||||||
|
case 'blog':
|
||||||
|
return 'blog'
|
||||||
|
case 'linkedin':
|
||||||
|
return 'linkedin'
|
||||||
|
default:
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="h-full w-full flex items-center justify-center text-sm text-neutral-400">
|
<div class="space-y-3">
|
||||||
Journal list placeholder (PC6)
|
<p v-if="loading" class="text-neutral-400 italic text-xs">
|
||||||
|
Chargement du journal...
|
||||||
|
</p>
|
||||||
|
<p v-else-if="errored" class="text-neutral-400 italic text-xs">
|
||||||
|
Journal en cours d'agregation. Reviens dans quelques heures.
|
||||||
|
</p>
|
||||||
|
<p v-else-if="empty" class="text-neutral-400 italic text-xs">
|
||||||
|
Aucun item agrege. Le cron nocturne tourne a 3h.
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
v-else-if="!visibleItems.length"
|
||||||
|
class="text-neutral-400 italic text-xs"
|
||||||
|
>
|
||||||
|
Aucun item ; ajuste les filtres.
|
||||||
|
</p>
|
||||||
|
<article
|
||||||
|
v-for="item in visibleItems"
|
||||||
|
:key="item.id"
|
||||||
|
class="border-b border-neutral-100 pb-3 last:border-b-0"
|
||||||
|
>
|
||||||
|
<a
|
||||||
|
:href="item.url"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
class="block group"
|
||||||
|
>
|
||||||
|
<div class="flex items-baseline gap-2 text-[11px] text-neutral-500 mb-1">
|
||||||
|
<time>{{ formatDate(item.date) }}</time>
|
||||||
|
<span class="font-mono text-neutral-700">{{ item.hashtag }}</span>
|
||||||
|
<span class="text-neutral-400">{{ platformLabel(item.platform) }}</span>
|
||||||
|
</div>
|
||||||
|
<h4
|
||||||
|
class="text-sm text-neutral-900 group-hover:text-neutral-600 leading-snug"
|
||||||
|
>
|
||||||
|
{{ item.titre }}
|
||||||
|
</h4>
|
||||||
|
<p
|
||||||
|
v-if="item.extrait"
|
||||||
|
class="text-xs text-neutral-500 mt-1 line-clamp-2"
|
||||||
|
>
|
||||||
|
{{ item.extrait }}
|
||||||
|
</p>
|
||||||
|
<img
|
||||||
|
v-if="item.thumbnail"
|
||||||
|
:src="item.thumbnail"
|
||||||
|
:alt="item.titre"
|
||||||
|
loading="lazy"
|
||||||
|
class="mt-2 w-full max-h-32 object-cover rounded"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</article>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
81
src/pages/a-propos.astro
Normal file
81
src/pages/a-propos.astro
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
|
||||||
|
import BaseLayout from '../layouts/BaseLayout.astro';
|
||||||
|
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
|
||||||
|
---
|
||||||
|
<BaseLayout
|
||||||
|
title="A propos - Jules Neny"
|
||||||
|
description="Architecte HMONP, ecrivain politique, facilitateur. Bagneres-de-Bigorre, Pyrenees."
|
||||||
|
>
|
||||||
|
<HamburgerMenu />
|
||||||
|
|
||||||
|
<main class="min-h-screen bg-white">
|
||||||
|
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24">
|
||||||
|
|
||||||
|
<header class="mb-10">
|
||||||
|
<p class="text-sm uppercase tracking-widest text-neutral-500 mb-3">
|
||||||
|
A propos
|
||||||
|
</p>
|
||||||
|
<h1 class="text-3xl md:text-4xl font-semibold text-neutral-900 leading-tight">
|
||||||
|
Jules Neny
|
||||||
|
</h1>
|
||||||
|
<p class="mt-3 text-neutral-600 text-base md:text-lg">
|
||||||
|
Architecte HMONP. Ecrivain politique. Facilitateur. Bagneres-de-Bigorre, Pyrenees.
|
||||||
|
</p>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="text-neutral-800 text-[17px] leading-[1.75] space-y-6">
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Je suis architecte, forme a l'ENSAPB (Paris-Belleville), avec un passage par Oxford Brookes et un bootcamp Perspectives. Je travaille en independant sous le nom <em>Transformations resilientes</em>, basee dans les Pyrenees. Mon metier de base : audit energetique (MAR), maitrise d'oeuvre en renovation et construction ecologique, accompagnement a l'autonomie (eau, energie, jardin).
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Mais l'architecture, pour moi, deborde du bati. C'est aussi une lecture des systemes ; un outil de diagnostic du corps social. La ou l'eau, l'energie, la justice, l'education, la sante, le logement et l'agriculture defaillent ; il y a des reconfigurations a inventer, situees, par territoire. Reprendre le pouvoir par la base. Ecrire, lentement, un nouveau contrat social.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Trois fils tissent mon travail :
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<ul class="list-disc list-outside pl-6 space-y-2">
|
||||||
|
<li>
|
||||||
|
<strong>Architecte</strong> ; reveler les problematiques complexes des systemes (diagnostic systemique, pratique de terrain).
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<strong>Ecrivain politique</strong> ; transmettre et problematiser via le manifeste <em>Architecture d'Ecologie Politique</em>, le blog trans-former.fr, et bientot Substack.
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<strong>Facilitateur</strong> ; accompagner la transformation democratique (intelligence collective, formations).
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Ma boussole : <em>me transformer pour contribuer a la transformation du monde</em>. Mes valeurs : emancipation, liens, amour. Mes pratiques quotidiennes : ecriture du matin, yoga Isha, marche en montagne, peinture a l'huile, cuisine.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Ce site est un journal vivant. Il agrege ce que je publie sur plusieurs plateformes (blog, Substack, LinkedIn, Instagram, Castopod, GitHub) ; et tente de rendre lisible une demarche qui se fait dans le temps long.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Pour me joindre :
|
||||||
|
<a
|
||||||
|
href="mailto:julesneny8@gmail.com"
|
||||||
|
class="underline underline-offset-2 hover:text-neutral-900"
|
||||||
|
>julesneny8@gmail.com</a>.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer class="mt-12 pt-8 border-t border-neutral-200 text-sm text-neutral-500">
|
||||||
|
<p>
|
||||||
|
<a href="/" class="underline underline-offset-2 hover:text-neutral-900">
|
||||||
|
← Retour au journal
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</article>
|
||||||
|
</main>
|
||||||
|
</BaseLayout>
|
||||||
65
src/pages/api/chatbot.ts
Normal file
65
src/pages/api/chatbot.ts
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
// PC7 — Endpoint proxy POST /api/chatbot
|
||||||
|
// Forward la question vers CHATBOT_UPSTREAM (default : aep.trans-former.fr/api/chatbot).
|
||||||
|
// Tunnel CORS (l'upstream ne sert pas de header CORS pour trans-former.fr) + timeout 25s.
|
||||||
|
|
||||||
|
import type { APIRoute } from 'astro'
|
||||||
|
|
||||||
|
export const prerender = false
|
||||||
|
|
||||||
|
const UPSTREAM =
|
||||||
|
import.meta.env.CHATBOT_UPSTREAM || 'https://aep.trans-former.fr/api/chatbot'
|
||||||
|
const TIMEOUT_MS = 25_000
|
||||||
|
|
||||||
|
const jsonResponse = (status: number, payload: unknown) =>
|
||||||
|
new Response(JSON.stringify(payload), {
|
||||||
|
status,
|
||||||
|
headers: { 'Content-Type': 'application/json; charset=utf-8' },
|
||||||
|
})
|
||||||
|
|
||||||
|
export const POST: APIRoute = async ({ request }) => {
|
||||||
|
let body: { question?: string; q?: string; history?: unknown }
|
||||||
|
|
||||||
|
try {
|
||||||
|
body = await request.json()
|
||||||
|
} catch {
|
||||||
|
return jsonResponse(400, { error: 'invalid_json' })
|
||||||
|
}
|
||||||
|
|
||||||
|
const question = (body.question || body.q || '').toString().trim()
|
||||||
|
if (!question) {
|
||||||
|
return jsonResponse(400, { error: 'missing_question' })
|
||||||
|
}
|
||||||
|
|
||||||
|
const ctrl = new AbortController()
|
||||||
|
const timer = setTimeout(() => ctrl.abort(), TIMEOUT_MS)
|
||||||
|
|
||||||
|
try {
|
||||||
|
const upstream = await fetch(UPSTREAM, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
question,
|
||||||
|
history: Array.isArray(body.history) ? body.history : [],
|
||||||
|
}),
|
||||||
|
signal: ctrl.signal,
|
||||||
|
})
|
||||||
|
clearTimeout(timer)
|
||||||
|
|
||||||
|
const text = await upstream.text()
|
||||||
|
const contentType =
|
||||||
|
upstream.headers.get('content-type') || 'application/json; charset=utf-8'
|
||||||
|
|
||||||
|
return new Response(text, {
|
||||||
|
status: upstream.status,
|
||||||
|
headers: { 'Content-Type': contentType },
|
||||||
|
})
|
||||||
|
} catch (e) {
|
||||||
|
clearTimeout(timer)
|
||||||
|
const err = e as Error
|
||||||
|
const aborted = err.name === 'AbortError'
|
||||||
|
return jsonResponse(aborted ? 504 : 502, {
|
||||||
|
error: aborted ? 'upstream_timeout' : 'upstream_failed',
|
||||||
|
detail: err.message || 'unknown',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
---
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
|
||||||
import BaseLayout from '../layouts/BaseLayout.astro';
|
import BaseLayout from '../layouts/BaseLayout.astro';
|
||||||
import ColJournal from '../components/astro/ColJournal.astro';
|
import ColJournal from '../components/astro/ColJournal.astro';
|
||||||
import ColCentre from '../components/astro/ColCentre.astro';
|
import ColCentre from '../components/astro/ColCentre.astro';
|
||||||
|
|||||||
@@ -1,9 +1,222 @@
|
|||||||
---
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
|
||||||
import BaseLayout from '../layouts/BaseLayout.astro';
|
import BaseLayout from '../layouts/BaseLayout.astro';
|
||||||
|
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
|
||||||
---
|
---
|
||||||
<BaseLayout title="Manifeste — trans-former.fr">
|
<BaseLayout
|
||||||
<main class="max-w-2xl mx-auto p-6">
|
title="Manifeste - Architecture d'Ecologie Politique"
|
||||||
<h1 class="text-2xl font-semibold mb-4">Manifeste</h1>
|
description="Manifeste AEP : un commun vivant pour une architecture d'ecologie politique. Architectes, allie-es, habitant-es."
|
||||||
<p class="text-sm text-neutral-500">Placeholder — PC2 importe le contenu manifeste depuis astro-site existant.</p>
|
>
|
||||||
|
<HamburgerMenu />
|
||||||
|
|
||||||
|
<main class="min-h-screen bg-white">
|
||||||
|
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24">
|
||||||
|
|
||||||
|
<!-- En-tete -->
|
||||||
|
<header class="mb-12 md:mb-16">
|
||||||
|
<p class="text-sm uppercase tracking-widest text-neutral-500 mb-3">
|
||||||
|
Manifeste
|
||||||
|
</p>
|
||||||
|
<h1 class="text-3xl md:text-4xl font-semibold text-neutral-900 leading-tight">
|
||||||
|
Architecture d'Ecologie Politique
|
||||||
|
</h1>
|
||||||
|
<p class="mt-4 text-neutral-600 text-base md:text-lg leading-relaxed">
|
||||||
|
Un commun vivant pour bifurquer ; ensemble, lentement, par accumulation de petits gestes situes.
|
||||||
|
</p>
|
||||||
|
<div class="mt-8">
|
||||||
|
<a
|
||||||
|
href="/manifeste/commander"
|
||||||
|
class="inline-block px-5 py-2.5 border border-neutral-900 text-neutral-900 rounded-lg font-medium hover:bg-neutral-900 hover:text-white transition-colors"
|
||||||
|
>
|
||||||
|
Commander la version imprimee
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<!-- Corps du manifeste -->
|
||||||
|
<div class="text-neutral-800 text-[17px] md:text-[18px] leading-[1.75] space-y-6">
|
||||||
|
|
||||||
|
<p class="italic text-neutral-700">
|
||||||
|
Un quart des architectes vivent sous le seuil de pauvrete. La moitie de nos heures, non facturees. Nos cotisations, parmi les plus lourdes des professions reglementees. Et le secteur du batiment, a lui seul, pese 34% des emissions mondiales de gaz a effet de serre.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Quelque chose s'est rompu ; pas dans nos vies, dans les cadres qui les contiennent.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Notre profession ne traverse pas une simple crise. Elle reflete l'effondrement d'un monde qui confond performance et destruction, signature et silence, expertise et soumission.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<hr class="border-neutral-200 my-8" />
|
||||||
|
|
||||||
|
<h2 class="text-xl md:text-2xl font-semibold text-neutral-900 mt-12 mb-4">
|
||||||
|
Ce que nous voyons.
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
A l'echelle du metier, une profession structurellement sous l'eau, qui absorbe les tensions d'un systeme extractiviste ; et porte la responsabilite quand d'autres captent la valeur.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
A l'echelle des corps, une culture qui rend l'exploitation desirable : metier-passion, modele starchitecte, isolement liberal, moteur critique delegitimant. Nous tenons. Nous payons.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
A l'echelle du monde, l'effondrement ecologique et social qui avance, pendant que notre voix s'efface du debat public. Notre silence le sert.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<hr class="border-neutral-200 my-8" />
|
||||||
|
|
||||||
|
<h2 class="text-xl md:text-2xl font-semibold text-neutral-900 mt-12 mb-4">
|
||||||
|
Ce que nous refusons.
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Nous ne signerons plus pour des projets qui detruisent.<br />
|
||||||
|
Nous n'isolerons plus celles et ceux qui doutent.<br />
|
||||||
|
Nous ne porterons plus seul-es ce qui doit se penser, se faire ; et se soigner ; ensemble.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<hr class="border-neutral-200 my-8" />
|
||||||
|
|
||||||
|
<!-- Pivot : centre emotionnel du texte (italique + retrait) -->
|
||||||
|
<blockquote class="my-12 md:my-16 px-6 md:px-10 py-6 border-l-4 border-neutral-900 italic text-neutral-800 text-lg md:text-xl leading-relaxed">
|
||||||
|
<p class="font-medium not-italic mb-3 text-neutral-900">
|
||||||
|
Et pourtant, quelque chose tient.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Pas l'espoir naif, ni la promesse heroique. Quelque chose de plus humble : la fatigue commune reconnue, et l'envie qui revient de ne plus economiser sa vie.
|
||||||
|
</p>
|
||||||
|
</blockquote>
|
||||||
|
|
||||||
|
<hr class="border-neutral-200 my-8" />
|
||||||
|
|
||||||
|
<h2 class="text-xl md:text-2xl font-semibold text-neutral-900 mt-12 mb-4">
|
||||||
|
Ce que nous tentons.
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<em class="font-semibold not-italic">Partager.</em> Nos parcours, nos doutes, nos bifurcations. Se former les un-es les autres. Se tendre la main. Documenter ce qui marche, ce qui rate. Le personnel devient politique quand il se met en commun.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<em class="font-semibold not-italic">Construire.</em> L'infrastructure collective qui nous a manque. Cartes d'entraide, communs documentes, gouvernance horizontale, financement transparent, infra souveraine. <strong>Architecture d'Ecologie Politique</strong> : un commun vivant, ouvert, bioregional, ancre.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<em class="font-semibold not-italic">Pratiquer une medecine du corps social.</em> Diagnostiquer les infrastructures qui defaillent ; l'education, la justice, la securite, l'energie, la sante, le logement, l'agriculture. Proposer des reconfigurations situees, territoire par territoire. Reprendre le pouvoir par la base. Ecrire, lentement, un nouveau contrat social.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<em class="font-semibold not-italic">Commencer par les marges.</em> La ou le corps social souffre le plus, la ou il est le plus pret a changer. Ne pas decider a la place ; faire emerger. Transparence totale, sur le process et sur l'argent. Tendresse militante : la lucidite sans le mepris, l'engagement sans la durete.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<hr class="border-neutral-200 my-8" />
|
||||||
|
|
||||||
|
<h2 class="text-xl md:text-2xl font-semibold text-neutral-900 mt-12 mb-4">
|
||||||
|
Architectes, allie-es, habitant-es.
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
Nous avons un travail a faire ensemble. Lentement, patiemment, par accumulation de petits gestes situes. Pas pour fuir ; pour bifurquer.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- Chute : italique, separee -->
|
||||||
|
<p class="mt-10 md:mt-12 italic text-neutral-900 text-lg md:text-xl leading-relaxed">
|
||||||
|
Nos metiers sont des medecines. Reprenons-en le pouls ; a mains nues, ensemble.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Diagramme ASCII : recap des mouvements -->
|
||||||
|
<section class="mt-16 md:mt-20 pt-10 border-t border-neutral-200">
|
||||||
|
<h2 class="text-xl md:text-2xl font-semibold text-neutral-900 mb-6">
|
||||||
|
Les mouvements
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<div class="space-y-6 text-sm">
|
||||||
|
<div class="border border-neutral-200 rounded-lg p-5 bg-neutral-50">
|
||||||
|
<h3 class="font-mono text-xs uppercase tracking-wider text-neutral-500 mb-3">
|
||||||
|
M / Profession
|
||||||
|
</h3>
|
||||||
|
<ul class="space-y-2">
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m1-echm/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Anatomie d'une profession sous l'eau
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m4-echxl/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
La filiere beton, dispositif d'extraction
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://aep.trans-former.fr" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Pratiques regeneratives (carto vivante)
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="border border-neutral-200 rounded-lg p-5 bg-neutral-50">
|
||||||
|
<h3 class="font-mono text-xs uppercase tracking-wider text-neutral-500 mb-3">
|
||||||
|
S / Individu
|
||||||
|
</h3>
|
||||||
|
<ul class="space-y-2">
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m2-echs/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Ce qui nous maintient sous l'eau
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m6-pivot/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Resistance et regeneration (M6)
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m8-action/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Action et poeme (M8)
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="border border-neutral-200 rounded-lg p-5 bg-neutral-50">
|
||||||
|
<h3 class="font-mono text-xs uppercase tracking-wider text-neutral-500 mb-3">
|
||||||
|
XL / Corps social
|
||||||
|
</h3>
|
||||||
|
<ul class="space-y-2">
|
||||||
|
<li>
|
||||||
|
<a href="https://trans-former.fr/aep-m4-echxl/" class="text-neutral-800 hover:text-neutral-900 underline underline-offset-2 decoration-neutral-300 hover:decoration-neutral-900 transition-colors">
|
||||||
|
Polycrise et logiciel patriarcal-imperial
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li class="text-neutral-400 italic">
|
||||||
|
Medecine du corps social ; esquisse [a venir, ete 2026]
|
||||||
|
</li>
|
||||||
|
<li class="text-neutral-400 italic">
|
||||||
|
Le nouveau contrat social ; Le Livre [horizon 10 ans]
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- En lire plus -->
|
||||||
|
<footer class="mt-16 pt-10 border-t border-neutral-200 text-center">
|
||||||
|
<p class="text-neutral-600 mb-4">En lire plus</p>
|
||||||
|
<a
|
||||||
|
href="https://www.trans-former.fr/"
|
||||||
|
class="inline-block px-5 py-2.5 bg-neutral-900 text-white rounded-lg font-medium hover:bg-neutral-700 transition-colors"
|
||||||
|
>
|
||||||
|
Blog trans-former.fr →
|
||||||
|
</a>
|
||||||
|
<p class="mt-4 text-xs text-neutral-400">
|
||||||
|
bientot sur Substack
|
||||||
|
</p>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</article>
|
||||||
</main>
|
</main>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
|
|||||||
114
src/pages/manifeste/commander.astro
Normal file
114
src/pages/manifeste/commander.astro
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
|
||||||
|
import BaseLayout from '../../layouts/BaseLayout.astro';
|
||||||
|
import HamburgerMenu from '../../components/astro/HamburgerMenu.astro';
|
||||||
|
---
|
||||||
|
<BaseLayout
|
||||||
|
title="Commander la version imprimee - Manifeste AEP"
|
||||||
|
description="Pre-inscription pour la version imprimee du manifeste Architecture d'Ecologie Politique."
|
||||||
|
>
|
||||||
|
<HamburgerMenu />
|
||||||
|
|
||||||
|
<main class="min-h-screen bg-white">
|
||||||
|
<article class="max-w-xl mx-auto px-6 py-16 md:py-24">
|
||||||
|
|
||||||
|
<header class="mb-10">
|
||||||
|
<p class="text-sm uppercase tracking-widest text-neutral-500 mb-3">
|
||||||
|
Manifeste imprime
|
||||||
|
</p>
|
||||||
|
<h1 class="text-3xl md:text-4xl font-semibold text-neutral-900 leading-tight">
|
||||||
|
Commander la version imprimee
|
||||||
|
</h1>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="text-neutral-700 text-base md:text-lg leading-relaxed space-y-5">
|
||||||
|
<p>
|
||||||
|
La version imprimee du manifeste sera disponible prochainement ; tirage limite, papier recycle, format A5.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Inscrivez-vous pour etre averti-e de la mise en vente. Vous serez prevenu-e en priorite, sans engagement.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Form pre-inscription (V1 stub, V1.1 cable Listmonk) -->
|
||||||
|
<form
|
||||||
|
id="manifeste-preinscription"
|
||||||
|
class="mt-10 flex flex-col gap-3"
|
||||||
|
novalidate
|
||||||
|
>
|
||||||
|
<label for="email" class="text-sm font-medium text-neutral-700">
|
||||||
|
Votre email
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
id="email"
|
||||||
|
name="email"
|
||||||
|
type="email"
|
||||||
|
required
|
||||||
|
autocomplete="email"
|
||||||
|
placeholder="prenom@exemple.fr"
|
||||||
|
class="px-4 py-3 border border-neutral-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-neutral-900 focus:border-neutral-900"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
class="mt-2 px-5 py-3 bg-neutral-900 text-white rounded-lg font-medium hover:bg-neutral-700 transition-colors"
|
||||||
|
>
|
||||||
|
Etre averti-e
|
||||||
|
</button>
|
||||||
|
<p
|
||||||
|
id="form-feedback"
|
||||||
|
class="text-sm text-neutral-600 hidden"
|
||||||
|
aria-live="polite"
|
||||||
|
></p>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<p class="mt-8 text-sm text-neutral-500">
|
||||||
|
<a
|
||||||
|
href="/manifeste"
|
||||||
|
class="underline underline-offset-2 hover:text-neutral-900 transition-colors"
|
||||||
|
>
|
||||||
|
← Retour au manifeste
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
</article>
|
||||||
|
</main>
|
||||||
|
</BaseLayout>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// V1 stub : capture email en localStorage, V1.1 = POST Listmonk
|
||||||
|
const form = document.getElementById('manifeste-preinscription') as HTMLFormElement | null;
|
||||||
|
const feedback = document.getElementById('form-feedback');
|
||||||
|
|
||||||
|
form?.addEventListener('submit', (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
const data = new FormData(form);
|
||||||
|
const email = String(data.get('email') || '').trim();
|
||||||
|
if (!email || !email.includes('@')) {
|
||||||
|
if (feedback) {
|
||||||
|
feedback.textContent = 'Merci de saisir un email valide.';
|
||||||
|
feedback.classList.remove('hidden');
|
||||||
|
feedback.classList.add('text-red-600');
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stockage local en attendant le cable Listmonk (V1.1)
|
||||||
|
try {
|
||||||
|
const existing = JSON.parse(
|
||||||
|
localStorage.getItem('tf-manifeste-preinscriptions') || '[]'
|
||||||
|
);
|
||||||
|
existing.push({ email, ts: Date.now() });
|
||||||
|
localStorage.setItem('tf-manifeste-preinscriptions', JSON.stringify(existing));
|
||||||
|
} catch {
|
||||||
|
// mode prive : on ignore
|
||||||
|
}
|
||||||
|
|
||||||
|
if (feedback) {
|
||||||
|
feedback.textContent = 'Merci ; vous serez prevenu-e des sa disponibilite.';
|
||||||
|
feedback.classList.remove('hidden', 'text-red-600');
|
||||||
|
feedback.classList.add('text-green-700');
|
||||||
|
}
|
||||||
|
form.reset();
|
||||||
|
});
|
||||||
|
</script>
|
||||||
94
src/pages/mentions-legales.astro
Normal file
94
src/pages/mentions-legales.astro
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
|
||||||
|
import BaseLayout from '../layouts/BaseLayout.astro';
|
||||||
|
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
|
||||||
|
---
|
||||||
|
<BaseLayout
|
||||||
|
title="Mentions legales - trans-former.fr"
|
||||||
|
description="Mentions legales du site trans-former.fr."
|
||||||
|
>
|
||||||
|
<HamburgerMenu />
|
||||||
|
|
||||||
|
<main class="min-h-screen bg-white">
|
||||||
|
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24">
|
||||||
|
|
||||||
|
<header class="mb-10">
|
||||||
|
<p class="text-sm uppercase tracking-widest text-neutral-500 mb-3">
|
||||||
|
Mentions
|
||||||
|
</p>
|
||||||
|
<h1 class="text-3xl md:text-4xl font-semibold text-neutral-900 leading-tight">
|
||||||
|
Mentions legales
|
||||||
|
</h1>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="text-neutral-800 text-[16px] leading-[1.7] space-y-8">
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h2 class="text-lg font-semibold text-neutral-900 mb-2">Editeur</h2>
|
||||||
|
<p>
|
||||||
|
Jules Neny, architecte HMONP.<br />
|
||||||
|
<em>Transformations resilientes</em><br />
|
||||||
|
Bagneres-de-Bigorre, Pyrenees, France.<br />
|
||||||
|
SIRET : 888 668 860 00016.<br />
|
||||||
|
TVA : non applicable, art. 293B du CGI.<br />
|
||||||
|
Inscription : Ordre des Architectes, CROA Occitanie.<br />
|
||||||
|
RC Pro : MAF n 71183.
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h2 class="text-lg font-semibold text-neutral-900 mb-2">Contact</h2>
|
||||||
|
<p>
|
||||||
|
<a
|
||||||
|
href="mailto:julesneny8@gmail.com"
|
||||||
|
class="underline underline-offset-2 hover:text-neutral-900"
|
||||||
|
>julesneny8@gmail.com</a>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h2 class="text-lg font-semibold text-neutral-900 mb-2">Hebergement</h2>
|
||||||
|
<p>
|
||||||
|
Hetzner Online GmbH<br />
|
||||||
|
Industriestr. 25, 91710 Gunzenhausen, Allemagne.<br />
|
||||||
|
<a
|
||||||
|
href="https://www.hetzner.com/"
|
||||||
|
class="underline underline-offset-2 hover:text-neutral-900"
|
||||||
|
>hetzner.com</a>
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h2 class="text-lg font-semibold text-neutral-900 mb-2">Donnees personnelles</h2>
|
||||||
|
<p>
|
||||||
|
Ce site n'utilise pas de cookies analytiques ni de traceurs publicitaires. Les donnees techniques de connexion (logs serveur) sont conservees a des fins de securite et de diagnostic, conformement aux obligations legales.
|
||||||
|
</p>
|
||||||
|
<p class="mt-3">
|
||||||
|
Les emails collectes via les formulaires de pre-inscription (manifeste imprime, newsletter) ne sont pas cedes a des tiers et servent uniquement aux finalites annoncees.
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h2 class="text-lg font-semibold text-neutral-900 mb-2">Propriete intellectuelle</h2>
|
||||||
|
<p>
|
||||||
|
Les contenus publies (textes, images, code) sont, sauf mention contraire, la propriete de Jules Neny. Reutilisation autorisee a des fins non commerciales avec attribution. Pour tout autre usage, contact prealable.
|
||||||
|
</p>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<p class="text-xs text-neutral-400 italic pt-4">
|
||||||
|
Page V1 ; les details (statut juridique a confirmer, politique cookies si ajoute, mediateur consommation) seront completes en V1.1.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer class="mt-12 pt-8 border-t border-neutral-200 text-sm text-neutral-500">
|
||||||
|
<p>
|
||||||
|
<a href="/" class="underline underline-offset-2 hover:text-neutral-900">
|
||||||
|
← Retour au journal
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</article>
|
||||||
|
</main>
|
||||||
|
</BaseLayout>
|
||||||
Reference in New Issue
Block a user