notBlank 约束不适用于文件输入 - 具有 OneToMany 关系的表单 Symfony2

notBlank constraint not working for File input - Form with OneToMany relationship Symfony2

我在 ArticleImage 实体之间有一对多关系。我创建了文章的形式,如:

class ArticleType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder

            ->add('articleTitle','text',array('required' => false))
            ->add('articlePrice','number',array('required' => false))
            ->add('images', new ImageType())

        ;

    }
....
}

class ImageType extends AbstractType
{

    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
        ->add('file', 'file',array('required' => false))

        ;
    }
....
}

我在验证文件属性时遇到问题。我的目标是 return 当用户没有选择任何文件时显示一条错误消息(至少应将一张图像与一篇文章相关联)。我尝试断言 NotBlank 但它不起作用。

<?php
....

use Symfony\Component\Validator\Constraints as Assert;


class Image
{


 /**
     * Image file
     *
     * @var File
     * @Assert\NotBlank
     */
    private $file;


   ....
}

我在创建表单时没有 embed collections,因为项目要求 AJAX 在提交整个表单之前分别上传每张图片。换句话说,绑定到输入 change 事件的 javascript 事件侦听器会创建 AJAX 调用,该调用会在验证后上传图像(即在控制器中验证文件的大小、扩展名) ).但是当我发送整个表单时,即使没有选择文件,也只会验证其他字段并提交表单(我在 headers 中看到除文件一之外的所有 formData 元素)。

我想注意到当我在浏览器中检查元素时,表单呈现正确。

我花了很多时间试图解决这个问题,但无济于事,你的帮助是一种解脱。

编辑: 根据 Nawfal Serrar 先生的要求

AJAX 调用的执行方式如下:

$(':file').change(function(){

        var file = this.files[0];

        var url= $('.article-form').attr('data-iu-url');

        var formData = new FormData($('form')[0]);

        $.ajax({
            url: url,
            type: 'POST',
            success: completeHandler,
            data: formData,
        });

    });

url 包含关于以下配置的路由 image_upload

image_upload:
    pattern:  /{id}/image_upload
    defaults: { _controller: ShopManagementBundle:Image:uploads}
    requirements: { _method: post|put }

控制器:

   public function uploadsAction(Request $request, $id) {

            if ($request->isMethod('POST')) {
                $image = $request->files->get('articletype')['images']['file'];

                $status='success';
                $message='';
                $uploadedURL='';
                $the_id=0;


                if (($image instanceof UploadedFile) && ($image->getError() == 0)) {

                    if ($image->getSize() < 50000000000) {
                        $originalName = $image->getClientOriginalName();
                        $name_array = explode('.', $originalName);
                        $extension = $name_array[sizeof($name_array) - 1];
                        $valid_file_types = array('jpeg', 'jpg', 'bmp', 'png', 'gif');
                        if (in_array(strtolower($extension), $valid_file_types)) {

                             $imagee= new Image();

                             $em = $this->getDoctrine()->getManager();
                             $imagee->setFile($image);

                             $imagee->setSubDir('hg');
                             $imagee->upload();

                             $entity = $em->getRepository('ShopManagementBundle:Article')->find($id);
                             $imagee->setAricle($entity);

                             $uploadedURL= $imagee->getUploadDir(). DIRECTORY_SEPARATOR . $imagee->getSubDir(). DIRECTORY_SEPARATOR . $image->getBasename();


                            $em->persist($entity);
                            $em->persist($imagee);
                            $em->flush();
                            $the_id=$imagee->getId();

                        } else {
                            $status = "fail";
                            $message = "extension problem";
                        }
                    } else {
                        $status = "fail";
                        $message = "Image size too big";
                    }
                } else {
                    $status = "fail";
                    $message = "Error uploading";
                }

                return $this->render('ShopManagementBundle:Image:image_portion.html.twig', array(
                    'status'      => $status,
                    'message' => $message,
                    'uploadedURL' => $uploadedURL,
                    'image_id'=>$the_id,

                   ));
            }
            else
                return new Response('RE try uploading');
        }

如您所见,我没有在控制器中使用 isValid 进行验证,而是假设文件已发送,我使用 if-else 语句进行验证。

要正确处理文件上传,请遵循以下方法:

http://symfony.com/doc/current/cookbook/doctrine/file_uploads.html

它是官方的,然后在上传方法的 return 检查是否没有文件 return 你的错误信息否则继续你的上传。

毕竟我们在评论区说了这么多,我建议您使用@Assert\File 进行文件上传验证。它会阻止你的 if/else 堆叠。

既然你告诉我你要将上传的文件附加到已经存在的文章中,那么使用嵌入式表单集合就是我提到的解决方案。

由于您将检索文章,然后用它创建一个表单类型,所有图像都将链接到它。

count constraint 添加到您的图像列表中:

/**
 * @Assert\Count(
 *      min = "1",
 *      minMessage = "You must upload at least one image",
 * )
 */
private $images;

由于您将检索已链接到已上传图像实例的文章,因此验证不会引发错误。否则,用户将尝试提交没有图像的表单。验证约束将使表单无效。

可能还需要做一些其他的小事才能使其发挥作用,但它应该可以帮助您继续前进。