19

我用 Symfony2 FormBuilder 创建了一个表单,我想禁用编辑视图中的一个字段。我实际上是用包装器(display:none)隐藏它,但我想知道是否有更好的方法来做到这一点。我的代码如下所示:

实体类型

public function buildForm(FormBuilderInterface $builder, array $options) {
    $builder->add('fieldToDisabledInEditView');
    // ...

实体控制器

public function newAction() {
    $entity = new Entity;
    $form = $this->createForm(new EntityType, $entity);
    // ...
}
public function editAction() {
    $entity = new Entity;
    $form = $this->createForm(new EntityType, $entity);
    // ...
}

新(树枝)模板

<form>
    {{ form_row(form.fieldToDisabledInEditView) }}
    {# ... #}

编辑(树枝)模板

<form>
    <span class="theValueOfTheHiddenField">{{ entity.fieldToDisabledInEditView }}</span>
    <div style="display:none">
        {{ form_row(form.fieldToDisabledInEditView) }}
    </div>
    {# ... #}
4

5 回答 5

29

我想您会发现创建和编辑之间会有其他差异,尤其是验证组。由于您的控制器知道正在执行哪个操作,因此请考虑创建两个表单类型 EditEntity 和 CreateEntity,然后使用公共基础来最大程度地减少重复代码。@cheesemackfly 展示了如何向元素添加禁用属性。

但是当然你可能会觉得有两种形式对于这样一个简单的区别来说是一种浪费。在这种情况下,向您的类添加一个意图标志并将其设置在控制器中

class EntityType
{
    public function __construct($intention)
    {
        $this->intention = $intention;

     ...
    // Use $this->intention to tweak the form

    }
}

// controller
$form = $this->createForm(new EntityType('create'), $entity);
OR
$form = $this->createForm(new EntityType('edit'), $entity);

如果您真的想进入它,请使用 di 注入意图。

 // controller
 $formType = $this->get('entity.create.formtype');
 OR
 $formType = $this->get('entity.edit.formtype');

通过使用服务,您可以从一种表单类型开始,然后当您最终将其拆分为两个(您将这样做)时,您的控制器仍将像以前一样工作。

还有一件事,假设您使用不同的模板进行编辑/创建,您实际上可以直接在 twig 中设置 disabled 属性。所以根本没有代码更改。

{{ form_row(form.yourField, { 'attr':{'disabled':'disabled'} }) }}

==================================================== ====================== 更新:2016 年 3 月 3 日

万一有人偶然发现这一点,请注意 Symfony 3 不再支持让一个类实现多种表单类型。您基本上必须有单独的表单类型类,即使它们几乎相同。并且永远不要将实例数据添加到您的表单类型中。

于 2013-08-16T14:37:33.230 回答
16

这是您可以将事件订阅者用于表单类的典型情况。
在你的情况下,它应该是:

// src/Acme/DemoBundle/Form/EventListener/AddfieldToDisabledInEditViewSubscriber.php
namespace Acme\DemoBundle\Form\EventListener;

use Symfony\Component\Form\FormEvent;
use Symfony\Component\Form\FormEvents;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;

class AddfieldToDisabledInEditViewSubscriber implements EventSubscriberInterface
{
    public static function getSubscribedEvents()
    {
        // Tells the dispatcher that you want to listen on the form.pre_set_data
        // event and that the preSetData method should be called.
        return array(FormEvents::PRE_SET_DATA => 'preSetData');
    }

    public function preSetData(FormEvent $event)
    {
        $data = $event->getData();
        $form = $event->getForm();

        // check if the object is "new"
        // If you didn't pass any data to the form, the data is "null".
        // This should be considered a new object
        if (!$data || !$data->getId()) {
            $form->add('fieldToDisabledInEditView');
        }
        else
        {
            $form->add('fieldToDisabledInEditView', null, array('disabled' => true));
            //If PHP >= 5.4
            //$form->add('fieldToDisabledInEditView', null, ['disabled' => true]);
        }
    }
}

并在您的表单类型中:

// src/Acme/DemoBundle/Form/Type/EntityType.php
namespace Acme\DemoBundle\Form\Type;

// ...
use Acme\DemoBundle\Form\EventListener\AddfieldToDisabledInEditViewSubscriber;

class EntityType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('your_field');
        //...

        $builder->addEventSubscriber(new AddfieldToDisabledInEditViewSubscriber());
    }

    // ...
}

http://symfony.com/doc/current/cookbook/form/dynamic_form_modification.html

于 2013-08-16T14:14:45.863 回答
11

这种方法一点也不优雅,但我使用它是因为它很简单:

实体控制器

public function newAction() {
    $entity = new Entity;
    $form = $this->createForm(new EntityType, $entity);
    // ...
}
public function editAction() {
    $entity = new Entity;
    $form = $this->createForm(new EntityType, $entity);
    $form->remove('fieldToDisabledInEditView');    
    // ...
}
于 2014-09-18T17:43:15.953 回答
7

对于那些在 Symfony 3 中寻找解决方案而不创建单独的表单类型类(用于添加和编辑)并且不使用表单事件的人,您可以定义一个自定义选项并在创建时将其传递给表单:

我在表单类型类中创建了一个is_edit具有默认值的选项:false

public function configureOptions(OptionsResolver $resolver)
{
    $resolver->setDefaults(array(
        'data_class' => SomeEntity::class,
        'is_edit' => false
    ));
}

您可以使用同一类方法中的$options数组访问此选项:buildForm

public function buildForm(FormBuilderInterface $builder, array $options)
{
    $builder->add('someField', TextType::class, array(
        'disabled' => $options['is_edit']
    ))
}

最后,您可以通过在创建表单时传递默认值来覆盖它:

$someForm = $this->createForm(
    SomeEntityType::class,
    $someEntity,
    array('is_edit' => true)
);

https://symfony.com/doc/3.4/form/form_dependencies.html

于 2017-09-27T06:43:15.193 回答
0

新(树枝)模板在单独渲染表单字段时不要忘记 {{ form_row(form._token) }}

<form>
    {{ form_row(form.fieldToDisabledInEditView) }}
    {{ form_row(form.field2) }}
    {{ form_row(form.field3) }}
    {# ... #}
    {{ form_row(form._token) }}

编辑(树枝)模板只是不要渲染 {{ form_row(form.fieldToDisabledInEditView) }} 并且不要忘记令牌。

<form>
    {{ form_row(form.field2) }}
    {{ form_row(form.field3) }}
    {# ... #}
    {{ form_row(form._token) }}
于 2013-10-24T00:56:26.547 回答