nepada / form-renderer
Latte template based form renderer for Nette forms with full support for Bootstrap 3, 4 & 5.
Installs: 181 406
Dependents: 0
Suggesters: 0
Security: 0
Stars: 9
Watchers: 3
Forks: 2
Open Issues: 0
Language:HTML
Requires
- php: >=8.1.0 <8.5
- latte/latte: ^3.0@dev
- nette/application: ^3.1.13@dev
- nette/forms: ^3.1@dev
- nette/utils: ^3.2@dev || ^4.0@dev
Requires (Dev)
- composer-runtime-api: ^2.0
- composer/semver: 3.4.3
- mockery/mockery: 1.6.12
- nepada/coding-standard: 7.14.0
- nepada/phpstan-nette-tester: 1.2.1
- nette/bootstrap: >=3.1@dev
- nette/component-model: >=3.0.2
- nette/di: ^3.0.6@dev
- nette/http: >=3.1@dev
- nette/neon: >=3.3.4@dev
- nette/robot-loader: *@dev
- nette/tester: v2.5.4
- php-parallel-lint/php-parallel-lint: 1.4.0
- phpstan/phpstan: 1.12.5
- phpstan/phpstan-mockery: 1.1.3
- phpstan/phpstan-nette: 1.3.8
- phpstan/phpstan-strict-rules: 1.6.1
- shipmonk/phpstan-rules: 3.2.1
- spaze/phpstan-disallowed-calls: 3.4.0
Suggests
- nette/di: for integration with Nette DI container
This package is auto-updated.
Last update: 2025-01-14 20:19:46 UTC
README
Installation
Via Composer:
$ composer require nepada/form-renderer
Register the extension in config.neon
:
extensions: formRenderer: Nepada\Bridges\FormRendererDI\FormRendererExtension
Usage
Nette gives you two options how to render a form:
- Render the whole form manually in Latte template using form macros. This way you have complete control over the rendering, but writing all the templates quickly gets repetitive.
- Render it using a form renderer, e.g.
DefaultFormRenderer
fromnette/forms
.DefaultFormRenderer
is very customizable, but it's hard to setup special rendering rules for only some controls of a form, or add support for rendering of new form control types.
nepada/form-renderer
is built on top of Latte templates and their powerful blocks, thus combining strengths of manual rendering with DRY principle of form renderers.
TemplateRenderer
You can use TemplateRendererFactory
service to create the renderer with preconfigured default template. It renders a form more or less the same way as DefaultFormRenderer
.
Customizing rendering
You can customize rendering by importing blocks from a template file - blocks imported later override the previously imported ones of the same name. You can also pass custom variables to the template.
/** @var Nepada\FormRenderer\TemplateRendererFactory $factory */ $renderer = $factory->create(); $renderer->importTemplate(__DIR__ . '/custom-form-rendering-blocks.latte'); $renderer->getTemlate()->foo = 'bar'; $form->setRenderer($renderer);
Tips:
- You can define special rendering for a specific control of a form in
#control-name-*
block (e.g.#control-name-container-subcontainer-foocontrol
). - If you need special rendering for both a control and its label, define it in
#pair-name-*
block. - Rendering of different control types (based on the value of
$control->getOption('type')
) is controlled by blocks#control-type-*
and#pair-type-*
. The default template actually uses this for buttons (rendering of consecutive buttons in one row). - You can also specify template files to be imported in your
config.neon
:formRenderer: default: imports: - %appDir%/templates/@form-extras1.latte - %appDir%/templates/@form-extras2.latte
For a complete overview of supported blocks and better understanding how the renderer works, please read the code of the template.
Creating custom TemplateRenderer setup from scratch
You don't need to use the default template, you can create one from scratch with blocks tailored to your needs. You can define factory for your custom setup like this:
services: customRenderer: implement: Nepada\FormRenderer\TemplateRendererFactory setup: - importTemplate('%appDir%/templates/@form.latte') - importTemplate('%appDir%/templates/@form-extras.latte')
Just make sure that one of your template files defines block named #form
- this is used as a starting point for the rendering.
Filter safeTranslate
in templates
For translations the templates may use custom safeTranslate
filter. The key differences from standard translate
filter are:
- It avoids translating instances of
Nette\Utils\IHtmlString
andLatte\Runtime\IHtmlString
. - It uses a translator from the form instance that is being rendered.
- If the form has no translator set, than it simply returns the passed string untranslated.
Improved version of n:class
macro
In all form templates there is also available an improved version of n:class
macro that supports merging of classes from Nette\Utils\Html
instances. You can do stuff like <input n:name="$control" n:class="$control->controlPrototype->class, form-control">
and don't need to worry if the class attribute is really represented as a string or array, it all just works.
Bootstrap5Renderer
Form renderer compatible with Bootstrap 5, it internally uses TemplateRenderer
with custom template.
The template supports three rendering modes:
/** @var Nepada\FormRenderer\Bootstrap5RendererFactory $factory */ $renderer = $factory->create(); $renderer->setBasicMode(); // Basic form $renderer->setInlineMode(); // Inline form $renderer->setHorizontalMode(4, 8); // Horizontal form (you can optionally set the size of label and control columns)
Use $renderer->setRenderValidState(true)
to enable/disable rendering of "valid" form control state for filled inputs after form submission.
In inline mode the error messages are always rendered as tooltips. In the other modes you can switch between standard and tooltip rendering by calling $renderer->setUseErrorTooltips(true)
.
You can enable floating labels by $renderer->setUseFloatingLabels(true)
(ignored in horizontal mode only). By default, all controls of text
, datetime
, textarea
and select
type are rendered with floating label, but you can manually override this on a specific control by setting $input->setOption('floatingLabel', false)
.
To render a checkbox as a switch, you need to set type option: $checkboxInput->setOption('type', 'switch')
.
To render radio or checkbox as a toggle button, add btn
class (and any desired button styling class) to label prototype: $radio->getItemLabelPrototype()->addClass('btn btn-outline-primary')
.
Bootstrap5Renderer
makes a couple of adjustments to the form before it is passed over to TemplateRenderer
:
- It adds
btn btn-primary
classes to the control prototype of firstSubmitButton
in the form, unless there already is such a control in the form. - It adds
btn btn-secondary
classes to the control prototype of everyButton
control, unless it already hasbtn
class set. - Changes
type
option on allCheckbox
,CheckboxList
,RadioList
controls setup to be rendered as toggle buttons fromcheckbox
/radio
totogglebutton
/togglebuttonlist
. - Changes
type
option on allCheckboxList
controls fromcheckbox
tocheckboxlist
. - When floating labels are enabled, it sets boolean
floatingLabel
option (unless already set) on all controls to indicate whether the floating label should be rendered.
You can change the default renderer configuration from your config.neon
:
formRenderer: bootstrap5: mode: horizontal renderValidState: true useErrorTooltips: true imports: - %appDir%/templates/@form-extras.latte
Bootstrap4Renderer
Form renderer compatible with Bootstrap 4, it internally uses TemplateRenderer
with custom template.
The template supports three rendering modes:
/** @var Nepada\FormRenderer\Bootstrap4RendererFactory $factory */ $renderer = $factory->create(); $renderer->setBasicMode(); // Basic form $renderer->setInlineMode(); // Inline form $renderer->setHorizontalMode(4, 8); // Horizontal form (you can optionally set the size of label and control columns)
Use $renderer->setRenderValidState(true)
to enable/disable rendering of "valid" form control state for filled inputs after form submission.
In inline mode the error messages are always rendered as tooltips. In the other modes you can switch between standard and tooltip rendering by calling $renderer->setUseErrorTooltips(true)
.
You can enable the use of custom form controls by $renderer->setUseCustomControls(true)
.
To render a checkbox as a switch, you need to set type option: $checkboxInput->setOption('type', 'switch')
.
Bootstrap4Renderer
makes a couple of adjustments to the form before it is passed over to TemplateRenderer
:
- It adds
btn btn-primary
classes to the control prototype of firstSubmitButton
in the form, unless there already is such a control in the form. - It adds
btn btn-secondary
classes to the control prototype of everyButton
control, unless it already hasbtn
class set. - Changes
type
option on allCheckboxList
controls fromcheckbox
tocheckboxlist
.
You can change the default renderer configuration from your config.neon
:
formRenderer: bootstrap4: mode: horizontal renderValidState: true useErrorTooltips: true useCustomControls: true imports: - %appDir%/templates/@form-extras.latte
Bootstrap3Renderer
Form renderer compatible with Bootstrap 3, it internally uses TemplateRenderer
with custom template.
The template supports three rendering modes:
/** @var Nepada\FormRenderer\Bootstrap3RendererFactory $factory */ $renderer = $factory->create(); $renderer->setBasicMode(); // Basic form $renderer->setInlineMode(); // Inline form $renderer->setHorizontalMode(4, 8); // Horizontal form (you can optionally set the size of label and control columns)
Use $renderer->setRenderValidState(true)
to enable/disable rendering of "valid" form control state for filled inputs after form submission.
Bootstrap3Renderer
makes a couple of adjustments to the form before it is passed over to TemplateRenderer
:
- It adds
btn btn-primary
classes to the control prototype of firstSubmitButton
in the form, unless there already is such a control in the form. - It adds
btn btn-default
classes to the control prototype of everyButton
control, unless it already hasbtn
class set. - Changes
type
option on allCheckboxList
controls fromcheckbox
tocheckboxlist
.
You can change the default renderer configuration from your config.neon
:
formRenderer: bootstrap3: mode: horizontal renderValidState: true imports: - %appDir%/templates/@form-extras.latte