Working with Django Forms

Working with Django Forms

Forms play a vital role in web applications, serving as a communication channel between you and your visitors. By collecting inputs from the visitors and transmitting them to the backend, forms enable effective interaction and collaboration.

Previously, we demonstrated how to build a basic CRUD operation using raw HTML forms. If you followed the linked tutorial, you will see that it is not an easy task.

You need to deal with different data types, validate the user input, match the user input to different model fields, set up CSRF protection, and so on.

This will increase the difficulty for future maintenance, especially when you need to reuse the same form in multiple webpages.

Django's built-in form functionality can significantly simplify this process and automate the majority of your work.

In this article, we are going to discuss how to create forms the Django way, how to create new resources, update existing resources, as well as how to upload files via Django forms.

Creating new resources using Django form

Let's start easy and consider this basic Post model:

python
1class Post(models.Model):
2    title = models.CharField(max_length=255)
3    content = models.TextField()
4    is_published = models.BooleanField(default=False)
5
6    def __str__(self):
7        return self.title

Instead of building the corresponding form using raw HTML, let's create a forms.py file and add the following code:

python
1from django import forms
2
3
4class PostForm(forms.Form):
5    title = forms.CharField(
6        max_length=255,
7    )
8    content = forms.CharField(
9        widget=forms.Textarea(),
10    )
11
12    is_published = forms.BooleanField(
13        required=False,
14    )

Here we created three form fields, each corresponds to one model field in the Post model. For the title field, forms.CharField matches models.CharField, with the same constraint (max_length=255). This form field also corresponds to an HTML input field in the rendered webpage, which you'll see later.

The content field is a bit more complex, as you must define an extra widget. For Django forms, you must define both a form field and a widget in order to properly render a field. The form field tells Django what data type it will be dealing with, and the widget tells Django what HTML input element the field should use.

These two concepts seem like the same thing, but Django separates them because sometimes the same form field requires different widgets. For example, the forms.CharField tells Django it should be expecting characters and texts, and by default, this field is tied to the forms.TextInput widget. That is the right widget for the title field, but for content, we should use a large textbox, so its widget is set to forms.Textarea.

Lastly, for the is_published field, forms.BooleanField uses the CheckboxInput widget. Since this field indicates the status of the post, either published or not published, you must allow the checkbox to be unchecked by setting required=False. This way the is_published field can return either True or False, instead of forcing it to be checked.

Besides the CharField and BooleanField, Django also offers other form fields such as: DateTimeField, EmailField, FileField, ImageField, ChoiceField, MultipleChoiceField, and so on. Please refer to the documentation for a full list of form fields available in Django.

Each of these fields also takes a number of extra arguments, such as the max_length, required, and widget arguments we just discussed. Different form fields may take different arguments, but there are some core arguments that are available to all fields.

  • required: by default, each field assumes the value is required. If the user pass an empty value, a ValidationError will raise. If you want to accept an empty value, set required=False.
  • label: allows you to define a custom label for the field.
html
1<label for="id_title">Custom Label:</label>
  • label_suffix: by default, the label suffix is a colon (:), but you can overwrite it by setting label_suffix to a different value.
  • initial: sets the initial value for the field when rendering the form. This is especially useful when you are creating a form for updating existing resources. We will discuss more about this later.
  • widget: specifies the corresponding widget for the field.
  • help_text: include a help text when rendering the form.
html
1<span class="helptext">100 characters max.</span>
  • error_messages: The default error message is This field is required., and this argument allows you to overwrite it.
  • validators: allows you to specify a custom validate method.
  • localize: enables the localization of form data input.
  • disabled: if set to True, the field will be rendered with a disabled attribute.

To render the form we just defined, create a view function.

python
1from django.shortcuts import render
2from .forms import PostForm
3
4
5def create(request):
6    if request.method == "GET":
7        return render(request, "post/create.html", {"form": PostForm})

And in the create.html template, print this form like a regular variable.

html
1{% extends 'layout.html' %} {% block title %}
2<title>Create</title>
3{% endblock %} {% block content %}
4<div class="w-96 mx-auto my-8">
5  <h2 class="text-2xl font-semibold underline mb-4">Create new post</h2>
6  <form action="{% url 'create' %}" method="POST">
7    {% csrf_token %} {{ form }}
8    <button type="submit" class=". . .">Submit</button>
9  </form>
10</div>
11{% endblock %}

The form will be outputted like this:

django form

html
1<form action=". . ." method="POST">
2  <label for="id_title">Title:</label>
3  <input type="text" name="title" maxlength="255" required="" id="id_title" />
4
5  <label for="id_content">Content:</label>
6  <textarea
7    name="content"
8    cols="40"
9    rows="10"
10    required=""
11    id="id_content"></textarea>
12
13  <label for="id_is_published">Is published:</label>
14  <input type="checkbox" name="is_published" id="id_is_published" />
15
16  <button type="submit" class=". . .">Submit</button>
17</form>

Styling your Django form

As you can see, the form does not look ideal. To improve that, you must edit the widget so that the rendered HTML input element would include class names.

python
1from django import forms
2
3
4class PostForm(forms.Form):
5    title = forms.CharField(
6        max_length=255,
7        widget=forms.TextInput(
8            attrs={
9                "class": "mb-4 p-2 w-full bg-gray-50 rounded border border-gray-300 focus:ring-3 focus:ring-blue-300"
10            }
11        ),
12    )
13    content = forms.CharField(
14        widget=forms.Textarea(
15            attrs={
16                "class": ". . ."
17            }
18        ),
19    )
20    is_published = forms.BooleanField(
21        required=False,
22        widget=forms.CheckboxInput(
23            attrs={
24                "class": ". . ."
25            }
26        ),
27    )

Notice that we added an attrs key, which stands for attributes. The specified attributes will be rendered as a part of the HTML input element. Of course, it does not have to be class, you can also add id, size, or something else, depending on the widget you are using.

The revised form should give an improved look:

form with style

html
1<form action="/create/" method="POST" enctype="multipart/form-data">
2  <input type="hidden" name="csrfmiddlewaretoken" value=". . ." />
3  <label for="id_title">Title:</label>
4  <input
5    type="text"
6    name="title"
7    class="mb-4 p-2 w-full bg-gray-50 rounded border border-gray-300 focus:ring-3 focus:ring-blue-300"
8    maxlength="255"
9    required=""
10    id="id_title" />
11
12  <label for="id_content">Content:</label>
13  <textarea
14    name="content"
15    cols="40"
16    rows="10"
17    class=". . ."
18    required=""
19    id="id_content"></textarea>
20
21  <label for="id_is_published">Is published:</label>
22  <input
23    type="checkbox"
24    name="is_published"
25    class="mb-4"
26    id="id_is_published" />
27  <button type="submit" class=". . .">Submit</button>
28</form>

Form submission

Of course, just rendering the form is not enough. You also need to deal with the user input and handle form submissions. To accomplish this, make sure the form has method=POST, then go back to the view method, and create a condition where a POST request is received.

python
1def create(request):
2    if request.method == "GET":
3        return render(request, "post/create.html", {"form": PostForm})
4    elif request.method == "POST":
5        form = PostForm(request.POST)
6        if form.is_valid():
7            post = Post(
8                title=form.cleaned_data["title"],
9                content=form.cleaned_data["content"],
10                is_published=form.cleaned_data["is_published"],
11            )
12            post.save()
13            return redirect("list")

The form.is_valid() method will validate the form inputs, making sure they match the requirements. Remember this step is necessary, or you will not be able to retrieve the inputs using form.cleaned_data[. . .]. And next, post=Post(. . .) creates a new instance of Post, and post.save() saves it to the database.

create new post

new post admin

Dealing with relations

In a real-life application, it is very common for one model to have relations with other models. For example, our Post could belong to a User, and have multiple Tags attached.

python
1from django.db import models
2from django.contrib.auth.models import User
3
4
5# Create your models here.
6class Tag(models.Model):
7    name = models.CharField(max_length=255)
8
9    def __str__(self):
10        return self.name
11
12
13class Post(models.Model):
14    title = models.CharField(max_length=255)
15    content = models.TextField()
16    is_published = models.BooleanField(default=False)
17
18    tags = models.ManyToManyField(Tag)
19    user = models.ForeignKey(User, on_delete=models.CASCADE, blank=True)
20
21    def __str__(self):
22        return self.title

How can we adjust the form so that it allows us to define relations? Django offers two form fields, ModelChoiceField and ModelMultipleChoiceField for this purpose. They are variants of the ChoiceField and the MultipleChoiceField. The first one has the default widget Select, which creates a single selection field.

python
1single_select = forms.ChoiceField(
2    choices=[
3        ("FR", "Freshman"),
4        ("SO", "Sophomore"),
5        ("JR", "Junior"),
6        ("SR", "Senior"),
7        ("GR", "Graduate"),
8    ],
9    widget=forms.Select(
10        attrs={. . .}
11    ),
12)

single select

html
1<label for="id_single_select">Single select:</label>
2<select name="single_select" class=". . ." id="id_single_select">
3  <option value="FR">Freshman</option>
4  <option value="SO">Sophomore</option>
5  <option value="JR">Junior</option>
6  <option value="SR">Senior</option>
7  <option value="GR">Graduate</option>
8</select>

The latter has the default widget SelectMultiple, which creates a multi-select field.

python
1multi_select = forms.MultipleChoiceField(
2    choices=[
3        ("FR", "Freshman"),
4        ("SO", "Sophomore"),
5        ("JR", "Junior"),
6        ("SR", "Senior"),
7        ("GR", "Graduate"),
8    ],
9    widget=forms.SelectMultiple(
10        attrs={. . .}
11    ),
12)

multi select

html
1<label for="id_multi_select">Multi select:</label>
2<select
3  name="multi_select"
4  class=". . ."
5  required=""
6  id="id_multi_select"
7  multiple="">
8  <option value="FR">Freshman</option>
9  <option value="SO">Sophomore</option>
10  <option value="JR">Junior</option>
11  <option value="SR">Senior</option>
12  <option value="GR">Graduate</option>
13</select>

The ModelChoiceField and ModelMultipleChoiceField are based on these choice fields, but instead of defining a choices argument, they can directly pull available choices from the database through models by specifying a queryset argument.

python
1user = forms.ModelChoiceField(
2    queryset=User.objects.all(),
3    widget=forms.Select(
4        attrs={
5            "class": "mb-4 p-2 w-full bg-gray-50 rounded border border-gray-300 focus:ring-3 focus:ring-blue-300"
6        }
7    ),
8)
9tags = forms.ModelMultipleChoiceField(
10    queryset=Tag.objects.all(),
11    widget=forms.SelectMultiple(
12        attrs={
13            "class": "mb-4 p-2 w-full bg-gray-50 rounded border border-gray-300 focus:ring-3 focus:ring-blue-300"
14        }
15    ),
16)

relation fields

Don't forget to edit the view function so that the relations can be saved.

python
1def create(request):
2    if request.method == "GET":
3        return render(request, "post/create.html", {"form": PostForm})
4    elif request.method == "POST":
5        form = PostForm(request.POST)
6        if form.is_valid():
7            post = Post(
8                title=form.cleaned_data["title"],
9                content=form.cleaned_data["content"],
10                is_published=form.cleaned_data["is_published"],
11            )
12            post.save()
13            user = form.cleaned_data["user"]
14            user.post_set.add(post)
15            post.tags.set(form.cleaned_data["tags"])
16            return redirect("list")

Uploading files using Django form

Sometimes when publishing a post, you might want to include a cover image to attract more audience. Django also provides an ImageField that enables you to upload images to your server.

python
1image = forms.ImageField(
2    widget=forms.ClearableFileInput(
3        attrs={. . .}
4    ),
5)

image field

Of course, you cannot save the image in the database. The image is stored on your server, and the path that points to the image will be saved to the database. To accomplish this, it is best to create a helper function.

python
1import uuid
2
3
4def upload_file(f):
5    path = "uploads//media/community/" + str(uuid.uuid4()) + ".png"
6    with open(path, "wb+") as destination:
7        for chunk in f.chunks():
8            destination.write(chunk)
9    return path

This function takes a file (f) as the input. It generates a random name for the file, saves it under the directory uploads//media/community/, and returns the file path as the output. You should make sure the directory exists, or the function will give an error.

Next, edit the view function like this:

python
1def create(request):
2    if request.method == "GET":
3        return render(request, "post/create.html", {"form": PostForm})
4    elif request.method == "POST":
5        form = PostForm(request.POST, request.FILES)
6        if form.is_valid():
7            path = upload_file(request.FILES["image"])
8            post = Post(
9                title=form.cleaned_data["title"],
10                content=form.cleaned_data["content"],
11                is_published=form.cleaned_data["is_published"],
12                image=path,
13            )
14            post.save()
15            user = form.cleaned_data["user"]
16            user.post_set.add(post)
17            post.tags.set(form.cleaned_data["tags"])
18            return redirect("list")

Line 5, files are transferred separately from the POST body, so here you must also include request.FILES.

Line 7, use the helper function to upload the file, the file path should be saved to the variable path.

Line 12, save the path to the database.

Lastly, you also need to ensure your form has enctype="multipart/form-data", or uploading files will not be allowed.

html
1<form action="{% url 'create' %}" method="POST" enctype="multipart/form-data">
2  {% csrf_token %} {{ form }}
3  <button type="submit" class=". . .">Submit</button>
4</form>

Updating existing resources using Django form

So far, we've only been talking about how to create new resources using Django forms, but what if you need to update existing resources? There are two major problems we need to tackle in order to achieve this. First of all, when displaying the form, we must include data from the old resource by setting the initial argument.

python
1def update(request, id):
2    post = Post.objects.get(pk=id)
3    if request.method == "GET":
4        return render(
5            request,
6            "post/update.html",
7            {
8                "form": PostForm(
9                    initial={
10                        "title": post.title,
11                        "content": post.content,
12                        "image": post.image,
13                        "is_published": post.is_published,
14                        "user": post.user,
15                        "tags": post.tags.all,
16                    }
17                ),
18                "post": post,
19            },
20        )

Post.objects.get(pk=id) retrieves the requested post based on its id.

The outputted form should look like this:

update form

The second problem with this form is that sometimes you don't need to update the image, but if you don't, Django will return a validation error. As we have mentioned before, Django assumes all form fields are required.

html
1<ul class="errorlist">
2  <li>
3    image
4    <ul class="errorlist">
5      <li>This field is required.</li>
6    </ul>
7  </li>
8</ul>

So you'll have to set required=False for the image field.

python
1is_published = forms.BooleanField(
2    required=False,
3    widget=forms.CheckboxInput(attrs={. . .}),
4)

And then add the condition request.method == "POST" for the view function.

python
1def update(request, id):
2    post = Post.objects.get(pk=id)
3    if request.method == "GET":
4        return render(
5            request,
6            "post/update.html",
7            {
8                "form": PostForm(
9                    initial={
10                        "title": post.title,
11                        "content": post.content,
12                        "image": post.image,
13                        "is_published": post.is_published,
14                        "user": post.user,
15                        "tags": post.tags.all,
16                    }
17                ),
18                "post": post,
19            },
20        )
21    elif request.method == "POST":
22        form = PostForm(request.POST, request.FILES)
23        if form.is_valid():
24            path = (
25                upload_file(request.FILES["image"])
26                if "image" in request.FILES
27                else post.image
28            )
29            Post.objects.update_or_create(
30                pk=id,
31                defaults={
32                    "title": form.cleaned_data["title"],
33                    "content": form.cleaned_data["content"],
34                    "is_published": form.cleaned_data["is_published"],
35                    "image": path,
36                },
37            )
38            user = form.cleaned_data["user"]
39            user.post_set.add(post)
40            post.tags.set(form.cleaned_data["tags"])
41            return redirect("list")

Line 24 to 28, since image might be None in this case, you have to account for this condition. If image is in request.FILES, the image is uploaded, and the file path is stored in the variable path. If image is not in request.FILES, path is set to the original file path.

ModelForm, a shortcut

As we've mentioned at the beginning of this article, the whole point of using Django forms is to simplify the form-building process. But as you can see, the create() and update() views demonstrated in this tutorial are not simple at all, even though we only have a very basic form.

Luckily, Django offers a shortcut, ModelForm, which allows you to create forms directly from models, and when saving the form, all you need to do is form.save(), without having to retrieve the user inputs one by one, and match them with each model field. And the best part is, it also works for relations and file uploads. Let's take a look at this example:

python
1class PostModelForm(forms.ModelForm):
2    class Meta:
3        model = Post
4        fields = ["title", "content", "image", "is_published", "tags", "user"]

Instead of setting up each field, you only need to tell Django the corresponding model, as well as the model fields you wish to be included in the form.

The view functions are a lot simpler too:

python
1def create(request):
2    if request.method == "GET":
3        return render(request, "post/create.html", {"form": PostModelForm})
4    elif request.method == "POST":
5        form = PostModelForm(request.POST, request.FILES)
6        if form.is_valid():
7            form.save()
8            return redirect("list")

Line 7, this is all you need to do to save the form input to the database.

python
1def update(request, id):
2    post = Post.objects.get(pk=id)
3    if request.method == "GET":
4        return render(
5            request,
6            "post/update.html",
7            {
8                "form": PostModelForm(instance=post),
9                "post": post,
10            },
11        )
12    elif request.method == "POST":
13        form = PostModelForm(request.POST, request.FILES, instance=post)
14        if form.is_valid():
15            form.save()
16            return redirect("list")

Line 8 and 13, pass the existing resource to the form.

In this article, we went over the basics of building web forms using Django's Form and ModelForm classes, which should significantly simplify your form-building process.

Interested in Learning Web Development?

Starting from HTML, CSS to JavaScript, from language syntaxes to frameworks, from the frontend to the backend, we'll guide you through every step of your coding journey.

Get started from here 🎉