Skip to main content

A simple application for Django to include (and fetch) asynchronous templates

Project description

django-async-include

License Maintenance made-with-python PyPI pyversions PyPI version gelidum PyPI status PyPI download month Maintainability Test Coverage

Asynchronous inclusion of Django templates

What's this?

This is a project to help the ajax load of chunks of HTML with minimal effort on the developer side, providing an easy way to improve web-site experience for your users by minimizing perceived loading times.

This is the development repository.

How does it work?

The async_include template tag sends the context to the server using an AJAX request.

In the case of model objects, it sends the model, application and id. In the case of QuerySets, it sends the encrypted parametrized SQL. Of course, in the case of safe values like strings, booleans or numbers this data are sent "as is".

The receiver is a basic view of this application that renders the template with the received context and returns it in the AJAX call.

Requirements

This application only depends on pycryptodome.

Of course, you will need Django version 1.10 or newer.

jQuery

No jQuery is required as of version 0.6.6.

Fontawesome (optional)

Fontawesome is the the-facto standard of font icons of our time. Include it in your project to see the spinner moving when loading the remote templates.

The easiest way to include it by using a CDN. For example, bootstrap CDN (not affiliated, nor they endorse this project) is one of the most known.

Default waiting spinner uses fontawesome. You can overwrite async_include/spinner.html template if don't want to use the default fontawesome style.

Installation

Using pip

This package is in pypi so you only have to write:

pip install django-async-include

to install it.

Install master version

Include this code in your requirements.txt file:

-e git://github.com/diegojromerolopez/django-async-include.git#egg=django_async_include

Installation in your Django project

Include the application in your project's settings.py:

INSTALLED_APPS = [
    ## ...
    'async_include',
]

Include django-async-include URLs

Include the URLs of Django-Async-Include in your urls.py project in the namespace async_include:

from django.urls import path, include

urlpatterns = [
    # ...
    path(r'async_include/', include('async_include.urls', namespace="async_include")),
]

Use

Load the async_include template tags at the top of your file and use the async_include template tag as a replacement of the django include template tag.

You have to pass the local context explicitly to the async included templates, so you can pass all variables you need in your included template as named parameters of the async_include template tag.

{# Load the async_include template tag at the top of your template file #}
{% load async_include %}

{# Call the async_include template tag indicating what objects needs to replace it #}
{% async_include "<path of the >" <object1_name>=<object1> <object2_name>=<object2> ... <objectN_name>=<objectN>  %}

There is also a repository with a full example: django-async-include-example.

Call javascript function whe load is completed

Pass the attribute onload with the name of the function you want to call after the request and the replacement has been completed. e.g.

{% async_include "boards/components/view/current_percentage_of_completion.html" board=board onload="load_board_style" %}

Warning and limitations

Object dynamic attributes

No dynamic attribute will be passed to the templates given that only a reference to it is passed from the caller to the included template callee. Don't use dynamic attributes inside an async_included template.

However, the full object will be passed to the async_included template, so you could call its methods and properties without any problem.

QuerySets

Each QuerySet is passed as encrypted SQL and converted on the receiver to a RawQuerySet.

Note that RawQuerySets have no len method so length filter returns always 0.

To fix this we have implemented a new version of the length filter that will be loaded in your template if you overwrite it.

{% load async_included %}

Note that this templatetag file is async_included, ending in ed.

Examples

Passing an object

{% load async_include %}

{# .. #}

{# Load the template and informs the board object is required for the included template  #}
{% async_include "boards/components/view/current_percentage_of_completion.html" board=board %}

Passing a QuerySet

{% load async_include %}

{# .. #}

{% async_include "boards/components/view/summary.html" board=board member=member next_due_date_cards=next_due_date_cards %}

Customization

Spinner

Overwrite async_include/spinner.html template if you want to change the spinner from fontawesome one (default) by a background image or an image. Otherwise, make sure you are loading fontawesome fonts.

Note that the spinner must have class async_included-spinner. Otherwise, the spinner behavior is going to be unpredictable.

Show/Hide spinner

Including the optional parameter spinner__visible=False when calling the async_include template tag will not show the spinner block.

{% load async_include %}

{# .. #}

{# Will not show the spinner #}
{% async_include "boards/components/view/last_comments.html" board=board spinner__visible=False %}

Customize spinner template per async_include template tag call

Use the optional parameter spinner__template_path to set a different template path for a specific async_include call in your templates.

{% load async_include %}

{# .. #}

{# Will not show the spinner #}
{% async_include "boards/components/view/last_comments.html" board=board spinner__template_path="templates/comments/last_comments_spinner.html" %}

Remember the spinner tag should contain the async_included-spinner class.

Block wrapper html tag

Wrapper tag is div and maybe you don't want that. Set html__tag optional parameter to the name of the tag you need in that particular context.

Example:

{% load async_include %}

{# .. #}

{# Will be replaced by <li></li> block instead of <div></div> #}
{% async_include "boards/components/view/last_comments.html" board=board html__tag='li' %}

Block wrapper html tag class

Customize the wrapper class by passing html__tag__class optional parameter to the template tag.

{% load async_include %}

{# .. #}

{# Will be replaced by <li></li> block instead of <div></div> #}
{# Class last_comments will be added to wrapper class #}
{% async_include "boards/components/view/last_comments.html" board=board html__tag='li' html__tag__class='last_comments' %}

## Request frequency

If do you want to make frequent requests, set request__frequency to the number of seconds you want to make the requests.

Example:

{% load async_include %}

{# .. #}

{# Update the last comments each 60 seconds #}
{% async_include "boards/components/view/last_comments.html" board=board request__frequency=60 %}

Development

To run the development tools, we provide a root-level Makefile with the following commands:

  • Run Unit Tests:
    make test-unit
    
  • Run E2E Tests (in Docker Compose):
    make test-e2e
    
  • Run Linters and Formatters:
    make lint
    

Main author

Diego J. Romero-López is a Software Engineer based on Madrid (Spain).

This project is in no way endorsed or related in any way to my past or current employers.

Contact me by email at diegojREMOVETHISromerolopezREMOVETHIS@gmail.com.

Contributors

  • Erik Telepovský:
    • Bug fixes.
    • Modern mobile browser support.
    • PyPy configuration fixes.
    • Python3 support.
    • Multi-language support.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django_async_include-0.8.0.tar.gz (14.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

django_async_include-0.8.0-py3-none-any.whl (13.5 kB view details)

Uploaded Python 3

File details

Details for the file django_async_include-0.8.0.tar.gz.

File metadata

  • Download URL: django_async_include-0.8.0.tar.gz
  • Upload date:
  • Size: 14.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for django_async_include-0.8.0.tar.gz
Algorithm Hash digest
SHA256 ff474e21c0be0e30ea03fad5de2d523c672f59ea6268d76a9d76438acb757442
MD5 9a1de7fd54f0cd20cdbe2de90ca9c48c
BLAKE2b-256 12300a05474a6892a5fffcef2f29d7e365db2339aa853afced06c2bf810f7cac

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_async_include-0.8.0.tar.gz:

Publisher: publish_on_pypi.yml on diegojromerolopez/django-async-include

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file django_async_include-0.8.0-py3-none-any.whl.

File metadata

File hashes

Hashes for django_async_include-0.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 cbcefce4c7bcd2a9170bcc1d3d438c8c83407175c576359a8136d001d1964932
MD5 2675b3ea219bc9b0d81b4295dbb223ac
BLAKE2b-256 5e8129c60801b2974604bfcbe33885b0ec033321a09b0ba41562b083414ec8c0

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_async_include-0.8.0-py3-none-any.whl:

Publisher: publish_on_pypi.yml on diegojromerolopez/django-async-include

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page