<p>OK the following assumes that you already have experience in creating and using HTML forms and that you know your way around HTTP requests (GET and POST) and HTTP responses (status + body). You may have used PHP, JSP or ASP for this in the past. If you haven't, then <ahref="https://duckduckgo.com/?q=http+get+and+response&atb=v457-1&ia=web">start here</a>.
<p>Django has several generations of quite different clever mechanisms to make creating HTML forms "easier". Yes, making them might be easier, but maintaining this opinionated stuff is a nightmare without adequately educating yourself how the architecture works. This will take time: do not hurry.
<p>WARNING: when reading the Django documentation on <ahref="https://docs.djangoproject.com/en/5.1/topics/forms/">Forms</a> (unbound and <ahref="https://docs.djangoproject.com/en/5.1/ref/forms/api/">bound</a>) and <ahref="https://docs.djangoproject.com/en/dev/topics/forms/modelforms/">ModelForms</a>, do not get diverted into looking at Formsets or ModelFormsets. We do not use any formsets in troggle - of any kind.
<li>In <em>all</em> our data entry webpages we use a Django Form object to work with the response that the form sends back to the server, where we have to interpret the data and make an update to the database. The form is <em>bound</em> to the data that the user has provided.
<li>In <em>many</em> (but not all) we use a Django Form object in a Django template to render [some or all of the] the HTML which displays the input fields for the user. An empty form can be rendered to the user using an <em>unbound</em> form.
</ul>
<p>This is now a hint that you need to refresh your knowledge of <ahref="https://docs.djangoproject.com/en/5.1/topics/templates/">the Django template system</a>. Fortunately you only need to know a tiny part of this to work with forms, basically just this idiom:
<p>Now have a look at the Django templates for some of the forms you will already be familiar with from using them as a caver entering data, e.g. <ahref="/logbookedit/2024-08-01c">Logbook entry editing</a> which uses the Django template <ahref="/repositories/troggle/.git/tree/templates/logbookform.html">logbookform.html</a>...
(This is one of those webpages where we do not use a Form object for writing the HTML, only for interpreting the results).
<p>Hah, that was a nasty shock wasn't it? OK, most of that stuff is not the stuff which manages the data entry form. Concentrate on just the bits between the <form> ....</form> tags.
<p>It might help you at this point to remind yourself how an ordinary webapge works, without a form. So look at <ahref="/logbookentry/2024-08-01/2024-08-01c">a logbook entry</a> which just displays the data and the corresponding template <ahref="/repositories/troggle/.git/tree/templates/logbookentry.html">logbookentry.html</a>... OK, not such a good idea: that is quite complicated. See <ahref="#forloop">For loop</a> below.
But if you haven't worked with HTML forms before, then you actually have <ahref="https://developer.mozilla.org/en-US/docs/Web/HTML/Element/form">a whole lot of HTML</a> you will need to learn from scratch: how the <code><form> ....</form></code> tag works, and how fields and labels and stuff works, and "input" and "textarea" and "submit", and how tag attributes such as "disabled" or "required" work.
You will also need to know about <ahref="https://docs.djangoproject.com/en/5.1/ref/forms/widgets/">Django Widgets</a>.
<p>So that's enough to get you started. Now you are on your own, apart from help on the Website room of the Matrix chat and the nerd email list of course.
<p>A few of our data entry pages use <ahref="https://docs.djangoproject.com/en/5.1/topics/forms/modelforms/">ModelForms</a>, these are where the Form object is automagically created from a Model class. If you can't find where something is initialised, it is probably because it was done automatically and invisibly by instantiating a ModelForm.
which illustrates the for loop syntax, but also the my_object.attribute syntax, where the attribute is a one-to-many link to other Objects (instances of a Class) and has the <ahref="https://dnmtechs.com/understanding-the-_set-attribute-in-django-querysets/"><em>function</em> "_set"</a> applied followed by the <ahref="https://docs.djangoproject.com/en/5.1/ref/models/querysets/#django.db.models.query.QuerySet.all">QuerySet function</a> ".all". The effect of the for loop is to iterate through all the "personlogentry" instances referenced by the specific "logbookentry" the page is looking at.