Skip to content
jasper-zanjani edited this page Oct 5, 2020 · 2 revisions

A typical Django project contains multiple apps, which are Python packages containing their own models, views, templates, and urls.

  • A model is the single definitive source of information about your data, and contains the essential fields and behaviors of the data you're storing.

  • Migrations are necessary when Model classes are updated. And for projects sufficiently advanced, migration scripts must be developed for any such changes.

  • Async Server Gateway Interface (ASGI) is the spiritual successor to, and superset of, WSGI. It implements the new Python standard for asynchronous web servers and applications, which resembles that of websockets. 👉 From WSGI to ASGI

  • WSGI is coupled tightly with the synchronous request-response model familiar from HTTP 1.1.

URLs

URL patterns (stored in the urlpatterns list defined in the project-level urls.py file) can be parameterized. Here, the template <int:x> specifies an integer to be assigned to the view parameter x.

from app.views import my_view

urlpatterns = [
  path('/example/<int:x>', my_view)
]

Forms

modelform_factory can be used to automatically produce a webform from a Model class.

# views.py

MeetingForm = modelform_factory(Meeting, exclude=[])

This can be placed into a template using the {{ form }} template tag. Note, a {% csrf_token %} template tag must also be present for a submit button to work.

{% block content %}
<h1>Plan a new meeting</h1>
<form method="POST">
  <table>
    {{form}}
  </table>
  {% csrf_token %}
  <button type="submit">Create</button>
</form>
{% endblock content %}

When the modelform_factory class has been defined, it is instantiated within the view function. This object exposes several methods:

  • is_valid data validation is strongly recommended for any form input
  • save imports the validated form data into the database
def new(request):
  if request.method == 'POST':
    form = MeetingForm(request.POST)
    if form.is_valid():
      form.save()
      return redirect("home")
  else:
    form = MeetingForm()
  
  return render(request, "meetings/new.html", {"form": form})

Template

Django templates are HTML files with additional markup to signify places where data can be dynamically inserted. The data used by the views file is called the template context.

Templates must be placed within the /templates folder within the app, and it is considered best practice to place templates within a nested subdirectory within it, e.g. /templates/app.

Django template tags are specified beween {% .. %} and allow for interpolation of data.

for loop

<ul>
  {% for m in meetings %}

  {% endfor %}
</ul>

Link building

URLs can be built by using the url template tag, specifying the name of a URL

urlpatterns = [
  path('', home, name='home')
]
<a href="{% url 'home' %}">Home</a>

Models

👉 Models

In Django, a Model class is mapped to a database table. Each object is a record in that table.

Model objects expose several attributes and methods

Model.objects.all

Get all objects

meetings = Meeting.objects.all()

Model.objects.count

Get count of objects in database

count = Meeting.objects.count()

Model.objects.get

Query for a specific object

meeting = Meeting.objects.get(pk=id)

👉 get_object_or_404 may be better for most cases

meeting = get_object_or_404(Meeting, pk=id)

Tasks

Adding a new app

python manage.py startapp app

Then add to settings.py in project directory

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

Bulma

There appears to be much flexibility in the arrangement of input controls in a form.

So long as the Submit button is child to the form element, tasks are accepted in the To-Do app.

Per Bulma documentation, the field class is intended as a container for label.labels, .controls, and optional p.help text. 👉

In contrast, control is a block container meant to enhance single form controls and can only contain input, select, button, or icon elements. 👉

form.field(method="POST", action="/")
	label.label Enter something to do
	.control
		| {{form.title}}
		| {% csrf_token %}
	button.button.is-primary(type="submit") Submit