表单实体:添加额外选项

Ale*_*sov 5 php symfony-forms symfony doctrine-orm

我使用 Symfony 表单来过滤一组数据。比如说,有一个“用户”下拉列表过滤日志记录表(每个记录可以与用户链接,但也可以不链接)。

换句话说,LogRecord <- many-to-one nullable -> User

问题是我希望它支持下拉列表中的 2 个选项:所有用户(禁用此字段的过滤)和“无用户”,它应该过滤字段 = NULL 的记录。但我不知道如何学习表单来区分这些情况,而不破坏表单验证过程。有任何想法吗?

public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder->setMethod('GET');

    $builder
        ->add('user', EntityType::class, array(
            'class' => 'AppBundle:User',
            'choice_label' => 'name',
            'placeholder' => 'All',
            'query_builder' => function (UserRepository $repo) {
                return $repo->findForOptionsQueryBuilder();
            },
            'label' => 'User',
            'required' => false,
        ));
}
Run Code Online (Sandbox Code Playgroud)

当前下拉列表:

  • 全部
  • 用户1
  • 用户2

所需的下拉菜单:

  • 全部
  • 无(空)
  • 用户1
  • 用户2

yce*_*uto 5

一种想法可以用ChoiceType它来代替。像这样的东西:

// all your users
$choices = $repo->findForOptionsQueryBuilder()->getQuery()->getResult();

// add None option
$choices = array_merge(['None' => 0], $choices);

$builder->add('user', ChoiceType::class, [
    'choices' => $choices,
    'placeholder' => 'All', // add All option to beginning
    'required' => false,
    'choice_label' => function ($value, $key) {
        return $value ?: $key;
    },
]);
Run Code Online (Sandbox Code Playgroud)

这应该呈现以下 HTML 输入:

<select id="form_user" name="form[user]">
    <option value="">All</option>
    <option value="0">None</option>
    <option value="1">User A</option>
    <option value="2">User B</option>
    <option value="3">User C</option>
</select>
Run Code Online (Sandbox Code Playgroud)

并且,在提交事件时:

  • If option "All" is selected then $form->get('user')->getData() is equal to null
  • If option "None" is selected then $form->get('user')->getData() is equal to 0
  • If option "User A" is selected then $form->get('user')->getData() is an instance of User

Thus you can distinguish these cases without breaking the form validation process.