4

私はSymfony2フォームコンポーネントを使用してフォームを構築および検証しています。ここで、単一のフィールド値に基づいてバリデーターグループを設定する必要がありますが、残念ながら、そこにあるすべての例はエンティティに基づいているようです-いくつかの理由で使用していません。

例:タスクが空の場合、すべての制約バリデーターを削除する必要がありますが、そうでない場合は、デフォルトのバリデーターのセット(またはバリデーターグループ)を使用する必要があります。

言い換えると、私が達成しようとしているのは、サブフォームをオプションにすることですが、キーフィールドに値が入力されている場合でも検証されます。

誰かが私にそれを設定する方法の例を教えてもらえますか?

<?php
namespace CoreBundle\Form\Type;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormInterface;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Symfony\Component\Validator\Constraints as Assert;
use CoreBundle\Form\Type\ItemGroupOption;

class ItemGroup extends AbstractType
{

    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('title', 'text', array(
            'label' => 'Titel',
            'attr' => array('class' => 'span10 option_rename'),
            'required' => false
        ));
        $builder->add('max_selections', 'integer', array(
            'label' => 'Max tilvalg',
            'constraints' => array(new Assert\Type('int', array('groups' => array('TitleProvided')))),
            'attr' => array('data-default' => 0)
        ));
        $builder->add('allow_multiple', 'choice', array(
            'label' => 'Tillad flere valg',
            'constraints' => array(new Assert\Choice(array(0,1))),
            'choices'  => array(0 => 'Nej', 1 => 'Ja')
        ));
        $builder->add('enable_price', 'choice', array(
            'label' => 'Benyt pris',
            'constraints' => array(new Assert\Choice(array(0,1))),
            'choices'  => array(0 => 'Nej', 1 => 'Ja'),
            'attr' => array('class' => 'option_price')
        ));
        $builder->add('required', 'choice', array(
            'label' => 'Valg påkrævet',
            'constraints' => array(new Assert\Choice(array(0,1))),
            'choices'  => array(0 => 'Nej', 1 => 'Ja')
        ));
        $builder->add('options', 'collection', array(
            'type' => new ItemGroupOption(),
            'allow_add' => true,
            'allow_delete' => true,
            'by_reference' => false
            )
        );
        $builder->add('sort', 'hidden');
    }

    public function getName()
    {
        return 'item_group';
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        global $app;

        $resolver->setDefaults(array(
            'validation_groups' => function(FormInterface $form) use ($app) {

                // Get submitted data
                $data = $form->getData();

                if (count($app['validator']->validateValue($data['title'], new Assert\NotBlank())) == 0) {
                    return array('TitleProvided');
                } else {
                    return false;
                }
            },
        ));
    }
}
4

1 に答える 1

7

2.1を使用している場合は、「送信されたデータに基づくグループ」を確認することをお勧めします。

アップデート

Symfony Standard Editionで提供される/demo/contactデフォルトのデモ連絡先ページを使用した例:AcmeDemoBundle

条件付き検証グループを持つフォームタイプ:

namespace Acme\DemoBundle\Form;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\Form\FormInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Symfony\Component\Validator\Constraints as Assert;

class ContactType extends AbstractType
{
    // Inject the validator so we can use it in the closure
    /**
     * @var Validator
     */
    private $validator;

    /**
     * @param Validator $validator
     */
    public function __construct(Validator $validator)
    {
        $this->validator = $validator;
    }

    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('email', 'email');
        $builder->add('message', 'textarea', array(
            // Added a constraint that will be applied if an email is provided
            'constraints' => new Assert\NotBlank(array('groups' => array('EmailProvided'))),
        ));
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        // This is needed as the closure doesn't have access to $this
        $validator = $this->validator;

        $resolver->setDefaults(array(
            'validation_groups' => function(FormInterface $form) use ($validator) {
                // Get submitted data
                $data = $form->getData();
                $email = $data['email'];

                // If email field is filled it will not be blank
                // Then we add a validation group so we can also check message field
                if (count($validator->validateValue($email, new Assert\NotBlank())) == 0) {
                    return array('EmailProvided');
                }
            },
        ));
    }

    public function getName()
    {
        return 'contact';
    }
}

validatorフォームタイプでサービスを挿入することを忘れないでください:

<?php

namespace Acme\DemoBundle\Controller;

//...

class DemoController extends Controller
{
    // ...

    public function contactAction()
    {
        $form = $this->get('form.factory')->create(new ContactType($this->get('validator')));

        // ...
    }
}

ご覧のとおり、フィールドの検証は、フィールドが入力されmessageている場合にのみトリガーさemailれます。

差分ツールを使用して、違いをキャッチします。

于 2012-12-26T09:45:00.163 に答える