How to Customize Form Rendering
Warning: You are browsing the documentation for Symfony 3.x, which is no longer maintained.
Read the updated version of this page for Symfony 7.1 (the current stable version).
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 be rendered
by using the form_row()
Twig function or the row
PHP helper method:
1
{{ form_row(form.age) }}
You can also render each of the three parts of the field individually:
1 2 3 4 5
<div>
{{ form_label(form.age) }}
{{ form_errors(form.age) }}
{{ form_widget(form.age) }}
</div>
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:
1 2 3 4 5 6 7
<div>
<label for="form_age">Age</label>
<ul>
<li>This field is required</li>
</ul>
<input type="number" id="form_age" name="form[age]"/>
</div>
To quickly prototype and test a form, you can render the entire form with just one line:
1 2 3 4 5
{# renders all fields #}
{{ form_widget(form) }}
{# renders all fields *and* the form start and end tags #}
{{ form(form) }}
The remainder of this recipe will explain how every part of the form's markup can be modified at several levels. For more information about form rendering in general, see How to Control the Rendering of a Form.
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 some 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
<div>
element. - form_table_layout.html.twig, wraps the entire form inside a
<table>
element and each form field inside a<tr>
element. - bootstrap_3_layout.html.twig, wraps each form field inside a
<div>
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).
- bootstrap_4_layout.html.twig, same as
bootstrap_3_layout.html.twig
, but updated for Bootstrap 4 CSS framework styles. - bootstrap_4_horizontal_layout.html.twig, same as
bootstrap_3_horizontal_layout.html.twig
but updated for Bootstrap 4 styles. - foundation_5_layout.html.twig, wraps each form field inside a
<div>
element with the appropriate CSS classes to apply the default Foundation CSS framework styles.
3.4
The Bootstrap 4 form themes were introduced in Symfony 3.4.
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()
.
Tip
Read more about the Bootstrap 4 form theme.
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
1
{{ form_widget(form.age) }}
renders:
1
<input type="number" id="form_age" name="form[age]" required="required" value="33"/>
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:
1 2 3 4 5
{# form_div_layout.html.twig #}
{% block integer_widget %}
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
{% endblock integer_widget %}
As you can see, this fragment itself renders another fragment - form_widget_simple
:
1 2 3 4 5
{# form_div_layout.html.twig #}
{% block form_widget_simple %}
{% set type = type|default('text') %}
<input type="{{ type }}" {{ block('widget_attributes') }} {% if value is not empty %}value="{{ value }}" {% endif %}/>
{% endblock form_widget_simple %}
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.
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
.
The integer
part comes from the class name: IntegerType
becomes integer
,
based on a standard.
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 How to Work with Form Themes.
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 | No need for an extra template file | 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.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
{% extends 'base.html.twig' %}
{% form_theme form _self %}
{% block integer_widget %}
<div class="integer_widget">
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
</div>
{% 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:
1 2 3 4 5 6 7
{# app/Resources/views/form/fields.html.twig #}
{% block integer_widget %}
<div class="integer_widget">
{% set type = type|default('number') %}
{{ block('form_widget_simple') }}
</div>
{% 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:
1 2 3
{% 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:
1 2 3
{% 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
.
Disabling Usage of Globally Defined Themes
Sometimes you may want to disable the use of the globally defined form themes in order to have more control over the rendering of a form. You might want this, for example, when creating an admin interface for a bundle which can be installed on a wide range of Symfony applications (and so you can't control what themes are defined globally).
You can do this by including the only
keyword after the list of form themes:
1 2 3
{% form_theme form with ['common.html.twig', 'form/fields.html.twig'] only %}
{# ... #}
Caution
When using the only
keyword, none of Symfony's built-in form themes
(form_div_layout.html.twig
, etc.) will be applied. In order to render
your forms correctly, you need to either provide a fully-featured form theme
yourself, or extend one of the built-in form themes with Twig's use
keyword instead of extends
to re-use the original theme contents.
1 2 3 4
{# app/Resources/views/common.html.twig #}
{% use "form_div_layout.html.twig" %}
{# ... #}
Child Forms
You can also apply a form theme to a specific child of your form:
1
{% 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:
1 2 3
{% form_theme form 'form/fields.html.twig' %}
{% form_theme form.a_child_form 'form/fields_child.html.twig' %}
Referencing base Form Blocks
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:
1
{% 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
:
1 2 3 4 5
{% block integer_widget %}
<div class="integer_widget">
{{ block('base_integer_widget') }}
</div>
{% 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:
1 2 3 4 5 6 7 8
{# app/Resources/views/form/fields.html.twig #}
{% extends 'form_div_layout.html.twig' %}
{% block integer_widget %}
<div class="integer_widget">
{{ parent() }}
</div>
{% 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.
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.
1 2 3 4 5
# app/config/config.yml
twig:
form_themes:
- 'form/fields.html.twig'
# ...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
<!-- app/config/config.xml -->
<?xml version="1.0" encoding="UTF-8" ?>
<container xmlns="http://symfony.com/schema/dic/services"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:twig="http://symfony.com/schema/dic/twig"
xsi:schemaLocation="http://symfony.com/schema/dic/services
https://symfony.com/schema/dic/services/services-1.0.xsd
http://symfony.com/schema/dic/twig
https://symfony.com/schema/dic/twig/twig-1.0.xsd">
<twig:config>
<twig:form-theme>form/fields.html.twig</twig:form-theme>
<!-- ... -->
</twig:config>
</container>
1 2 3 4 5 6 7 8
// app/config/config.php
$container->loadFromExtension('twig', [
'form_themes' => [
'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:
1 2 3 4 5
# app/config/config.yml
twig:
form_themes:
- 'form_table_layout.html.twig'
# ...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
<!-- app/config/config.xml -->
<?xml version="1.0" encoding="UTF-8" ?>
<container xmlns="http://symfony.com/schema/dic/services"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:twig="http://symfony.com/schema/dic/twig"
xsi:schemaLocation="http://symfony.com/schema/dic/services
https://symfony.com/schema/dic/services/services-1.0.xsd
http://symfony.com/schema/dic/twig
https://symfony.com/schema/dic/twig/twig-1.0.xsd">
<twig:config>
<twig:form-theme>form_table_layout.html.twig</twig:form-theme>
<!-- ... -->
</twig:config>
</container>
1 2 3 4 5 6 7 8
// app/config/config.php
$container->loadFromExtension('twig', [
'form_themes' => [
'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:
1
{% 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.
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:
1 2 3 4 5 6 7 8 9
{% form_theme form _self %}
{% block _product_name_widget %}
<div class="text_widget">
{{ block('form_widget_simple') }}
</div>
{% endblock %}
{{ form_widget(form.name) }}
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:
1 2 3 4 5 6 7 8 9 10 11
use Symfony\Component\Form\Extension\Core\Type\TextType;
use Symfony\Component\Form\FormBuilderInterface;
public function buildForm(FormBuilderInterface $builder, array $options)
{
// ...
$builder->add('name', TextType::class, [
'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:
1 2 3 4 5 6 7 8 9 10 11
{% form_theme form _self %}
{% block _product_name_row %}
<div class="name_row">
{{ form_label(form) }}
{{ form_errors(form) }}
{{ form_widget(form) }}
</div>
{% endblock %}
{{ form_row(form.name) }}
How to Customize a Collection Prototype
When using a 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:
1 2 3 4 5 6 7 8
{% form_theme form _self %}
{% block _tasks_entry_widget %}
<tr>
<td>{{ form_widget(form.task) }}</td>
<td>{{ form_widget(form.dueDate) }}</td>
</tr>
{% endblock %}
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 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 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 How to Customize Form Rendering 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 validation.
There are many 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:
1
{{ form_errors(form.age) }}
By default, the errors are rendered inside an unordered list:
1 2 3
<ul>
<li>This field is required</li>
</ul>
To override how errors are rendered for all fields, simply copy, paste
and customize the form_errors
fragment.
1 2 3 4 5 6 7 8 9 10 11 12 13 14
{% form_theme form _self %}
{# form_errors.html.twig #}
{% block form_errors %}
{% spaceless %}
{% if errors|length > 0 %}
<ul>
{% for error in errors %}
<li>{{ error.message }}</li>
{% endfor %}
</ul>
{% endif %}
{% endspaceless %}
{% endblock form_errors %}
Tip
See How to Customize Form Rendering 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 How to Work with Form Themes 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:
1
{{ form_errors(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.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
{% form_theme form _self %}
{# form_errors.html.twig #}
{% block form_errors %}
{% spaceless %}
{% if errors|length > 0 %}
{% if compound %}
<ul>
{% for error in errors %}
<li>{{ error.message }}</li>
{% endfor %}
</ul>
{% else %}
{# ... display the errors for a single field #}
{% endif %}
{% endif %}
{% endspaceless %}
{% endblock form_errors %}
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:
1 2 3 4 5 6 7 8
{# form_row.html.twig #}
{% block form_row %}
<div class="form_row">
{{ form_label(form) }}
{{ form_errors(form) }}
{{ form_widget(form) }}
</div>
{% endblock form_row %}
Tip
See How to Customize Form Rendering 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.
If you're making the form customization inside the same template as your
form, modify the use
tag and add the following:
1 2 3 4 5 6 7 8 9
{% use 'form_div_layout.html.twig' with form_label as base_form_label %}
{% block form_label %}
{{ block('base_form_label') }}
{% if label is not same as(false) and required %}
<span class="required" title="This field is required">*</span>
{% endif %}
{% endblock %}
If you're making the form customization inside a separate template, use the following:
1 2 3 4 5 6 7 8 9
{% extends 'form_div_layout.html.twig' %}
{% block form_label %}
{{ parent() }}
{% if label is not same as(false) and required %}
<span class="required" title="This field is required">*</span>
{% endif %}
{% endblock %}
Tip
See How to Customize Form Rendering for how to apply this customization.
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:
1 2 3
label.required:before {
content: "* ";
}
Adding "help" Messages
You can also customize your form widgets to have an optional "help" message.
If you're making the form customization inside the same template as your
form, modify the use
tag and add the following:
1 2 3 4 5 6 7 8 9
{% 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 %}
<span class="help-block">{{ help }}</span>
{% endif %}
{% endblock %}
If you're making the form customization inside a separate template, use the following:
1 2 3 4 5 6 7 8 9
{% extends 'form_div_layout.html.twig' %}
{% block form_widget_simple %}
{{ parent() }}
{% if help is defined %}
<span class="help-block">{{ help }}</span>
{% endif %}
{% endblock %}
To render a help message below a field, pass in a help
variable:
1
{{ form_widget(form.title, {'help': 'foobar'}) }}
Tip
See How to Customize Form Rendering 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:
1 2
{# render a widget, but add a "foo" class to it #}
{{ form_widget(form.name, { 'attr': {'class': 'foo'} }) }}
The array passed as the second argument contains form "variables". For more details about this concept in Twig, see Twig Template Form Function and Variable Reference.