diff --git a/docs/reference/core/mixins.md b/docs/reference/core/mixins.md new file mode 100644 index 00000000..d08c1ee8 --- /dev/null +++ b/docs/reference/core/mixins.md @@ -0,0 +1,10 @@ +::: core.views.mixins + handler: python + options: + heading_level: 3 + members: + - TabedViewMixin + - QuickNotifMixin + - AllowFragment + - FragmentMixin + - UseFragmentsMixin \ No newline at end of file diff --git a/docs/tutorial/fragments.md b/docs/tutorial/fragments.md index 007d4c0d..d2b154cc 100644 --- a/docs/tutorial/fragments.md +++ b/docs/tutorial/fragments.md @@ -1,40 +1,358 @@ -Pour utiliser HTMX, on a besoin de renvoyer des fragments depuis le backend. -Le truc, c'est que tout est optimisé pour utiliser `base.jinja` qui est assez gros. +from django.utils.safestring import SafeStringfrom django.utils.safestring import SafeString -Dans beaucoup de scénario, on veut pouvoir renvoyer soit la vue complète, soit -juste le fragment. En particulier quand on utilise l'attribut `hx-history` de htmx. +## Qu'est-ce qu'un fragment -Pour remédier à cela, il existe le mixin [AllowFragment][core.views.AllowFragment]. +Une application web django traditionnelle suit en général +le schéma suivant : -Une fois ajouté à une vue Django, il ajoute le boolean `is_fragment` dans les -templates jinja. Sa valeur est `True` uniquement si HTMX envoie la requête. -Il est ensuite très simple de faire un if/else pour hériter de -`core/base_fragment.jinja` au lieu de `core/base.jinja` dans cette situation. +1. l'utilisateur envoie une requête au serveur +2. le serveur renvoie une page HTML, + qui contient en général des liens et/ou des formulaires +3. lorsque l'utilisateur clique sur un lien ou valide + un formulaire, on retourne à l'étape 1 -Exemple d'utilisation d'une vue avec fragment: +C'est un processus qui marche, mais qui est lourd : +générer une page entière demande du travail au serveur +et effectuer le rendu de cette page en demande également +beaucoup au client. +Or, des temps de chargement plus longs et des +rechargements complets de page peuvent nuire +à l'expérience utilisateur, en particulier +lorsqu'ils interviennent lors d'opérations simples. + +Pour éviter ce genre de rechargement complet, +on peut utiliser AlpineJS pour rendre la page +interactive et effectuer des appels à l'API. +Cette technique fonctionne particulièrement bien +lorsqu'on veut afficher des objets ou des listes +d'objets de manière dynamique. + +En revanche, elle est moins efficace pour certaines +opérations, telles que la validation de formulaire. +En effet, valider un formulaire demande beaucoup +de travail de nettoyage des données et d'affichage +des messages d'erreur appropriés. +Or, tout ce travail existe déjà dans django. + +On veut donc, dans ces cas-là, ne pas demander +toute une page HTML au serveur, mais uniquement +une toute petite partie, que l'on utilisera +pour remplacer la partie qui correspond sur la page actuelle. +Ce sont des fragments. + + +## HTMX + +Toutes les fonctionnalités d'interaction avec les +fragments, côté client, s'appuient sur la librairie htmx. + +L'usage qui en est fait est en général assez simple +et ressemblera souvent à ça : + +```html+jinja +
+``` + +C'est la majorité de ce que vous avez besoin de savoir +pour utiliser HTMX sur le site. + +Bien entendu, ce n'est pas tout, il y a d'autres +options et certaines subtilités, mais pour ça, +consultez [La doc officielle d'HTMX](https://htmx.org/docs/). + +## La surcouche du site + +Pour faciliter et standardiser l'intégration d'HTMX +dans la base de code du site AE, +nous avons créé certains mixins à utiliser +dans les vues basées sur des classes. + +### [AllowFragment][core.views.mixins.AllowFragment] + +`AllowFragment` est extrêmement simple dans son +concept : il met à disposition la variable `is_fragment`, +qui permet de savoir si la vue est appelée par HTMX, +ou si elle provient d'un autre contexte. + +Grâce à ça, on peut écrire des vues qui +fonctionnent dans les deux contextes. + +Par exemple, supposons que nous avons +une `EditView` très simple, contenant +uniquement un formulaire. +On peut écrire la vue et le template de la manière +suivante : + +=== "`views.py`" + + ```python + from django.views.generic import UpdateView + + + class FooUpdateView(UpdateView): + model = Foo + fields = ["foo", "bar"] + pk_url_kwarg = "foo_id" + template_name = "app/foo.jinja" + ``` + +=== "`app/foo.jinja`" + + ```html+jinja + {% if is_fragment %} + {% extends "core/base_fragment.jinja" %} + {% else %} + {% extends "core/base.jinja" %} + {% endif %} + + {% block content %} + + {% endblock %} + ``` + +Lors du chargement initial de la page, le template +entier sera rendu, mais lors de la soumission du formulaire, +seul le fragment html de ce dernier sera changé. + +### [FragmentMixin][core.views.mixins.FragmentMixin] + +Il arrive des situations où le résultat que l'on +veut accomplir est plus complexe. +Dans ces situations, pouvoir décomposer une vue +en plusieurs vues de fragment permet de ne plus +raisonner en termes de condition, mais en termes +de composition : on n'a pas un seul template +qui peut changer les situations, on a plusieurs +templates que l'on injecte dans un template principal. + +Supposons, par exemple, que nous n'avons plus un, +mais deux formulaires à afficher sur la page. +Dans ce cas, nous pouvons créer deux templates, +qui seront alors injectés. + +=== "`urls.py`" + + ```python + from django.urls import path + + from app import views + + urlpatterns = [ + path("", FooCompositeView.as_view(), name="main"), + path("create/", FooUpdateFragment.as_view(), name="update_foo"), + path("update/", FooCreateFragment.as_view(), name="create_foo"), + ] + ``` + +=== "`view.py`" + + ```python + from django.views.generic import CreateView, UpdateView, TemplateView + from core.views.mixins import FragmentMixin + + + class FooCreateFragment(FragmentMixin, CreateView): + model = Foo + fields = ["foo", "bar"] + template_name = "app/fragments/create_foo.jinja" + + + class FooUpdateFragment(FragmentMixin, UpdateView): + model = Foo + fields = ["foo", "bar"] + pk_url_kwarg = "foo_id" + template_name = "app/fragments/update_foo.jinja" + + + class FooCompositeFormView(TemplateView): + template_name = "app/foo.jinja" + + def get_context_data(**kwargs): + return super().get_context_data(**kwargs) | { + "create_fragment": FooCreateFragment.as_fragment()(), + "update_fragment": FooUpdateFragment.as_fragment()(foo_id=1) + } + ``` + +=== "`app/fragment/create_foo.jinja`" + + ```html+jinja + + ``` + +=== "`app/fragment/update_foo.jinja`" + + ```html+jinja + + ``` + +=== "`app/foo.jinja`" + + ```html+jinja + {% extends "core/base.html" %} + + {% block content %} +