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.
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:
We want to invert the order, having the partial around the for loop:
Let's perform the swap in the template core/templates/index.html:
<ul> <p>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:<br> </p> <pre class="brush:php;toolbar:false">... 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.
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.
Let's change the template accordingly:
<ul> <p>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</p> <p>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. <p>We can now move to the views file.</p> <p>As a recap, here's how the GET /tasks view looks like by now; it's always returning the full template.<br> </p> <pre class="brush:php;toolbar:false">@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.
To wrap it up, we can add a test to ensure pagination is working as intended
<ul> <p>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:<br> </p> <pre class="brush:php;toolbar:false">... 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 }, )
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!
The above is the detailed content of Creating a To-Do app with HTMX and Django, part infinite scroll. For more information, please follow other related articles on the PHP Chinese website!