Flask will be looking for our HTML pages in a "templates" directory by default. So let's create a new subdirectory in the "web_app" folder called "templates", with new HTML files called "home.html", "about.html", and "hello.html", with the following contents inside, respectively.
Inside "web_app/templates/home.html":
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>My App</title>
</head>
<body>
<h1>Welcome Home</h1>
<p>This is a paragraph on the "home" page.</p>
<nav>
<ul>
<li><a href="/">Home</a></li>
<li><a href="/about">About</a></li>
<li><a href="/hello">Hello</a></li>
</ul>
</nav>
<footer>
<p>My footer</p>
</footer>
</body>
</html>
Inside "web_app/templates/about.html":
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>My App</title>
</head>
<body>
<h1>About Me</h1>
<p>This is a paragraph on the "about" page.</p>
<nav>
<ul>
<li><a href="/">Home</a></li>
<li><a href="/about">About</a></li>
<li><a href="/hello">Hello</a></li>
</ul>
</nav>
<footer>
<p>My footer</p>
</footer>
</body>
</html>
Inside "web_app/templates/hello.html":
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>My App</title>
</head>
<body>
<h1>{{ message }}</h1>
<p>This is a paragraph on the "hello" page.</p>
<nav>
<ul>
<li><a href="/">Home</a></li>
<li><a href="/about">About</a></li>
<li><a href="/hello">Hello</a></li>
</ul>
</nav>
<footer>
<p>My footer</p>
</footer>
</body>
</html>
Notice each HTML page has a unique heading, and a shared navigation and footer. We'll refactor the duplicate / shared HTML code later, but for right now it's fine.
Let's update the home routes to render these HTML pages:
# web_app/routes/home_routes.pyfrom flask import Blueprint, request, render_templatehome_routes =Blueprint("home_routes", __name__)@home_routes.route("/")@home_routes.route("/home")defindex():print("HOME...")returnrender_template("home.html")@home_routes.route("/about")defabout():print("ABOUT...")returnrender_template("about.html")@home_routes.route("/hello")defhello_world():print("HELLO...", dict(request.args))# NOTE: `request.args` is dict-like, so below we're using the dictionary's `get()` method,# ... which will return None instead of throwing an error if key is not present# ... see also: https://www.w3schools.com/python/ref_dictionary_get.asp name = request.args.get("name")or"World" message =f"Hello, {name}!"returnrender_template("hello.html", message=message)
Notice we're passing a variable called message from the router to the "hello" page, and using the "Jinja" template language to reference it (i.e. {{ message }}).
Restart the server and view the app in the browser and navigate between the three HTML pages. Use URL params to customize the name on the "hello" page. Cool!
Shared Layouts
Right now each template is a complete HTML file. If we wanted to continue to implement common navigation and footer across each of the files, we'd have to copy and paste the same header and footer into all the files. This is not ideal from a maintainability perspective.
Luckily we have the ability to define the shared aspects of each page, like the header and footer, and inherit them from a common "layout".
Let's create a new file in the "templates" directory called "layout.html", and place inside the following contents:
`), which is a placeholder for the respective page contents.
Let's update the "home.html", "about.html", and "hello.html" templates to leverage this shared layout, using the following contents, respectively:
Inside "web_app/templates/home.html":
{% extends "layout.html" %}
{% block content %}
<h1>Welcome Home</h1>
<p>This is a paragraph on the "home" page.</p>
{% endblock %}
Inside "web_app/templates/about.html":
{% extends "layout.html" %}
{% block content %}
<h1>About Me</h1>
<p>This is a paragraph on the "about" page.</p>
{% endblock %}
Inside "web_app/templates/hello.html":
{% extends "layout.html" %}
{% block content %}
<h1>{{ message }}</h1>
<p>This is a paragraph on the "hello" page.</p>
{% endblock %}
Notice each of these templates is inheriting from the "layout.html" template, and specifying some unique page contents to overwrite the "content" block.
Restart your server and view your app in the browser and use the HTML links to navigate between pages. Observe the consistent header and footer. Nice!