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



ArticleImage实体之间存在OneToMany关系。我创建了文章的形式,如:

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))
        ;
    }
....
}

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

<?php
....
use SymfonyComponentValidatorConstraints as Assert;

class Image
{

 /**
     * Image file
     *
     * @var File
     * @AssertNotBlank
     */
    private $file;

   ....
}

在创建表单时,我不嵌入集合,因为项目要求在提交整个表单之前,必须分别对每个图像进行AJAX上传。换句话说,绑定到输入change事件的javascript事件监听器创建AJAX调用,该调用在验证后上传图像(即,在控制器中验证文件的大小、扩展名)。但是,当我发送整个表单时,即使没有选择任何文件,也只验证其他字段并提交表单(我在页眉中看到除文件外的所有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语句进行验证。

在评论区说了这么多之后,我建议您使用@AssertFile进行文件上传验证。它将阻止if/else堆栈。

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

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

在您的:的图像列表上添加计数限制

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

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

也许还有其他小事情要做才能让它发挥作用,但它应该会帮助你继续前进。

为了更好地正确处理文件上传,请遵循以下方法:

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

这是官方的,然后在返回上传方法时,您检查是否没有文件返回您的错误消息,否则继续上传。

最新更新