Skip to main content

Dynamically update choice fields based on dependent fields in django

Project description

Usage

  • pip install django-dynamic-form-fields

  • Add dynamic_fields to INSTALLED_APPS

  • Add to your form as a widget:

from typing import Type
from dynamic_fields.fields import DynamicChoicesWidget

from myapp.models import MyModel, Item

def get_item_choices(model: Type[Item], value: int):
    choices = []

    item = model.objects.get(id=value)

    for thing in item.things:
        # do something with item
        choices.append({
            'value': thing.id,
            'label': thing.name,
       })

    return choices


class MyForm(forms.ModelForm):
    class Meta:
        model = MyModel
        fields = ('item', 'dependent_item')
        widgets = {
            'dependent_item': DynamicChoicesWidget(
                depends_field='item',
                model=Item,
                callback=get_item_choices,
                no_value_disable=True,
                include_empty_choice=True,
                empty_choice_label="Please choose an option",
            )
        }
  • Ensure that form media is included with the rest of your view’s media

Requirements

  • Django 1.11-2.2

  • Python 3.5+ (built/tested with 3.6.2)

  • jQuery (built/tested with 1.12.4)

How does it work?

A view is injected into the project’s urlpatterns which is the handler for the ajax callback when the dependent field changes. When the DynamicChoicesWidget template is rendered, it serializes the various provided values into Python dotted strings to send to the template. It uses HTML5 data attributes in the markup to pass these values to the Javascript in order to send to the view to tell it what to do next.

The model is serialized into app.model, and the callback is serialized into the full dotted path to the method within your code, e.g; app.form.my_form.get_choices. When the dependent field’s value is changed, the ajax calls the view with the relevant information from above, then this goes and imports the path, grabs the model class for what was provided from django’s model loader, then calls your method with the class itself, and the value provided by the dependent field. The output of the callback method is serialized back into JSON, returned to the Javascript making the call, which iterates over the options and resets the select box’s options to those provided.

These can be chained (theoretically) for as many fields as required. It should also (though untested) work with multiple select boxes.

Change Log

0.3.2

  • No changes.

0.3.1

  • Changing formatting of documentation since PyPI doesn’t like bog-standard markdown these days.

0.3.0

  • Bump allowed django to 2.2+

  • Fixed issue found in django 2.2 where the choices JS was being called twice, causing multiple loads and the selected value to not persist across page reloads.

0.1.0/0.2.0

  • Initial version

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django-dynamic-form-fields-0.3.2.tar.gz (6.5 kB view details)

Uploaded Source

File details

Details for the file django-dynamic-form-fields-0.3.2.tar.gz.

File metadata

  • Download URL: django-dynamic-form-fields-0.3.2.tar.gz
  • Upload date:
  • Size: 6.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/41.2.0 requests-toolbelt/0.9.1 tqdm/4.44.1 CPython/3.7.6

File hashes

Hashes for django-dynamic-form-fields-0.3.2.tar.gz
Algorithm Hash digest
SHA256 18a747e01891ec91049ba7b7a2317e561d60243de8bb254912d169b48bfa479c
MD5 9c2ea4c429b69fd68429c4a98d025f02
BLAKE2b-256 87b57387331fcc7f36f421f6e92f07616dce42fa0e0231daeb4dc1e51e200d27

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page