Django’s forms framework can be used within Wagtail admin views just like in any other Django app. However, Wagtail also provides various admin-specific form widgets, such as date/time pickers and choosers for pages, documents, images and snippets. By constructing forms using wagtail.admin.forms.models.WagtailAdminModelForm
as the base class instead of django.forms.models.ModelForm
, the most appropriate widget will be selected for each model field. For example, given the model and form definition:
from django.db import models from wagtail.admin.forms.models import WagtailAdminModelForm from wagtail.images.models import Image class FeaturedImage(models.Model): date = models.DateField() image = models.ForeignKey(Image, on_delete=models.CASCADE) class FeaturedImageForm(WagtailAdminModelForm): class Meta: model = FeaturedImage
the date
and image
fields on the form will use a date picker and image chooser widget respectively.
If you have implemented a form widget of your own, you can configure WagtailAdminModelForm
to select it for a given model field type. This is done by calling the wagtail.admin.forms.models.register_form_field_override
function, typically in an AppConfig.ready
method.
For example, if the app wagtail.videos
implements a Video
model and a VideoChooser
form widget, the following AppConfig definition will ensure that WagtailAdminModelForm
selects VideoChooser
as the form widget for any foreign keys pointing to Video
:
from django.apps import AppConfig from django.db.models import ForeignKey class WagtailVideosAppConfig(AppConfig): name = 'wagtail.videos' label = 'wagtailvideos' def ready(self): from wagtail.admin.forms.models import register_form_field_override from .models import Video from .widgets import VideoChooser register_form_field_override(ForeignKey, to=Video, override={'widget': VideoChooser})
Wagtail’s edit views for pages, snippets and ModelAdmin use WagtailAdminModelForm
as standard, so this change will take effect across the Wagtail admin; a foreign key to Video
on a page model will automatically use the VideoChooser
widget, with no need to specify this explicitly.
Panels (also known as edit handlers until Wagtail 3.0) are Wagtail’s mechanism for specifying the content and layout of a model form without having to write a template. They are used for the editing interface for pages and snippets, as well as the ModelAdmin and site settings contrib modules.
See Panel types for the set of panel types provided by Wagtail. All panels inherit from the base class wagtail.admin.panels.Panel
. A single panel object (usually ObjectList
or TabbedInterface
) exists at the top level and is the only one directly accessed by the view code; panels containing child panels inherit from the base class wagtail.admin.panels.PanelGroup
and take care of recursively calling methods on their child panels where appropriate.
A view performs the following steps to render a model form through the panels mechanism:
edit_handler
property and falling back on an ObjectList
consisting of children given by the model’s panels
property. However, it may come from elsewhere - for example, the ModelAdmin module allows defining it on the ModelAdmin configuration object.PanelsGroup
s permissions do not allow a user to see this panel, then nothing more will be done.bind_to_model
on the top-level panel, passing the model class, and this returns a clone of the panel with a model
property. As part of this process the on_model_bound
method is invoked on each child panel, to allow it to perform additional initialisation that requires access to the model (for example, this is where FieldPanel
retrieves the model field definition).The view then calls get_form_class
on the top-level panel to retrieve a ModelForm subclass that can be used to edit the model. This proceeds as follows:
base_form_class
property, falling back on wagtail.admin.forms.WagtailAdminModelForm
get_form_options
on each child panel - which returns a dictionary of properties including fields
and widgets
- and merge the results into a single dictionaryMeta
class.get_bound_panel
on the top-level panel, passing instance
, form
and request
as keyword arguments. This returns a BoundPanel
object, which follows the template component API. Finally, the BoundPanel
object (and its media definition) is rendered onto the template.New panel types can be defined by subclassing wagtail.admin.panels.Panel
- see Panel API.
© 2014-present Torchbox Ltd and individual contributors.
All rights are reserved.
Licensed under the BSD License.
https://docs.wagtail.org/en/stable/extending/forms.html