How do I create a Django form that displays a checkbox label to the right of the checkbox?
Asked Answered
S

9

39

When I define a Django form class similar to this:

def class MyForm(forms.Form):
    check = forms.BooleanField(required=True, label="Check this")

It expands to HTML that looks like this:

<form action="." id="form" method=POST>
<p><label for="check">Check this:</label> <input type="checkbox" name="check" id="check" /></p>
<p><input type=submit value="Submit"></p>
</form>

I would like the checkbox input element to have a label that follows the checkbox, not the other way around. Is there a way to convince Django to do that?

[Edit]

Thanks for the answer from Jonas - still, while it fixes the issue I asked about (checkbox labels are rendered to the right of the checkbox) it introduces a new problem (all widget labels are rendered to the right of their widgets...)

I'd like to avoid overriding _html_output() since it's obviously not designed for it. The design I would come up with would be to implement a field html output method in the Field classes, override the one for the Boolean field and use that method in _html_output(). Sadly, the Django developers chose to go a different way, and I would like to work within the existing framework as much as possible.

CSS sounds like a decent approach, except that I don't know enough CSS to pull this off or even to decide whether I like this approach or not. Besides, I prefer markup that still resembles the final output, at least in rendering order.

Furthermore, since it can be reasonable to have more than one style sheet for any particular markup, doing this in CSS could mean having to do it multiple times for multiple styles, which pretty much makes CSS the wrong answer.

[Edit]

Seems like I'm answering my own question below. If anyone has a better idea how to do this, don't be shy.

Sandwich answered 21/2, 2009 at 4:18 Comment(0)
S
2

Here's what I ended up doing. I wrote a custom template stringfilter to switch the tags around. Now, my template code looks like this:

{% load pretty_forms %}
<form action="." method="POST">
{{ form.as_p|pretty_checkbox }}
<p><input type="submit" value="Submit"></p>
</form>

The only difference from a plain Django template is the addition of the {% load %} template tag and the pretty_checkbox filter.

Here's a functional but ugly implementation of pretty_checkbox - this code doesn't have any error handling, it assumes that the Django generated attributes are formatted in a very specific way, and it would be a bad idea to use anything like this in your code:

from django import template
from django.template.defaultfilters import stringfilter
import logging

register=template.Library()

@register.filter(name='pretty_checkbox')
@stringfilter
def pretty_checkbox(value):
    # Iterate over the HTML fragment, extract <label> and <input> tags, and
    # switch the order of the pairs where the input type is "checkbox".
    scratch = value
    output = ''
    try:
        while True:
            ls = scratch.find('<label')
            if ls > -1:
                le = scratch.find('</label>')
                ins = scratch.find('<input')
                ine = scratch.find('/>', ins)
                # Check whether we're dealing with a checkbox:
                if scratch[ins:ine+2].find(' type="checkbox" ')>-1:
                    # Switch the tags
                    output += scratch[:ls]
                    output += scratch[ins:ine+2]
                    output += scratch[ls:le-1]+scratch[le:le+8]
                else:
                    output += scratch[:ine+2]
                scratch = scratch[ine+2:]
            else:
                output += scratch
                break
    except:
        logging.error("pretty_checkbox caught an exception")
    return output

pretty_checkbox scans its string argument, finds pairs of <label> and <input> tags, and switches them around if the <input> tag's type is "checkbox". It also strips the last character of the label, which happens to be the ':' character.

Advantages:

  1. No futzing with CSS.
  2. The markup ends up looking the way it's supposed to.
  3. I didn't hack Django internals.
  4. The template is nice, compact and idiomatic.

Disadvantages:

  1. The filter code needs to be tested for exciting values of the labels and input field names.
  2. There's probably something somewhere out there that does it better and faster.
  3. More work than I planned on doing on a Saturday.
Sandwich answered 22/2, 2009 at 6:25 Comment(4)
Can you provide the code for pretty_checkbox? My own preferred markup for this is <label for="check"><input type="checkbox" name="check" id="check" />&nbsp;Check this</label>Gabriello
I don't have the code handy right now. It wasn't very elaborate, if I recall - it's simply a function that's fed a short HTML snippet, and swaps the label and input around if the input is a checkbox. The markup is generated by Django, so any preference you might have is probably irrelevant.Sandwich
you have got to be kidding. This is very simple in HTML. So I use Django to make things hard, because I am a real programmer?? Django check forms are broken. pure and simple.Utilitarian
It's even simpler in Smalltalk.Sandwich
R
33

Here's a solution I've come up with (Django v1.1):

{% load myfilters %}

[...]

{% for field in form %}
    [...]
    {% if field.field.widget|is_checkbox %}
      {{ field }}{{ field.label_tag }}
    {% else %}
      {{ field.label_tag }}{{ field }}
    {% endif %}
    [...]
{% endfor %}

You'll need to create a custom template tag (in this example in a "myfilters.py" file) containing something like this:

from django import template
from django.forms.fields import CheckboxInput

register = template.Library()

@register.filter(name='is_checkbox')
def is_checkbox(value):
    return isinstance(value, CheckboxInput)

More info on custom template tags available here.

Edit: in the spirit of asker's own answer:

Advantages:

  1. No futzing with CSS.
  2. The markup ends up looking the way it's supposed to.
  3. I didn't hack Django internals. (but had to look at quite a bunch)
  4. The template is nice, compact and idiomatic.
  5. The filter code plays nice regardless of the exact values of the labels and input field names.

Disadvantages:

  1. There's probably something somewhere out there that does it better and faster.
  2. Unlikely that the client will be willing to pay for all the time spent on this just to move the label to the right...
Redford answered 11/1, 2010 at 22:6 Comment(1)
I like this answer. The implementation is simple, which is always a plus. It's clear what's going on just from reading the template. The only downside (in my biased and admittedly lazy view) is that my solution makes for shorter templates, and is closer to what you could have expected Django to have done in the first place.Sandwich
M
15

I took the answer from romkyns and made it a little more general

def field_type(field, ftype):
    try:
        t = field.field.widget.__class__.__name__
        return t.lower() == ftype
    except:
        pass
    return False

This way you can check the widget type directly with a string

{% if field|field_type:'checkboxinput' %}
    <label>{{ field }} {{ field.label }}</label>
{% else %}
    <label> {{ field.label }} </label> {{ field }}
{% endif %}
Misdemean answered 22/2, 2010 at 3:52 Comment(0)
F
12

All presented solutions involve template modifications, which are in general rather inefficient concerning performance. Here's a custom widget that does the job:

from django import forms
from django.forms.fields import BooleanField
from django.forms.util import flatatt
from django.utils.encoding import force_text
from django.utils.html import format_html
from django.utils.translation import ugettext as _


class PrettyCheckboxWidget(forms.widgets.CheckboxInput):
    def render(self, name, value, attrs=None):
        final_attrs = self.build_attrs(attrs, type='checkbox', name=name)
        if self.check_test(value):
            final_attrs['checked'] = 'checked'
        if not (value is True or value is False or value is None or value == ''):
            final_attrs['value'] = force_text(value)
        if 'prettycheckbox-label' in final_attrs:
            label = _(final_attrs.pop('prettycheckbox-label'))
        else:
            label = ''
        return format_html('<label for="{0}"><input{1} /> {2}</label>', attrs['id'], flatatt(final_attrs), label)


class PrettyCheckboxField(BooleanField):
    widget = PrettyCheckboxWidget
    def __init__(self, *args, **kwargs):
        if kwargs['label']:
            kwargs['widget'].attrs['prettycheckbox-label'] = kwargs['label']
            kwargs['label'] = ''
        super(PrettyCheckboxField, self).__init__(*args, **kwargs)


# usage in form
class MyForm(forms.Form):
    my_boolean = PrettyCheckboxField(label=_('Some label'), widget=PrettyCheckboxWidget())

I have PrettyCheckboxWidget and PrettyCheckboxField in an extra file, so they may be imported where needed. If you don't need translations, you can remove the ugettext parts. This code works on Django 1.5 and is untested for lower versions.

Advantages:

  • Highly performant, needs no template modifications
  • Easy to use as a custom widget

Disadvantages:

  • "as_table" renders the checkbox + label inside the second column
  • {{ field.label }} inside the template is empty. The label is instead bound to {{ field }}
  • More work than I planned on doing on a Saturday ;-)
Fragmentary answered 9/3, 2013 at 7:18 Comment(0)
K
4

I know that the user excluded CSS, but considering the top answers take about half hour of work to do such a small thing, but knowing that details like these are important on a website, I'd settle for the CSS solution.

checkbox.css

input[type="checkbox"] {
    float: left;
    margin-right: 10px;
    margin-top: 4px;
}

forms.py

class MyForm(forms.ModelForm):
    # ...
    class Media:
    css = {
        'all': 'checkbox.css',
    }

template.html

{{ form.media }}
{{ form.as_p }}

Advantages:

  • fast!
  • no futzing with template tags (just form.as_p)
  • no new damned widgets
  • the CSS file is automatically included in every form

Disadvantages:

  • the HTML doesn't reflect the presentation (but is good enough!)
  • your frontendist could complain
Keddah answered 23/7, 2013 at 11:27 Comment(1)
You can also add this css snippet into your current css file and you won't need to use {{ form.media }}Overzealous
S
2

Here's what I ended up doing. I wrote a custom template stringfilter to switch the tags around. Now, my template code looks like this:

{% load pretty_forms %}
<form action="." method="POST">
{{ form.as_p|pretty_checkbox }}
<p><input type="submit" value="Submit"></p>
</form>

The only difference from a plain Django template is the addition of the {% load %} template tag and the pretty_checkbox filter.

Here's a functional but ugly implementation of pretty_checkbox - this code doesn't have any error handling, it assumes that the Django generated attributes are formatted in a very specific way, and it would be a bad idea to use anything like this in your code:

from django import template
from django.template.defaultfilters import stringfilter
import logging

register=template.Library()

@register.filter(name='pretty_checkbox')
@stringfilter
def pretty_checkbox(value):
    # Iterate over the HTML fragment, extract <label> and <input> tags, and
    # switch the order of the pairs where the input type is "checkbox".
    scratch = value
    output = ''
    try:
        while True:
            ls = scratch.find('<label')
            if ls > -1:
                le = scratch.find('</label>')
                ins = scratch.find('<input')
                ine = scratch.find('/>', ins)
                # Check whether we're dealing with a checkbox:
                if scratch[ins:ine+2].find(' type="checkbox" ')>-1:
                    # Switch the tags
                    output += scratch[:ls]
                    output += scratch[ins:ine+2]
                    output += scratch[ls:le-1]+scratch[le:le+8]
                else:
                    output += scratch[:ine+2]
                scratch = scratch[ine+2:]
            else:
                output += scratch
                break
    except:
        logging.error("pretty_checkbox caught an exception")
    return output

pretty_checkbox scans its string argument, finds pairs of <label> and <input> tags, and switches them around if the <input> tag's type is "checkbox". It also strips the last character of the label, which happens to be the ':' character.

Advantages:

  1. No futzing with CSS.
  2. The markup ends up looking the way it's supposed to.
  3. I didn't hack Django internals.
  4. The template is nice, compact and idiomatic.

Disadvantages:

  1. The filter code needs to be tested for exciting values of the labels and input field names.
  2. There's probably something somewhere out there that does it better and faster.
  3. More work than I planned on doing on a Saturday.
Sandwich answered 22/2, 2009 at 6:25 Comment(4)
Can you provide the code for pretty_checkbox? My own preferred markup for this is <label for="check"><input type="checkbox" name="check" id="check" />&nbsp;Check this</label>Gabriello
I don't have the code handy right now. It wasn't very elaborate, if I recall - it's simply a function that's fed a short HTML snippet, and swaps the label and input around if the input is a checkbox. The markup is generated by Django, so any preference you might have is probably irrelevant.Sandwich
you have got to be kidding. This is very simple in HTML. So I use Django to make things hard, because I am a real programmer?? Django check forms are broken. pure and simple.Utilitarian
It's even simpler in Smalltalk.Sandwich
S
2

Changing checkbox position in Django admin can be quite tricky, but luckily there is a simple solution using custom widget:

from django.forms.widgets import Widget, CheckboxInput, boolean_check

class RightCheckbox(Widget):
    render = CheckboxInput().render

    def __init__(self, attrs=None, check_test=None):
        super(RightCheckbox, self).__init__(attrs)
        self.check_test = boolean_check if check_test is None else check_test

Django uses left position only when widget is subclass of CheckboxInput

Sidwel answered 15/2, 2015 at 19:11 Comment(0)
T
1

Order of inputs and labels is provided via normal_row parameter of the form and there are no different row patterns for checkboxes. So there are two ways to do this (in 0.96 version exactly):
1. override _html_output of the form
2. use CSS to change position of the label and checkbox

Tight answered 21/2, 2009 at 6:52 Comment(0)
A
0

I had this problem too and I just used this method :

post-detail.html

{% if field.html_name == "remember_my_password" %}
    {{ field }}
    <label for="{{ field.id_for_label }}" class="form-label">
        {{ field.label }}
    </label>
{% else %}
   <label for="{{ field.id_for_label }}" class="form-label">
       {{ field.label }}
   </label>
  {{ field }}
{% endif %}
Audiometer answered 22/4, 2023 at 14:55 Comment(1)
Your answer could be improved with additional supporting information. Please edit to add further details, such as citations or documentation, so that others can confirm that your answer is correct. You can find more information on how to write good answers in the help center.Attainment
C
0

In your Django template, you can check if the widget is a checkbox by looking at field.field.widget.input_type, like this:

{% for field in form %}

  {% if field.field.widget.input_type == "checkbox" %}

    {{ field }}

    <label for="{{ field.id_for_label }}">{{ field.label }}</label>

    {{ field.errors }}

  {% else %}

    ...

  {% endif %}

{% endfor %}
Cressida answered 13/6, 2024 at 19:59 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.