我正在使用Symfony <4,并且我在使用与参数的多对多关系中的其他表单时遇到问题。
你会在我的FeatureForm下面找到:
->add('tags',CollectionType::class,
array(
'entry_type' =>TagFeatureType::class,
'allow_add' => true,
'allow_delete' => true,
'data' => $datas,
'entry_options' => array(
'label' => false,
)
)
)
现在我的TagFeatureType:
->add('tag', EntityType::class,
array(
'choice_label' => 'name',
'class' => 'AppBundle:Tag',
'query_builder' => function(TagRepository $tr){
return $tr->findObjectNotMine();
}
)
)
我想在findOBjectNotMine中注入一个参数但是我无法从控制器传递参数,因为TagFeatureType是由FeatureForm创建的。在buildForm函数内部,我无法传递任何额外的参数。
我看到2个可能性,第一个我修改默认选项以允许额外的选项,但它有点恶心。 2,我可以使用会话参数并在构造函数中注入会话服务......但它看起来更像是一种解决方法而不是正确的方式......
您是否知道从FormType中的buildForm函数向表单注入参数的优雅方法?
最好的祝福
如果您需要将容器已知参数传递给自定义表单类型,您可以采用上面尝试的方式(显然通过参数注入)。但是,如果要将数据从控制器传递到表单类型,可以通过$options
(last)参数(在buildForm
中)传递:
FeatureForm
public function buildForm(FormBuilderInterface $builder, array $options)
// ....
$builder->add('tags', CollectionType::class, array (
'entry_type' => TagFeatureType::class,
'allow_add' => true,
'allow_delete' => true,
'data' => $datas,
'entry_options' => array(
'label' => false,
'some_custom_param' => $options['some_custom_param']
)
)
);
// ....
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => Task::class
));
$resolver->setRequired(array(
'some_custom_param'
));
}
然后,在TagFeatureType
应该配置选项:
TagFeatureType
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setRequired(array(
'some_custom_param'
));
}
最后,将它包含在buildForm
中:
public function buildForm(FormBuilderInterface $builder, array $options)
// ....
$someCustomParam = $options['some_custom_param'];
// ....
$builder->add('tag', EntityType::class, array(
'choice_label' => 'name',
'class' => 'AppBundle:Tag',
'query_builder' => function(TagRepository $tr) use ($someCustomParam)
{
return $tr->findObjectNotMine($someCustomParam);
}
);
// ....
}
明显的缺点是需要所有形式的路径有setRequired
或setDefault
。
希望这可以帮助...
在您的TagFeatureType中,您可以通过构造函数注入传递参数:
class TagFeatureType extends AbstractType
{
private $tagRepository;
public function __construct(TagRepository $tagRepository)
{
$this->tagRepository = $tagRepository;
}
public function buildForm(FormBuilderInterface $builder, array $options)
{
$tr = $this->tagRepository;
// ...
->add('tag', EntityType::class,
array(
'choice_label' => 'name',
'class' => 'AppBundle:Tag',
'query_builder' => function(TagRepository $tr){
return $tr->findObjectNotMine();
}
)
)
}
}
有一个DependencyInjectionExtension form extension将检查表单类型是否在服务容器中注册然后注入它,而不是创建一个新实例。这应该确保标记存储库存在。