Extra Templates
Now that we’ve scaffolded our blog templates, we can turn to some supporting features.
# About Page
Our original content model included “About” page, which we’ll set up as another section.
# Fields
Let’s by creating some fields for the page. We need a place to put a paragraph or two of text, and way to upload a profile photo.
# Bio
- In the control panel, navigate to Settings → Fields;
- Click + New group in the sidebar, and name it About;
- Click + New field in the upper-right corner;
- Provide these settings:
- Name: Bio
- Handle:
bio
- Field Type: Plain Text
- Allow line breaks: Enabled
- Initial rows:
8
- Click Save or press Ctrl/Command + S;
# Profile Photo
Picking up where you left off (in the About field group), follow these steps:
- Click + New field in the upper-right corner;
- Provide these settings:
- Name: Profile Photo
- Handle:
profilePhoto
- Field Type: Assets
- Sources: Images only
- Max relations:
1
- View mode: Large Thumbnails
- Click Save or press Ctrl/Command + S;
# Single Section
Continuing in the control panel…
- Navigate to Settings → Sections;
- Click + New section;
- Provide these settings:
- Name: About
- Handle:
about
- Section Type: Single
- URI:
about
- Template:
_singles/about
- Click Save and edit entry types or press Ctrl/Command + S;
- Select the auto-generated entry type (it should also be named About);
- Drag fields from the About group into the field layout in whatever order you prefer;
- Click Save press Ctrl/Command + S;
With that, our new page is configured! Let’s head over to Entries → Singles, then click About to add some content.
# Template
Create templates/_singles/about.twig
(the Template path we used when setting up the section) and add the following to it:
{% extends '_layout' %}
{% set profilePhoto = entry.profilePhoto.one() %}
{% block content %}
<h1>{{ entry.title }}</h1>
<div class="about">
{% if profilePhoto %}
<div class="photo">
{{ profilePhoto.getImg() }}
</div>
{% endif %}
<div class="bio">
{{ entry.bio | md }}
</div>
</div>
{% endblock %}
This template builds upon features we’ve already seen in other places—layouts, element queries, if
tags, and filters. We’ve interspersed a bit of HTML so that we can come back and apply CSS, later.
Navigate to /about
(the About section’s URI) in the front-end—you should see our page’s title
and the profile image we uploaded:
These big images really are a hassle! We’ll look at Craft’s built-in tools for transforming images in the optimization section.
# Global Features
Our site is taking shape, but it’s difficult to move from page-to-page, outside of the blog.
# Page Structure
Our pages’ content is currently output directly into the <body>
element, but it would be nice to have a bit more structure—and carve out some space for a header and footer.
Back in templates/_layout.twig
, make these updates:
<!DOCTYPE html>
<html lang="{{ craft.app.language }}">
<head>
<meta charset="utf-8"/>
<title>{{ siteName }}</title>
<meta content="width=device-width, initial-scale=1.0" name="viewport">
{% do craft.app.view.registerCssFile('@web/styles.css') %}
</head>
<body>
<header>
<a class="home" href="{{ siteUrl }}">{{ siteName }}</a>
</header>
<main>
{% block content %}
{# Nothing here, yet! #}
{% endblock %}
</main>
<footer>
<div class="copyright">©{{ now | date('Y') }} {{ siteName }}</div>
<div class="colophon">Built with <a href="https://craftcms.com/" target="_blank">Craft CMS</a></div>
</footer>
</body>
</html>
This small change gives our content a dedicated region, and separates it from any global features we might add… like navigation!
# Navigation
The new <header>
region is a great place to put a globally-available menu. Just below the anchor tag (<a>
), add a new <nav>
element:
<a class="home" href="{{ siteUrl }}">{{ siteName }}</a>
<nav>
<ul>
<li>
<a href="{{ url('blog') }}">Blog</a>
</li>
<li>
<a href="{{ url('about') }}">About</a>
</li>
</ul>
</nav>
What a relief—we can finally click around the site, and every page that extends _layout.twig
gets our centrally-defined menu without any additional work!
# Footer
The updates above included a couple of items in the footer, but it’s worth taking a moment to make this space even more useful.
# Global Sets
Global sets are Craft’s way of storing a data that should be accessible everywhere in our templates. They function almost exactly like the Single section we defined for the About page, except that global sets don’t get their own URLs.
In the control panel, navigate to Settings → Fields, then follow these steps:
- Click + New group and name it Global;
- Click + New field;
- Provide these field settings:
- Name: Description
- Handle:
description
- Field type: Plain Text
- Allow line breaks: Enabled
- Click Save or press Ctrl/Command + S;
With the field created, navigate to Settings → Global Sets:
- Click + New global set;
- Name the set Site Info and use
siteInfo
for the handle; - Click + New Tab in the field layout designer, and name it Settings, Content, or something else generic (when there’s only a single tab, Craft will hide it—but it can’t be empty);
- Click Save or press Ctrl/Command + S;
A new Globals item should appear in the main navigation. Click that, and add a description:
Back in templates/_layout.twig
, output that blurb in the footer:
<footer>
<div class="description">
{{ siteInfo.description | md }}
</div>
<div class="copyright">©{{ now | date('Y') }} {{ siteName }}</div>
<div class="colophon">Built with <a href="https://craftcms.com/" target="_blank">Craft CMS</a></div>
</footer>
Let’s put HTML aside for a moment and get some styles applied!