Laravel

Un framework qui rend heureux

Voir cette catégorie
Vers le bas
Cours Laravel 6 – les vues
Samedi 12 octobre 2019 14:23

Je vous ai déjà parlé des vues dans ce cours et de Blade et on a eu plusieurs exemples de code. Dans ce chapitre je vais faire un peu le point et montrer des possibilités intéressantes qui n'ont pas encore été évoquées.

Pour les besoins d'illustrer ce chapitre on va repartir de l'application des films que j'ai créée pour ce cours et dont vous pouvez récupérer la version finale ici. Il suffit de décompresser dans un dossier et ensuite de lancer cette commande :

composer install
Il faut aussi créer une base de données et bien renseigner dans le fichier .env :
DB_DATABASE=laravel6
DB_USERNAME=root
DB_PASSWORD=
Ensuite lancer les migrations et la population :
php artisan migrate --seed
Ensuite l'application doit fonctionner avec l'url .../films.

Le cache

Les vues générées par Blade sont en PHP et sont en cache jusqu'à ce qu'elles soient modifiées, ce qui assure de bonnes performances. Le cache se situe ici :

Si vous avez besoin de nettoyer ce cache ce n'est pas la peine d'aller supprimer ces fichiers dans le dossier, il y a une commande Artisan pour ça :

php artisan view:clear

On en a besoin parfois en cours de développement lorsqu'une vue n'est pas régénérée correctement.

L'héritage

On a vu qu'une vue peut en étendre une autre, c'est un héritage. Ainsi pour les vues de l'application on a un template de base : Ce template comporte la structure globale des pages et est déclaré comme parent par les autres vues :
@extends('template')
Dans le template on prévoit un emplacement (@yield) pour que les vues enfants puissent placer leur code :
<main class="section">
    <div class="container">
        @yield('content')
    </div>
</main>
Ainsi dans la vue index.blade.php on utilise cet emplacement :
@section('content')
   // Code de la vue
@endsection

On peut également déclarer une section dans le template avec @section lorsqu'on a du code dans le template qu'on veut soit utiliser, soit surcharger.

L'inclusion

On peut faire beaucoup de choses avec l'héritage mais il est souvent utile de pouvoir inclure une vue dans une autre, classiquement on parle de vue partielle (partial).

Si on regarde de près la vue index on a un tableau avec comme corps des lignes avec les films :

On peut mettre le code qui génère ces lignes dans une vue partielle : En récupérant le code de la vue index :
@foreach($films as $film)
    <tr @if($film->deleted_at) class="has-background-grey-lighter" @endif>
        <td><strong>{{ $film->title }}</strong></td>
            <td>
                @if($film->deleted_at)
                    <form action="{{ route('films.restore', $film->id) }}" method="post">
                        @csrf
                        @method('PUT')
                        <button class="button is-primary" type="submit">Restaurer</button>
                    </form>
                @else
                    <a class="button is-primary" href="{{ route('films.show', $film->id) }}">Voir</a>
                @endif
            </td>
            <td>
                @if($film->deleted_at)
                @else
                    <a class="button is-warning" href="{{ route('films.edit', $film->id) }}">Modifier</a>
                @endif
            </td>
        <td>
            <form action="{{ route($film->deleted_at? 'films.force.destroy' : 'films.destroy', $film->id) }}" method="post">
                @csrf
                @method('DELETE')
                <button class="button is-danger" type="submit">Supprimer</button>
            </form>
        </td>
    </tr>
@endforeach
Et dans la vue index on se contente d'inclure cette vue :
<tbody>
    @include('partials.lines')
</tbody>

On dispose aussi des directives @includeIf, @includeWhen et @includeFirst.

Une autre possibilité consiste à gérer l'itération à partir de la vue principale :
@each('partials.lines', $films, 'film')
Et évidemment on ne conserve dans la vue partielle que le code inclus dans la boucle :
<tr @if($film->deleted_at) class="has-background-grey-lighter" @endif>
    ...
</tr>

Les composants

Comme alternative à l'inclusion on dispose des composants. Voyons de quoi il s'agit. Dans la vue index on a une partie du code consécrée à la notification lorsqu'un action a bien abouti comme la modification d'un film : Voici le code correspondant :
@if(session()->has('info'))
    <div class="notification is-success">
        {{ session('info') }}
    </div>
@endif
On va créer un composant pour cette notification : Avec ce code :
<div class="notification is-success">
    {{ $text }}
</div>
La variable $text va recevoir ce qu'on va injecter dans le composant. Voici le code modifié dans la vue index :
@if(session()->has('info'))
    @component('components.notification', ['text' => session('info')]) @endcomponent
@endif

On peut aussi prévoir une variable {{ $slot }} dans le composant et injecter directement du contenu HTML.

Les structures de contrôle

@if

La directive @if permet d'introduire une condition. Par exemple dans la vue partielle lines qu'on a créée ci-dessus on trouve ce code :
@if($film->deleted_at) class="has-background-grey-lighter" @endif
Ce qui permet de griser le fond de la ligne pour un film dans la corbeille :

Les directives @else et @unless permettent de compléter la logique.

Remarquez au passage la directive @php qui permet d'insérer du code PHP dans la vue.

@isset et @empty

La directive @isset est l'équivalent de la fonction PHP isset(). La directive @empty est l'équivalent de la fonction PHP empty().

@auth et @guest

Il est fréquent d'avoir à tester si un utilisateur est connecté et on peut écrire :
@if (Auth::check())
Mais il est plus élégant d'utiliser @auth. de la même manière on dispose de @guest.

Il existe aussi la directive @switch équivalente de la directive PHP switch.

Condition personnalisée

On peut aussi se construire une condition personnalisée. On pourrait prévoir ce code dans AppServiceProvider:
public function boot()
{
    ...

    Blade::if('admin', function () {
        return auth()->user()->role === 'admin';
    });

    Blade::if('redac', function () {
        return auth()->user()->role === 'redac';
    });

    Blade::if('request', function ($url) {
        return request()->is($url);
    });
}
On crée ainsi 3 directives :
  • admin : vérifie qu'un utilisateur est administrateur
  • redac : vérifie qu'un utilisateur est rédacteur
  • request : vérifie que la requête correspond à une certaine url
Du coup on peut écrire ce genre de code :
@admin
  <li>
    <a href="{{ url('admin') }}">@lang('Administration')</a>
  </li>
@endadmin
@redac
  <li>
    <a href="{{ url('admin/posts') }}">@lang('Administration')</a>
  </li>
@endredac
Et dans une vue :
@request('password/reset')
  <li class="current">
    <a href="{{ request()->url() }}">@lang('Password')</a>
  </li>
@endrequest
@request('password/reset/*')
  <li class="current">
    <a href="{{ request()->url() }}">@lang('Password')</a>
  </li>
@endrequest

Les boucles

Pour les boucles on dispose de @for, @foreach, @each, @forelse et @while. La logique est la même que les directives équivalentes de PHP. Voyons un exemple dans la vue show :
@foreach($film->categories as $category)
    <li>{{ $category->name }}</li>
@endforeach
C'est la partie du code qui remplit la liste des catégories quand on affiche un film  : Avec la directive @foreach on dispose de la variable interne $loop qui offre des informations sur l'index en cours. Voici tout ce qui est disponible :
Propriété Description
$loop->index L'index en cours (commence à 0)
$loop->iteration L'itération courante (commence à 1)
$loop->remaining Les itérations restantes
$loop->count Le nombre total d'itérations
$loop->first Indique si c'est le premier élément
$loop->last Indique si c'est le dernier élément
$loop->depth Le niveau d'imbrication actuel
$loop->parent La variable d'itération parente si imbrication

Les composeurs de vues (view composers)

Les composeurs de vues sont des fonctions de rappel ou des classes qui sont appelée lorsqu'une vue est générée, ce qui permet de localiser du code. Voyons une exemple dans l'application.

Regardez ce code dans AppServiceProvider (on aurait pu créer un provider spécifique) :

public function boot()
{
    View::composer(['index', 'create', 'edit'], function ($view) {
        $view->with('categories', Category::all());
        $view->with('actors', Actor::all());
    });
}
On dit là que lorsqu'on affiche une des vues index, create ou edit il faut envoyer les variables $categories et $actors.

J’ai prévu un ZIP récupérable ici qui contient le code de cet article.

En résumé

Avec Blade on peut :
  • utiliser l'héritage de vue,
  • inclure une vue,
  • utiliser un composant,
  • utiliser de nombreuses directives pour organiser le code,
  • utiliser des boucles,
  • ...
D'autre part les composeurs de vues permettent de localiser du code facilement.


Par bestmomo

Nombre de commentaires : 2