Symfony数组作为表单集合

时间:2018-04-26 00:07:27

标签: php symfony

我几天都在Symfony遇到问题,在谷歌上找不到任何东西。

我有两张桌子。

帐户:

AppBundle\Entity\Account:
type: entity
fields:
    name:
        type: string
        nullable: true
        length: 255
oneToMany:
    towns:
        targetEntity: AppBundle\Entity\Town
        mappedBy: owner
        cascade: ['persist']

镇:

AppBundle\Entity\Town:
type: entity
fields:
    name:
        type: string
        nullable: true
        length: 255
manyToOne:
    owner:
        targetEntity: AppBundle\Entity\Account
        inversedBy: towns

我有一个名字的数组:

$names = ['New York','Berlin'];

我想要一个表单,用户可以在其中检查数组中的名称(复选框)。当用户检查“纽约”并提交表单时,我想在Town字段中添加一个带有“纽约”的新实体name。如果用户取消选中“纽约”,我希望删除该实体。

到目前为止,我使用EntityTypeCollectionTypeChoiceType进行了尝试。我能得到的最好的是ChoiceType。

public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder
        ->add('towns', ChoiceType::class,
            [
                'choices' =>
                    [
                        new Town('New York'),
                        new Town('Berlin'),
                    ],
                'choice_label' => function (Town $town, $key, $index) {
                    return $town->getName();
                },
                'choice_value' => function (Town $town) {
                    return $town->getName();
                },
                'expanded' => TRUE,
                'multiple' => TRUE,
                'required' => FALSE,
            ]
        );
}

但每次用户提交任何已检查城镇的表单时,它都会添加一个新实体,并且不会删除未选中的城市......

1 个答案:

答案 0 :(得分:0)

不要使用数组,将城镇存储在数据库中 然后将towns字段类型设置为EntityType

->add(
    'towns',
    EntityType::class,
    [
        'class' => Town::class,
        'multiple' => true,
        'expanded' => true,
        'required' => false,
    ]
)

将此方法添加到Town类,以便在必须转换为字符串的任何位置自动显示名称:

/**
 * @return string
 */
public function __toString()
{
    return $this->name;
}

如果数组的想法是过滤显示的选项以选择城镇,您可以在query_builder选项中使用它:

public function buildForm(FormBuilderInterface $builder, array $options)
{
    $names = ['New York', 'Berlin']; // Or pass it from the controller in $options

    $builder
        ->add('towns',
            EntityType::class,
            [
                'class' => Town::class,
                'query_builder' => function (EntityRepository $er) use ($names) {
                    return $er->createQueryBuilder('town')
                        ->where('town.name IN (:names)')
                        ->setParameter('names', $names);
                },
                'multiple' => true,
                'expanded' => true,
                'required' => false,
            ]
        );
}