我的Symfony2项目中有一个实体类型表单字段。
$builder = $this->createFormBuilder();
$projects = $this->getProjects();
$builder->add('project', 'entity',
array(
'class' => 'MyBundle:Project',
'required' => false,
'choices' => $projects,
));
我遇到的问题是,当getProjects()
方法返回空结果集时,下拉列表将在项目表中全部项目。
有没有办法禁用此行为?
答案 0 :(得分:5)
我认为出现意外行为是因为您混淆了Choice和Entity表单字段类型的使用。
您正在指定实体字段($ builder-> add()的第二个参数),然后尝试使用'choices'选项用值填充它。但是,'choices'选项并不直接适用于Entity field type,尽管据说它继承自Choice。而是,Entity字段旨在为您自动加载数据库中的选项。如果仅设置实体“类”,则会按升序主键顺序使用表中的所有实体填充该字段。为了加载实体的子集和/或以特定顺序加载它们,您可以设置'query_builder'函数。
例如,要按升序名称顺序创建所有国家/地区的下拉列表:
$builder->add('country',
'entity',
array('class' => 'My\Bundle\Entity\Country',
'property' => 'name',
'query_builder' => function(EntityRepository $er) {
return $er->createQueryBuilder('country')
->orderBy('country.name', 'ASC');
},
'required' => true,
'empty_value' => false));
查询可以根据需要简单或复杂。请参阅Using Doctrine's Query Builder。
我怀疑在问题中创建项目字段的方式导致下拉列表的基础选择设置两次 - 首先设置'class'选项时,设置为所有可用的项目实体,其次是'choices'选项设置为$ this-> getProjects()的结果。据推测,如果后者是一个空数组,它不会覆盖前者,因此所有项目都会出现在列表中。
如果由于某种原因您无法使用查询构建器来获取下拉列表的项目,则可以使用“选择”字段类型并手动将项目数据映射到“选项”选项中。例如,像这样:
$builder = $this->createFormBuilder();
$projects = $this->getProjects();
$projectChoices = array();
foreach ($projects as $project) {
$key = $project->getId();
$value = $project->getName();
$projectChoices[$key] = $value;
}
$builder->add('project',
'choice',
array('choices' => $projectChoices,
'required' => false));
请注意,在这种情况下,'project'字段的值将是Project id,而对于Entity字段,它将是一个实际的Project实体,这是为什么最好使用Entity字段的另一个原因。