How to use related select boxes in Symfony ?
Let's say, I have a select list containing compagnies and another containing employees of the selected company. How do I define those in Symfony? I have already created all the Javascript related code but when submitting my form and having errors on some fields, the all "sub" select fields are reset to null.
Any ideas?
Thanks,
EDIT : As the question seems to be misunderstood, I'll add some precisions :
Not working, there is no way out of the box to do this. Even non out of the box solutions...
But when the form is created, the values are empty. The values are only set on bindRequest.
This actually works, but I think there is no other word than really really ugly programming here.
Question has been asked here, on the Symfony2 mailing-list, on Twitter and the officiel Symfony 2 forum. I have of course searched each of those several times before posting my questions.
$('#type'). change(function() { alert('Value changed to ' + $(this). attr('value')); }); This will give you the value of the selected option tag.
The multiple attribute is a boolean attribute. When present, it specifies that multiple options can be selected at once. Selecting multiple options vary in different operating systems and browsers: For windows: Hold down the control (ctrl) button to select multiple options.
Yes. Add an onChange property to the first select, then use it to call a javascript function you have written elsewhere.
Concerning what you already tried, I think you should retry your first/second ideas:
My first idea was to use the form entity type, thinking the component could be binded somehow on another field. ie. update list of employees based on the value of the selected company.
You can populate the employees select box using the entity
type. All you have to do is to define the good options:
class FooType extends AbstractType { public function buildForm(FormBuilder $builder, array $options) { $builder ->add('employee', 'entity', array( 'class' => 'Entity\Employee', 'query_builder' => function ($repository) use($options) { return $repository ->createQueryBuilder('e') ->where('e.company = :company') ->setParameter('company', $options['companyId']) ; }, )) ; } public function getDefaultOptions(array $options) { return array('data_class' => 'Entity\Foo', 'companyId' => null); } }
Then I thought about manually passing the selected company as parameter to the query builder of the second dropdown list.
The example here filters the employees list based on the companyId form's option. You can modify this behavior by filtering directly on the company present in the form's data.
public function buildForm(FormBuilder $builder, array $options) { $companyId = $builder->getData()->getCompanyId(); $builder ->add('employee', 'entity', array( 'class' => 'Entity\Employee', 'query_builder' => function ($repository) use ($companyId) { return $repository ->createQueryBuilder('e') ->where('e.company = :company') ->setParameter('company', $companyId) ; }, )) ; }
You still have to implement the getEmployee()
and setEmployee()
methods in your Entity\Foo
class.
But when the form is created, the values are empty. The values are only set on bindRequest.
No. Values are set when you create you form using form factory (third argument), OR when you call $form->setData($foo);
. Data is modified when you bind
new inputs to the form.
There could be a problem with this approach: it's possible that the employee id bound to the form is not available in the form choice list, because you changed the company (and thus the employees).
I had the same problem. You must use form events. My code example with country, region, city relations.
namespace Orfos\UserBundle\Form\Type; ///import form events namespace use Symfony\Component\EventDispatcher\EventSubscriberInterface; use Symfony\Component\Form\FormEvents; use Symfony\Component\Form\Event\DataEvent; class RegistrationFormType extends BaseType { private $request; public function __construct($class, $request, $doctrine) { parent::__construct($class); $this->request = $request; $this->doctrine = $doctrine; } public function buildForm(FormBuilder $builder, array $options) { parent::buildForm($builder, $options); //other fields $locale = $this->request->getLocale(); $builder->add('country', 'entity', array( 'class' => 'Orfos\CoreBundle\Entity\Country', 'property' => $locale . 'name', 'label' => 'register.country.label', 'query_builder' => function(EntityRepository $er) { return $er->createQueryBuilder('c') ->select('c', 't') ->join('c.translations', 't'); }, )); $factory = $builder->getFormFactory(); $refreshRegion = function ($form, $country) use ($factory, $locale) { $form->add($factory->createNamed('entity', 'region', null, array( 'class' => 'Orfos\CoreBundle\Entity\Region', 'property' => $locale . 'name', 'label' => 'register.region.label', 'query_builder' => function (EntityRepository $repository) use ($country) { $qb = $repository->createQueryBuilder('region') ->select('region', 'translation') ->innerJoin('region.country', 'country') ->join('region.translations', 'translation'); if ($country instanceof Country) { $qb = $qb->where('region.country = :country') ->setParameter('country', $country); } elseif (is_numeric($country)) { $qb = $qb->where('country.id = :country_id') ->setParameter('country_id', $country); } else { $qb = $qb->where('country.id = 1'); } return $qb; } ))); }; $factory = $builder->getFormFactory(); $refreshCity = function($form, $region) use ($factory, $locale) { $form->add($factory->createNamed('entity', 'city', null, array( 'class' => 'Orfos\CoreBundle\Entity\City', 'property' => $locale . 'name', 'label' => 'register.city.label', 'query_builder' => function (EntityRepository $repository) use ($region) { $qb = $repository->createQueryBuilder('city') ->select('city', 'translation') ->innerJoin('city.region', 'region') ->innerJoin('city.translations', 'translation'); if ($region instanceof Region) { $qb = $qb->where('city.region = :region') ->setParameter('region', $region); } elseif (is_numeric($region)) { $qb = $qb->where('region.id = :region_id') ->setParameter('region_id', $region); } else { $qb = $qb->where('region.id = 1'); } return $qb; } ))); }; $builder->addEventListener(FormEvents::PRE_SET_DATA, function (DataEvent $event) use ($refreshRegion, $refreshCity) { $form = $event->getForm(); $data = $event->getData(); if ($data == null){ $refreshRegion($form, null); $refreshCity($form, null); } if ($data instanceof Country) { $refreshRegion($form, $data->getCountry()->getRegions()); $refreshCity($form, $data->getRegion()->getCities()); } }); $builder->addEventListener(FormEvents::PRE_BIND, function (DataEvent $event) use ($refreshRegion, $refreshCity) { $form = $event->getForm(); $data = $event->getData(); if (array_key_exists('country', $data)) { $refreshRegion($form, $data['country']); } if (array_key_exists('region', $data)) { $refreshCity($form, $data['region']); } }); } }
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With