如何在Symfony2/3中根据用户角色自定义表单字段?

13

根据请求表单的用户角色自定义表单有正确的方法吗?

我的场景非常简单:如果用户没有被授予ROLE_ADMIN,我需要隐藏一些字段。我尝试在Twig中避免字段显示,但是

  {% if is_granted('ROLE_ADMIN') %}
              {{form_row(form.field)}}
  {% endif %}

无法正常工作,因为表单构建器绕过了此检查。

Symfony版本:2.8.2

编辑

感谢@Rooneyl的建议,我找到了解决方案:

首先,您需要将“role”键添加到选项参数中。 因此,在configureOptions()中,$options ['role']始终为 ROLE_USER

/**
 * @param OptionsResolver $resolver
 */
public function configureOptions(OptionsResolver $resolver)
{
    $resolver->setDefaults(array(
        'data_class' => 'MyBundle\Entity\Ticket',
        'role' => 'ROLE_USER'
    ));
}

然后在控制器中,您需要传递 getRoles() 数组:

$user_roles = $this->getUser()->getRoles();
$form = $this->createForm('MyBundle\Form\TicketType', $ticket, array('role' => $user_roles));

已更新代码。 - Rooneyl
2个回答

19

你可以在你的表单中做到这一点。

为你的表单创建一个服务。

app.form.type.task:
    class: AppBundle\Form\FormType
    arguments: ["@security.authorization_checker"]
    tags:
        - { name: form.type }
在你的FormType中,添加一个构造函数以获取你的服务。
private $authorization;
public function __construct(AuthorizationChecker $authorizationChecker)
{
    $this->authorization = $authorizationChecker;
}

然后,在您的构建器中,您将能够检查用户权限。

$builder->add('foo');
if($this->authorization->isGranted('ROLE_ADMIN'))
{
   $builder->add('bar');
}

最后,您可以呈现您的表单。

{% if form.formbar is defined %}
    {{ form_row(form.formbar ) }}
{% endif %}
请注意,这意味着您的字段可能为空。因为您可能希望某些用户能看到其中一些字段,而其他用户则不能看到。
否则,您可以在实体构造方法中设置默认值,以确保如果用户不填写或无法填写,则值不会为空。

非常棒。谢谢。 - Diego Montero

7

您可以使用传递给表单构建器的选项来指定生成哪些元素。
这样,您就可以更改所执行的内容和验证(使用validation_groups)。
例如,您的控制器(假设roles是一个数组);
您的控制器;

$form = $this->createForm(new MyType(), $user, ['role' => $this->getUser()->getRoles()]);

并且您的表单:

<?php 
namespace AppBundle\Form\Entity;

use AppBundle\Entity\UserRepository;
use Symfony\Component\Form\AbstractType,
    Symfony\Component\Form\FormBuilderInterface,
    Symfony\Component\OptionsResolver\OptionsResolver;

class MyType extends AbstractType 
{
    /**
     * @param OptionsResolver $resolver
     */
    public function configureOptions(OptionsResolver $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'AppBundle\Entity\User',
            'validation_groups' => ['create'],
            'role' => ['ROLE_USER']
        ));
    }

    /**
     * @param FormBuilderInterface $builder
     * @param array $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        // dump($options['roles']);
        if (in_array('ROLE_ADMIN', $options['role'])) {
            // do as you want if admin
            $builder
                ->add('name', 'text');
        } else {
            $builder
                ->add('supername', 'text');
        }
    }

    /**
     * @return string
     */
    public function getName()
    {
        return 'appbundle_my_form';
    }

}

当我尝试使用选项参数时,Symfony返回此错误:选项"role"不存在。为什么? - sentenza
我认为 $options['role'] 应该是 getUser()->getRoles() - sentenza
代码可能不完整,我现在正在使用移动设备,无法进行测试。稍后我会运行它并提供完整的解决方案,但理论是正确的。 - Rooneyl
主要问题似乎与$options数组有关。无法添加role键。 - sentenza
我编辑了Rooneyl的代码。现在它使用role而不是roles。对我来说它起作用了。我已经测试过了。 - sentenza
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接