我认为出现意外行为是因为您混淆了“选择”和“实体”表单字段类型的使用。
您正在指定一个实体字段($builder->add() 的第二个参数),然后尝试使用 'choices' 选项为其填充值。然而,'choices' 选项并不直接适用于Entity field type,尽管据说它继承自Choice。相反,实体字段旨在为您自动加载数据库中的选项。如果您仅设置实体“类”,则该字段将使用表中的所有实体按主键升序填充。为了加载实体子集和/或以特定顺序加载它们,您可以设置“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。
我怀疑在问题中创建项目字段的方式会导致下拉菜单的基础选择被设置两次 - 首先是在设置“类”选项时,对所有可用的项目实体,其次是当'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));
请注意,在这种情况下,“项目”字段的值将是项目 ID,而对于实体字段,它将是实际的项目实体,这也是最好使用实体字段的另一个原因。