How it works

First we have to discuss, what is the problem with Symfony forms and vue.js (or any other javascript based render library). In Symfony we have a quite mighty form api which help us to translate a model into a form with view data, which can be used by the twig engine to render a html form. On the other hand the form api can use the input from a request to translate it back to our model with changed data.


If we use a javascript render library like vue, we will generate the html code with this library on the client side, so we are not able to use the provided twig functions by symfony. We have to replace this part with our own solution. This bundle and the corresponding js package provide a solution and give us a small api to extend the behavior for your own forms.

Let’s have a closer look into how symfony is handle the things with twig and how this bundle is working.


On the top of the above image you can see the Symfony workflow. The Form will create a FormView which will be used by twig. I guess you are already familiar with this. The FormView contains all necessary data to render the form. For our workflow you can see below, we use the view data as input and normalize them to data we need for our vue application.


Why we don’t just use the symfony serializer? In the normalize step we also have to add some additional data, which is not provided by the FormView

Here is a small code example to show how easy it is to use the VueForm inside a controller.

namespace App\Controller;

use App\Form\MyForm;
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\HttpFoundation\JsonResponse;

class HelloController extends AbstractController
    public function __construct(
        private VueForm $vueForm,

    public function formAction()
        $form = $this->formFactory->create(MyForm::class);
        $formData = $this->vueForm->createData($form->createView());

        return new JsonResponse(['form' => $formData])


A vue type helps us to translate the data from the FormView into data we need for our vue application.

namespace Enhavo\Bundle\VueFormBundle\Form\VueType;

use Enhavo\Bundle\VueFormBundle\Form\AbstractVueType;
use Enhavo\Bundle\VueFormBundle\Form\VueData;
use Symfony\Component\Form\FormView;

class TextVueType extends AbstractVueType
    public static function supports(FormView $formView): bool
        return in_array('text', $formView->vars['block_prefixes']);

    public function buildView(FormView $view, VueData $data)
        $data['component'] = 'form-simple';

    public function finishView(FormView $view, VueData $data)


This VueType reacts only if a text block prefix is used and add the component which should be used later.


In contrast to build the variables in the FormView with twig we have to take care what kind of data we will expose, because this data can be read by the user.

Client side

On the client side we can use the output from our controller to render the view in the browser.

// entrypoint

import {createApp, reactive} from "vue";
import {FormFactory} from "@enhavo/vue-form/form/FormFactory";
import VueForm from "@enhavo/vue-form/index";
import ApplicationComponent from "./component/Application";

let formFactory = new FormFactory();

  .then((response) => response.json())
  .then((data) => {
    let form = formFactory.create(data.form);

    const app = createApp(ApplicationComponent, {
        form: reactive(form)

The VueForm plugin provide components similar to the twig function from Symfony.

// ./component/Application

    <!-- like {{ form(form) }} in twig -->
    <form-form :form="form" />

    <!-- like {{ form_row( }} in twig -->
    <form-row :form="form.get('property')" />

    <!-- like {{ form_widget( }} in twig -->
    <form-widget :form="form.get('property')" />

    <!-- like {{ form_label( }} in twig -->
    <form-label :form="form.get('property')" />

    <!-- like {{ form_errors( }} in twig -->
    <form-errors :form="form.get('property')" />

    <!-- like {{ form_help( }} in twig -->
    <form-help :form="form.get('property')" />