Appearance
Layout component: part 1
- A Layout component is a simple, yet powerful Blade engine to maintain the same general layout across various pages
- Blade is Laravels rendering engine to parse HTML and PHP to the view
- Blade adds no overhead to your application: all Blade views are compiled into plain PHP code and cached until they are modified
- Blade view files use the .blade.php file extension and are stored in the resources/views folder
WARNING
In Laravel, there are two ways to create a layout:
- Component based layouts: this is the method we discuss in this course
- Template inheritance based layouts: this was the primary way of building applications prior to the introduction of components in Laravel 7.x
- Since most web applications maintain the same layout across various pages, it's convenient to define this layout as a single Blade file
- The layout component or "master page" consists of:
- regular HTML code that is common to all pages (header, footer, navigation, ...)
- one or more slots or "gaps" which can be filled in separately for each page
- one or more stacks that allow you to push data to, which can be rendered somewhere else in another view or layout
WARNING
- Within the app/View/Components/ folder, you already see two classes:
- AppLayout.php renders the app.blade.php component
- GuestLayout.php renders the guest.blade.php component
- The components itself are inside the resources/views/layouts/ folder:
- app.blade.php is the layout Jetstream uses for logged-in users
- guest.blade.php a very basic layout for non-logged-in users (guest users)
- We are now creating a third, universal, layout component that will be used by all our pages (logged-in or not)
Create a new layout component
- Create a new component with
php artisan make:component VinylshopLayout
- This command created two files:
- A component class in app/View/Components/VinylshopLayout.php
- A Blade file in resources/views/components/vinylshop-layout.blade.php
- Move vinylshop-layout.blade.php from the components folder to the layouts folder and rename it to vinylshop.blade.php
- Open app/View/Components/VinylshopLayout.php and update the path inside the render methode
(from'components.vinylshop-layout'
to'layouts.vinylshop'
)
Common code
- For the layout we choose Tailwind CSS for styling:
you can keep it simple for now and link the necessary CSS file via CDN - Replace the code in vinylshop.blade.php with this code:
html
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<script src="https://cdn.tailwindcss.com"></script>
<meta name="description" content="Welcome to the Vinyl Shop">
<title>The vinyl Shop</title>
</head>
<body class="font-sans antialiased">
<div class="flex flex-col space-y-4 min-h-screen text-gray-800 bg-gray-100">
<header class="shadow bg-white/70 sticky inset-0 backdrop-blur-sm z-10">
{{-- Navigation --}}
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
</header>
<main class="container mx-auto p-4 flex-1 px-4">
{{-- Title --}}
<h1 class="text-3xl mb-4">
</h1>
{{-- Main content --}}
</main>
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
</div>
</body>
</html>
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<script src="https://cdn.tailwindcss.com"></script>
<meta name="description" content="Welcome to the Vinyl Shop">
<title>The vinyl Shop</title>
</head>
<body class="font-sans antialiased">
<div class="flex flex-col space-y-4 min-h-screen text-gray-800 bg-gray-100">
<header class="shadow bg-white/70 sticky inset-0 backdrop-blur-sm z-10">
{{-- Navigation --}}
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
</header>
<main class="container mx-auto p-4 flex-1 px-4">
{{-- Title --}}
<h1 class="text-3xl mb-4">
</h1>
{{-- Main content --}}
</main>
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
</div>
</body>
</html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
REMARKS
- You can add comments in Blade by putting them between
{{--
and--}}
- These Blade comments are not included in the rendered HTML
- In PhpStorm you can use the shortcuts
Ctrl
+/
(single comment line) orCtrl
+Shift
+/
(multiple comment lines)
Slot
- For now, we create four slots in the layout:
- Line 27: one default slot
{{ $slot }}
for the content of the page - Line 8: a named slot
{{ $description }}
for the description meta tag - Line 9: a named slot
{{ $title }}
for the page title - Line 24: a named slot
{{ $subtitle }}
for the main heading on the page
- Line 27: one default slot
WARNING
Use the ternary operator to give EVERY NAMED SLOT a default value:
- Line 7:
{{ $description ?? 'Welcome to the Vinyl Shop' }}
us Welcome to the Vinyl Shop as value is no description value is given - Line 8:
{{ $title ?? 'The Vinyl Shop' }}
us The Vinyl Shop as value is no title value is given - Line 23:
{{ $subtitle ?? $title ?? "This page has no (sub)title" }}
us This page has no (sub)title as value is no subtitle or title value is given
html
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<script src="https://cdn.tailwindcss.com"></script>
<meta name="description" content="{{ $description ?? 'Welcome to the Vinyl Shop' }}">
<title>The vinyl Shop: {{ $title ?? 'The Vinyl Shop' }}</title>
</head>
<body class="font-sans antialiased">
<div class="flex flex-col space-y-4 min-h-screen text-gray-800 bg-gray-100">
<header class="shadow bg-white/70 sticky inset-0 backdrop-blur-sm z-10">
{{-- Navigation --}}
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
</header>
<main class="container mx-auto p-4 flex-1 px-4">
{{-- Title --}}
<h1 class="text-3xl mb-4">
{{ $subtitle ?? $title ?? "This page has no (sub)title" }}
</h1>
{{-- Main content --}}
{{ $slot }}
</main>
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
</div>
</body>
</html>
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<script src="https://cdn.tailwindcss.com"></script>
<meta name="description" content="{{ $description ?? 'Welcome to the Vinyl Shop' }}">
<title>The vinyl Shop: {{ $title ?? 'The Vinyl Shop' }}</title>
</head>
<body class="font-sans antialiased">
<div class="flex flex-col space-y-4 min-h-screen text-gray-800 bg-gray-100">
<header class="shadow bg-white/70 sticky inset-0 backdrop-blur-sm z-10">
{{-- Navigation --}}
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
</header>
<main class="container mx-auto p-4 flex-1 px-4">
{{-- Title --}}
<h1 class="text-3xl mb-4">
{{ $subtitle ?? $title ?? "This page has no (sub)title" }}
</h1>
{{-- Main content --}}
{{ $slot }}
</main>
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
</div>
</body>
</html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
Stack
- A stack can best be described as a 'reference' on the page where you can later add additional code
- A stack is especially useful to add page specific JavaScript to the code
- Just before the closing body tag, add a stack with the name
script
html
<html lang="en">
<head> ... </head>
<body>
...
@stack('script')
</body>
</html>
<html lang="en">
<head> ... </head>
<body>
...
@stack('script')
</body>
</html>
1
2
3
4
5
6
7
2
3
4
5
6
7
Link a page to the layout component
- A component will be rendered from a
<x-***> ... </x-***>
element, where***
refers to the layout component name - Open the file resources/views/home.blade.php
Default slot
- Wrap the content inside an
<x-vinylshop-layout> ... </x-vinylshop-layout>
element - All content inside the
<x-vinylshop-layout>
element will be rendered inside the default slot - Visit http://vinyl_shop.test
html
<x-vinylshop-layout>
<h1>The Vinyl Shop</h1>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
</x-vinylshop-layout>
<x-vinylshop-layout>
<h1>The Vinyl Shop</h1>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
</x-vinylshop-layout>
1
2
3
4
5
2
3
4
5
Named slots
- Remove the h1 tag and move the text to the
title
slot - Update the description (
description
slot)
html
<x-vinylshop-layout>
{{-- <h1>The Vinyl Shop</h1> --}}
<x-slot name="description">New description</x-slot>
<x-slot name="title">Welcome to the Vinyl Shop</x-slot>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
</x-vinylshop-layout>
<x-vinylshop-layout>
{{-- <h1>The Vinyl Shop</h1> --}}
<x-slot name="description">New description</x-slot>
<x-slot name="title">Welcome to the Vinyl Shop</x-slot>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
</x-vinylshop-layout>
1
2
3
4
5
6
7
2
3
4
5
6
7
Push to stack
- Push a simple JavaScript to the
script
stack
html
<x-vinylshop-layout>
<x-slot name="description">New description</x-slot>
<x-slot name="title">Welcome to the Vinyl Shop</x-slot>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
@push('script')
<script>
console.log('The Vinyl Shop JavaScript works! 🙂')
</script>
@endpush
</x-vinylshop-layout>
<x-vinylshop-layout>
<x-slot name="description">New description</x-slot>
<x-slot name="title">Welcome to the Vinyl Shop</x-slot>
<p>Welcome to the website of The Vinyl Shop, a large online store with lots of (classic) vinyl records.</p>
@push('script')
<script>
console.log('The Vinyl Shop JavaScript works! 🙂')
</script>
@endpush
</x-vinylshop-layout>
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
EXERCISE 1: Refactor all pages
- Link the views contact.blade.php and admin/records/index.blade.php to the layout component
- Test your pages (http://vinyl_shop.test/contact and http://vinyl_shop.test/admin/records) in a browser
Child components
- In order to not overload the template, it is best practice to place larger blocks of code (e.g. navigation, footer, ...) in separate components
- Create three new components:
- resources/views/components/layout/nav.blade.php
- resources/views/components/layout/footer.blade.php
- resources/views/components/layout/favicons.blade.php
Navigation
- Move the entire navigation to the nav component
- Refer in the template to the nav component
php
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
<nav class="container mx-auto p-4 flex justify-between items-center">
<a href="{{ route('home') }}" class="underline">Home</a>
<a href="{{ route('contact') }}" class="underline">Contact</a>
<a href="{{ route('admin.records') }}" class="underline">Records</a>
</nav>
1
2
3
4
5
2
3
4
5
Footer
- Move the entire footer to the footer component
- Refer in the template to the footer component
php
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
<footer class="container mx-auto p-4 text-sm border-t flex justify-between items-center">
<div>The Vinyl Shop - © {{ date('Y') }}</div>
<div>Build with Laravel {{ app()->version() }}</div>
</footer>
1
2
3
4
2
3
4
EXERCISE2: Favicons
- Add a real favicon (https://realfavicongenerator.net/) to all the pages of your project
- Search for a suitable (and original) favicon picture and place the generated files in (a subfolder assets/icons in) the public folder
- Do not to overload the template and place the generated icon code in a separate component resources/views/components/layout/favicons.blade.php
- Watch the result in a browser by revisiting http://vinyl_shop.test/