PARTIE IV — Pour le développeur intégrateur
- CHAPITRE 17 — Préparer un site Dolibarr Website
- CHAPITRE 18 — Annoter un template avec des slots
- CHAPITRE 19 — La grammaire des slots en détail
- CHAPITRE 20 — Insérer des données Dolibarr (shortcodes)
- CHAPITRE 21 — Gérer le multilingue (pages sœurs)
- CHAPITRE 22 — Créer ses propres gabarits de page
- CHAPITRE 23 — Le catalogue produit dynamique en profondeur
- CHAPITRE 24 — Les formulaires publics
CHAPITRE 17 — Préparer un site Dolibarr Website
Avant de pouvoir éditer un site via le module, ce site doit exister côté Dolibarr Website. Ce chapitre s'adresse au développeur qui démarre un nouveau projet : créer le site, le câbler proprement, et préparer le terrain pour le Studio.
Étape 1 — Créer le site dans Dolibarr Website
- Connectez-vous à Dolibarr en tant qu'administrateur.
- Rendez-vous dans Accueil → Sites web.
- Cliquez sur Nouveau site.
- Renseignez les informations suivantes :
- Référence : identifiant court sans espace, par exemple
monsiteoukeaticweb. C'est l'identifiant interne, utilisé partout (URLs des wrappers, dossiers de données, constantes de configuration). - Description : libellé facultatif.
- Virtualhost principal : URL publique, par exemple
https://monsite.com. - Langue principale : code de la locale, par exemple
fr_FR. - Autres langues : valeurs séparées par des virgules, par exemple
en_US,de_DE,es_ES.
- Référence : identifiant court sans espace, par exemple
- Enregistrez.
Convention de référence — Privilégiez un identifiant court et stable. Cette référence apparaîtra dans les chemins de fichiers, les URLs internes et les constantes de configuration. La modifier ultérieurement nécessite plusieurs ajustements.
Étape 2 — Configurer le virtualhost Apache
Le module Website ne configure pas Apache pour vous. Vous devez créer un VirtualHost qui pointe sur le docroot du site.
Arborescence type
/var/www/monsite/ # docroot Apache du site
├── index.php # wrapper page d'accueil
├── <alias>.php # autres wrappers générés
├── master.inc.php # bootstrap (inclut DOL_DOCUMENT_ROOT)
├── medias -> lien symbolique vers DOL_DATA_ROOT/<entity>/medias/
└── ...
DOL_DATA_ROOT/<entity>/website/monsite/ # dossier de données du site
├── page1.tpl.php # gabarits des pages
├── page2.tpl.php
├── ...
└── htmlheader.html # en-tête commun éventuel
Exemple de VirtualHost minimal
<VirtualHost *:443>
ServerName monsite.com
DocumentRoot /var/www/monsite
SSLEngine on
SSLCertificateFile /etc/letsencrypt/live/monsite.com/fullchain.pem
SSLCertificateKeyFile /etc/letsencrypt/live/monsite.com/privkey.pem
<Directory /var/www/monsite>
Options FollowSymLinks
AllowOverride None
Require all granted
</Directory>
# PHP-FPM dédié recommandé
<FilesMatch \.php$>
SetHandler "proxy:unix:/run/php/php8.2-fpm.monsite.sock|fcgi://localhost"
</FilesMatch>
</VirtualHost>
Note — Un pool PHP-FPM dédié est recommandé pour isoler les performances et les variables d'environnement par site. Le pool peut tourner sous un utilisateur spécifique pour faciliter la gestion des permissions.
Étape 3 — Le lien symbolique medias
Pour que les images téléversées soient servies directement par Apache (mode média « native »), il faut un lien symbolique :
cd /var/www/monsite
ln -sfn /mnt/data/dolibarr/<entity>/medias medias
ls -la medias # doit pointer sur le dossier de données Dolibarr
Conseil — Si vous ne pouvez pas créer ce lien symbolique, basculez le site en mode média module dans la configuration du module. Les images seront servies via document.php, avec un coût de performance modéré.
Étape 4 — Le master.inc.php multicompany
Si votre installation Dolibarr fonctionne en multicompany, le fichier master.inc.php du site doit définir DOLENTITY avant le chargement :
<?php
// /var/www/monsite/master.inc.php
define('DOLENTITY', 2); // l'entity correspondant au client
require_once '/var/www/dolibarr/htdocs/master.inc.php';
Avertissement — Sur une instance multicompany sans DOLENTITY, le site renverra une erreur 503 ou affichera les données de la mauvaise entity. C'est l'erreur la plus fréquemment rencontrée lors de la mise en service.
Étape 5 — Créer la page d'accueil
Toujours dans le module Website Dolibarr :
- Sélectionnez votre site.
- Cliquez sur Nouvelle page.
- Renseignez :
- Page URL :
home,indexou autre. - Title : Accueil.
- Type container :
page. - Lang :
fr_FR. - Status : Brouillon (le passage à Publié interviendra plus tard).
- Page URL :
- Enregistrez.
- Dolibarr crée automatiquement les fichiers
/var/www/monsite/home.phpetpage<N>.tpl.php.
Étape 6 — Activer le site dans le module
La procédure complète est détaillée au Chapitre 7. En résumé :
- Outils → InfraSStudio → Configuration.
- Cochez votre site dans la liste.
- Choisissez le mode média (native recommandé).
- Enregistrez.
Étape 7 — Annoter les premières pages avec des slots
Cette étape fait l'objet du Chapitre 18. En quelques mots : modifiez le fichier page<N>.tpl.php pour ajouter des tokens {{slot:nom|type=...}} aux endroits que vous souhaitez rendre éditables.
Étape 8 — Vérifier avec la page Diagnostic
Avant de livrer le site au client, lancez le diagnostic (Outils → InfraSStudio → Diagnostic) :
- Tous les voyants verts dans Environnement, Schéma SQL, Stockage et Intégration Dolibarr.
- Le site géré apparaît dans la section Sites avec docroot résolu, mode média correct et dossier de données accessible en écriture.
Récapitulatif
Votre site est prêt si :
- Le site est créé dans Dolibarr Website (référence, virtualhost, langues).
- Le VirtualHost Apache pointe sur le bon docroot.
- Le lien symbolique
mediasest en place (mode native), ou la constanteINFRASSTUDIO_SITE_<id>_MEDIA_MODE=moduleest définie. - Le fichier
master.inc.phpdéfinitDOLENTITYen multicompany. - Au moins une page existe (l'accueil).
- Le site est coché dans la configuration du module.
- La page Diagnostic est entièrement verte.
Le chapitre suivant aborde l'annotation des templates avec des slots.
CHAPITRE 18 — Annoter un template avec des slots
Le slot est l'unité de base du module. Ce chapitre vous montre comment transformer un HTML statique en HTML annoté, prêt à être édité par le client. Il s'agit de l'opération la plus fréquente dans votre travail d'intégrateur.
Le principe en deux phrases
- Vous écrivez votre HTML normalement, comme vous le feriez sans le module.
- Aux endroits que vous voulez rendre éditables, vous remplacez le contenu en dur par un token
{{slot:...}}.
C'est tout. Le module se charge du reste : détection automatique, persistance, rendu, interface d'édition.
Premier exemple complet
Avant — page non éditable
<section class="hero">
<h1>Bienvenue chez Keatic</h1>
<p>Votre partenaire numérique de confiance depuis 2010.</p>
<a href="/contact" class="btn">Nous contacter</a>
</section>
Après — page éditable via le Studio
<section class="hero">
<h1>{{slot:hero_title|type=text|default=Bienvenue chez Keatic|label=Titre du hero|group=hero}}</h1>
<p>{{slot:hero_lead|type=textarea|default=Votre partenaire numérique de confiance depuis 2010.|label=Accroche|group=hero}}</p>
<a href="{{slot:hero_cta_url|type=url|default=/contact|label=URL du bouton|group=hero}}" class="btn">
{{slot:hero_cta_label|type=text|default=Nous contacter|label=Libellé du bouton|group=hero}}
</a>
</section>
Résultat — Le client voit dans le Studio un panneau « Hero » comportant quatre champs (titre, accroche, URL du bouton, libellé du bouton). Aucun HTML à modifier de son côté.
Anatomie d'un token de slot
{{slot:<identifiant>|type=<type>|attribut1=valeur1|attribut2=valeur2}}
Élément | Obligatoire | Description |
|---|---|---|
| Oui | Préfixe fixe qui déclare un slot. |
Identifiant | Oui | Unique par page. Caractères acceptés : minuscules, chiffres, tiret bas. 64 caractères maximum. |
| Oui | text, textarea, richtext, image, url, number, select, bool, icon, color. |
| Recommandé | Valeur de repli si le slot n'est pas encore édité. |
| Recommandé | Libellé affiché à l'éditeur dans le Studio. |
| Facultatif | Regroupe les slots dans une section de l'interface. |
| Facultatif | Aide affichée sous le champ. |
| Facultatif | Limite de caractères pour text et textarea. |
| Si type=select | Liste CSV des valeurs possibles. |
Conseil — La grammaire complète est détaillée au Chapitre 19.
Où placer un slot
Un slot peut être placé à de nombreux endroits dans votre HTML.
Dans le contenu d'une balise (cas le plus fréquent)
<h1>{{slot:title|type=text|default=Bienvenue}}</h1>
Dans un attribut HTML
<img src="{{slot:hero_image|type=image|default=/medias/hero.jpg}}" alt="...">
<a href="{{slot:cta_url|type=url|default=/contact}}">...</a>
<section style="background-color:{{slot:bg_color|type=color|default=#19052d}}">
Dans une chaîne de classes CSS
<i class="fa-solid {{slot:hero_icon|type=text|default=fa-rocket}}"></i>
À ne pas faire — Ne placez pas un slot dans un commentaire HTML, dans un bloc <script>, ou dans du JSON intégré. Le scanner détecte tous les tokens et peut générer des slots fantômes.
Convention de nommage des identifiants
L'identifiant est libre, mais une convention claire facilite la maintenance.
Schéma recommandé
<section>_<champ> ou <page>_<section>_<champ>
Bon | À éviter |
|---|---|
|
(trop générique, risque de collision) |
|
(incompréhensible) |
|
(les majuscules sont rejetées par le scanner) |
Recommandé — Si votre site comporte cinq sections (hero, features, stats, testimonials, footer), préfixez tous vos slots par leur section. Le client retrouvera ainsi plus rapidement ce qu'il cherche dans le Studio.
Utiliser le groupe pour structurer l'interface
L'attribut group détermine sous quelle section le slot apparaît dans le panneau « Contenu de la page » du Studio.
{{slot:hero_title|type=text|group=hero|...}}
{{slot:hero_lead|type=textarea|group=hero|...}}
{{slot:hero_image|type=image|group=hero|...}}
{{slot:features_card_1_title|type=text|group=features|...}}
{{slot:features_card_2_title|type=text|group=features|...}}
{{slot:features_card_3_title|type=text|group=features|...}}
{{slot:footer_copyright|type=text|group=footer|...}}
Dans le Studio, ces slots seront regroupés visuellement sous trois sections : Hero, Features et Footer.
Le rescan après ajout
Ajouter un slot dans un fichier tpl.php ne suffit pas : le module doit le détecter et l'enregistrer en base. Pour cela, lancez un rescan.
Méthode A — Via l'interface Studio
- Outils → InfraSStudio → Contenu des pages.
- Sélectionnez votre site.
- Cliquez sur Rescanner en haut.
- Le module parcourt tous les fichiers
tpl.phpet synchronise les slots.
Méthode B — En ligne de commande
php htdocs/custom/infrasstudio/scripts/rescan_slots.php <ref-du-site> --entity=<N>
Sortie typique :
Rescanning website #1 — monsite (entity 2)
Pages scanned : 22
Slots added : 3
Slots updated : 40
Slots orphaned : 0
Mode de validation — Ajoutez l'option --lint pour détecter les erreurs de syntaxe avant de les valider :php rescan_slots.php <ref> --lint
Le code de sortie est:
- 0 en l'absence d'erreur
- 1 en présence d'avertissements
- 2 en cas d'erreurs.
Cette commande peut être intégrée dans un crochet de pré-commit.
Slots orphelins
Lorsque vous supprimez un slot d'un fichier tpl.php, son enregistrement en base passe au statut « orphelin ». Le module le conserve pendant 30 jours avant de le supprimer automatiquement (via une tâche planifiée).
Cette politique présente un avantage : si vous restaurez le slot dans le HTML pendant cette période, sa valeur précédente est récupérée.
Note — Pour forcer la purge immédiate des slots orphelins :php rescan_slots.php <ref> --purge-orphans
Liste de contrôle d'annotation
Avant de livrer une page annotée :
- Chaque slot possède un
typeexplicite. - Chaque slot possède une valeur
default, garantissant la lisibilité de la page si la base venait à se vider. - Chaque slot possède un
labelen français clair, destiné à l'éditeur. - Les slots sont regroupés par section avec
group. - Les identifiants suivent la convention
<section>_<champ>. - Le rescan a été lancé sans avertissement ni erreur.
- Vous avez ouvert la page dans le Studio et vérifié la présence de tous les slots.
Le chapitre suivant détaille la grammaire complète des slots, avec tous les types et leurs comportements.
CHAPITRE 19 — La grammaire des slots en détail
Référence exhaustive de tous les types de slot et de leurs attributs. Gardez ce chapitre à portée de main lors de l'annotation d'un template.
Format général
{{slot:<name>|type=<type>|default=<value>|label=<text>|group=<section>|help=<text>|maxlength=<int>|options=<csv>}}
Tous les attributs, à l'exception de name et type, sont facultatifs. L'ordre est libre. Le séparateur est le caractère pipe.
Règles syntaxiques
Règle | Détail |
|---|---|
Identifiant |
, 64 caractères maximum. Pas de tirets, pas de majuscules. |
Unicité | Unique par page. Deux slots avec le même nom sur la même page produisent un avertissement au scan. |
Pas d'espaces | Aucun espace de part et d'autre du signe
. Aucun espace dans les noms d'attributs. |
Échappement | Le pipe à l'intérieur d'une valeur n'est pas pris en charge. Évitez-le dans les valeurs
. |
Une seule ligne | Un token de slot tient sur une seule ligne. Aucun saut de ligne ne doit y être présent. |
Les dix types de slot
Type text — Texte court
Champ d'une seule ligne, sans mise en forme.
<h1>{{slot:hero_title|type=text|default=Bienvenue|label=Titre principal|maxlength=80}}</h1>
- Interface : champ de saisie simple.
- Idéal pour : titres, libellés, boutons d'appel à l'action, copyright.
- Compteur de caractères affiché si
maxlengthest défini.
Type textarea — Texte long brut
Multi-lignes, sans mise en forme HTML.
<p class="lead">{{slot:hero_lead|type=textarea|default=Une accroche.|label=Accroche|maxlength=300}}</p>
- Interface : zone de texte multi-lignes.
- Les retours à la ligne sont automatiquement convertis en balises
<br>au rendu. - Idéal pour : accroches, paragraphes simples.
Type richtext — Texte riche WYSIWYG
Mise en forme complète via CKEditor.
<div class="post-body">
{{slot:post_body|type=richtext|default=<p>Contenu de l'article.</p>|label=Corps de l'article}}
</div>
- Interface : éditeur CKEditor (Format, gras, italique, listes, liens, couleurs, alignement, etc.).
- Idéal pour : corps d'articles, descriptions longues, conditions générales.
- La valeur stockée contient du HTML (
<p>,<strong>, etc.).
Type image — Média image
<img src="{{slot:hero_image|type=image|default=/medias/hero.jpg|label=Image hero}}" alt="...">
- Interface : champ texte, bouton de sélection dans la bibliothèque, bouton de suppression et vignette d'aperçu.
- Stocke soit une URL absolue, soit un shortcode
{{media:ref=X.url}}. - Non traduisible par défaut : la même image est servie dans toutes les langues.
Type url — Lien
<a href="{{slot:hero_cta_url|type=url|default=/contact|label=URL du bouton}}">...</a>
- Interface : champ texte avec validation URL légère.
- Idéal pour : URL de bouton, liens vers les réseaux sociaux, URL de partage.
Type icon — Icône FontAwesome
{{slot:feature_1_icon|type=icon|default=fa-solid fa-rocket|label=Icône feature 1}}
- Interface : champ classe, sélecteur de couleur, vingt icônes proposées en accès rapide, aperçu en direct.
- Stockage en JSON :
{"class":"fa-solid fa-star","color":"#fbbf24"}. - Rendu :
<i class="fa-solid fa-star" style="color:#fbbf24"></i>. - Sécurisé : liste blanche sur la classe (a-zA-Z0-9_-) et expression régulière hexadécimale sur la couleur.
Type color — Couleur
<section style="background-color:{{slot:section_bg|type=color|default=#19052d|label=Couleur de fond}}">
- Interface : sélecteur de couleur HTML5, champ hexadécimal, bouton de retour à la valeur par défaut.
- Format strict :
#RRGGBBou#RRGGBBAA(avec transparence). - Non traduisible.
Type number — Nombre
<span class="stat">{{slot:stats_clients|type=number|default=42|label=Nombre de clients}}</span>
Type bool — Booléen
<?php if ('{{slot:hero_show_badge|type=bool|default=1|label=Afficher le badge}}'): ?>
<span class="badge">Nouveau</span>
<?php endif; ?>
- Interface : case à cocher.
- Valeur : 0 ou 1.
Type select — Liste déroulante
<section class="hero hero-{{slot:hero_layout|type=select|options=light,dark,gradient|default=light|label=Style du hero}}">
Valeurs par défaut spéciales
default=@lang:KEY — Résolution via les fichiers de langue
Lorsque votre site utilise déjà des fichiers .lang Dolibarr (cas typique pour les sites multilingues existants), vous pouvez réutiliser les clés au lieu de les dupliquer dans les slots :
<h1>{{slot:hero_title|type=text|default=@lang:HeroTitle|label=Titre du hero}}</h1>
Au moment du rendu, le module résout via $langs->trans('HeroTitle') selon la langue du visiteur. Les fichiers .lang du site présents dans DOL_DATA_ROOT/<entity>/website/<ref>/langs/ sont chargés automatiquement.
Cas d'usage — Migration progressive d'un site existant : vous conservez vos fichiers .lang en l'état, vous ajoutez les slots, et le client peut soit éditer dans le Studio (surcharge), soit conserver la traduction du .lang (canonique).
Récapitulatif des types et de leur traduisibilité
Type | Interface | Traduisible |
|---|---|---|
| Champ de saisie | Oui |
| Zone de texte | Oui |
| CKEditor | Oui |
| Sélecteur de média | Non |
| Champ de saisie | Non |
| Classe et sélecteur de couleur | Non |
| Sélecteur de couleur | Non |
| Champ numérique | Oui (rare) |
| Case à cocher | Non |
| Menu déroulant | Non |
Pièges fréquents
Slots dans des attributs JSON intégrés — Si vous avez du JSON intégré dans une balise (par exemple data-config="{...}"), n'y placez pas de slot. Les doubles accolades sont aussi un délimiteur Mustache et risquent de casser les bibliothèques JavaScript qui parseraient ce JSON.
Identifiant trop long — La limite est de 64 caractères. Au-delà, le scanner rejette silencieusement, ce que met en évidence l'option --lint.
Espaces autour des = — Le scanner rejette type = text. Utilisez toujours type=text.
Caractère pipe dans une valeur — default=A|B casse l'analyseur. Utilisez une autre séparation, ou laissez default vide et saisissez la valeur via le Studio.
Récapitulatif
Vous savez désormais :
- La syntaxe complète d'un token de slot.
- Les dix types disponibles et leurs interfaces respectives.
- La résolution
@lang:KEYpour réutiliser les fichiers .lang Dolibarr. - Quels types sont traduisibles ou non.
- Les pièges syntaxiques fréquents.
Le chapitre suivant aborde les shortcodes, qui injectent des données Dolibarr dans le HTML.
CHAPITRE 20 — Insérer des données Dolibarr (shortcodes)
Les slots stockent du contenu éditable. Les shortcodes, eux, affichent des données Dolibarr lues en direct (produits, catégories, informations société, médias). Ils sont résolus au moment du rendu, à chaque consultation de page.
Distinction entre slot et shortcode
Slot | Shortcode |
|---|---|
Stocke du contenu éditable par le client. | Affiche des données Dolibarr lues en direct. |
Persistance dans
. | Aucune persistance — résolu à chaque requête. |
Exemple :
| Exemple :
|
Syntaxe générale
{{<namespace>:<sélecteur>.<champ>}}
- namespace : la source de données (
product,category,dict,mysoc,extrafield,media). - sélecteur : la manière d'identifier l'élément (le plus souvent
ref=xxxouid=N). - champ : la donnée à extraire de l'élément.
Namespace product
Lit un produit dans la table llx_product.
<h2>{{product:ref=supplyflow-pro.label}}</h2>
<p>{{product:ref=supplyflow-pro.description}}</p>
<span class="price">{{product:ref=supplyflow-pro.price}} €</span>
<span>{{product:ref=supplyflow-pro.ef_tagline}}</span>
<img src="{{product:ref=supplyflow-pro.ef_hero_image}}" alt="...">
Champs disponibles
Champ | Source |
|---|---|
,
,
| Champs natifs (traduits selon la langue du visiteur via
) |
,
,
| Prix HT, TTC, coût (formaté selon la langue) |
,
| Référence, statut commercialisable |
| Tout champ personnalisé (
,
, etc.) |
Le marqueur $current
Pour un même gabarit utilisé par plusieurs produits (par exemple solution-detail servi par les wrappers solution-<ref>.php), utilisez ref=$current :
<h1>{{product:ref=$current.label}}</h1>
<p>{{product:ref=$current.ef_tagline}}</p>
Le module résout $current via la variable globale $infrasstudio_current_product_ref, posée par le wrapper.
Namespace category
Lit une catégorie dans la table llx_categorie.
<h3>{{category:id=5.label}}</h3>
<p>{{category:id=5.description}}</p>
Namespace dict
Lit une entrée d'un dictionnaire Dolibarr (c_country, c_currencies, etc.).
<span>{{dict:c_country.code=FR.label}}</span>
<span>{{dict:c_currencies.code_iso=EUR.label}}</span>
Namespace mysoc
Lit les informations de la société courante ($mysoc).
<footer>
© {{mysoc.name}} — {{mysoc.address}}, {{mysoc.zip}} {{mysoc.town}}
Tél : {{mysoc.phone}} — Courriel : {{mysoc.email}}
SIRET {{mysoc.idprof2}} — TVA {{mysoc.tva_intra}}
</footer>
Champs disponibles
name, address, zip, town, country_code, phone, fax, email, url, capital, tva_intra, idprof1 à idprof6, logo, logo_small, logo_squarred.
Namespace extrafield
Lit un champ personnalisé d'un objet Dolibarr quelconque.
{{extrafield:table=product|ref=supplyflow.field=tagline}}
{{extrafield:table=societe|id=42|field=segment}}
{{extrafield:table=product|ref=$current|field=tagline}}
Plus générique que {{product:ref=X.ef_tagline}} mais plus verbeux. À utiliser lorsque le namespace dédié n'existe pas (par exemple pour societe ou contact).
Namespace media
Lit un média de la bibliothèque du module (llx_infrasstudio_media).
<img src="{{media:ref=hero-1.url}}" alt="{{media:ref=hero-1.alt}}">
<img src="{{media:ref=hero-1.thumb}}">
<img src="{{media:ref=hero-1.card}}">
<img src="{{media:ref=hero-1.wide}}">
Champs disponibles
Champ | Description |
|---|---|
| URL du fichier original |
| Variante 200 × 200 |
| Variante 640 × 480 |
| Variante 1600 × 1200 |
| Texte alternatif (résolu selon la langue) |
,
,
| Métadonnées |
Combiner shortcodes et slots
Vous pouvez intégrer un shortcode dans la valeur par défaut d'un slot, ou dans le contenu d'un slot de type texte riche :
<!-- Shortcode dans le default d'un slot -->
<p>{{slot:greeting|type=text|default=Bienvenue chez {{mysoc.name}}}}</p>
<!-- Shortcode dans un slot richtext (l'éditeur peut le saisir librement) -->
<div>{{slot:long_intro|type=richtext|default=Notre équipe à {{mysoc.town}} vous accompagne.}}</div>
Cas d'usage — Une page de remerciements après commande qui dit « Merci, votre commande sera livrée à... ». Le HTML reste statique côté gabarit, mais le rendu est personnalisé pour chaque visiteur.
Étendre avec un namespace personnalisé
Pour ajouter un namespace propre à votre projet, déposez un fichier dans htdocs/custom/infrasstudio/shortcodes/ :
// shortcodes/myorg.shortcode.php
function infrasstudio_shortcode_myorg_resolve($args, $context)
{
global $db;
$id = isset($args['id']) ? (int) $args['id'] : 0;
$field = isset($args['_field']) ? $args['_field'] : 'name';
if ($id <= 0) { return ''; }
$sql = "SELECT name, sector FROM ".MAIN_DB_PREFIX."myorg WHERE rowid = ".$id;
$resql = $db->query($sql);
if (!$resql) { return ''; }
$obj = $db->fetch_object($resql);
$db->free($resql);
return isset($obj->$field) ? (string) $obj->$field : '';
}
Utilisable dans n'importe quel gabarit :
<h2>{{myorg:id=12.name}}</h2>
<p>Secteur : {{myorg:id=12.sector}}</p>
Pièges et performance
Shortcode dans une boucle — Si vous résolvez cent fois le même shortcode dans une page, vous effectuez cent requêtes SQL. Mettez vos données en cache dans une variable locale avant la boucle, ou utilisez les fonctions natives Dolibarr (Product::fetch) en PHP.
Shortcode introuvable — Si {{product:ref=inexistant.label}} ne résout rien, le module retourne une chaîne vide silencieusement. Aucune erreur n'est affichée. Testez en local avant la livraison.
Shortcodes dans les attributs — Les shortcodes fonctionnent dans tous les contextes HTML, attributs compris : <img src="{{media:ref=X.url}}">, <a href="{{slot:cta_url|...}}">, etc.
Récapitulatif
Vous savez désormais :
- Distinguer slots (contenu éditable) et shortcodes (données Dolibarr).
- Utiliser les six namespaces livrés (product, category, dict, mysoc, extrafield, media).
- Utiliser
$currentdans les gabarits partagés entre plusieurs produits. - Combiner shortcodes et slots.
- Ajouter votre propre namespace via un fichier dans
shortcodes/. - Anticiper les pièges (boucles, valeurs absentes).
CHAPITRE 21 — Gérer le multilingue (pages sœurs)
Dolibarr Website propose deux modèles multilingues différents. Identifier le modèle que vous utilisez est une étape préalable à toute écriture de code. Ce chapitre vous présente les deux options et la manière dont le module s'y intègre.
Les deux modèles multilingues
Modèle | Caractéristique |
|---|---|
A. Slot par langue (recommandé) | Un seul fichier tpl.php sert toutes les langues. Les slots disposent de surcharges par langue. |
B. Pages sœurs (legacy) | Un fichier tpl.php par langue (
,
,
). Modèle hérité des sites Dolibarr Website classiques. |
Recommandé — Adoptez le modèle A pour tout nouveau site. Le modèle B reste pris en charge pour la migration progressive de sites existants. Le module fournit un outil de consolidation B vers A.
Modèle A — Slot par langue (le standard moderne)
Fonctionnement
Une seule page Dolibarr Website existe par concept (par exemple about). Le fichier tpl.php est unique. Les slots possèdent leur valeur canonique (FR par défaut) et des surcharges par langue stockées dans llx_infrasstudio_slot.
<!-- about.tpl.php — UN SEUL fichier pour FR/EN/DE/ES/IT/PT/NL/PL -->
<h1>{{slot:about_title|type=text|default=À propos de nous|label=Titre About}}</h1>
<p>{{slot:about_lead|type=textarea|default=Notre histoire en quelques mots.|label=Accroche}}</p>
Récupérer la langue du visiteur dans le gabarit
Le module résout les slots automatiquement selon la langue détectée. Pour vos propres traitements PHP (afficher la date dans la bonne langue, choisir une image différente par langue), récupérez la langue via :
<?php
// helpers fournis par le module
$iso2 = infrasstudio_current_lang(); // 'fr', 'en', 'de', ...
$locale = infrasstudio_current_locale(); // 'fr_FR', 'en_US', 'de_DE', ...
?>
Cascade de détection
La fonction infrasstudio_current_lang() détecte la langue dans cet ordre :
- Constante
INFRASSTUDIO_LANG_ISOdéfinie par le site (souvent danslang.inc.php). - Paramètre URL
?lang=xx. - Suffixe sur le slug (
about-en.phpdonne en). - Cookie de persistance (
INFRASSTUDIO_LANG_COOKIE). - Valeur de
$langs->defaultlang. - En-tête HTTP
Accept-Languagedu visiteur.
Modèle B — Pages sœurs (legacy)
Il s'agit du modèle Dolibarr Website classique : pour chaque page, vous créez un fichier tpl.php par langue.
about.php # FR (canonique)
about-en.php # EN
about-de.php # DE
about-es.php # ES
...
Si vous reprenez un site existant qui suit ce modèle, deux options s'offrent à vous :
- Conserver le modèle avec des stubs du module.
- Migrer vers le modèle A grâce à l'outil de consolidation en ligne de commande.
Conserver le modèle B avec des stubs
Le module fournit un helper sister_stub.tpl.php qui réduit chaque page sœur à trois lignes et permet de partager le HTML avec la canonique.
Procédure
La page canonique reste un tpl.php classique :
// page5.tpl.php — canonique FR
<?php // bootstrap dolibarr ... ?>
<html>...<h1>{{slot:about_title|type=text|default=À propos|...}}</h1>...</html>
Chaque page sœur devient un stub :
<?php
// page42.tpl.php — sister EN
$_infrasstudio_sister_canonical = 'page5.tpl.php';
$_infrasstudio_sister_lang = 'en';
include dol_buildpath('/infrasstudio/core/tpl/sister_stub.tpl.php', 0);
Le stub force $_GET['lang'] = 'en' puis délègue le rendu à la canonique. Les surcharges EN sont automatiquement utilisées pour résoudre les slots.
Migrer du modèle B vers A en ligne de commande
Un script consolide une famille de pages sœurs en une page canonique et plusieurs stubs :
php htdocs/custom/infrasstudio/scripts/consolidate_sister_pages.php <ref-site> \
[--entity=N] \
[--base-slug=about] \
[--dry-run] \
[--extractor=/path/to/extractor.php]
Ce que fait le script
- Détecte les groupes de pages sœurs parmi les codes ISO2 pris en charge (en, de, es, it, pt, nl, pl).
- Pour chaque groupe :
- Conserve la page tpl FR comme canonique.
- Extrait les valeurs traduites depuis chaque page sœur (par expression régulière ou via un extracteur personnalisé).
- Insère les surcharges dans
llx_infrasstudio_slot. - Réécrit la canonique avec des slots
{{slot:...}}. - Remplace chaque page sœur par un stub de trois lignes.
- Sauvegarde de chaque tpl original avec l'extension
.bak.
Conseil — Lancez d'abord avec --dry-run pour visualiser ce que le script va faire sans rien écrire. Lancez en mode réel uniquement après vérification.
hreflang : déclarer les alternates au navigateur
Pour que Google sache que vos pages sont des traductions les unes des autres, vous devez émettre des balises <link rel="alternate" hreflang="..."> dans l'en-tête HTML.
Le module fournit un helper qui les génère automatiquement :
<head>
...
<?php echo infrasstudio_hreflang_tags($website, $WEBSITE_PAGE); ?>
...
</head>
Sortie type :
<link rel="alternate" hreflang="fr" href="https://exemple.com/about.php" />
<link rel="alternate" hreflang="en" href="https://exemple.com/about-en.php" />
<link rel="alternate" hreflang="de" href="https://exemple.com/about-de.php" />
<link rel="alternate" hreflang="x-default" href="https://exemple.com/about.php" />
Compatibilité avec les deux modèles — Pour le modèle A, le helper émet une seule URL canonique avec les langues différenciées par ?lang=. Pour le modèle B, il émet une URL par page sœur.
Sélecteur de langue côté gabarit
Pour permettre au visiteur de changer de langue, vous devez exposer un sélecteur. Le helper infrasstudio_translated_url($iso2) génère l'URL équivalente de la page courante dans une autre langue :
<nav class="lang-switcher">
<?php foreach (array('fr', 'en', 'de', 'es') as $iso): ?>
<a href="<?php echo infrasstudio_translated_url($iso); ?>">
<?php echo strtoupper($iso); ?>
</a>
<?php endforeach; ?>
</nav>
Charger les fichiers de langue du site
Si votre site utilise des fichiers .lang Dolibarr (par exemple pour les libellés de menu), chargez-les en début de tpl :
<?php
require_once __DIR__ . '/lang.inc.php';
$langs->loadLangs(array('website_main', 'website_blog'));
?>
Et utilisez les clés via $langs->trans('Key') ou via le slot avec default=@lang:Key.
Récapitulatif
Vous savez désormais :
- Distinguer le modèle A (slot par langue, recommandé) du modèle B (pages sœurs, legacy).
- Récupérer la langue du visiteur via
infrasstudio_current_lang(). - Convertir un site existant utilisant le modèle B en utilisant le helper
sister_stub.tpl.php. - Migrer en bloc B vers A grâce à l'outil
consolidate_sister_pages.php. - Émettre les balises hreflang automatiquement avec
infrasstudio_hreflang_tags. - Construire un sélecteur de langue avec
infrasstudio_translated_url.
CHAPITRE 22 — Créer ses propres gabarits de page
Lorsque votre client clique sur « + Nouvelle page » dans le Studio, il choisit un gabarit de départ. Ce chapitre vous explique comment créer vos propres gabarits adaptés à la charte de votre site.
À quoi sert un gabarit
Un gabarit est un squelette de page qui comporte :
- Du HTML structuré (sections, classes CSS, balises sémantiques).
- Des slots pré-déclarés pour les zones éditables.
- Des valeurs par défaut, pour disposer d'une page complète dès la création.
- Des métadonnées (titre par défaut, slug, type de conteneur).
Lorsqu'un client crée une page depuis un gabarit, le module :
- Crée une entrée dans la table
llx_website_page. - Génère un fichier
page<N>.tpl.phpà partir du squelette. - Crée le wrapper Apache
<slug>.php. - Lance un rescan des slots pour les détecter immédiatement.
Structure d'un gabarit
Un gabarit est un dossier dans htdocs/custom/infrasstudio/templates/ :
templates/mon-gabarit/
├── meta.php # Métadonnées du gabarit
└── skeleton.tpl.php # Le squelette HTML avec slots
Le fichier meta.php
Il retourne un tableau de configuration :
<?php
return array(
'code' => 'mon-gabarit', // identifiant unique
'label' => 'Mon Gabarit', // affiché dans l'assistant
'category' => 'page', // page | landing | blog
'description' => "Description courte affichée sous le titre.",
'icon' => 'fa-file-lines', // icône FontAwesome
'type_container' => 'page', // type Dolibarr Website
'default_slug' => 'nouvelle-page', // slug suggéré
);
Champ | Description |
|---|---|
| Identifiant unique. Doit correspondre au nom du dossier. |
| Texte affiché dans la grille de sélection de l'assistant. |
| Permet le regroupement visuel (page, landing, blog). |
| Classe FontAwesome de l'icône de la tuile. |
| Valeur écrite dans
. Valeurs standards :
,
,
,
. |
| Slug suggéré dans le formulaire. Le client peut le modifier. |
Le fichier skeleton.tpl.php
Il s'agit du squelette HTML, identique à un fichier tpl.php standard, à la différence qu'il contient des marqueurs qui seront remplacés au moment de la création.
Marqueurs disponibles
Marqueur | Remplacé par |
|---|---|
| L'identifiant Dolibarr de la nouvelle page (par exemple 42). |
| Le slug URL final (par exemple
). |
| Le code ISO2 de la langue principale (
,
). |
Exemple complet — gabarit page-libre
meta.php
<?php
return array(
'code' => 'page-libre',
'label' => 'Page libre',
'category' => 'page',
'description' => 'Une page simple avec un titre et un grand champ de texte riche.',
'icon' => 'fa-file-lines',
'type_container' => 'page',
'default_slug' => 'nouvelle-page',
);
skeleton.tpl.php
<?php // BEGIN PHP
$websitekey = basename(__DIR__);
if (! defined('USEDOLIBARRSERVER') && ! defined('USEDOLIBARREDITOR')) {
require_once __DIR__ . '/master.inc.php';
}
require_once DOL_DOCUMENT_ROOT . '/core/lib/website.lib.php';
require_once DOL_DOCUMENT_ROOT . '/core/website.inc.php';
ob_start();
try {
// END PHP ?>
<html lang="@@ISO2@@">
<head>
<title>{{slot:page_title|type=text|default=Nouvelle page|label=Titre SEO|group=seo}}</title>
<meta name="description" content="{{slot:page_meta_description|type=text|default=|label=Meta description|group=seo}}" />
<link rel="canonical" href="<?php echo $website->virtualhost; ?>/@@PAGEURL@@.php" />
</head>
<body>
<?php includeContainer('header'); ?>
<main class="container">
<h1>{{slot:page_h1|type=text|default=Titre principal|label=H1|group=hero}}</h1>
<div class="content">
{{slot:page_body|type=richtext|default=<p>Contenu de la page.</p>|label=Contenu}}
</div>
</main>
<?php includeContainer('footer'); ?>
</body>
</html>
<?php // BEGIN PHP
} catch (Exception $e) { print $e->getMessage(); }
include dol_buildpath('/infrasstudio/core/tpl/website_output.tpl.php', 0);
// END PHP ?>
Le bloc final — N'oubliez pas la ligne include dol_buildpath('/infrasstudio/core/tpl/website_output.tpl.php', 0);. C'est elle qui déclenche la résolution des slots et shortcodes au moment du rendu.
Localiser vos gabarits hors du module
Si vous souhaitez livrer des gabarits avec votre projet client sans modifier le module, définissez la constante :
// htdocs/conf/conf.php ou via dolibarr_set_const
$conf->global->INFRASSTUDIO_TEMPLATE_EXTRA_DIR = '/var/www/monsite/templates';
Le module scanne ce répertoire en complément de htdocs/custom/infrasstudio/templates/.
Gabarits livrés par défaut
Code | Description |
|---|---|
| Page libre avec titre et un grand champ texte riche. Pour les pages ponctuelles. |
| Article de blog générique avec hero, introduction, corps et appel à l'action. |
| Article de blog au design moderne (hero CSS, accroche en italique, image secondaire, articles liés). Adaptable à votre charte. |
| Page de destination produit complète (environ 70 slots). Hero, problème, solution, fonctionnalités, contact, FAQ. |
Conseil — Inspirez-vous de example-landing pour comprendre comment structurer un gabarit complexe avec environ 70 slots organisés en sections.
Bonnes pratiques pour vos gabarits
- Préfixez tous les slots du gabarit par un identifiant commun (par exemple
landing_) pour éviter les collisions entre gabarits. - Regroupez les slots avec
group=par section logique (hero, fonctionnalités, contact, etc.). - Donnez des valeurs par défaut représentatives. Le rédacteur dispose ainsi d'un exemple à modifier plutôt que d'une page vide intimidante.
- Incluez les slots SEO (
page_title,page_meta_description) dans tout gabarit de type page. - Incluez les balises Open Graph dans l'en-tête HTML pour le partage social.
- Incluez le helper hreflang si le site est multilingue.
- Testez le gabarit en créant une page réelle depuis l'assistant et vérifiez le rendu public.
Récapitulatif
Vous savez désormais :
- Comprendre l'utilité d'un gabarit (squelette, slots, valeurs par défaut).
- Créer un dossier
templates/<code>/avecmeta.phpetskeleton.tpl.php. - Renseigner les métadonnées (code, label, category, icon, type_container, default_slug).
- Utiliser les marqueurs
@@PAGEID@@,@@PAGEURL@@et@@ISO2@@. - Localiser vos gabarits en dehors du module via
INFRASSTUDIO_TEMPLATE_EXTRA_DIR. - Suivre les bonnes pratiques (préfixe, regroupement, valeurs par défaut, SEO, multilingue).
Le dernier chapitre de la Partie IV détaille le catalogue produit dynamique en profondeur.
CHAPITRE 23 — Le catalogue produit dynamique en profondeur
Le catalogue produit dynamique transforme votre table llx_product en pages web générées automatiquement. Ce chapitre détaille l'ensemble du fonctionnement côté développeur : architecture, classes, configuration et points d'extension.
Architecture en couches
Couche | Rôle |
|---|---|
StudioProductCatalog | Lit les produits Dolibarr (filtres, tris, multilingue) et expose un format normalisé. |
StudioSolutionWrapper | Génère et supprime les wrappers Apache
en fonction des produits publiés. |
Le gabarit solution-detail (page<X>.tpl.php) | Le gabarit unique qui affiche un produit. Une seule page Dolibarr Website pour N produits. |
Trigger PRODUCT et CATEGORY | Régénère automatiquement les wrappers à chaque modification de produit ou de catégorie. |
Tâche planifiée horaire | Filet de sécurité : relance la génération si un trigger a été manqué. |
Le modèle de données produit
Chaque produit Dolibarr utilisé dans le catalogue exploite plusieurs tables :
Table | Données |
|---|---|
| Champs natifs : ref, label, description, prix, tosell. |
| Traductions natives (label, description par langue). |
| Champs personnalisés canoniques : tagline, hero_image, badge, cta_label, cta_url, deployment, compatibility, support, languages, features (JSON), pricing_tiers (JSON), infrasstudio_published. |
| Surcharges par langue des champs personnalisés traduisibles. |
| Liens entre produits et catégories (utilisés pour la cartographie d'univers). |
Provisionner les champs personnalisés
Les onze champs personnalisés nécessaires ne sont pas livrés en standard avec Dolibarr. Pour les créer en une commande, utilisez le script de provisionnement générique :
php htdocs/custom/infrasstudio/scripts/preset_default.php \
htdocs/custom/infrasstudio/presets/keaticweb.json
Le fichier JSON livre les définitions par défaut (voir Chapitre 28 pour le format). Vous pouvez créer votre propre preset si vous avez besoin d'autres champs personnalisés.
Utiliser StudioProductCatalog
La classe expose plusieurs méthodes pour interroger les produits :
dol_include_once('/infrasstudio/class/studioproductcatalog.class.php');
$catalog = new StudioProductCatalog($db);
// Tous les produits publiés, dans la langue courante
$products = $catalog->fetchPublishedProducts(array(), $iso2);
// Filtres
$products = $catalog->fetchPublishedProducts(array(
'univers' => 'supply-chain',
'type' => 'saas',
), $iso2);
// Un produit par référence
$product = $catalog->fetchProductBySlug('supplyflow-pro', $iso2);
// Univers utilisés (pour le filtre du catalogue)
$univers = $catalog->fetchUsedUnivers();
Format normalisé d'un produit
array(
'id' => 5,
'ref' => 'supplyflow-pro',
'label' => 'SupplyFlow Pro', // résolu selon la langue
'description' => '<p>Description...</p>',
'price' => 290.00,
'badge' => 'Nouveau',
'hero_image' => '/medias/...',
'cta_label' => 'Demander une démo',
'cta_url' => '/contact',
'tagline' => 'Optimisez votre supply chain...',
'deployment' => 'Cloud SaaS',
'compatibility'=> 'SAP, Oracle, Sage',
'support' => '24/7',
'languages' => 'fr,en,de',
'features' => array('Prévision IA', 'Multi-entrepôts'),
'pricing_tiers'=> array(...),
'univers' => 'supply-chain',
'type' => 'saas',
)
Cartographie catégorie vers univers
Les produits sont rattachés à des catégories Dolibarr. Le module associe chaque catégorie à un univers (concept éditorial : Supply Chain, Health, Legal, etc.).
Cartographie par défaut
// Les six univers livrés par défaut
'supply-chain' => array(...catégories supply chain),
'health' => array(...catégories santé),
'legal' => array(...catégories juridique),
'digital-humanities' => array(...catégories sciences humaines),
'transversal' => array(...catégories outils transversaux),
'fsm' => array(...catégories field service)
Surcharge par JSON
Pour personnaliser, définissez la constante :
// Via dolibarr_set_const ou via la page de configuration admin
$conf->global->INFRASSTUDIO_PRODUCT_UNIVERS_MAP = json_encode(array(
'mon-univers' => array(12, 13, 14), // identifiants de catégories
'autre' => array(15, 16),
));
StudioSolutionWrapper
Cette classe gère la génération des wrappers Apache à partir des produits publiés.
Génération manuelle
dol_include_once('/infrasstudio/class/studiosolutionwrapper.class.php');
require_once DOL_DOCUMENT_ROOT . '/website/class/website.class.php';
$website = new Website($db);
$website->fetch(0, 'monsite');
$wrapper = new StudioSolutionWrapper($db);
$stats = $wrapper->rebuildAll($website, '/var/www/monsite');
print_r($stats);
// array('created' => 3, 'updated' => 2, 'deleted' => 1, 'skipped' => 5)
Anatomie d'un wrapper généré
<?php
// /var/www/monsite/solution-supplyflow-pro.php
// Generated by StudioSolutionWrapper — do not edit manually.
$solution_ref = 'supplyflow-pro';
$infrasstudio_current_product_ref = 'supplyflow-pro';
global $dolibarr_main_data_root, $conf;
if (empty($dolibarr_main_data_root)) {
$res = include './page42.tpl.php'; // page solution-detail
} else {
$res = include $dolibarr_main_data_root
. ($conf->entity > 1 ? '/' . $conf->entity : '')
. '/website/monsite/page42.tpl.php';
}
if ($res === false) { http_response_code(500); print 'Failed to make include'; }
Sécurité anti-collision — Le module ne touche qu'aux wrappers qu'il a lui-même créés (présence du marqueur « StudioSolutionWrapper » dans l'en-tête du fichier). Les pages Dolibarr standards portant un slug commençant par solution- sont préservées.
Configuration du générateur de wrappers
Constante | Rôle |
|---|---|
| Référence du site cible. |
| Docroot Apache absolu (par exemple
). |
| Préfixe des wrappers (par défaut
). Exemples :
,
. |
| Slug du gabarit solution-detail (par défaut
). |
Le gabarit solution-detail
Il s'agit d'une page Dolibarr Website classique (avec son slug et son tpl.php), mais utilisée par tous les wrappers. Elle reçoit la variable globale $solution_ref qui identifie le produit à afficher.
Squelette type
<?php
// page42.tpl.php (slug='solution-detail', type_container='page')
require_once __DIR__ . '/master.inc.php';
require_once DOL_DOCUMENT_ROOT . '/core/lib/website.lib.php';
require_once DOL_DOCUMENT_ROOT . '/core/website.inc.php';
dol_include_once('/infrasstudio/class/studioproductcatalog.class.php');
$catalog = new StudioProductCatalog($db);
$iso2 = infrasstudio_current_lang();
$product = $catalog->fetchProductBySlug($solution_ref, $iso2);
if (!$product) {
http_response_code(404);
print '<h1>Produit introuvable</h1>';
exit;
}
ob_start();
try {
?>
<html lang="<?php echo $iso2; ?>">
<head>
<title><?php echo dol_escape_htmltag($product['label']); ?></title>
</head>
<body>
<section class="hero">
<h1><?php echo dol_escape_htmltag($product['label']); ?></h1>
<p><?php echo dol_escape_htmltag($product['tagline']); ?></p>
<img src="<?php echo $product['hero_image']; ?>" alt="...">
</section>
<section class="features">
<ul>
<?php foreach ($product['features'] as $feature): ?>
<li><?php echo dol_escape_htmltag($feature); ?></li>
<?php endforeach; ?>
</ul>
</section>
</body>
</html>
<?php
} catch (Exception $e) { print $e->getMessage(); }
include dol_buildpath('/infrasstudio/core/tpl/website_output.tpl.php', 0);
Mélanger slots et données produit — Vous pouvez combiner les deux : les zones éditoriales (par exemple un titre marketing au-dessus de la grille) en slots, et les zones produit (label, prix, fonctionnalités) lues via StudioProductCatalog. Les deux cohabitent sans conflit.
Le trigger PRODUCT et CATEGORY
Le module installe un trigger qui écoute :
PRODUCT_CREATEPRODUCT_MODIFYPRODUCT_DELETEPRODUCT_PRICE_MODIFYCATEGORY_LINK(uniquement si l'objet lié est un produit)CATEGORY_UNLINK
À chaque événement, le trigger appelle StudioSolutionWrapper::rebuildAll() sur le site configuré.
Note — Si INFRASSTUDIO_WEBSITE_KEY ou INFRASSTUDIO_PUBLIC_DOCROOT ne sont pas configurées, le trigger se termine silencieusement. Ni bruit, ni erreur. Cette discrétion convient aux instances Dolibarr qui n'utilisent pas le catalogue dynamique.
La tâche planifiée horaire (filet de sécurité)
Une tâche planifiée s'exécute toutes les heures et appelle StudioSolutionWrapper::rebuildAllConfigured(). Elle lit les constantes et relance la régénération.
Cette tâche apporte les bénéfices suivants :
- Rattrapage si un trigger a été manqué (import en masse, modification SQL directe, etc.).
- Synchronisation après une migration de serveur.
- Garantie de cohérence sans intervention manuelle.
Reconstruction manuelle en ligne de commande
php htdocs/custom/infrasstudio/scripts/rebuild_solution_wrappers.php \
<ref-site-ou-id> <docroot-public> [entity]
# Exemple
php htdocs/custom/infrasstudio/scripts/rebuild_solution_wrappers.php \
monsite /var/www/monsite 2
Aides visuelles dans le Studio
Deux indicateurs visuels accompagnent l'administrateur dans la prise en main et la maintenance d'une instance qui utilise le catalogue. Ils sont là pour signaler immédiatement les problèmes de configuration et les actions ponctuelles qui n'ont de sens qu'à un moment précis du cycle de vie d'un site.
Bandeau « Produits de démonstration » dans l'éditeur Produits
Quand un site a été créé depuis un template préfabriqué (Vibrant ou Editorial), le seed pose six produits factices catégorisés pour que le catalogue soit immédiatement peuplé et visualisable. Ces produits ont vocation à être remplacés par les vrais produits du client. L'éditeur Produits du Studio détecte automatiquement leur présence — via le préfixe de référence demo-* posé par le seed — et affiche un bandeau jaune en haut de la grille avec deux actions en un clic :
- Dépublier en masse — bascule
infrasstudio_published=0sur les six produits démo. Ils restent dans Dolibarr (cohérents avec une éventuelle migration ultérieure) mais disparaissent du catalogue public et de l'index du site. Réversible en un clic. - Supprimer en masse — efface complètement les six produits, leurs traductions
llx_product_lang, leurs extrafieldsllx_product_extrafieldset leurs liaisons à la catégorie démo. Irréversible. Une modale de confirmation Studio garde le doigt sur le bouton.
Le bandeau disparaît automatiquement dès qu'aucun produit démo n'est détecté dans l'entité courante. Aucun risque d'effacer des produits réels : la détection se base strictement sur le préfixe demo- et sur la liaison à la catégorie démo créée par le seed.
Badge d'alerte dans la barre du Studio
La barre supérieure du Studio (visible sur toutes les pages de l'éditeur) intègre un badge d'alerte qui clignote en rouge lorsque la configuration du catalogue présente un problème bloquant. Le badge agrège plusieurs vérifications croisées et synthétise leur état en un seul point d'attention :
- Module InfraSStudio désactivé sur l'entité courante (cas typique d'une migration multi-entité partiellement appliquée).
- Tables InfraSStudio manquantes — la classe
StudioProductCataloga besoin dellx_infrasstudio_slot,llx_infrasstudio_mediaetllx_infrasstudio_product_translation; le badge s'allume si l'une d'elles est absente (échec de migration, restauration partielle). - Docroot Apache introuvable — la constante
INFRASSTUDIO_PUBLIC_DOCROOTpointe vers un chemin qui n'existe pas physiquement, ce qui empêche les wrapperssolution-*.phpd'être écrits. - Référence de site cassée — la constante
INFRASSTUDIO_WEBSITE_KEYrenvoie vers un site supprimé ou inexistant. - Champs personnalisés produit manquants — un ou plusieurs des onze extrafields utilisés par
StudioProductCatalogn'existent pas dansllx_extrafields, ce qui dégraderait le rendu des fiches produit.
Au survol, le badge déplie une mini liste détaillée des incidents avec un lien direct vers la page Diagnostic complète, qui les résout un par un. Cette approche évite que l'administrateur découvre un dysfonctionnement par un client qui voit une page blanche, en remontant le problème avant que les visiteurs ne soient impactés.
Récapitulatif
Vous savez désormais :
- Comprendre l'architecture du catalogue (StudioProductCatalog, StudioSolutionWrapper, gabarit, trigger, tâche planifiée).
- Provisionner les champs personnalisés produit avec le preset générique.
- Interroger les produits via
StudioProductCatalog. - Cartographier les catégories Dolibarr vers vos univers via
INFRASSTUDIO_PRODUCT_UNIVERS_MAP. - Personnaliser la cartographie des types de produit et le préfixe d'URL des fiches via les constantes catalogue.
- Activer ou désactiver le filtre
infrasstudio_publishedselon que votre site utilise ou non la notion de brouillon catalogue. - Comprendre comment
StudioSolutionWrapper::rebuildAll()génère les wrappers Apache. - Configurer le préfixe et le slug du gabarit par site.
- Écrire le gabarit
solution-detailqui sert tous les produits. - Comprendre le rôle du trigger automatique et de la tâche planifiée de sécurité.
- Lancer une reconstruction manuelle en ligne de commande.
- Utiliser le bandeau de gestion des produits de démonstration livrés avec les templates Vibrant et Editorial.
- Identifier les incidents de configuration du catalogue grâce au badge d'alerte de la barre du Studio.
Fin de la Partie IV — Vous maîtrisez désormais l'intégration côté développeur : préparer un site, annoter avec des slots, comprendre la grammaire, brancher des données Dolibarr, gérer le multilingue, créer vos gabarits et exploiter le catalogue dynamique. Vous êtes en mesure de livrer un site clé en main.
La Partie V aborde l'administration et la maintenance. Si vous êtes administrateur Dolibarr, c'est votre prochaine destination.
Fin de la Partie IV — Vous maîtrisez désormais l'intégration côté développeur : préparer un site, annoter avec des slots, comprendre la grammaire, brancher des données Dolibarr, gérer le multilingue, créer vos gabarits et exploiter le catalogue dynamique. Vous êtes en mesure de livrer un site clé en main.
La Partie V aborde l'administration et la maintenance. Si vous êtes administrateur Dolibarr, c'est votre prochaine destination.
CHAPITRE 24 — Les formulaires publics
InfraSStudio embarque un moteur de formulaires publics conçu pour transformer la moindre saisie web en lead qualifié dans Dolibarr. Un formulaire de contact rempli sur un site géré par le module ne déclenche pas un simple envoi d'email : il alimente automatiquement le CRM, ouvre un ticket, prévient les équipes commerciales et déclenche, au besoin, un rappel en agenda. Ce chapitre détaille la mécanique interne, la configuration côté administrateur et l'intégration dans un site Dolibarr Website.
Architecture du moteur
L'architecture repose sur trois pièces : un descripteur JSON par formulaire, un point d'entrée unique côté serveur, et une chaîne d'adapters exécutés à chaque soumission. Aucun code PHP n'est nécessaire pour ajouter ou modifier un formulaire : il suffit d'éditer le descripteur depuis l'interface d'administration ou directement depuis l'éditeur Studio.
Pièce | Rôle |
|---|---|
Endpoint unique |
|
Descripteur JSON | Une ligne dans |
Pipeline d'adapters | Six étapes activables à la carte : validation, anti-spam, tiers, contact, ticket, notification, agenda. |
Le pipeline étape par étape
Chaque soumission acceptée traverse la chaîne ci-dessous. Chaque étape s'active indépendamment dans le descripteur, ce qui permet de couvrir aussi bien un formulaire de newsletter minimaliste qu'une demande de démonstration commerciale complète.
Adapter | Effet sur la soumission |
|---|---|
antispam | Honeypot, délai minimum de remplissage, rate-limit par adresse IP, captcha délégué au gestionnaire Dolibarr actif. |
tiers | Recherche ou création d'une société dans |
contact | Recherche ou création d'un contact dans |
ticket | Ouverture d'un ticket Dolibarr avec sujet, message, catégorie et sévérité. |
notification | Envoi d'un email interne à l'équipe commerciale et d'un accusé de réception au visiteur. |
agenda | Création d'un événement de rappel lié au tiers et au ticket. |
Configurer un formulaire
Deux interfaces complémentaires sont disponibles. L'administration avancée se trouve dans Outils → InfraS → Formulaires : grille des configurations, statistiques de soumission, accès au descripteur JSON brut. L'éditeur Studio propose désormais un onglet Formulaires avec un inspector inline qui couvre la majorité des réglages courants sans repasser par l'administration Dolibarr.
Créer un nouveau formulaire
Le bouton « + Nouveau formulaire » de l'onglet Formulaires de l'éditeur Studio ouvre une modale en quatre champs :
- le libellé affiché en administration (facultatif) ;
- l'identifiant technique (a-z, 0-9, tiret, underscore — utilisé dans le HTML, unique par entité) ;
- le type (contact, newsletter, demo ou generic) — le type définit les champs par défaut et le pipeline par défaut ;
- un modèle de design (facultatif) — voir la section Les starter design templates ci-dessous.
Le formulaire fraîchement créé est immédiatement sélectionné dans l'inspector et prêt à recevoir d'éventuels ajustements (champs, anti-spam, design, pipeline). L'identifiant technique sert d'ancrage tout au long du cycle de vie : il est référencé dans le HTML, dans le viewer de soumissions et dans les journaux.
Gérer les champs depuis l'éditeur
Dans l'onglet Formulaires de l'éditeur Studio, la section Champs permet d'ajouter, de réordonner (boutons monter / descendre) et de supprimer un champ sans toucher au JSON. L'ordre défini ici est exactement l'ordre d'affichage du formulaire public. Chaque champ porte un interrupteur Obligatoire : lorsqu'il est activé, un astérisque * est ajouté automatiquement à côté du libellé sur le formulaire rendu. Les saisies en cours (libellé, type, placeholder) sont préservées lors d'un déplacement, d'un ajout ou d'une suppression.
Supprimer un formulaire
La section « Zone dangereuse » en bas de l'inspector expose un bouton Supprimer ce formulaire. La suppression efface uniquement la configuration : les soumissions déjà enregistrées dans llx_infrasstudio_form_submission sont conservées (traçabilité RGPD préservée). Une modale de confirmation Studio garde le doigt sur le bouton — la fenêtre confirm() native du navigateur n'est jamais utilisée. Le même contrôle est disponible depuis Outils → InfraS → Formulaires via le bouton Supprimer de chaque ligne.
Le descripteur JSON
Le cœur de la configuration est un descripteur JSON qui décrit le formulaire sous une forme structurée. L'éditeur d'administration valide la syntaxe à l'enregistrement et propose une référence dépliable de toutes les clés supportées pour éviter d'avoir à mémoriser la grammaire.
{
"antispam": { "honeypot": true, "min_fill_seconds": 3, "rate_limit_per_hour": 5, "captcha": true },
"consent": { "required": true, "field_name": "consent", "text": "..." },
"fields": {
"name": { "required": true, "type": "text", "maxlength": 100 },
"email": { "required": true, "type": "email", "maxlength": 200 },
"message": { "required": true, "type": "text", "maxlength": 5000 }
},
"tiers": { "enabled": true, "lookup_by_email": true, "category_label": "Lead web" },
"ticket": { "enabled": true, "category_code": "COMMERCIAL" },
"notification": { "autoreply_enabled": true, "autoreply_template_label": "Accusé de réception" },
"template_override": "site:contact.tpl.php"
}
La clé template_override est facultative — sans elle, le moteur utilise le template par défaut du type. Voir la section Intégrer le formulaire dans un site pour les trois modes de résolution.
Tester un formulaire avant mise en ligne
La fiche admin de chaque formulaire (Outils → InfraS → Formulaires → ouvrir un formulaire) expose un bouton « Tester ce formulaire » qui simule une soumission de bout en bout sans passer par le site public. Le tester forge un payload factice à partir des champs déclarés dans la configuration, traverse le pipeline complet (tiers, contact, ticket, notification, agenda) et restitue un rapport détaillé étape par étape.
Les entités créées en mode test (Société, Contact, Ticket, événement d'agenda) sont :
- marquées explicitement : leur libellé est préfixé
[TEST]pour être reconnaissables au premier coup d'œil dans Dolibarr ; - flaguées en base : la soumission correspondante porte
is_test=1dansllx_infrasstudio_form_submission; - purgeables en un clic : le bouton Purger les données de test supprime en cascade tous les enregistrements de test (événement → ticket → contact → société → soumission), avec un filet de sécurité qui empêche d'effacer une société qui aurait reçu des soumissions de production entre-temps.
En mode test, le captcha est volontairement bypassé par le moteur — l'administrateur est déjà authentifié dans Dolibarr, exiger un code image en plus n'aurait pas de sens. Les autres contrôles antispam (honeypot, délai minimum de remplissage, rate-limit) restent actifs et sont exercés avec des valeurs forgées correctement par le tester, ce qui permet de valider la chaîne complète.
L'usage typique : après chaque modification importante d'un formulaire (ajout d'un champ, changement de catégorie ticket, nouveau template d'accusé), cliquer une fois sur Tester ce formulaire, vérifier dans le rapport que chaque adapter s'est exécuté avec succès, puis purger. Cinq secondes de validation qui évitent de découvrir un bug en production via un vrai client.
Connecter le formulaire à votre CRM
L'intérêt principal du moteur réside dans la chaîne de traitement qu'il déclenche au moment de la soumission. Chaque adapter peut être activé indépendamment selon les besoins.
Tiers et contact
Lorsque l'adapter tiers est actif, le moteur recherche d'abord une société existante dont l'adresse email correspond à celle saisie ; à défaut, il regarde si l'email appartient à un contact rattaché à une société, puis tente une correspondance par nom d'entreprise. Si aucune correspondance n'est trouvée, un nouveau tiers est créé, catégorisé automatiquement et associé à un canal d'origine (extrafield origine renseigné depuis le dictionnaire c_input_reason). La même logique existe pour les contacts, utile notamment pour les inscriptions à la newsletter qui ne nécessitent pas la création d'une société.
Ouverture d'un ticket
L'adapter ticket ouvre un ticket Dolibarr rattaché au tiers résolu. Le sujet et le message sont construits soit à partir des champs du formulaire, soit à partir de gabarits permettant d'injecter dynamiquement le nom du visiteur, sa demande, l'adresse IP d'origine ou tout autre élément du contexte. Catégorie et sévérité sont déterminées par les codes du dictionnaire Dolibarr (llx_c_ticket_category, llx_c_ticket_severity).
Notification et accusé de réception
Deux emails partent automatiquement à chaque soumission acceptée : une notification interne adressée à l'équipe pour traiter la demande, et un accusé de réception destiné au visiteur pour le rassurer. Les deux sont indépendants — il est possible d'envoyer uniquement la notification interne, uniquement l'accusé de réception, ou les deux. Tous les paramètres sont accessibles depuis l'éditeur Studio, onglet Pipeline, section Notifications email, sans avoir à manipuler le JSON brut.
Les champs disponibles dans le wizard
Chaque champ porte une tooltip d'aide affichée en gris sous l'input. Le tableau ci-dessous récapitule leur rôle.
Champ | Effet |
|---|---|
Email destinataire admin | Adresse(s) qui reçoivent l'alerte interne. Plusieurs destinataires séparés par une virgule. Laisser vide pour désactiver la notification interne. |
Modèle du sujet admin | Objet du mail interne. Accepte les variables
,
,
. |
Corps de la notification admin | Corps du mail interne. Laisser vide pour générer automatiquement un récapitulatif de tous les champs saisis. Permet de mettre en forme une notification riche (tableau, branding, lien vers la fiche soumission). |
Le corps admin est du HTML | Interrupteur (toggle). Si activé, le corps est interprété comme du HTML ; sinon il part en texte brut et les balises apparaissent telles quelles. |
Champ payload utilisé comme Reply-To | Nom du champ contenant l'email du visiteur (défaut :
). Un clic sur Répondre dans la messagerie répondra directement au visiteur, pas à l'expéditeur technique. |
Activer l'accusé de réception | Interrupteur. Si activé, le visiteur reçoit un mail de confirmation à l'adresse qu'il a saisie. |
Champ payload de l'adresse visiteur | Nom du champ du formulaire contenant l'adresse du visiteur. Par défaut
. |
Sujet de l'accusé de réception | Objet du mail reçu par le visiteur. Ignoré si un modèle est sélectionné ci-dessous. |
Corps de l'accusé de réception | Corps du mail reçu par le visiteur. Ignoré si un modèle est sélectionné. Laisser vide pour un texte générique de remerciement. |
Le corps de l'accusé est du HTML | Interrupteur (toggle) équivalent côté visiteur. |
Modèle d'accusé de réception | Liste déroulante des modèles d'email Dolibarr de type
. Quand un modèle est sélectionné, il prend la main sur les champs Sujet et Corps ci-dessus. Avantage : le texte est centralisé dans Configuration → Emails → Modèles d'e-mails , partagé entre tous les formulaires qui pointent dessus. |
Adresse expéditeur (From) / Nom expéditeur (From) | Surcharge per-formulaire de l'adresse et du nom apparaissant comme expéditeur des deux mails. Vides : utilise
et le nom de la société configurée dans Dolibarr. |
Variables disponibles dans les templates
Les sujets et corps de mail (admin et accusé) acceptent une substitution simple. Aucune logique conditionnelle : les marqueurs inconnus sont laissés en clair, ce qui facilite le debug.
Variable | Valeur |
|---|---|
| Valeur du champ
saisi par le visiteur. Raccourci :
. |
| Identifiant technique du formulaire actif. |
| Numéro de la soumission, utile pour bâtir un lien admin direct. |
| IP du visiteur. |
,
,
| Coordonnées de la société configurée dans Dolibarr. |
| Clé de premier niveau du JSON config courant. |
Exemple — notification interne mise en forme
Sujet :
[{{form_name}}] Nouvelle demande de {{payload.name}} ({{payload.company}})
Corps HTML (interrupteur Le corps admin est du HTML activé) :
<div style="font-family:Arial,sans-serif;max-width:600px;color:#333">
<div style="background:#0f172a;color:#fff;padding:16px 20px;border-radius:6px 6px 0 0">
<h2 style="margin:0;font-size:18px">Nouvelle demande de contact</h2>
<div style="font-size:12px;opacity:.8;margin-top:4px">
Formulaire : {{form_name}} · Soumission #{{submission_id}}
</div>
</div>
<div style="border:1px solid #e2e8f0;border-top:0;padding:20px;border-radius:0 0 6px 6px">
<table style="width:100%;border-collapse:collapse;font-size:14px">
<tr><td style="color:#64748b;width:140px">Nom</td><td><strong>{{payload.name}}</strong></td></tr>
<tr><td style="color:#64748b">Société</td><td>{{payload.company}}</td></tr>
<tr><td style="color:#64748b">Email</td><td><a href="mailto:{{payload.email}}">{{payload.email}}</a></td></tr>
</table>
<p style="color:#64748b;font-size:13px;margin:20px 0 6px">Message</p>
<div style="background:#f8fafc;border-left:3px solid #6366f1;padding:12px 14px;white-space:pre-wrap">{{payload.message}}</div>
<div style="margin-top:24px;padding-top:14px;border-top:1px solid #e2e8f0;font-size:12px;color:#94a3b8">
IP : {{ip}} · Reçu via {{mysoc.name}}
</div>
</div>
</div>
Côté boîte mail, l'équipe reçoit un message présenté en tableau avec une bande de couleur, le nom et l'email cliquables, le contenu du message dans un encadré, et un pied technique discret avec l'IP.
Cohabitation avec le module Gestionnaire de tickets
Quand l'adapter ticket d'InfraSStudio crée un ticket Dolibarr à la suite d'une soumission, le module Gestionnaire de tickets enverrait normalement son propre mail natif TICKET_CREATE au tiers et sa propre alerte interne. Pour éviter les doublons, InfraSStudio positionne deux drapeaux dans le contexte de création :
disableticketemail = 1— supprime le mail natif Dolibarr au tiers (l'accusé de réception InfraSStudio prend le relais avec un message bien plus soigné) ;notify_tiers_at_create = 0— désactive aussi la notification native du tiers (configurable via la cléticket.notify_tiers_at_createdu JSON si besoin).
La répartition est donc claire :
Module | Périmètre |
|---|---|
InfraSStudio | Accusé de réception au visiteur + notification interne à la création initiale du ticket par soumission web. |
Gestionnaire de tickets Dolibarr | Tout l'échange ultérieur sur le ticket : réponses manuelles d'un agent, mises à jour de statut, fermeture. Utilise ses propres adresses From , Reply-To et destinataire interne configurées dans la page d'admin du module. |
Concrètement : une soumission de formulaire web déclenche exactement deux mails (accusé visiteur + notification équipe, gérés par InfraSStudio), jamais quatre. Quand un agent répond ensuite au ticket depuis Dolibarr, c'est le module natif qui reprend la main — les emails configurés dans Accueil → Configuration → Modules → Gestionnaire de tickets s'appliquent à ce moment-là.
Rappel automatique en agenda
Pour les formulaires à fort enjeu commercial (typiquement une demande de démonstration), un événement de rappel peut être créé dans l'agenda. Le délai est configurable, les week-ends peuvent être évités automatiquement, et l'événement est lié au ticket et au tiers pour garantir la traçabilité.
Intégrer le formulaire dans un site
Une fois la configuration en place, il reste à exposer le formulaire dans le site. Trois approches sont possibles selon le degré de personnalisation souhaité.
Le shortcode {{form:name=...}}
La voie la plus simple, surtout pour un rédacteur. Dans n'importe quelle page du site (slot richtext ou directement dans le tpl) :
{{form:name=contact-site}}
{{form:name=…}} est un shortcode résolu au rendu par le moteur de contenu d'InfraSStudio (StudioContentEngine::applyToPage, via le registre de shortcodes — provider shortcodes/form.shortcode.php), qui appelle infrasstudio_render_public_form pour produire le HTML complet du formulaire — anti-spam et style scopé inclus. Aucune ligne de PHP à toucher côté site. (Il n'existe pas de hook completeHtmlOutput ; les hooks websitepage / websitenav sont déclarés mais ne servent pas à ce rendu.)
L'helper de rendu unifié
Pour passer des options dynamiques (référence produit sur une landing, libellés sur mesure, etc.), appeler directement le helper depuis un tpl.php :
dol_include_once('/infrasstudio/core/lib/infrasstudio.lib.php');
infrasstudio_render_public_form('contact-site', array(
'fk_website' => $website->id,
'fk_page' => $object->id,
'extra' => array('productRef' => 'monproduit'),
));
Les starter design templates
Quatre templates « clé en main » sont livrés dans templates/forms/_starter-*.tpl.php. Chacun est auto-suffisant (CSS inline scopé, rendu dynamique des champs déclarés dans config.fields) — idéal pour démarrer rapidement sur un nouveau site, avant d'éventuellement basculer sur du HTML maison.
Starter | Style |
|---|---|
| Sobre, labels au-dessus, focus indigo. |
| Carte avec ombre douce et bouton en gradient. |
| Inputs alignés horizontalement, compact, idéal newsletter/footer. |
| Coins arrondis, fond teinté, gradient bouton, labels uppercase. |
Le starter choisi dans la modale « + Nouveau formulaire » est automatiquement posé comme template_override dans la configuration. Il peut être changé à tout moment via l'éditeur JSON avancé.
Conserver un design existant — préfixe site:
Pour intégrer le formulaire dans une charte graphique déjà existante (classes CSS du site, structure HTML spécifique), le mécanisme officiel consiste à livrer son propre template depuis le dossier source du site Dolibarr Website et à le référencer via le préfixe site: dans la configuration :
- Créer un dossier
forms/dans le source du site, soitDOL_DATA_ROOT/<entity>/website/<ref>/forms/. - Y déposer un fichier
contact.tpl.phpqui réutilise les classes CSS du site et appelleinfrasstudio_render_form_fields()pour générer ses champs (voir la section Le rendu des champs — helper unique ci-dessous). - Dans le descripteur du formulaire, poser :
"template_override": "site:contact.tpl.php".
Au rendu, le moteur résout vers DOL_DATA_ROOT/<entity>/website/<ref>/forms/contact.tpl.php en utilisant le fk_website du contexte. Cette approche permet à chaque site de livrer ses propres templates sans déposer le moindre fichier dans le module générique — qui reste 100% indépendant du métier de chaque client.
Alternative : si le design impose seulement quelques champs cachés à injecter dans un <form> déjà existant, le helper infrasstudio_render_form_security($formName, $fkWebsite, $fkPage, $_SERVER['PHP_SELF']) émet d'un seul tenant le form_name, le timestamp anti-bot, le honeypot et le captcha conditionnel.
Le rendu des champs — helper unique
Quel que soit le template (générique, starter ou site:), le rendu des champs est centralisé dans le helper infrasstudio_render_form_fields($cfg, $opts) : ordre de déclaration des champs, mapping type → input HTML, hint autocomplete, placeholder, et surtout l'astérisque automatique des champs obligatoires. Un template de formulaire — y compris un template site: — doit appeler ce helper plutôt que coder ses <input> en dur : sinon l'ajout, le réordonnancement ou le passage en obligatoire d'un champ depuis l'éditeur ne se reflète pas côté site.
Le markup reste entièrement paramétrable pour conserver la charte du site. Options principales : row_class, label_class, input_class, textarea_class, req_html (markup de l'astérisque), id_prefix, label_wrap (libellé englobant l'input), show_label (libellés masqués pour les formulaires compacts), exclude (champs cachés gérés à part, ex. product_ref), labels (overrides i18n field_<clé> / placeholder_<clé>).
Exemple dans un template site: réutilisant les classes CSS du site :
print infrasstudio_render_form_fields($cfg, array(
'row_class' => 'form-row',
'label_class' => 'form-label',
'input_class' => 'form-input',
'req_html' => ' <span class="req">*</span>',
'id_prefix' => '',
));
Suivre et auditer les soumissions
Chaque soumission acceptée est persistée dans llx_infrasstudio_form_submission avec son contenu sanitisé, l'adresse IP d'origine, l'agent utilisateur, la page de provenance et la trace du consentement RGPD. Le viewer d'administration (Outils → InfraS → Soumissions) permet de filtrer par formulaire, statut ou plage de dates, d'ouvrir le détail complet d'une soumission et d'accéder en un clic au tiers, au contact, au ticket et à l'événement d'agenda qui en ont découlé. Cette traçabilité complète est précieuse à la fois pour le suivi commercial et pour répondre aux demandes RGPD des visiteurs.
Récapitulatif
Le moteur de formulaires d'InfraSStudio transforme un simple formulaire web en véritable point d'entrée du CRM. Configurable sans code, sécurisé par défaut et entièrement intégré à l'écosystème Dolibarr, il évite la fragmentation des outils tout en gardant la souplesse nécessaire à chaque projet. Pour aller plus loin, voir le Chapitre 28 (constantes), le Chapitre 31 (modèle SQL des trois tables llx_infrasstudio_form_*) et l'Annexe B (FAQ) pour les questions opérationnelles courantes.