7

我有一个用户注册表格,表格中只有用户名字段。在我的表单中,我希望只允许用户输入用户名。Nicename 将与注册时的用户名相同。

此表单绑定到用户实体,即在我的表单类型类中:

public function setDefaultOptions(OptionsResolverInterface $resolver)
{
    $resolver->setDefaults(array(
        'data_class' => 'Some\Bundle\Entity\User',
    ));
}

实体 User,它为 username 和 nicename 设置了 NotBlank 约束。

namespace Some\Bundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Constraints;

//...

class User
{
    //...

    /**
     * @var string $username
     *
     * @ORM\Column(name="user_login", type="string", length=60, unique=true)
     * @Constraints\NotBlank()
     */
    private $username;

    /**
     * @var string $nicename
     *
     * @ORM\Column(name="user_nicename", type="string", length=64)
     * @Constraints\NotBlank()
     */
    private $nicename;

    //...

但是,如果我构建一个只有用户名但没有 nicename 的表单,则在验证时,即$form->isValid()它无法验证。

为了绕过这个,我想出了以下内容:

namespace Some\Bundle\Controller;

use Symfony\Bundle\FrameworkBundle\Controller\Controller;
use Some\Bundle\Form\Type\RegisterType;
//...

class UserController extends Controller
{
   //...
public function registerAction()
{
    //...
    $request = $this->getRequest();
    $form = $this->createForm(new RegisterType());

    if ($request->getMethod() == 'POST') {
        // force set nicename to username.
        $registerFields = $request->request->get('register');
        $registerFields['nicename'] = $registerFields['username'];
        $request->request->set('register', $registerFields);

        $form->bind($request);

        if ($form->isValid()) {
            $user = $form->getData();
            //persist $user, etc...

在表单类型中,我将其添加到我的buildForm方法中:

$builder->add('nicename', 'hidden');

但是我觉得这很不优雅,给控制器留下了一些负担(从请求对象中提取,放入数据,然后将其放回请求对象,哎哟!),如果用户要检查隐藏字段,则可以看到生成的 HTML 的源代码。

无论如何,至少任何使用表单类型的控制器都不需要执行上述操作,同时保留实体约束?

我无法更改备份用户实体的表架构,并且我想保留 NotBlank 约束。


编辑:经过长时间的麻烦,我决定使用验证组并且它起作用了。

class User
{
//...

/**
 * @var string $username
 *
 * @ORM\Column(name="user_login", type="string", length=60, unique=true)
 * @Constraints\NotBlank(groups={"register", "edit"})
 */
private $username;

/**
 * @var string $nicename
 *
 * @ORM\Column(name="user_nicename", type="string", length=64)
 * @Constraints\NotBlank(groups={"edit"})
 */
private $nicename;

表格类型:

public function setDefaultOptions(OptionsResolverInterface $resolver)
{
    $resolver->setDefaults(array(
        'data_class' => 'Some\Bundle\Entity\User',
        'validation_groups' => array('register', 'Default')
    ));
}

'Default'是必需的,否则它会忽略我在表单类型方法中添加的所有其他约束buildForm......请注意,它区分大小写:'default'不起作用。

虽然,我发现这还不够(很抱歉我没有把它放在我原来的问题中),因为当我坚持时,我需要在我的控制器中这样做:

$user->setNicename($user->getUsername());

作为奖励,我通过添加表单事件订阅者将其从控制器移动到表单类型级别

在表单类型buildForm方法中:

$builder->addEventSubscriber(new RegisterPostBindListener($builder->getFormFactory()));

和 RegisterPostBindListener 类

<?php
namespace Some\Bundle\Form\EventListener;

use Symfony\Component\Form\Event\DataEvent;
use Symfony\Component\Form\FormFactoryInterface;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\Form\FormEvents;

class RegisterPostBindListener implements EventSubscriberInterface
{

public function __construct(FormFactoryInterface $factory)
{

}

public static function getSubscribedEvents()
{
    return array(FormEvents::POST_BIND => 'setNames');
}

public function setNames(DataEvent $event)
{
    $data = $event->getData();

    $data->setNicename($data->getUsername());
}
}
4

3 回答 3

2

我认为您应该使用验证组

于 2012-11-07T07:10:06.217 回答
0

在这种情况下,您应该使用回调断言来创建自定义验证规则。

于 2012-11-07T11:38:55.210 回答
0

在您的用户实体中,您可以判断哪个字段可以为空:

/**
 *@ORM\Column(type="string", length=100, nullable=TRUE)
 */
 protected $someVar;

这样你的视图控制器就不需要做任何事情了。

忘了提。您还可以定义一个 PrePersist 条件来初始化您的 nicename 变量:

// you need to first tell your User entity class it has LifeCycleCallBacks:

/**
 * @ORM\Entity()
 * @ORM\HasLifecycleCallbacks()
 */
class User
{

     ...

    /**
     *@ORM\PrePersist
     */
     public function cloneName()
     {
         $this->nicename = $this->username;
     }
}
于 2012-11-07T04:45:56.200 回答