search
HomeBackend DevelopmentPython TutorialCreating a To-Do app with HTMX and Django, part 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:

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

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

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

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


    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", # 



<p>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.</p>

<h2>
  
  
  Implementing infinite scroll
</h2>

<p>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.</p>

<p><img src="/static/imghwm/default1.png" data-src="https://img.php.cn/upload/article/000/000/000/173613850692024.jpg?x-oss-process=image/resize,p_40" class="lazy" alt="Creating a To-Do app with HTMX and Django, part infinite scroll"></p>

<p>Let's change the template accordingly:<br>
</p>

<pre class="brush:php;toolbar:false">    

    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

  • 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.

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

    Adding tests

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

    
    

      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", # 
    
    
    
    <p>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. </p>
    
    <p>For the next post I'm considering adding some client state management with AlpineJS, or maybe add a "due date" feature. See you!</p>
    
    
              
    
                
            

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!

Statement
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
How do you slice a Python list?How do you slice a Python list?May 02, 2025 am 12:14 AM

SlicingaPythonlistisdoneusingthesyntaxlist[start:stop:step].Here'showitworks:1)Startistheindexofthefirstelementtoinclude.2)Stopistheindexofthefirstelementtoexclude.3)Stepistheincrementbetweenelements.It'susefulforextractingportionsoflistsandcanuseneg

What are some common operations that can be performed on NumPy arrays?What are some common operations that can be performed on NumPy arrays?May 02, 2025 am 12:09 AM

NumPyallowsforvariousoperationsonarrays:1)Basicarithmeticlikeaddition,subtraction,multiplication,anddivision;2)Advancedoperationssuchasmatrixmultiplication;3)Element-wiseoperationswithoutexplicitloops;4)Arrayindexingandslicingfordatamanipulation;5)Ag

How are arrays used in data analysis with Python?How are arrays used in data analysis with Python?May 02, 2025 am 12:09 AM

ArraysinPython,particularlythroughNumPyandPandas,areessentialfordataanalysis,offeringspeedandefficiency.1)NumPyarraysenableefficienthandlingoflargedatasetsandcomplexoperationslikemovingaverages.2)PandasextendsNumPy'scapabilitieswithDataFramesforstruc

How does the memory footprint of a list compare to the memory footprint of an array in Python?How does the memory footprint of a list compare to the memory footprint of an array in Python?May 02, 2025 am 12:08 AM

ListsandNumPyarraysinPythonhavedifferentmemoryfootprints:listsaremoreflexiblebutlessmemory-efficient,whileNumPyarraysareoptimizedfornumericaldata.1)Listsstorereferencestoobjects,withoverheadaround64byteson64-bitsystems.2)NumPyarraysstoredatacontiguou

How do you handle environment-specific configurations when deploying executable Python scripts?How do you handle environment-specific configurations when deploying executable Python scripts?May 02, 2025 am 12:07 AM

ToensurePythonscriptsbehavecorrectlyacrossdevelopment,staging,andproduction,usethesestrategies:1)Environmentvariablesforsimplesettings,2)Configurationfilesforcomplexsetups,and3)Dynamicloadingforadaptability.Eachmethodoffersuniquebenefitsandrequiresca

How do you slice a Python array?How do you slice a Python array?May 01, 2025 am 12:18 AM

The basic syntax for Python list slicing is list[start:stop:step]. 1.start is the first element index included, 2.stop is the first element index excluded, and 3.step determines the step size between elements. Slices are not only used to extract data, but also to modify and invert lists.

Under what circumstances might lists perform better than arrays?Under what circumstances might lists perform better than arrays?May 01, 2025 am 12:06 AM

Listsoutperformarraysin:1)dynamicsizingandfrequentinsertions/deletions,2)storingheterogeneousdata,and3)memoryefficiencyforsparsedata,butmayhaveslightperformancecostsincertainoperations.

How can you convert a Python array to a Python list?How can you convert a Python array to a Python list?May 01, 2025 am 12:05 AM

ToconvertaPythonarraytoalist,usethelist()constructororageneratorexpression.1)Importthearraymoduleandcreateanarray.2)Uselist(arr)or[xforxinarr]toconvertittoalist,consideringperformanceandmemoryefficiencyforlargedatasets.

See all articles

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

Video Face Swap

Video Face Swap

Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Tools

Zend Studio 13.0.1

Zend Studio 13.0.1

Powerful PHP integrated development environment

MantisBT

MantisBT

Mantis is an easy-to-deploy web-based defect tracking tool designed to aid in product defect tracking. It requires PHP, MySQL and a web server. Check out our demo and hosting services.

SublimeText3 Chinese version

SublimeText3 Chinese version

Chinese version, very easy to use

SublimeText3 Linux new version

SublimeText3 Linux new version

SublimeText3 Linux latest version

SecLists

SecLists

SecLists is the ultimate security tester's companion. It is a collection of various types of lists that are frequently used during security assessments, all in one place. SecLists helps make security testing more efficient and productive by conveniently providing all the lists a security tester might need. List types include usernames, passwords, URLs, fuzzing payloads, sensitive data patterns, web shells, and more. The tester can simply pull this repository onto a new test machine and he will have access to every type of list he needs.