2 Commits

Author SHA1 Message Date
Jules Neny
090fb581ee feat(v1.6): ajout LinkedIn API V2 comme 3e source journal
- Node Fetch-linkedin (LinkedIn REST API v2/posts, OAuth2 predefined credential)
- Parser parseLinkedIn() dans Normalise : extrait commentary + UGC text, URL post, thumbnail
- Fonction safeJson() parallèle à safeText() pour payloads JSON (vs XML)
- Variable LINKEDIN_MEMBER_ID (env n8n) dans l'URL du endpoint
- .env.example documenté avec LINKEDIN_MEMBER_ID + comment récupération
- counts.linkedin ajouté dans le payload journal.json

Prerequis L.1 bloquants (humain Jules) :
- Redirect URL n8n dans LinkedIn Developer Portal
- Credential LinkedIn OAuth2 créé dans n8n UI
- Member ID Jules récupéré via /v2/me et stocké dans env n8n

Branche depuis feat/v1.5-E-rsshub (V1.5 pas encore mergé sur main).
Pas de modif Astro/Vue — JournalFeed (JournalList.vue) déjà platform-agnostic.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-13 08:37:47 +02:00
Jules Neny
458a152525 V1.5-E RSSHub journal feed (E.1-E.8)
Bascule Behold -> RSSHub self-host (rss.trans-former.fr) :

- InstaFeed.vue : consomme PUBLIC_JOURNAL_URL (journal unifie n8n)
  au lieu de feeds.behold.so direct. Filtre platform=instagram +
  account dans l'url. Plus de feedId Behold a passer.
- ColInsta.astro : supprime props feedId, passe seulement account
  handle (aep.politique, julesneny) pour filtrage cote client.
- .env.example : PUBLIC_BEHOLD_* deprecies (conservees pour compat),
  PUBLIC_JOURNAL_URL note V1.5-E source RSSHub.
- docs/n8n-workflow-journal-aggregate-v2-rsshub.json : workflow V2
  a importer dans n8n. Remplace 3 nodes Behold par : Substack natif
  (julesneny.substack.com/feed) + RSSHub Insta x2 (rss.trans-former.fr/
  instagram/user/*). Cron 3h -> 4h UTC (anti-rate-limit). Parser RSS
  ajoute (CDATA, dedup par GUID).
- carte-o.json : regen prebuild auto.

Ops VPS realisees hors repo (E.1-E.3, E.8 conf) :
- docker pull diygod/rsshub:latest (image officielle, pas rsshub/rsshub)
- /opt/rsshub/docker-compose.yml : RSSHub + Redis cache, port 3006 local
- Caddy : bloc rss.trans-former.fr -> reverse_proxy 127.0.0.1:3006

CHECKPOINT E.4 BLOQUANT : DNS OVH 'rss' IN A 178.104.106.195 a pousser
manuellement par Jules. Sans DNS, rss.trans-former.fr inaccessible.

LIMITE FONCTIONNELLE detectee : RSSHub /instagram/* requiert config
Instagram cookie (ConfigNotFoundError 503). Tant que les env vars
INSTAGRAM_USERNAME/PASSWORD ou cookie ne sont pas fournis a RSSHub,
les routes Insta retourneront vide. Le journal aura 0 item insta et
ColInsta affichera fallbackBio. Substack et Gitea sources OK.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-13 00:56:36 +02:00
16 changed files with 591 additions and 108 deletions

View File

@@ -1,17 +1,24 @@
# Kit (ex-ConvertKit) - newsletter infolettre # Kit (ex-ConvertKit) - newsletter infolettre
KIT_API_SECRET_V4=kit_xxx KIT_API_SECRET_V4=kit_xxx
# Behold.so feed IDs (voir docs/BEHOLD-SETUP.md) # Behold.so : DEPRECATED V1.5-E — remplace par RSSHub self-host (rss.trans-former.fr).
# 1) Inscris-toi sur https://behold.so/dashboard # InstaFeed.vue consomme desormais PUBLIC_JOURNAL_URL (filtre platform=instagram).
# 2) Connecte les 2 comptes Insta (@aep.politique + @julesneny) # Les 2 vars ci-dessous ne sont plus lues ; conservees pour compat.env.local existant.
# 3) Recupere les feed IDs et copie ce fichier vers .env.local puis remplis ci-dessous
PUBLIC_BEHOLD_AEP= PUBLIC_BEHOLD_AEP=
PUBLIC_BEHOLD_JULESNENY= PUBLIC_BEHOLD_JULESNENY=
# Journal unifie (PC6) - URL JSON agrege par n8n cron nocturne # Journal unifie (V1.6) - URL JSON agrege par n8n cron 4h UTC
# Sources : RSSHub self-host (Insta @aep.politique + @julesneny) + Substack natif
# + Atom Gitea natif (git.trans-former.fr/jules.atom) + LinkedIn API V2
# Override en local : pointer vers un mock /public/data/journal.json par exemple # Override en local : pointer vers un mock /public/data/journal.json par exemple
PUBLIC_JOURNAL_URL=https://data.trans-former.fr/journal.json PUBLIC_JOURNAL_URL=https://data.trans-former.fr/journal.json
# LinkedIn (V1.6) - Member ID du profil Jules (format numerique, sans urn: prefix)
# Recuperer via curl -H "Authorization: Bearer TOKEN" https://api.linkedin.com/v2/me | jq .id
# Stocke comme variable d'env n8n (Settings -> Variables) sous le nom LINKEDIN_MEMBER_ID
# Le workflow l'utilise comme : urn:li:person:${LINKEDIN_MEMBER_ID}
LINKEDIN_MEMBER_ID=
# Chatbot upstream (PC7) - URL backend chatbot AEP # Chatbot upstream (PC7) - URL backend chatbot AEP
# V1 : chatbot AEP classique (Mistral Small + 120 fiches) # V1 : chatbot AEP classique (Mistral Small + 120 fiches)
# V1.5 : switch vers LightRAG-PE (1 ligne) # V1.5 : switch vers LightRAG-PE (1 ligne)

File diff suppressed because one or more lines are too long

View File

@@ -1,6 +1,6 @@
{ {
"version": "1.1", "version": "1.1",
"generatedAt": "2026-05-12T09:28:20.972Z", "generatedAt": "2026-05-12T22:53:33.094Z",
"nodes": [ "nodes": [
{ {
"id": "contrat-social-medecine-corps-social", "id": "contrat-social-medecine-corps-social",

View File

@@ -1,23 +1,21 @@
--- ---
import InstaFeed from '../vue/InstaFeed.vue'; import InstaFeed from '../vue/InstaFeed.vue';
// Feed IDs Behold a remplir apres inscription Behold (voir docs/BEHOLD-SETUP.md) // V1.5-E : Behold remplacé par RSSHub self-host. InstaFeed lit désormais
const FEED_AEP = import.meta.env.PUBLIC_BEHOLD_AEP || 'PLACEHOLDER_AEP'; // le journal unifié (PUBLIC_JOURNAL_URL → data.trans-former.fr/journal.json)
const FEED_JULESNENY = import.meta.env.PUBLIC_BEHOLD_JULESNENY || 'PLACEHOLDER_JULESNENY'; // agrégé par n8n depuis rss.trans-former.fr. Plus de feedId Behold à passer.
--- ---
<div class="h-full overflow-y-auto"> <div class="h-full overflow-y-auto">
<InstaFeed <InstaFeed
client:visible client:visible
feedId={FEED_AEP} account="aep.politique"
account="@aep.politique"
accountUrl="https://www.instagram.com/aep.politique/" accountUrl="https://www.instagram.com/aep.politique/"
fallbackBio="Carrousels manifeste AEP ; pensee politique eco-architecture" fallbackBio="Carrousels manifeste AEP ; pensee politique eco-architecture"
/> />
<InstaFeed <InstaFeed
client:visible client:visible
feedId={FEED_JULESNENY} account="julesneny"
account="@julesneny"
accountUrl="https://www.instagram.com/julesneny/" accountUrl="https://www.instagram.com/julesneny/"
fallbackBio="Peinture, poesie, Corse ; archives visuelles personnelles" fallbackBio="Peinture, poesie, Corse ; archives visuelles personnelles"
/> />

View File

@@ -58,14 +58,13 @@ const categories = [
}, },
]; ];
--- ---
<div class="h-full flex flex-col px-4 pb-4 pt-1 md:pt-6 gap-3 md:gap-5"> <div class="h-full flex flex-col p-4 pt-20 md:pt-6 gap-5">
<!-- Catégories accordeon (V1.5-A.4 : "Hashtags" -> "Catégories") <!-- Hashtags accordeon -->
V1.5-A.5 : ouvert par defaut sur mobile + desktop (collé sous tab actif). --> <details id="hashtags-accordion" class="border-t border-neutral-200 pt-4">
<details id="hashtags-accordion" class="md:border-t md:border-neutral-200 md:pt-4" open>
<summary class="font-semibold cursor-pointer select-none flex items-center justify-between"> <summary class="font-semibold cursor-pointer select-none flex items-center justify-between">
<span>Catégories</span> <span>Hashtags</span>
<span class="text-xs text-neutral-400 font-normal">4 catégories</span> <span class="text-xs text-neutral-400 font-normal">4 categories</span>
</summary> </summary>
<div class="mt-3 flex flex-wrap gap-2" id="category-badges"> <div class="mt-3 flex flex-wrap gap-2" id="category-badges">
@@ -122,10 +121,10 @@ const categories = [
</a> </a>
</details> </details>
<!-- Publication chrono (V1.5-A.2 : "Journal" -> "Publication") --> <!-- Journal chrono -->
<section class="border-t border-neutral-200 pt-4 flex-1 overflow-y-auto"> <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"> <h2 class="font-semibold mb-3 flex items-center justify-between">
<span>Publication</span> <span>Journal</span>
<span class="text-xs text-neutral-400 font-normal">chrono</span> <span class="text-xs text-neutral-400 font-normal">chrono</span>
</h2> </h2>
<div id="journal-list" class="space-y-3 text-sm"> <div id="journal-list" class="space-y-3 text-sm">
@@ -135,11 +134,13 @@ const categories = [
</div> </div>
<script> <script>
// V1.5-A.5 : Catégories ouvert par défaut sur mobile ET desktop.
// (Auparavant : forcé ouvert desktop seulement, replié sur mobile.)
// L'attribut `open` SSR garantit l'état initial — script JS retiré pour ne pas
// refermer l'accordéon au resize cross-breakpoint.
const accordion = document.getElementById('hashtags-accordion') as HTMLDetailsElement | null; 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);
}
const closeBtn = document.getElementById('accordion-close'); const closeBtn = document.getElementById('accordion-close');
if (closeBtn && accordion) { if (closeBtn && accordion) {
@@ -312,25 +313,14 @@ const categories = [
dispatchPlatform(platformFilters['politique']); dispatchPlatform(platformFilters['politique']);
} }
// V1.3-E + V1.5-A.6 : intercept clics liens Manifeste -> preview centrale (uniquement sur racine /). // V1.3-E : intercept clics liens Manifeste -> preview centrale (uniquement sur racine /)
// Mobile : on dispatche aussi `mobile-tab-scroll` pos:1 pour swiper vers la col centre // Tagger les liens [data-manifeste-link] et router vers event 'preview-open' si on est sur /.
// (sinon la preview ouvre en background sur la col journal — invisible utilisateur). // Sinon : laisser la navigation native vers /manifeste.
// SwipeContainer écoute `mobile-tab-scroll` et fait scrollTo(1).
document.querySelectorAll<HTMLAnchorElement>('a[data-manifeste-link]').forEach((el) => { document.querySelectorAll<HTMLAnchorElement>('a[data-manifeste-link]').forEach((el) => {
el.addEventListener('click', (e) => { el.addEventListener('click', (e) => {
if (window.location.pathname === '/') { if (window.location.pathname === '/') {
e.preventDefault(); e.preventDefault();
// Mobile : naviguer d'abord vers tab Carte (pos:1) puis ouvrir preview après le swipe.
const isMobile = window.matchMedia('(max-width: 767px)').matches;
if (isMobile) {
document.dispatchEvent(new CustomEvent('mobile-tab-scroll', { detail: { pos: 1 } }));
// Laisse le temps au swipe Embla de se positionner avant d'ouvrir la preview
setTimeout(() => {
window.dispatchEvent(new CustomEvent('preview-open', { detail: { type: 'manifeste' } })); window.dispatchEvent(new CustomEvent('preview-open', { detail: { type: 'manifeste' } }));
}, 350);
} else {
window.dispatchEvent(new CustomEvent('preview-open', { detail: { type: 'manifeste' } }));
}
} }
// sinon : navigation normale vers /manifeste (fallback SEO + no-JS) // sinon : navigation normale vers /manifeste (fallback SEO + no-JS)
}); });

View File

@@ -0,0 +1,103 @@
---
// HamburgerMenu - drawer slide-in left avec liens nav (PC2)
// Astro vanilla + script inline, pas besoin d'island Vue
---
<button
id="hamburger-trigger"
type="button"
class="fixed top-4 right-4 z-50 p-3 bg-white/95 border border-neutral-200 rounded-lg shadow-md hover:bg-white transition-colors md:hidden"
aria-label="Ouvrir le menu"
aria-expanded="false"
aria-controls="hamburger-drawer"
>
<span class="block w-5 h-0.5 bg-neutral-800 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-800"></span>
</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"
>
&times;
</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>

View File

@@ -1,22 +1,20 @@
--- ---
// MobileTabBar - indicateur de colonne active sur mobile (V1.1-D.1) // MobileTabBar - indicateur de colonne active sur mobile (V1.1-D.1)
// V1.5-A.3 : suppression border-b (ligne séparatrice sous header mobile supprimée).
// V1.5-A.2 : "Journal" -> "Publication" (vocabulaire visiteur).
// Ecoute l'event "swipe-position-change" emis par SwipeContainer.vue // Ecoute l'event "swipe-position-change" emis par SwipeContainer.vue
// Affiche : Publication | Carte | Insta - colonne active surlignee // Affiche : Journal | Carte | Insta - colonne active surlignee
--- ---
<nav <nav
id="mobile-tab-bar" id="mobile-tab-bar"
aria-label="Navigation colonnes" aria-label="Navigation colonnes"
class="fixed top-12 left-0 right-0 z-30 h-11 bg-white flex items-stretch justify-around md:hidden" class="fixed top-12 left-0 right-0 z-30 h-11 bg-white border-b border-neutral-200 flex items-stretch justify-around md:hidden"
> >
<button <button
type="button" type="button"
data-tab-index="0" data-tab-index="0"
class="mobile-tab flex-1 text-sm px-2 border-b-2 transition-colors" class="mobile-tab flex-1 text-sm px-2 border-b-2 transition-colors"
aria-label="Aller à la publication" aria-label="Aller au journal"
> >
Publication Journal
</button> </button>
<button <button
type="button" type="button"

View File

@@ -12,7 +12,7 @@
import '@fontsource-variable/roboto-condensed/wght.css'; import '@fontsource-variable/roboto-condensed/wght.css';
--- ---
<header <header
class="site-header w-full md:border-b md:border-[#E5E7EB] bg-[#FAFAF7] text-[#0F172A] px-4 md:px-6 flex items-center" class="site-header w-full border-b border-[#E5E7EB] bg-[#FAFAF7] text-[#0F172A] px-4 md:px-6 flex items-center"
role="banner" role="banner"
> >
<!-- Bloc identite (gauche) --> <!-- Bloc identite (gauche) -->

View File

@@ -1,7 +1,4 @@
<script setup lang="ts"> <script setup lang="ts">
// V1.5-A.7 : fallback mobile retiré — la Carte O D3-force fonctionne sur viewport <768px
// (testé tel ≥6"). Auparavant : message "optimisée desktop" + miniature SVG statique.
// Pourquoi : décision τ.15 PILOTE trop agressive — la mindmap tourne, juste plus dense.
import { ref, onMounted, computed } from 'vue' import { ref, onMounted, computed } from 'vue'
import CarteO from './CarteO.vue' import CarteO from './CarteO.vue'
import CarteOContextMenu from './CarteOContextMenu.vue' import CarteOContextMenu from './CarteOContextMenu.vue'
@@ -47,6 +44,7 @@ const error = ref<string | null>(null)
const selectedNode = ref<CarteNode | null>(null) const selectedNode = ref<CarteNode | null>(null)
const contextX = ref(0) const contextX = ref(0)
const contextY = ref(0) const contextY = ref(0)
const isMobileScreen = ref(false)
const familyColors = computed(() => const familyColors = computed(() =>
data.value?.meta?.familyColors || { data.value?.meta?.familyColors || {
@@ -65,6 +63,7 @@ function onNodeClick(payload: { node: CarteNode; x: number; y: number }) {
} }
onMounted(async () => { onMounted(async () => {
isMobileScreen.value = window.innerWidth < 768
try { try {
const res = await fetch(props.src) const res = await fetch(props.src)
if (!res.ok) throw new Error(`HTTP ${res.status}`) if (!res.ok) throw new Error(`HTTP ${res.status}`)
@@ -73,13 +72,37 @@ onMounted(async () => {
console.error('[CarteO] failed to load', e) console.error('[CarteO] failed to load', e)
error.value = e?.message || 'Erreur de chargement' error.value = e?.message || 'Erreur de chargement'
} }
window.addEventListener('resize', () => {
isMobileScreen.value = window.innerWidth < 768
})
}) })
</script> </script>
<template> <template>
<div class="wrapper"> <div class="wrapper">
<!-- Mobile fallback (V1) -->
<div v-if="isMobileScreen" class="mobile-fallback">
<div class="mini-map">
<svg viewBox="0 0 200 120" xmlns="http://www.w3.org/2000/svg" aria-hidden="true">
<circle cx="100" cy="60" r="8" fill="#3b82f6" />
<circle cx="50" cy="35" r="5" fill="#10b981" />
<circle cx="150" cy="35" r="5" fill="#f59e0b" />
<circle cx="55" cy="90" r="5" fill="#ef4444" />
<circle cx="145" cy="90" r="5" fill="#8b5cf6" />
<line x1="100" y1="60" x2="50" y2="35" stroke="#94a3b8" stroke-width="0.8" opacity="0.5" />
<line x1="100" y1="60" x2="150" y2="35" stroke="#94a3b8" stroke-width="0.8" opacity="0.5" />
<line x1="100" y1="60" x2="55" y2="90" stroke="#94a3b8" stroke-width="0.8" opacity="0.5" />
<line x1="100" y1="60" x2="145" y2="90" stroke="#94a3b8" stroke-width="0.8" opacity="0.5" />
</svg>
</div>
<p class="msg">
Carte O optimisee desktop. Retournez sur grand ecran pour explorer la mindmap interactive.
</p>
</div>
<!-- Loading state --> <!-- Loading state -->
<div v-if="!data && !error" class="state"> <div v-else-if="!data && !error" class="state">
<span>Chargement de la Carte O...</span> <span>Chargement de la Carte O...</span>
</div> </div>
@@ -125,4 +148,34 @@ onMounted(async () => {
.state.error { .state.error {
color: #dc2626; color: #dc2626;
} }
.mobile-fallback {
height: 100%;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 1.5rem;
text-align: center;
gap: 1rem;
}
.mini-map {
width: 70%;
max-width: 240px;
}
.mini-map svg {
width: 100%;
height: auto;
}
.msg {
color: #6b7280;
font-size: 0.85rem;
line-height: 1.45;
font-style: italic;
max-width: 24rem;
margin: 0;
}
</style> </style>

View File

@@ -1,16 +1,9 @@
<script setup lang="ts"> <script setup lang="ts">
// V1.5-A.9 : iframe AEP responsive sur mobile.
// AEP est confirmé responsive (viewport meta + media queries Tailwind md:).
// Mobile (<768px) : on retire le hack scale 0.42 + viewport simulé 1440px,
// l'iframe rend à width:100% / height:100% nativement.
// Desktop (≥768px) : on conserve le rendu desktop forcé (scale dynamique) pour garder
// la mise en page riche de la carte AEP dans la col centre étroite (~480px).
import { ref, computed, onMounted, onUnmounted } from 'vue' import { ref, computed, onMounted, onUnmounted } from 'vue'
const iframeRef = ref<HTMLIFrameElement | null>(null) const iframeRef = ref<HTMLIFrameElement | null>(null)
const wrapperRef = ref<HTMLDivElement | null>(null) const wrapperRef = ref<HTMLDivElement | null>(null)
const skeletonHidden = ref(false) const skeletonHidden = ref(false)
const isMobile = ref(false)
// Force rendu desktop de l'iframe AEP : viewport simulee 1440px + scale dynamique // Force rendu desktop de l'iframe AEP : viewport simulee 1440px + scale dynamique
const VIEWPORT_W = 1440 const VIEWPORT_W = 1440
@@ -23,24 +16,12 @@ const updateScale = () => {
if (w > 0) iframeScale.value = w / VIEWPORT_W if (w > 0) iframeScale.value = w / VIEWPORT_W
} }
const iframeStyle = computed(() => { const iframeStyle = computed(() => ({
if (isMobile.value) {
// Mobile : rendu natif responsive, pas de scale
return {
width: '100%',
height: '100%',
transform: 'none',
transformOrigin: '0 0',
}
}
// Desktop : viewport simulé 1440px + scale dynamique (workaround col étroite)
return {
width: VIEWPORT_W + 'px', width: VIEWPORT_W + 'px',
height: (100 / iframeScale.value) + '%', height: (100 / iframeScale.value) + '%',
transform: `scale(${iframeScale.value})`, transform: `scale(${iframeScale.value})`,
transformOrigin: '0 0', transformOrigin: '0 0',
} }))
})
let fallbackTimer: ReturnType<typeof setTimeout> | null = null let fallbackTimer: ReturnType<typeof setTimeout> | null = null
@@ -52,13 +33,7 @@ const revealIframe = () => {
skeletonHidden.value = true skeletonHidden.value = true
} }
const detectMobile = () => {
isMobile.value = window.matchMedia('(max-width: 767px)').matches
}
onMounted(() => { onMounted(() => {
detectMobile()
window.addEventListener('resize', detectMobile)
if (wrapperRef.value && typeof ResizeObserver !== 'undefined') { if (wrapperRef.value && typeof ResizeObserver !== 'undefined') {
updateScale() updateScale()
resizeObs = new ResizeObserver(updateScale) resizeObs = new ResizeObserver(updateScale)
@@ -73,7 +48,6 @@ onMounted(() => {
onUnmounted(() => { onUnmounted(() => {
resizeObs?.disconnect() resizeObs?.disconnect()
resizeObs = null resizeObs = null
window.removeEventListener('resize', detectMobile)
if (fallbackTimer) { if (fallbackTimer) {
clearTimeout(fallbackTimer) clearTimeout(fallbackTimer)
fallbackTimer = null fallbackTimer = null

View File

@@ -1,46 +1,80 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted } from 'vue'; import { ref, onMounted } from 'vue';
interface BeholdPost { // V1.5-E : on consomme désormais le journal unifié agrégé par n8n
// (sources RSSHub self-host → data.trans-former.fr/journal.json),
// au lieu de l'API Behold directe (rate-limit, plafond 6 posts gratuit).
interface JournalItem {
id: string;
platform: string;
date: string;
titre: string;
extrait?: string;
url: string;
thumbnail: string | null;
}
interface JournalPayload {
generatedAt: string;
items: JournalItem[];
}
interface InstaPost {
id: string; id: string;
permalink: string; permalink: string;
mediaUrl: string; thumbnailUrl: string | null;
thumbnailUrl?: string; caption: string;
caption?: string;
mediaType: 'IMAGE' | 'VIDEO' | 'CAROUSEL_ALBUM';
timestamp: string; timestamp: string;
} }
const props = defineProps<{ const props = defineProps<{
feedId: string; /** handle Instagram sans @ (ex: 'aep.politique', 'julesneny') — sert à filtrer le journal */
account: string; account: string;
accountUrl: string; accountUrl: string;
fallbackBio?: string; fallbackBio?: string;
/** nombre max de posts affichés (défaut 6, parité avec ancien Behold) */
max?: number;
}>(); }>();
const posts = ref<BeholdPost[]>([]); const posts = ref<InstaPost[]>([]);
const loading = ref(true); const loading = ref(true);
const error = ref<string | null>(null); const error = ref<string | null>(null);
const isPlaceholder = (id: string) => !id || id.startsWith('PLACEHOLDER_'); 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 accountHandle = (props.account || '').replace(/^@/, '').toLowerCase();
const limit = props.max ?? 6;
onMounted(async () => { onMounted(async () => {
if (isPlaceholder(props.feedId)) {
loading.value = false;
error.value = 'no-feed-id';
return;
}
try { try {
const controller = new AbortController(); const controller = new AbortController();
const timeoutId = setTimeout(() => controller.abort(), 5000); const timeoutId = setTimeout(() => controller.abort(), 8000);
const res = await fetch(`https://feeds.behold.so/${props.feedId}`, { const res = await fetch(JOURNAL_URL, {
signal: controller.signal, signal: controller.signal,
cache: 'no-store',
}); });
clearTimeout(timeoutId); clearTimeout(timeoutId);
if (!res.ok) throw new Error(`Behold returned ${res.status}`); if (!res.ok) throw new Error(`Journal returned ${res.status}`);
const data = await res.json(); const data = (await res.json()) as JournalPayload;
const items: BeholdPost[] = Array.isArray(data) ? data : (data.posts ?? []); const all = Array.isArray(data?.items) ? data.items : [];
posts.value = items.slice(0, 6); posts.value = all
.filter(
(it) =>
it.platform === 'instagram' &&
typeof it.url === 'string' &&
it.url.toLowerCase().includes(`/${accountHandle}`),
)
.slice(0, limit)
.map((it) => ({
id: it.id,
permalink: it.url,
thumbnailUrl: it.thumbnail ?? null,
caption: it.titre || it.extrait || '',
timestamp: it.date,
}));
if (!posts.value.length) error.value = 'no-posts';
} catch (e) { } catch (e) {
error.value = (e as Error).message || 'fetch-error'; error.value = (e as Error).message || 'fetch-error';
} finally { } finally {
@@ -71,24 +105,28 @@ onMounted(async () => {
/> />
</div> </div>
<div <div v-else-if="posts.length" class="grid grid-cols-2 gap-1 p-1">
v-else-if="posts.length"
class="grid grid-cols-2 gap-1 p-1"
>
<a <a
v-for="post in posts" v-for="post in posts"
:key="post.id" :key="post.id"
:href="post.permalink" :href="post.permalink"
target="_blank" target="_blank"
rel="noopener" rel="noopener"
class="block aspect-square overflow-hidden group" class="block aspect-square overflow-hidden group bg-neutral-100"
> >
<img <img
:src="post.thumbnailUrl || post.mediaUrl" v-if="post.thumbnailUrl"
:src="post.thumbnailUrl"
:alt="post.caption?.slice(0, 80) || account" :alt="post.caption?.slice(0, 80) || account"
loading="lazy" loading="lazy"
class="w-full h-full object-cover group-hover:scale-105 transition-transform" class="w-full h-full object-cover group-hover:scale-105 transition-transform"
/> />
<span
v-else
class="w-full h-full flex items-center justify-center text-[10px] text-neutral-400 p-2 text-center"
>
{{ post.caption?.slice(0, 60) || account }}
</span>
</a> </a>
</div> </div>

View File

@@ -2,11 +2,14 @@
export const prerender = true; export const prerender = true;
import BaseLayout from '../layouts/BaseLayout.astro'; import BaseLayout from '../layouts/BaseLayout.astro';
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
--- ---
<BaseLayout <BaseLayout
title="A propos - Jules Neny" title="A propos - Jules Neny"
description="Architecte HMONP, ecrivain politique, facilitateur. Bagneres-de-Bigorre, Pyrenees." description="Architecte HMONP, ecrivain politique, facilitateur. Bagneres-de-Bigorre, Pyrenees."
> >
<HamburgerMenu />
<main class="h-full overflow-y-auto bg-white"> <main class="h-full overflow-y-auto bg-white">
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24"> <article class="max-w-2xl mx-auto px-6 py-16 md:py-24">

View File

@@ -6,10 +6,12 @@ import ColJournal from '../components/astro/ColJournal.astro';
import ColCentre from '../components/astro/ColCentre.astro'; import ColCentre from '../components/astro/ColCentre.astro';
import ColInsta from '../components/astro/ColInsta.astro'; import ColInsta from '../components/astro/ColInsta.astro';
import SwipeContainer from '../components/vue/SwipeContainer.vue'; import SwipeContainer from '../components/vue/SwipeContainer.vue';
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
import MobileTabBar from '../components/astro/MobileTabBar.astro'; import MobileTabBar from '../components/astro/MobileTabBar.astro';
import PopupOnboarding from '../components/astro/PopupOnboarding.astro'; import PopupOnboarding from '../components/astro/PopupOnboarding.astro';
--- ---
<BaseLayout title="trans-former.fr"> <BaseLayout title="trans-former.fr">
<HamburgerMenu />
<MobileTabBar /> <MobileTabBar />
<PopupOnboarding /> <PopupOnboarding />

View File

@@ -2,6 +2,7 @@
export const prerender = true; export const prerender = true;
import BaseLayout from '../layouts/BaseLayout.astro'; import BaseLayout from '../layouts/BaseLayout.astro';
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
--- ---
<BaseLayout <BaseLayout
title="Manifeste - Architecture d'Ecologie Politique" title="Manifeste - Architecture d'Ecologie Politique"
@@ -10,6 +11,8 @@ import BaseLayout from '../layouts/BaseLayout.astro';
articleDate="2026-05-01" articleDate="2026-05-01"
articleDescription="Manifeste fondateur des Agences d'Ecologie Politique - un commun vivant pour bifurquer ensemble." articleDescription="Manifeste fondateur des Agences d'Ecologie Politique - un commun vivant pour bifurquer ensemble."
> >
<HamburgerMenu />
<main class="h-full overflow-y-auto bg-white"> <main class="h-full overflow-y-auto bg-white">
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24"> <article class="max-w-2xl mx-auto px-6 py-16 md:py-24">

View File

@@ -2,11 +2,14 @@
export const prerender = true; export const prerender = true;
import BaseLayout from '../../layouts/BaseLayout.astro'; import BaseLayout from '../../layouts/BaseLayout.astro';
import HamburgerMenu from '../../components/astro/HamburgerMenu.astro';
--- ---
<BaseLayout <BaseLayout
title="Commander la version imprimee - Manifeste AEP" title="Commander la version imprimee - Manifeste AEP"
description="Pre-inscription pour la version imprimee du manifeste Architecture d'Ecologie Politique." description="Pre-inscription pour la version imprimee du manifeste Architecture d'Ecologie Politique."
> >
<HamburgerMenu />
<main class="h-full overflow-y-auto bg-white"> <main class="h-full overflow-y-auto bg-white">
<article class="max-w-xl mx-auto px-6 py-16 md:py-24"> <article class="max-w-xl mx-auto px-6 py-16 md:py-24">

View File

@@ -2,11 +2,14 @@
export const prerender = true; export const prerender = true;
import BaseLayout from '../layouts/BaseLayout.astro'; import BaseLayout from '../layouts/BaseLayout.astro';
import HamburgerMenu from '../components/astro/HamburgerMenu.astro';
--- ---
<BaseLayout <BaseLayout
title="Mentions legales - trans-former.fr" title="Mentions legales - trans-former.fr"
description="Mentions legales du site trans-former.fr." description="Mentions legales du site trans-former.fr."
> >
<HamburgerMenu />
<main class="h-full overflow-y-auto bg-white"> <main class="h-full overflow-y-auto bg-white">
<article class="max-w-2xl mx-auto px-6 py-16 md:py-24"> <article class="max-w-2xl mx-auto px-6 py-16 md:py-24">