Courses

[FREE] Laravel 11 For Beginners: Your First Project

Main Menu, Blade Structure and Static Pages

In this lesson, we will add two more pages: "About" and "Contact".

Of course, we could use File -> Save as in the IDE and change text where needed, but this would make a lot of duplicate code, and changing something in the future would require changing every file.

Instead, we need to create a main layout and extend it on every individual page. Layouts can be created in two ways:

This is a personal preference, but in this course, we will use the second option, layouts inheritance. Mainly to get to the result quicker and to avoid explaining another new term of Blade components. My course goal is usually to help get students to the actual practical result as soon as possible.


Main Layout and First Page

First, we must create a View file: a main layout. It will go inside the resources/views/layouts folder (but you can place it wherever you want inside the resources/views) and be called app.blade.php.

In the layout, we only need to add parts of the template that must be reused on every page. In our case, it's these elements:

  • Everything in the head tag
  • JavaScript files
  • Navigation with header and footer

And then, instead of placing the page content in the layout, we must use the @yield Blade directive and give that section a name. That name will be used on every page within the @section Blade directive. So, the main layout would look like this:

resources/views/layouts/app.blade.php:

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
<meta name="description" content="" />
<meta name="author" content="" />
<title>Blog Home - Start Bootstrap Template</title>
<!-- Favicon-->
<link rel="icon" type="image/x-icon" href="assets/favicon.ico" />
<!-- Core theme CSS (includes Bootstrap)-->
<link href="{{ asset('css/styles.css') }}" rel="stylesheet" />
</head>
<body>
<!-- Responsive navbar-->
<nav class="navbar navbar-expand-lg navbar-dark bg-dark">
<div class="container">
<a class="navbar-brand" href="#!">Start Bootstrap</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation"><span class="navbar-toggler-icon"></span></button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav ms-auto mb-2 mb-lg-0">
<li class="nav-item"><a class="nav-link" href="#">Home</a></li>
<li class="nav-item"><a class="nav-link" href="#!">About</a></li>
<li class="nav-item"><a class="nav-link" href="#!">Contact</a></li>
<li class="nav-item"><a class="nav-link active" aria-current="page" href="#">Blog</a></li>
</ul>
</div>
</div>
</nav>
 
@yield('content')
 
<!-- Footer-->
<footer class="py-5 bg-dark">
<div class="container"><p class="m-0 text-center text-white">Copyright &copy; Your Website 2023</p></div>
</footer>
<!-- Bootstrap core JS-->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.bundle.min.js"></script>
<!-- Core theme JS-->
<script src="{{ asset('js/scripts.js') }}"></script>
</body>
</html>

Then, we extend this layout in every individual page, placing the actual content inside the @section block. With that, we don't need to repeat all the HTML tags and navigation with the footer.

To extend the main layout, we use the @extends Blade directive at the start of the View file and provide where the layout is.

resources/views/home.blade.php:

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
<meta name="description" content="" />
<meta name="author" content="" />
<title>Blog Home - Start Bootstrap Template</title>
<!-- Favicon-->
<link rel="icon" type="image/x-icon" href="assets/favicon.ico" />
<!-- Core theme CSS (includes Bootstrap)-->
<link href="{{ asset('css/styles.css') }}" rel="stylesheet" />
</head>
<body>
<!-- Responsive navbar-->
<nav class="navbar navbar-secondexpand-lg navbar-dark bg-dark">
<div class="container">
<a class="navbar-brand" href="#!">Start Bootstrap</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation"><span class="navbar-toggler-icon"></span></button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav ms-auto mb-2 mb-lg-0">
<li class="nav-item"><a class="nav-link" href="#">Home</a></li>
<li class="nav-item"><a class="nav-link" href="#!">About</a></li>
<li class="nav-item"><a class="nav-link" href="#!">Contact</a></li>
<li class="nav-item"><a class="nav-link active" aria-current="page" href="#">Blog</a></li>
</ul>
</div>
</div>
</nav>
@extends('layouts.app')
 
@section('content')
 
// The content of the page goes here...
 
<!--
<footer class="py-5 bg-dark">
<div class="container"><p class="m-0 text-center text-white">Copyright &copy; Your Website 2023</p></div>
</footer>
<!-- Bootstrap core JS-->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.3/dist/js/bootstrap.bundle.min.js"></script>
<!-- Core theme JS-->
<script src="{{ asset('js/scripts.js') }}"></script>
</body>
</html>
@endsection

After refreshing the home page, it should look the same.


Two More Pages: Contact and About

Now, let's create two new pages. First, create two new View files for each page and extend the main layout.

resources/views/contact.blade.php:

@extends('layouts.app')
 
@section('content')
<!-- Page header with logo and tagline-->
<header class="py-5 bg-light border-bottom mb-4">
<div class="container">
<div class="text-center my-5">
<h1 class="fw-bolder">Contact us</h1>
</div>
</div>
</header>
<!-- Page content-->
<div class="container mb-4">
<div class="row">
<!-- Blog entries-->
<div class="col-lg-12">
<p class="lead mb-0">Contact us text</p>
</div>
</div>
</div>
@endsection

resources/views/about.blade.php:

@extends('layouts.app')
 
@section('content')
<!-- Page header with logo and tagline-->
<header class="py-5 bg-light border-bottom mb-4">
<div class="container">
<div class="text-center my-5">
<h1 class="fw-bolder">About us</h1>
</div>
</div>
</header>
<!-- Page content-->
<div class="container mb-4">
<div class="row">
<!-- Blog entries-->
<div class="col-lg-12">
<p class="lead mb-0">About us text</p>
</div>
</div>
</div>
@endsection

Next, we need Routes. In this case, we only show static View files without more logic, so we can use Route::view() instead of Route::get().

routes/web.php:

Route::get('/', function () {
return view('home');
});
Route::view('/', 'home');
Route::view('contact', 'contact');
Route::view('about', 'about');

You should see a corresponding page after visiting /contact or /about.


Menu Navigation

Now, we should make these pages accessible via navigation.

First, we should add names to the Routes. It's a good practice: when using names, if the URL is changed in the future, you won't need to change the URL on every file where it was used. Instead, after changing only in the Routes file, everywhere else, it will change automatically.

routes/web.php:

Route::view('/', 'home')->name('home');
Route::view('contact', 'contact')->name('contact');
Route::view('about', 'about')->name('about');

In the main layout, where the navigation is, we can use these Route names within the Laravel helper route().

resources/views/layouts/app.blade.php:

// ...
<nav class="navbar navbar-expand-lg navbar-dark bg-dark">
<div class="container">
<a class="navbar-brand" href="#!">Start Bootstrap</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation"><span class="navbar-toggler-icon"></span></button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav ms-auto mb-2 mb-lg-0">
<li class="nav-item"><a class="nav-link" href="#">Home</a></li>
<li class="nav-item"><a class="nav-link" href="#!">About</a></li>
<li class="nav-item"><a class="nav-link" href="#!">Contact</a></li>
<li class="nav-item"><a class="nav-link" href="{{ route('home') }}">Home</a></li>
<li class="nav-item"><a class="nav-link" href="{{ route('about') }}">About</a></li>
<li class="nav-item"><a class="nav-link" href="{{ route('contact') }}">Contact</a></li>
<li class="nav-item"><a class="nav-link active" aria-current="page" href="#">Blog</a></li>
</ul>
</div>
</div>
</nav>
 
// ...

The navigation now works; we can click the links and go to three pages with different content!

avatar

There it says: "After refreshing the home page, it should look the same.". Hm, no. The section content is => // The content of the page goes here...
this is what i see, not the downloaded bootstrap/template content.

avatar

If this happened, there is an issue with the layout template. Check that and make sure it is correct