Partial Templates in Django

Creating a Partial Template

  1. Create the Partial Template File: Create a separate HTML file for the partial template. For instance, you might create a file called _header.html for the header of your web pages.
    <!-- templates/partials/_header.html -->
    <header>
        <h1>My Website Header</h1>
        <nav>
            <ul>
                <li><a href="/">Home</a></li>
                <li><a href="/about/">About</a></li>
                <li><a href="/contact/">Contact</a></li>
            </ul>
        </nav>
    </header>
    
  2. Include the Partial Template in Your Main Template: Use the {% include %} template tag to include the partial template in your main templates.
    <!-- templates/base.html -->
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <title>My Website</title>
    </head>
    <body>
        {% include "partials/_header.html" %}
        
        <main>
            {% block content %}
            <!-- Main content goes here -->
            {% endblock %}
        </main>
        
        <footer>
            <p>&copy; 2024 My Website</p>
        </footer>
    </body>
    </html>
    
  3. Extending the Base Template: If you are using a base template that other templates will extend, make sure to extend it properly in your child templates.
    <!-- templates/home.html -->
    {% extends "base.html" %}
    
    {% block content %}
    <h2>Welcome to My Website!</h2>
    <p>This is the homepage.</p>
    {% endblock %}

Using Context in Partial Templates

If you need to pass context variables to a partial template, you can include it with additional context using the with keyword.

  1. Update the Partial Template:
    <!-- templates/partials/_header.html -->
    <header>
        <h1>{{ site_name }}</h1>
        <nav>
            <ul>
                <li><a href="/">Home</a></li>
                <li><a href="/about/">About</a></li>
                <li><a href="/contact/">Contact</a></li>
            </ul>
        </nav>
    </header>
    
  2. Include the Partial Template with Context:
    <!-- templates/base.html -->
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <title>{{ title }}</title>
    </head>
    <body>
        {% include "partials/_header.html" with site_name="My Awesome Website" %}
        
        <main>
            {% block content %}
            <!-- Main content goes here -->
            {% endblock %}
        </main>
        
        <footer>
            <p>&copy; 2024 My Awesome Website</p>
        </footer>
    </body>
    </html>
    

By breaking your templates into partials and using the {% include %} tag, you can keep your templates clean, modular, and easy to manage. This approach is particularly useful for repetitive components like headers, footers, navigation bars, and sidebars.

Template Inheritance in Django

Basic Concepts

  1. Base Template: A general template that includes common elements (like headers, footers, and navigation bars) used across multiple pages.
  2. Block Tags: Special tags in Django templates that define sections of the template that can be overridden by child templates.

Steps to Implement Template Inheritance

1. Create a Base Template

First, create a base template that includes the common layout for your website. For example, base.html:

<!DOCTYPE html>
<html>
<head>
    <title>{% block title %}My Site{% endblock %}</title>
    <link rel="stylesheet" type="text/css" href="{% static 'css/styles.css' %}">
</head>
<body>
    <header>
        <h1>Welcome to My Site</h1>
    </header>

    <nav>
        <ul>
            <li><a href="{% url 'home' %}">Home</a></li>
            <li><a href="{% url 'about' %}">About</a></li>
            <li><a href="{% url 'contact' %}">Contact</a></li>
        </ul>
    </nav>

    <div class="content">
        {% block content %}{% endblock %}
    </div>

    <footer>
        <p>&copy; 2024 My Site</p>
    </footer>
</body>
</html>

2. Create a Child Template

Next, create a child template that extends the base template. For example, home.html:

{% extends 'base.html' %}

{% block title %}Home - My Site{% endblock %}

{% block content %}
    <h2>Home Page</h2>
    <p>Welcome to the home page of my awesome site!</p>
{% endblock %}

3. Render the Template in Views

In your Django views, render the child template. For example, in views.py:

from django.shortcuts import render

def home(request):
    return render(request, 'home.html')

4. URL Configuration

Ensure you have the appropriate URL configuration in urls.py:

from django.urls import path
from . import views

urlpatterns = [
    path('', views.home, name='home'),
    # other paths
]

Key Points to Remember

  • {% extends 'base.html' %}: This tag is used in the child template to specify the base template it extends.
  • Block Tags ({% block content %}{% endblock %}): Define sections in the base template that can be overridden by child templates.
  • Static Files: Use the {% static %} template tag to include static files like CSS and JavaScript.

Example Directory Structure

Here’s an example of what your directory structure might look like:

myproject/
    myapp/
        templates/
            base.html
            home.html
        views.py
        urls.py
    myproject/
        settings.py
        urls.py
        wsgi.py

By using template inheritance, you can efficiently manage and update the layout and design of your website while ensuring consistency across different pages.

url Tag in Django Template

In Django, the {% url %} template tag is used to generate URLs by reversing the URL patterns defined in your urls.py file. This approach promotes cleaner code and helps avoid hard-coding URLs in your templates.

Here’s how you can use the {% url %} tag:

Syntax: The {% url %} tag syntax is:

{% url 'viewname' arg1 arg2 %}

Here, 'viewname' is the name of the URL pattern you defined in your urls.py. arg1, arg2, etc., are the arguments that the URL pattern expects.

Example: Suppose you have a URL pattern defined in your urls.py like this:

# urls.py
from django.urls import path
from . import views

urlpatterns = [
    path('article/<int:year>/', views.year_archive, name='news-year-archive'),
]

You can use the {% url %} tag in your template like this:

<!-- In your Django template -->
<a href="{% url 'news-year-archive' 2022 %}">View 2022 Archive</a>

This will resolve to the URL for viewing the 2022 archive, assuming your app is hosted at the root.

Using the {% url %} tag helps ensure your templates remain easy to maintain, particularly if you later decide to change URL patterns. The links in your templates will update automatically to reflect the new patterns.

References
https://docs.djangoproject.com/en/5.0/ref/templates/builtins/#url

Django Template Filters

Django template filters are simple Python functions that modify the values of variables for display in templates. They are used within Django’s template language to transform data before rendering it to the user. Filters can be applied using the pipe (|) symbol and can take optional arguments.

Here’s a basic example of a Django template filter:

Suppose you have a variable called name in your template, and you want to display it in all uppercase letters. Django has a built-in filter called upper that can do this. Here’s how you might use it in a template:

<!-- Assuming 'name' has the value 'Mahmood' -->
<p>Your name in uppercase is: {{ name|upper }}</p>

This would render as:

<p>Your name in uppercase is: MAHMOOD</p>

This example demonstrates how to use the upper filter to convert the text of a variable to uppercase before rendering it on a webpage. Django provides a variety of built-in filters for common tasks like formatting dates, handling strings, and manipulating arrays. You can also create your own custom filters if you need functionality that isn’t provided by the built-in options.

References
https://docs.djangoproject.com/en/5.0/ref/templates/builtins/#built-in-filter-reference

Pass Parameters from Views to Templates in Django

In Django, you pass parameters from views to templates through the context dictionary. This dictionary serves as the medium for transferring data from your view to your template. Here’s a step-by-step guide on how to do this:

Define Your View

First, you define your view in your views.py file. Django views can be either functions or classes. Here’s an example of a simple function-based view:

from django.shortcuts import render

def my_view(request):
    my_variable = "Welcome to Django!"
    context = {'my_variable': my_variable}
    return render(request, 'my_template.html', context)

In this example, my_variable is the parameter we want to pass to the template. The context dictionary is used to pass this parameter.

Create Your Template

Next, create your template file (my_template.html), where you’ll use the variable passed from the view. Here’s how you might use it in the template:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>My Page</title>
</head>
<body>
    <h1>{{ my_variable }}</h1>
</body>
</html>

In the template, {{ my_variable }} will be replaced with “Welcome to Django!” when the template is rendered.

Add and Register templates in Django

Setting Up Template Directories

First, you need to define where Django should look for templates. You can set this up in your settings file (settings.py).

  1. Template Directory: By default, Django looks for a templates folder in each app directory. You can also set up a global templates directory by adding it to the DIRS option in the TEMPLATES configuration in settings.py. Here’s an example:
    TEMPLATES = [
        {
            'BACKEND': 'django.template.backends.django.DjangoTemplates',
            'DIRS': [BASE_DIR / 'templates'],  # Path to global templates
            'APP_DIRS': True,
            'OPTIONS': {
                # ... some options here ...
            },
        },
    ]
    

    In this configuration, BASE_DIR / 'templates' refers to a templates directory at the root of your project.

Creating Templates

Create HTML template files in the appropriate templates folder. If you’re using app-specific templates, it’s a good practice to create a subdirectory within the templates folder that matches the name of your app to avoid naming collisions between apps.

For example, if you have an app named blog, your directory structure might look like this:

blog/
│
├── templates/
│   └── blog/
│       └── post_detail.html

Using Templates in Views

To use a template in a view, you need to load it and render it with a context. Here’s a basic example of how to do this in a view function:

from django.shortcuts import render

def post_detail(request):
    context = {'post': Post.objects.get(id=1)}
    return render(request, 'blog/post_detail.html', context)

In this example, post_detail.html will be loaded from the blog/templates/blog/ directory, and the post object will be passed to it.

Use render_to_string

The render_to_string function in Django is very useful for rendering a template into a string. It’s often used when you need the rendered output of a template but don’t want to return an HTTP response directly, such as when sending emails or generating PDF files from HTML. Here’s a basic example to illustrate how to use render_to_string:

from django.template.loader import render_to_string

def send_welcome_email(user):
    # Context to pass to the template
    context = {'user': user}
    
    # Render the template into a string
    email_html_message = render_to_string('users/welcome_email.html', context)
    
    # Send the email using Django's email functionality (assuming configuration is set)
    send_mail(
        'Welcome to Our Site!',
        strip_tags(email_html_message),  # Optional: plain text content
        'from@example.com',  # From email
        [user.email],  # To email list
        html_message=email_html_message,  # HTML content
    )

 

Return a redirect response in Django

from django.http import HttpResponseRedirect

def my_view(request):
    # your logic here
    return HttpResponseRedirect('/another-page/')

Or, if you want to redirect to a URL resolved by a named URL pattern, you’d typically use the reverse function to dynamically build the URL:

# urls.py

from django.urls import path
from . import views

urlpatterns = [
    path('home/', views.home, name='home'),
    path('profile/<str:username>/', views.profile, name='profile'),
]
# views.py

from django.http import HttpResponseRedirect
from django.urls import reverse

def my_view(request):
    # Example logic here
    # Suppose we determine the username as follows:
    username = 'john_doe'
    
    # Using reverse to dynamically generate the URL
    url = reverse('profile', args=[username])
    
    # Redirecting to the dynamically generated URL
    return HttpResponseRedirect(url)

def home(request):
    # Logic for home view
    pass

def profile(request, username):
    # Logic for profile view
    return HttpResponse(f"Profile page of {username}")

 

Return a 404 response in Django

To return a 404 response in Django, you can use the Http404 exception or the get_object_or_404() utility function, which are both simple and efficient ways to handle cases where a resource is not found. Here’s how you can use each method:

1. Using Http404 Exception

You can raise the Http404 exception in your views when a certain condition is not met (e.g., when an object does not exist in the database). Here’s an example:

from django.http import Http404

def my_view(request, id):
    try:
        item = MyModel.objects.get(pk=id)
    except MyModel.DoesNotExist:
        raise Http404("Item does not exist")

    # Proceed with the rest of the view
    return render(request, 'template.html', {'item': item})

2. Using get_object_or_404()

Django provides a shortcut function get_object_or_404() that encapsulates the try-except block used with Http404. It tries to get an object from the model and if it fails, it automatically raises Http404. Here’s how to use it:

from django.shortcuts import get_object_or_404, render

def my_view(request, id):
    item = get_object_or_404(MyModel, pk=id)
    return render(request, 'template.html', {'item': item})