En django existen dos formas de crear un formulario, una es crear cada campo manualmente y la otra es crearlos automáticamente basados en el modelo de datos.
Ya que casi siempre los formulario se usan para agregar/editar una tabla en una BD, y los campos del formulario reflejan los campos de la tabla, seria un desperdicio de tiempo crearlos manualmente.
Dejemos de hablar y empecemos a escribir código, editemos el archivo forms.py en el directorio de nuestra aplicación:
- from django.forms import ModelForm #1
- from blog.models import Comment #2
- class CommentForm(ModelForm): #3
- """Formulario de comentarios"""
- class Meta: #4
- model = Comment #5
- exclude = ("post","deleted") #6
Hora de explicar lo que hemos hecho:
- Importamos ModelForm, la base para los formularios basados en modelos de datos.
- Importamos el modelo para el que se creará el formulario
- Creamos una subclase de ModelForm.
- En la clase Meta especificamos como crear el formulario
- model: modelo en el que se basa el formulario.
- exclude: campos a excluir, en este ejemplo post lo asignamos según el articulo desde el que se envió el comentario y deleted lo decidimos en la pagina de administración si es un comentario insultante. Si son muchos los campos a excluir podemos hacer lo contrario y especificar los que se deben incluir.
- # Archivo views.py
- from blog.models import Post, Tag, Comment
- from blog.forms import CommentForm
- ...
- # la función recibe el numero capturado
- def single(request, id=0):
- # recuperar post según id
- post = Post.objects.get(id=int(id))
- # Crear formulario
- form = CommentForm()
- return render_to_response("single.html", locals())
- {# plantilla single.html #}
- {% extends 'base.html' %}
- {% block main_content %}
- {% include 'include/article.html' %}
- {# necesitamos crear el tag form #}
- <form action="" method="form">
- {{ form }}
- </form>
- {% endblock %}
- # Archivo views.py
- from blog.models import Post, Tag, Comment
- from blog.forms import CommentForm
- # importamos el decorador
- from django.views.decorators.csrf import csrf_protect
- # importamos render
- from django.shortcuts import render
- ...
- # aplicamos el decorador función que recibe datos de formulario
- @csrf_protect
- def single(request, id=0):
- # recuperar post según id
- post = Post.objects.get(id=int(id))
- form = CommentForm()
- # usamos render en lugar de render_to_response
- return render(request, "single.html", locals())
{% csrf_token %}
dentro del formulario en la plantilla single.html. Esto crea un elemento input oculto con un token único que se verifica al enviar datos al servidor, si este no coincide, se produce un error 403 (prohibido) y tenemos una app algo mas segura.Ok, ya podemos enviar datos del formulario al servidor, hora debemos validarlos y guardarlos en la BD.
- # Archivo views.py
- @csrf_protect
- def single(request, id=0):
- # recuperar post según id
- post = Post.objects.get(id=int(id))
- # recuperar comentarios de este post
- comments = Comment.objects.filter(post=post)
- # Si la petición es POST es porque enviaron el formulario
- if request.method == "POST":
- # Creamos una instancia del modelo Comment asignado el post actual
- comment = Comment(post=post)
- # Creamos una instancia del formulario con los datos recibidos
- form = CommentForm(request.POST, instance=comment)
- # Validamos los datos
- if form.is_valid():
- # Guardamos el comentario en la BD
- form.save()
- # Enviamos al usuario de nuevo al post
- return HttpResponseRedirect("/post/{0}".format(slug))
- # De lo contrario creamos un formulario vacío
- else:
- form = CommentForm()
- return render(request, "single.html", locals())
- Validamos el formulario con el método is_valid que no definimos. ¿como es esto posible?; Al crear el modelo Comment especificamos el tipo de datos admitidos, el formulario "sabe" que si esto datos no corresponden, el formulario contiene errores.
- Para guardar los datos usamos form.save y no comment.save. Esto se debe a que hay datos que no están incluidos en el formulario (post) pero si en la instancia comment y viceversa, por esto pasamos comment al formulario en el momento de crearlo.
- {# plantilla single.html #}
- {% extends 'base.html' %}
- {% block main_content %}
- {% include 'include/article.html' %}
- {% for comment in comments %}
- <div class="comment">
- <div class="user">{{ comment.name }} dijo:</div>
- <div class="body">{{ comment.body }}</div>
- <small>{{ comment.date|date:"d-m-Y" }}</small>
- </div>
- {% endfor %}
- {# necesitamos crear el tag form #}
- <form action="" method="form">
- {{ form }}
- </form>
- {% endblock %}