In this tutorial, we will learn about a key concept in Flask known as Template Inheritance. Template inheritance allows you to build a base "skeleton" template that contains all the common elements of your site and defines blocks that child templates can override.
By the end of this tutorial, you will:
- Understand the concept of template inheritance
- Know how to create a base template in Flask
- Learn how to extend this base template in your child templates
Prerequisites:
- Basic knowledge of Python
- Basic understanding of HTML and CSS
- Familiarity with Flask
Template inheritance is a feature in Flask which allows the reusability of HTML code. We can create a base template with common HTML structures (like header, footer, navigation bar, etc.) and then extend this base template on our other templates.
The {% block blockname %} and {% endblock %} statements in Jinja2 are used to denote blocks in your templates that can be overridden by child templates.
Let's create a base template named base.html.
<!DOCTYPE html>
<html lang="en">
<head>
    <title>{% block title %}My Website{% endblock %}</title>
</head>
<body>
    <header>
        {% block header %}Default Header{% endblock %}
    </header>
    <main>
        {% block content %}{% endblock %}
    </main>
    <footer>
        {% block footer %}Default Footer{% endblock %}
    </footer>
</body>
</html>
In the above code, we have defined several blocks (title, header, content, and footer) that can be overridden in our child templates.
Now, let's create a child template named home.html that extends base.html.
{% extends "base.html" %}
{% block title %}Home{% endblock %}
{% block header %}
    <h1>Welcome to My Website!</h1>
{% endblock %}
{% block content %}
    <p>This is the home page.</p>
{% endblock %}
{% block footer %}
    <p>Copyright 2022.</p>
{% endblock %}
In this child template, we are overriding all the blocks defined in base.html.
In this tutorial, we've learned about template inheritance in Flask, created a base template, and extended it in a child template. This concept helps us to avoid code duplication and maintain consistency across our web pages.
For your next steps, try creating more complex base templates and child templates. Practice overriding different blocks and experiment with including default content in your blocks.
Exercise 1: Create a base template with a navigation bar and extend this base template in a child template.
Solution: 
Base template (base.html):
<!DOCTYPE html>
<html lang="en">
<head>
    <title>{% block title %}My Website{% endblock %}</title>
</head>
<body>
    <nav>
        {% block nav %}
            <a href="/">Home</a> |
            <a href="/about">About</a> |
            <a href="/contact">Contact</a>
        {% endblock %}
    </nav>
    <main>
        {% block content %}{% endblock %}
    </main>
</body>
</html>
Child template (home.html):
{% extends "base.html" %}
{% block title %}Home{% endblock %}
{% block content %}
    <h1>Welcome to the home page!</h1>
{% endblock %}
Exercise 2: Create a child template that only overrides the title block of the base template.
Solution: 
Child template (about.html):
{% extends "base.html" %}
{% block title %}About{% endblock %}
In the above code, about.html inherits all content from base.html but overrides the title block.