Logo

dev-resources.site

for different kinds of informations.

Creating a To-Do app with HTMX and Django, part 7: infinite scroll

Published at
1/6/2025
Categories
htmx
django
python
tailwindcss
Author
rodbv
Categories
4 categories in total
htmx
open
django
open
python
open
tailwindcss
open
Creating a To-Do app with HTMX and Django, part 7: infinite scroll

This is part 7 of the series in which I'm documenting my learning process of HTMX with Django, in which we will follow HTMX's documentation to implement an infinite scroll feature for the todo items.

If you want to check the rest of the series, have a look at dev.to/rodbv for the complete list.

Updating the partial template to load several items

When we implement infinite scroll, we will have to return several todo items (the next "page" of items) and load them in the partial template we currently have. This means changing a bit how our partial template is composed; it's currently set as described in the diagram below, in which the partial template is responsible for rendering a single todo item:

Diagram of the partial template before: the partial template renders a single item, inside a loop for all todo items

We want to invert the order, having the partial around the for loop:

Diagram of the partial template after: the partial template renders a list of items, arouund the for loop

Let's perform the swap in the template core/templates/index.html:

<ul id="todo-items" class="list bg-base-100 rounded-box shadow-md">
  {% partialdef todo-items-partial inline %}
    {% for todo in todos %} 
        <li class="list-row">
           ... code inside the li, unchanged
        </li>
      {% endfor %}
    {% endpartialdef %} 
</ul>

Soon we will get back to the template to add the hx-get ... hx-trigger="revealed" bit that performs the infinite scroll, but first let's just change the view to return several items instead of one on the toggle and create operations:

... previous code 

def _create_todo(request):
    title = request.POST.get("title")
    if not title:
        raise ValueError("Title is required")

    todo = Todo.objects.create(title=title, user=request.user)

    return render(
        request,
        "tasks.html#todo-items-partial", # <-- CHANGED
        {"todos": [todo]}, # <-- CHANGED
        status=HTTPStatus.CREATED,
    )

... previous code 


@login_required
@require_http_methods(["PUT"])
def toggle_todo(request, task_id):
    todo = request.user.todos.get(id=task_id)
    todo.is_completed = not todo.is_completed
    todo.save()

    return render(
        request,
        "tasks.html#todo-items-partial",  # <-- CHANGED
        {
            "todos": [todo], # <-- CHANGED
        },
    )

The tests checking for the content being still pass, and the page looks the same, so we're good to implement the infinite scroll itself.

Implementing infinite scroll

On the template, we need to setup a hx-get request to /tasks, with hx-trigger="revealed", which means the GET request is only fired when the element is about to enter become visible on screen; this means we want it to be set after the last element in the list, and we also need to indicate which "page" of data we want to load. In our case we'll show 20 items at a time.

Loader element

Let's change the template accordingly:

    <ul id="todo-items" class="list bg-base-100 rounded-box shadow-md" hx-indicator=".indicator">
      {% partialdef todo-items-partial inline %}
        {% for todo in todos %} 
          <li class="list-row">
                ...code inside the li is unchanged
          </li>
        {% endfor %}
        {% if next_page_number %}
          <p id="loading" 
              hx-get="{% url 'tasks' %}?page={{next_page_number}}" 
              hx-trigger="revealed" 
              hx-swap="outerHTML" 
              class="mx-auto my-4 loading loading-dots"></p>
        {% endif %}
        {% endpartialdef %} 
    </ul>

There's an if next_page_number check around the "loading" icon at the bottom of the list, it will have two purposes: one is to indicate when we're loading more data, but more importantly, when the loader is revealed (it appears on the visible part of the page), it will trigger the hx-get call to /tasks, passing the page number to be retrieved. The attribute next_page_number will also be provided by the context

The directive hx-swap:outerHTML indicates that we will replace the outerHTML of this element with the set of <li>s we get from the server, which is great because not only we show the new data we got, but we also get rid of the loading icon.

We can now move to the views file.

As a recap, here's how the GET /tasks view looks like by now; it's always returning the full template.

@require_http_methods(["GET", "POST"])
@login_required
def tasks(request):
    if request.method == "POST":
        return _create_todo(request)

    # GET /tasks
    context = {
        "todos": request.user.todos.all().order_by("-created_at"),
        "fullname": request.user.get_full_name() or request.user.username,
    }

    return render(request, "tasks.html", context)

There's a change done in the code above already, which is to sort by newest todos first; now that we expect to have a long list, it doesn't make sense to add new items at the bottom and mix it with infinite scroll - the new item will end up mixed in the middle of the list.

We now need to differentiate regular GET requests from HTMX requests, for which we will return just a list of todos and our partial template. There is a library called django-htmx which is very handy, as it extends the request parameter with attributes like request.htmx and the values of all hx-* attributes, but that's overkill at the moment; let's just check for the HTMX header by now, and handle paging using Django's paginator.

# core/views.py

... previous code

PAGE_SIZE = 20

...previous code

@require_http_methods(["GET", "POST"])
@login_required
def tasks(request):
    if request.method == "POST":
        return _create_todo(request)

    page_number = int(request.GET.get("page", 1))

    all_todos = request.user.todos.all().order_by("-created_at")
    paginator = Paginator(all_todos, PAGE_SIZE)
    curr_page = paginator.get_page(page_number)

    context = {
        "todos": curr_page.object_list,
        "fullname": request.user.get_full_name() or request.user.username,
        "next_page_number": page_number + 1 if curr_page.has_next() else None,
    }

    template_name = "tasks.html"

    if "HX-Request" in request.headers:
        template_name += "#todo-items-partial"

    return render(request, template_name, context)

The first thing we do is to check the page param, and set it to 1 if it's not present.

We check for the HX-Request header in the request, which will inform us whether the incoming request is from HTMX, and lets us return the partial template or the full template accordingly.

This code requires some tests for sure, but before that let's just give it a go. Have a look at the network tool, how the requests are fired as the page is scrolled, until we reach the last page. You can also see the animated "loading" icon showing for a brief moment; I've throttled the network speed to 4g to make it visible for longer.

Scrolling in action

Adding tests

To wrap it up, we can add a test to ensure pagination is working as intended

# core/tests/test_view_tasks.py

... previous tests

@pytest.mark.django_db
def test_tasks_pagination(client, make_todo, make_user):
    user = make_user()
    client.force_login(user)

    # create 2 pages of data
    for i in range(PAGE_SIZE + 3):
        make_todo(title=f"Todo #{i}", user=user)

    response = client.get(reverse("tasks"))

    context = response.context
    assert context["next_page_number"] == 2
    assert len(context["todos"]) == PAGE_SIZE

    # add header to ensure this is processed as an HTMX request
    response = client.get(reverse("tasks") + "?page=2", HTTP_HX_Request="true")

    content = response.content.decode().strip()
    assert content.startswith("<li")
    assert content.endswith("</li>")

    assert ">Todo #22<" not in content
    assert ">Todo #1<" in content

That's it by now! This was by far the most fun I've had with HTMX so far. The full code for this post is here.

For the next post I'm considering adding some client state management with AlpineJS, or maybe add a "due date" feature. See you!

Featured ones: