.. index::
single: Form; Custom form rendering
How to Customize Form Rendering
===============================
Symfony gives you a wide variety of ways to customize how a form is rendered.
In this guide, you'll learn how to customize every possible part of your
form with as little effort as possible whether you use Twig or PHP as your
templating engine.
Form Rendering Basics
---------------------
Recall that the label, error and HTML widget of a form field can easily
be rendered by using the ``form_row()`` Twig function or the ``row`` PHP helper
method:
.. configuration-block::
.. code-block:: twig
{{ form_row(form.age) }}
.. code-block:: php
row($form['age']); ?>
You can also render each of the three parts of the field individually:
.. configuration-block::
.. code-block:: html+twig
In both cases, the form label, errors and HTML widget are rendered by using
a set of markup that ships standard with Symfony. For example, both of the
above templates would render:
.. code-block:: html
This field is required
To quickly prototype and test a form, you can render the entire form with
just one line:
.. configuration-block::
.. code-block:: twig
{# renders all fields #}
{{ form_widget(form) }}
{# renders all fields *and* the form start and end tags #}
{{ form(form) }}
.. code-block:: php
widget($form) ?>
form($form) ?>
The remainder of this recipe will explain how every part of the form's markup
can be modified at several different levels. For more information about form
rendering in general, see :doc:`/form/rendering`.
.. _form-customization-form-themes:
What are Form Themes?
---------------------
Symfony uses form fragments - a small piece of a template that renders just
one part of a form - to render each part of a form - field labels, errors,
``input`` text fields, ``select`` tags, etc.
The fragments are defined as blocks in Twig and as template files in PHP.
A *theme* is nothing more than a set of fragments that you want to use when
rendering a form. In other words, if you want to customize one portion of
how a form is rendered, you'll import a *theme* which contains a customization
of the appropriate form fragments.
Symfony comes with four **built-in form themes** that define each and every
fragment needed to render every part of a form:
* `form_div_layout.html.twig`_, wraps each form field inside a ``
`` element.
* `form_table_layout.html.twig`_, wraps the entire form inside a ``
``
element and each form field inside a ``
`` element.
* `bootstrap_3_layout.html.twig`_, wraps each form field inside a ``
`` element
with the appropriate CSS classes to apply the default `Bootstrap 3 CSS framework`_
styles.
* `bootstrap_3_horizontal_layout.html.twig`_, it's similar to the previous theme,
but the CSS classes applied are the ones used to display the forms horizontally
(i.e. the label and the widget in the same row).
.. caution::
When you use the Bootstrap form themes and render the fields manually,
calling ``form_label()`` for a checkbox/radio field doesn't show anything.
Due to Bootstrap internals, the label is already shown by ``form_widget()``.
In the next section you will learn how to customize a theme by overriding
some or all of its fragments.
For example, when the widget of an ``integer`` type field is rendered, an ``input``
``number`` field is generated
.. configuration-block::
.. code-block:: html+twig
{{ form_widget(form.age) }}
.. code-block:: php
widget($form['age']) ?>
renders:
.. code-block:: html
Internally, Symfony uses the ``integer_widget`` fragment to render the field.
This is because the field type is ``integer`` and you're rendering its ``widget``
(as opposed to its ``label`` or ``errors``).
In Twig that would default to the block ``integer_widget`` from the `form_div_layout.html.twig`_
template.
In PHP it would rather be the ``integer_widget.html.php`` file located in
the ``FrameworkBundle/Resources/views/Form`` folder.
The default implementation of the ``integer_widget`` fragment looks like this:
.. configuration-block::
.. code-block:: twig
{# form_div_layout.html.twig #}
{% block integer_widget %}
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
{% endblock integer_widget %}
.. code-block:: html+php
block($form, 'form_widget_simple', array('type' => isset($type) ? $type : "number")) ?>
As you can see, this fragment itself renders another fragment - ``form_widget_simple``:
.. configuration-block::
.. code-block:: html+twig
{# form_div_layout.html.twig #}
{% block form_widget_simple %}
{% set type = type|default('text') %}
{% endblock form_widget_simple %}
.. code-block:: html+php
value="escape($value) ?>"
block($form, 'widget_attributes') ?>
/>
The point is, the fragments dictate the HTML output of each part of a form. To
customize the form output, you just need to identify and override the correct
fragment. A set of these form fragment customizations is known as a form "theme".
When rendering a form, you can choose which form theme(s) you want to apply.
In Twig a theme is a single template file and the fragments are the blocks defined
in this file.
In PHP a theme is a folder and the fragments are individual template files in
this folder.
.. _form-customization-sidebar:
.. sidebar:: Knowing which Block to Customize
In this example, the customized fragment name is ``integer_widget`` because
you want to override the HTML ``widget`` for all ``integer`` field types. If
you need to customize ``textarea`` fields, you would customize ``textarea_widget``.
As you can see, the fragment name is a combination of the field type and
which part of the field is being rendered (e.g. ``widget``, ``label``,
``errors``, ``row``). As such, to customize how errors are rendered for
just input ``text`` fields, you should customize the ``text_errors`` fragment.
More commonly, however, you'll want to customize how errors are displayed
across *all* fields. You can do this by customizing the ``form_errors``
fragment. This takes advantage of field type inheritance. Specifically,
since the ``text`` type extends from the ``form`` type, the Form component
will first look for the type-specific fragment (e.g. ``text_errors``) before
falling back to its parent fragment name if it doesn't exist (e.g. ``form_errors``).
For more information on this topic, see :ref:`form-template-blocks`.
.. _form-theming-methods:
Form Theming
------------
To see the power of form theming, suppose you want to wrap every input ``number``
field with a ``div`` tag. The key to doing this is to customize the
``integer_widget`` fragment.
Form Theming in Twig
--------------------
When customizing the form field block in Twig, you have two options on *where*
the customized form block can live:
+--------------------------------------+-----------------------------------+-------------------------------------------+
| Method | Pros | Cons |
+======================================+===================================+===========================================+
| Inside the same template as the form | Quick and easy | Can't be reused in other templates |
+--------------------------------------+-----------------------------------+-------------------------------------------+
| Inside a separate template | Can be reused by many templates | Requires an extra template to be created |
+--------------------------------------+-----------------------------------+-------------------------------------------+
Both methods have the same effect but are better in different situations.
Method 1: Inside the same Template as the Form
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The easiest way to customize the ``integer_widget`` block is to customize it
directly in the template that's actually rendering the form.
.. code-block:: html+twig
{% extends 'base.html.twig' %}
{% form_theme form _self %}
{% block integer_widget %}
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
{% endblock %}
{% block content %}
{# ... render the form #}
{{ form_row(form.age) }}
{% endblock %}
By using the special ``{% form_theme form _self %}`` tag, Twig looks inside
the same template for any overridden form blocks. Assuming the ``form.age``
field is an ``integer`` type field, when its widget is rendered, the customized
``integer_widget`` block will be used.
The disadvantage of this method is that the customized form block can't be
reused when rendering other forms in other templates. In other words, this method
is most useful when making form customizations that are specific to a single
form in your application. If you want to reuse a form customization across
several (or all) forms in your application, read on to the next section.
Method 2: Inside a separate Template
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
You can also choose to put the customized ``integer_widget`` form block in a
separate template entirely. The code and end-result are the same, but you
can now re-use the form customization across many templates:
.. code-block:: html+twig
{# app/Resources/views/form/fields.html.twig #}
{% block integer_widget %}
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
{% endblock %}
Now that you've created the customized form block, you need to tell Symfony
to use it. Inside the template where you're actually rendering your form,
tell Symfony to use the template via the ``form_theme`` tag:
.. code-block:: html+twig
{% form_theme form 'form/fields.html.twig' %}
{{ form_widget(form.age) }}
When the ``form.age`` widget is rendered, Symfony will use the ``integer_widget``
block from the new template and the ``input`` tag will be wrapped in the
``div`` element specified in the customized block.
Multiple Templates
..................
A form can also be customized by applying several templates. To do this, pass the
name of all the templates as an array using the ``with`` keyword:
.. code-block:: html+twig
{% form_theme form with ['common.html.twig', 'form/fields.html.twig'] %}
{# ... #}
The templates can also be located in different bundles, use the Twig namespaced
path to reference these templates, e.g. ``@AcmeFormExtra/form/fields.html.twig``.
Child Forms
...........
You can also apply a form theme to a specific child of your form:
.. code-block:: html+twig
{% form_theme form.a_child_form 'form/fields.html.twig' %}
This is useful when you want to have a custom theme for a nested form that's
different than the one of your main form. Just specify both your themes:
.. code-block:: html+twig
{% form_theme form 'form/fields.html.twig' %}
{% form_theme form.a_child_form 'form/fields_child.html.twig' %}
Form Theming in PHP
-------------------
When using PHP as a templating engine, the only method to customize a fragment
is to create a new template file - this is similar to the second method used by
Twig.
The template file must be named after the fragment. You must create a ``integer_widget.html.php``
file in order to customize the ``integer_widget`` fragment.
.. code-block:: html+php
Now that you've created the customized form template, you need to tell Symfony
to use it. Inside the template where you're actually rendering your form,
tell Symfony to use the theme via the ``setTheme()`` helper method::
setTheme($form, array(':form')); ?>
widget($form['age']) ?>
When the ``form.age`` widget is rendered, Symfony will use the customized
``integer_widget.html.php`` template and the ``input`` tag will be wrapped in
the ``div`` element.
If you want to apply a theme to a specific child form, pass it to the ``setTheme()``
method::
setTheme($form['child'], ':form'); ?>
.. note::
The ``:form`` syntax is based on the functional names for templates:
``Bundle:Directory``. As the form directory lives in the
``app/Resources/views`` directory, the ``Bundle`` part is empty, resulting
in ``:form``.
Referencing base Form Blocks (Twig specific)
--------------------------------------------
So far, to override a particular form block, the best method is to copy
the default block from `form_div_layout.html.twig`_, paste it into a different template,
and then customize it. In many cases, you can avoid doing this by referencing
the base block when customizing it.
This is easy to do, but varies slightly depending on if your form block customizations
are in the same template as the form or a separate template.
Referencing Blocks from inside the same Template as the Form
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Import the blocks by adding a ``use`` tag in the template where you're rendering
the form:
.. code-block:: twig
{% use 'form_div_layout.html.twig' with integer_widget as base_integer_widget %}
Now, when the blocks from `form_div_layout.html.twig`_ are imported, the
``integer_widget`` block is called ``base_integer_widget``. This means that when
you redefine the ``integer_widget`` block, you can reference the default markup
via ``base_integer_widget``:
.. code-block:: html+twig
{% block integer_widget %}
{{ block('base_integer_widget') }}
{% endblock %}
Referencing base Blocks from an external Template
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
If your form customizations live inside an external template, you can reference
the base block by using the ``parent()`` Twig function:
.. code-block:: html+twig
{# app/Resources/views/form/fields.html.twig #}
{% extends 'form_div_layout.html.twig' %}
{% block integer_widget %}
{{ parent() }}
{% endblock %}
.. note::
It is not possible to reference the base block when using PHP as the
templating engine. You have to manually copy the content from the base block
to your new template file.
Making Application-wide Customizations
--------------------------------------
If you'd like a certain form customization to be global to your application,
you can accomplish this by making the form customizations in an external
template and then importing it inside your application configuration.
Twig
~~~~
By using the following configuration, any customized form blocks inside the
``form/fields.html.twig`` template will be used globally when a form is
rendered.
.. configuration-block::
.. code-block:: yaml
# app/config/config.yml
twig:
form_themes:
- 'form/fields.html.twig'
# ...
.. code-block:: xml
form/fields.html.twig
.. code-block:: php
// app/config/config.php
$container->loadFromExtension('twig', array(
'form_themes' => array(
'form/fields.html.twig',
),
// ...
));
By default, Twig uses a *div* layout when rendering forms. Some people, however,
may prefer to render forms in a *table* layout. Use the ``form_table_layout.html.twig``
resource to use such a layout:
.. configuration-block::
.. code-block:: yaml
# app/config/config.yml
twig:
form_themes:
- 'form_table_layout.html.twig'
# ...
.. code-block:: xml
form_table_layout.html.twig
.. code-block:: php
// app/config/config.php
$container->loadFromExtension('twig', array(
'form_themes' => array(
'form_table_layout.html.twig',
),
// ...
));
If you only want to make the change in one template, add the following line to
your template file rather than adding the template as a resource:
.. code-block:: html+twig
{% form_theme form 'form_table_layout.html.twig' %}
Note that the ``form`` variable in the above code is the form view variable
that you passed to your template.
PHP
~~~
By using the following configuration, any customized form fragments inside the
``app/Resources/views/Form`` folder will be used globally when a
form is rendered.
.. configuration-block::
.. code-block:: yaml
# app/config/config.yml
framework:
templating:
form:
resources:
- 'AppBundle:Form'
# ...
.. code-block:: xml
AppBundle:Form
.. code-block:: php
// app/config/config.php
// PHP
$container->loadFromExtension('framework', array(
'templating' => array(
'form' => array(
'resources' => array(
'AppBundle:Form',
),
),
),
// ...
));
By default, the PHP engine uses a *div* layout when rendering forms. Some people,
however, may prefer to render forms in a *table* layout. Use the ``FrameworkBundle:FormTable``
resource to use such a layout:
.. configuration-block::
.. code-block:: yaml
# app/config/config.yml
framework:
templating:
form:
resources:
- 'FrameworkBundle:FormTable'
.. code-block:: xml
FrameworkBundle:FormTable
.. code-block:: php
// app/config/config.php
$container->loadFromExtension('framework', array(
'templating' => array(
'form' => array(
'resources' => array(
'FrameworkBundle:FormTable',
),
),
),
// ...
));
If you only want to make the change in one template, add the following line to
your template file rather than adding the template as a resource:
.. code-block:: html+php
setTheme($form, array('FrameworkBundle:FormTable')); ?>
Note that the ``$form`` variable in the above code is the form view variable
that you passed to your template.
How to Customize an individual Field
------------------------------------
So far, you've seen the different ways you can customize the widget output
of all text field types. You can also customize individual fields. For example,
suppose you have two ``text`` fields in a ``product`` form - ``name`` and
``description`` - but you only want to customize one of the fields. This can be
accomplished by customizing a fragment whose name is a combination of the field's
``id`` attribute and which part of the field is being customized. For example, to
customize the ``name`` field only:
.. configuration-block::
.. code-block:: html+twig
{% form_theme form _self %}
{% block _product_name_widget %}
Here, the ``_product_name_widget`` fragment defines the template to use for the
field whose *id* is ``product_name`` (and name is ``product[name]``).
.. tip::
The ``product`` portion of the field is the form name, which may be set
manually or generated automatically based on your form type name (e.g.
``ProductType`` equates to ``product``). If you're not sure what your
form name is, just view the source of your generated form.
If you want to change the ``product`` or ``name`` portion of the block
name ``_product_name_widget`` you can set the ``block_name`` option in your
form type::
use Symfony\Component\Form\FormBuilderInterface;
public function buildForm(FormBuilderInterface $builder, array $options)
{
// ...
$builder->add('name', 'text', array(
'block_name' => 'custom_name',
));
}
Then the block name will be ``_product_custom_name_widget``.
You can also override the markup for an entire field row using the same method:
.. configuration-block::
.. code-block:: html+twig
{% form_theme form _self %}
{% block _product_name_row %}
.. _form-custom-prototype:
How to Customize a Collection Prototype
---------------------------------------
When using a :doc:`collection of forms `,
the prototype can be overridden with a completely custom prototype by
overriding a block. For example, if your form field is named ``tasks``, you
will be able to change the widget for each task as follows:
.. configuration-block::
.. code-block:: html+twig
{% form_theme form _self %}
{% block _tasks_entry_widget %}
{{ form_widget(form.task) }}
{{ form_widget(form.dueDate) }}
{% endblock %}
.. code-block:: html+php
widget($form->task) ?>
widget($form->dueDate) ?>
Not only can you override the rendered widget, but you can also change the
complete form row or the label as well. For the ``tasks`` field given above,
the block names would be the following:
================ =======================
Part of the Form Block Name
================ =======================
``label`` ``_tasks_entry_label``
``widget`` ``_tasks_entry_widget``
``row`` ``_tasks_entry_row``
================ =======================
Other common Customizations
---------------------------
So far, this recipe has shown you several different ways to customize a single
piece of how a form is rendered. The key is to customize a specific fragment that
corresponds to the portion of the form you want to control (see
:ref:`naming form blocks `).
In the next sections, you'll see how you can make several common form customizations.
To apply these customizations, use one of the methods described in the
:ref:`form-theming-methods` section.
Customizing Error Output
~~~~~~~~~~~~~~~~~~~~~~~~
.. note::
The Form component only handles *how* the validation errors are rendered,
and not the actual validation error messages. The error messages themselves
are determined by the validation constraints you apply to your objects.
For more information, see the article on :doc:`validation `.
There are many different ways to customize how errors are rendered when a
form is submitted with errors. The error messages for a field are rendered
when you use the ``form_errors()`` helper:
.. configuration-block::
.. code-block:: twig
{{ form_errors(form.age) }}
.. code-block:: php
errors($form['age']); ?>
By default, the errors are rendered inside an unordered list:
.. code-block:: html
This field is required
To override how errors are rendered for *all* fields, simply copy, paste
and customize the ``form_errors`` fragment.
.. configuration-block::
.. code-block:: html+twig
{% form_theme form _self %}
{# form_errors.html.twig #}
{% block form_errors %}
{% spaceless %}
{% if errors|length > 0 %}
.. tip::
See :ref:`form-theming-methods` for how to apply this customization.
You can also customize the error output for just one specific field type.
To customize *only* the markup used for these errors, follow the same directions
as above but put the contents in a relative ``_errors`` block (or file in case
of PHP templates). For example: ``text_errors`` (or ``text_errors.html.php``).
.. tip::
See :ref:`form-template-blocks` to find out which specific block or file you
have to customize.
Certain errors that are more global to your form (i.e. not specific to just one
field) are rendered separately, usually at the top of your form:
.. configuration-block::
.. code-block:: twig
{{ form_errors(form) }}
.. code-block:: php
render($form); ?>
To customize *only* the markup used for these errors, follow the same directions
as above, but now check if the ``compound`` variable is set to ``true``. If it
is ``true``, it means that what's being currently rendered is a collection of
fields (e.g. a whole form), and not just an individual field.
.. configuration-block::
.. code-block:: html+twig
{% form_theme form _self %}
{# form_errors.html.twig #}
{% block form_errors %}
{% spaceless %}
{% if errors|length > 0 %}
{% if compound %}
{% for error in errors %}
{{ error.message }}
{% endfor %}
{% else %}
{# ... display the errors for a single field #}
{% endif %}
{% endif %}
{% endspaceless %}
{% endblock form_errors %}
.. code-block:: html+php
getMessage() ?>
Customizing the "Form Row"
~~~~~~~~~~~~~~~~~~~~~~~~~~
When you can manage it, the easiest way to render a form field is via the
``form_row()`` function, which renders the label, errors and HTML widget of
a field. To customize the markup used for rendering *all* form field rows,
override the ``form_row`` fragment. For example, suppose you want to add a
class to the ``div`` element around each row:
.. configuration-block::
.. code-block:: html+twig
{# form_row.html.twig #}
{% block form_row %}
.. tip::
See :ref:`form-theming-methods` for how to apply this customization.
Adding a "Required" Asterisk to Field Labels
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
If you want to denote all of your required fields with a required asterisk (``*``),
you can do this by customizing the ``form_label`` fragment.
In Twig, if you're making the form customization inside the same template as your
form, modify the ``use`` tag and add the following:
.. code-block:: html+twig
{% use 'form_div_layout.html.twig' with form_label as base_form_label %}
{% block form_label %}
{{ block('base_form_label') }}
{% if required %}
*
{% endif %}
{% endblock %}
In Twig, if you're making the form customization inside a separate template, use
the following:
.. code-block:: html+twig
{% extends 'form_div_layout.html.twig' %}
{% block form_label %}
{{ parent() }}
{% if required %}
*
{% endif %}
{% endblock %}
When using PHP as a templating engine you have to copy the content from the
original template:
.. code-block:: html+php
humanize($name); } ?>
*
.. tip::
See :ref:`form-theming-methods` for how to apply this customization.
.. sidebar:: Using CSS only
By default, ``label`` tags of required fields are rendered with a
``required`` CSS class. Thus, you can also add an asterisk using CSS only:
.. code-block:: css
label.required:before {
content: "* ";
}
Adding "help" Messages
~~~~~~~~~~~~~~~~~~~~~~
You can also customize your form widgets to have an optional "help" message.
In Twig, if you're making the form customization inside the same template as your
form, modify the ``use`` tag and add the following:
.. code-block:: html+twig
{% use 'form_div_layout.html.twig' with form_widget_simple as base_form_widget_simple %}
{% block form_widget_simple %}
{{ block('base_form_widget_simple') }}
{% if help is defined %}
{{ help }}
{% endif %}
{% endblock %}
In Twig, if you're making the form customization inside a separate template, use
the following:
.. code-block:: html+twig
{% extends 'form_div_layout.html.twig' %}
{% block form_widget_simple %}
{{ parent() }}
{% if help is defined %}
{{ help }}
{% endif %}
{% endblock %}
When using PHP as a templating engine you have to copy the content from the
original template:
.. code-block:: html+php
value="escape($value) ?>"
block($form, 'widget_attributes') ?>
/>
escape($help) ?>
To render a help message below a field, pass in a ``help`` variable:
.. configuration-block::
.. code-block:: twig
{{ form_widget(form.title, {'help': 'foobar'}) }}
.. code-block:: php
widget($form['title'], array('help' => 'foobar')) ?>
.. tip::
See :ref:`form-theming-methods` for how to apply this customization.
Using Form Variables
--------------------
Most of the functions available for rendering different parts of a form (e.g.
the form widget, form label, form errors, etc.) also allow you to make certain
customizations directly. Look at the following example:
.. configuration-block::
.. code-block:: twig
{# render a widget, but add a "foo" class to it #}
{{ form_widget(form.name, { 'attr': {'class': 'foo'} }) }}
.. code-block:: php
widget($form['name'], array(
'attr' => array(
'class' => 'foo',
),
)) ?>
The array passed as the second argument contains form "variables". For
more details about this concept in Twig, see :ref:`twig-reference-form-variables`.
.. _`form_div_layout.html.twig`: https://github.com/symfony/symfony/blob/master/src/Symfony/Bridge/Twig/Resources/views/Form/form_div_layout.html.twig
.. _`form_table_layout.html.twig`: https://github.com/symfony/symfony/blob/master/src/Symfony/Bridge/Twig/Resources/views/Form/form_table_layout.html.twig
.. _`bootstrap_3_layout.html.twig`: https://github.com/symfony/symfony/blob/master/src/Symfony/Bridge/Twig/Resources/views/Form/bootstrap_3_layout.html.twig
.. _`bootstrap_3_horizontal_layout.html.twig`: https://github.com/symfony/symfony/blob/master/src/Symfony/Bridge/Twig/Resources/views/Form/bootstrap_3_horizontal_layout.html.twig
.. _`Bootstrap 3 CSS framework`: http://getbootstrap.com/