Intégration optionnelle de Vite.js et Tailwind CSS v4 pour Astral MVC.
- ⚡ Démarrage dev < 300ms — ESM natif, aucun bundling à froid
- 🔥 HMR instantané — CSS et JS mis à jour sans rechargement de page
- 📦 Build production optimisé — tree-shaking Rollup, fichiers hashés
- 🎨 Tailwind CSS v4 — plugin officiel Vite, zéro config PostCSS
- 🔌 Intégration en 4 étapes — un
ServiceProvider, une variable$vite
- Prérequis
- Installation
- Utilisation quotidienne
- API de ViteAssets
- Variables d'environnement
- Stubs fournis
- Comment ça fonctionne
- Ajouter Vue 3 ou React
- Structure du projet après installation
- Licence
| Outil | Version minimale |
|---|---|
| PHP | 8.0+ |
| Astral MVC | 1.1+ |
| Node.js | 18+ |
| npm | 9+ |
composer require astral-php/astral-viteOuvrir config/dependencies.php et ajouter ViteServiceProvider après FrameworkServiceProvider :
// config/dependencies.php
use Core\Providers\FrameworkServiceProvider;
use Core\Providers\DatabaseServiceProvider;
use AstralVite\ViteServiceProvider; // ← importer
use App\Providers\AppServiceProvider;
return [
FrameworkServiceProvider::class,
DatabaseServiceProvider::class,
ViteServiceProvider::class, // ← ajouter ici
AppServiceProvider::class,
];Pourquoi après
FrameworkServiceProvider?ViteServiceProviderappelleView::share('vite', ...). La classeViewest instanciée parFrameworkServiceProvider— elle doit donc être enregistrée en premier.
Le dossier vendor/astral-php/astral-vite/stubs/ contient tous les fichiers
prêts à l'emploi. Copiez-les à la racine de votre projet :
# Config Vite + npm
cp vendor/astral-php/astral-vite/stubs/vite.config.js .
cp vendor/astral-php/astral-vite/stubs/package.json .
# Sources front-end
cp -r vendor/astral-php/astral-vite/stubs/resources .
# Layout PHP (adapter selon votre layout existant)
cp vendor/astral-php/astral-vite/stubs/views/layouts/app.php views/layout/npm installCela installe : vite, tailwindcss, @tailwindcss/vite.
Les valeurs par défaut conviennent pour la plupart des projets.
Pour personnaliser, ajouter dans .env :
# Défauts — à modifier seulement si nécessaire
VITE_DEV_SERVER=http://localhost:5173
VITE_BUILD_DIR=/build$vite est automatiquement disponible dans toutes les vues (partagé via View::share()).
Tu as deux options :
- Supprimer l'import Tailwind CDN (si présent)
- Ajouter l'appel à
$vite->tags()dans le<head>:
<!-- views/layout/main.php -->
<head>
<meta charset="UTF-8">
<title><?= htmlspecialchars($title ?? 'Mon App') ?></title>
<?= $vite->tags([
'resources/js/app.js',
'resources/css/app.css',
]) ?>
</head>- Copier
stubs/views/layouts/app.phpversviews/layout/ - Adapter le contenu HTML (nav, footer, etc.)
C'est tout. Le package gère le reste automatiquement selon APP_ENV.
Si tu préfères n’avoir qu’un seul entry point :
- Dans
resources/js/app.js:
import '../css/app.css'- Dans le layout :
<?= $vite->tags('resources/js/app.js') ?>Lancer deux terminaux en parallèle :
# Terminal 1 — Serveur PHP
php -S localhost:8000 -t public
# Terminal 2 — Serveur Vite (HMR)
npm run devAccéder à http://localhost:8000.
- PHP génère le HTML et les données
- Vite sert les assets JS/CSS avec mise à jour instantanée (HMR)
- Toute modification CSS ou JS est répercutée sans rechargement de la page
# Compiler les assets
npm run buildCela génère dans public/build/ :
public/build/
├── manifest.json ← carte src → fichier hashé (lu par ViteAssets)
└── assets/
├── app-Bz3kT9mQ.js ← JS minifié + tree-shaké
└── app-Cx4mPrN8.css ← CSS Tailwind purgé + minifié
En production (APP_ENV=production), $vite->tags() lit manifest.json et
génère automatiquement les balises avec les noms de fichiers hashés.
La variable $vite est disponible dans toutes les vues. Elle peut aussi être
injectée dans un contrôleur ou un service via le container.
Génère les balises HTML pour un ou plusieurs entry points. C'est la méthode principale — à utiliser dans le layout.
// Un seul entry point
<?= $vite->tags('resources/js/app.js') ?>
// Plusieurs entry points
<?= $vite->tags([
'resources/js/app.js',
'resources/css/app.css',
]) ?>En développement (APP_ENV=development), génère :
<script type="module" src="http://localhost:5173/@vite/client"></script>
<script type="module" src="http://localhost:5173/resources/js/app.js"></script>En production, génère :
<link rel="stylesheet" href="/build/assets/app-Cx4mPrN8.css">
<script type="module" src="/build/assets/app-Bz3kT9mQ.js"></script>Retourne l'URL publique d'un asset statique (image, SVG, font…).
<img src="<?= $vite->asset('resources/images/logo.svg') ?>" alt="Logo">
<link rel="icon" href="<?= $vite->asset('resources/images/favicon.png') ?>">Indique si le mode développement est actif. Utile pour du rendu conditionnel.
<?php if ($vite->isDev()): ?>
<div class="fixed bottom-2 right-2 bg-indigo-600 text-white text-xs px-2 py-1 rounded">
⚡ Vite HMR
</div>
<?php endif; ?>use AstralVite\ViteAssets;
final class HomeController extends AbstractController
{
public function __construct(
private View $view,
private ViteAssets $vite, // ← injecter depuis le container
) {}
public function index(): Response
{
$logoUrl = $this->vite->asset('resources/images/logo.svg');
return $this->render('home/index', compact('logoUrl'));
}
}Enregistrer dans app/Providers/AppServiceProvider.php :
$container->bind(HomeController::class, fn($c) => new HomeController(
view: $c->make(View::class),
vite: $c->make(ViteAssets::class), // ← déjà enregistré par ViteServiceProvider
));| Variable | Défaut | Description |
|---|---|---|
VITE_DEV_SERVER |
http://localhost:5173 |
URL du serveur Vite en mode développement |
VITE_BUILD_DIR |
/build |
Chemin public vers le dossier de build (relatif à public/) |
Ces variables sont optionnelles. Les valeurs par défaut conviennent pour la grande majorité des projets.
Le dossier stubs/ contient des fichiers de démarrage prêts à l'emploi :
stubs/
├── vite.config.js ← Config Vite calibrée pour Astral MVC
├── package.json ← Dépendances npm (vite, tailwindcss)
├── .env.example ← Variables Vite à ajouter dans .env
├── resources/
│ ├── js/
│ │ ├── app.js ← Entry point JS
│ │ └── components/
│ │ ├── flash.js ← Auto-dismiss des messages flash Astral
│ │ └── confirm.js ← Confirmation avant suppression (DELETE)
│ └── css/
│ └── app.css ← Entry point CSS avec Tailwind v4
└── views/
└── layouts/
└── app.php ← Layout PHP complet avec $vite->tags()
Anime et auto-dismiss les messages flash d'Astral MVC après 4 secondes.
Utilise l'attribut data-flash sur vos éléments de flash :
<!-- views/partials/flash.php -->
<?php if ($session->hasFlash('success')): ?>
<div data-flash="success"
class="flex items-center gap-3 p-4 rounded-lg bg-green-50 text-green-800 border border-green-200 mb-4">
<?= htmlspecialchars($session->getFlash('success')) ?>
<button data-flash-close class="ml-auto text-green-600 hover:text-green-900">✕</button>
</div>
<?php endif; ?>Ajoute une demande de confirmation avant la soumission des formulaires de suppression. Compatible avec le verb spoofing DELETE d'Astral MVC :
<form method="POST" action="/users/<?= $user->id ?>/delete"
data-confirm="Supprimer <?= htmlspecialchars($user->name) ?> ?">
<input type="hidden" name="_method" value="DELETE">
<?= $csrf->field() ?>
<button type="submit" class="text-red-600 hover:underline text-sm">
Supprimer
</button>
</form>Navigateur → http://localhost:8000/
↓
PHP (port 8000)
génère le HTML :
<script type="module" src="http://localhost:5173/@vite/client"></script>
<script type="module" src="http://localhost:5173/resources/js/app.js"></script>
↓
Vite (port 5173)
sert les modules ES natifs
WebSocket HMR → mise à jour instantanée
Le navigateur charge les modules JS directement depuis Vite via les ES Modules natifs. Aucun bundling à froid — c'est pour ça que le démarrage est quasi-instantané.
npm run build
↓
public/build/manifest.json ←── ViteAssets lit ce fichier
public/build/assets/app-[hash].js
public/build/assets/app-[hash].css
PHP génère :
<link rel="stylesheet" href="/build/assets/app-Cx4mP.css">
<script type="module" src="/build/assets/app-Bz3kT.js"></script>
Le build Rollup effectue le tree-shaking (supprime le code mort), le code splitting (chunks partagés chargés une seule fois) et la minification. Tailwind CSS v4 purge automatiquement les classes non utilisées.
npm install vue
npm install -D @vitejs/plugin-vue// vite.config.js
import { defineConfig } from 'vite'
import tailwindcss from '@tailwindcss/vite'
import vue from '@vitejs/plugin-vue'
export default defineConfig({
root: 'resources',
plugins: [tailwindcss(), vue()],
// ... reste inchangé
})// resources/js/app.js
import { createApp } from 'vue'
import MonComposant from './components/MonComposant.vue'
createApp(MonComposant).mount('#app')<!-- Dans une vue Astral -->
<div id="app"></div>npm install react react-dom
npm install -D @vitejs/plugin-react// vite.config.js
import react from '@vitejs/plugin-react'
export default defineConfig({
root: 'resources',
plugins: [tailwindcss(), react()],
// ... reste inchangé
})votre-projet/
├── resources/ ← Sources front-end (créées depuis stubs/)
│ ├── js/
│ │ ├── app.js ← Entry point JS
│ │ └── components/
│ │ ├── flash.js
│ │ └── confirm.js
│ └── css/
│ └── app.css ← Entry point CSS + Tailwind v4
│
├── public/
│ ├── index.php
│ └── build/ ← Généré par npm run build (gitignored)
│ ├── manifest.json
│ └── assets/
│
├── vendor/
│ └── astral-php/
│ └── astral-vite/ ← Ce package
│ ├── src/
│ │ ├── ViteAssets.php
│ │ └── ViteServiceProvider.php
│ └── stubs/
│
├── vite.config.js ← Copié depuis stubs/
├── package.json ← Copié depuis stubs/
├── node_modules/ ← Généré par npm install (gitignored)
└── .env ← VITE_DEV_SERVER + VITE_BUILD_DIR (optionnel)
MIT — © 2026 astral-php