簡體   English   中英

Symfony2-上傳文件(編輯表單)

[英]Symfony2 - Upload File (Edit form)

我的問題是,當我使用編輯表單上傳文件時,如果實體的字段為空,則文件完美上傳,但是,如果實體的字段不為空(以前我已上傳文件),則引用該文件的字段上傳沒有更新,文件也沒有上傳。

下一個是我的實體關系:

  • 類文檔=>表,用於存儲名稱和路徑。
  • 類MyEntity(任何具有編輯操作的實體)=>具有引用Documents.id的int字段的實體

MyEntityController.php

public function editTeamAction($comp, $id) {
    if(!$id)
        throw $this->createNotFoundException('ID is needed, '. $id);
    $em = $this->getDoctrine()->getEntityManager();
    $entity = $em->getRepository('MyBundle:MyEntity')->find($id);
    if(!$entity)
        throw $this->createNotFoundException('Entity not found with id '. $id);
    $editForm = $this->createEditForm($entity);

    return $this->render("MyBundle::edit.html.twig", array("entity"=> $entity, "comp" => $comp, "edit_form"=>$editForm->createView(), "path_form" => "ent_update"));
}

private function createEditForm(MyEntity $entity) {
    $form = $this->createForm(new MyEntityType(), $entity);
    $form->add('submit', 'submit', array('label' => 'Update'));

    return $form;
}

public function updateTeamAction(Request $request, $comp, $id) {
    $em = $this->getDoctrine()->getEntityManager();
    $entity = $em->getRepository('MyBundle:MyEntity')->find($id);
    if(!$entity)
        throw $this->createNotFoundException('Entity not found with id '. $id);

    $editForm = $this->createEditForm($entity);
    $editForm->handleRequest($request);

    if($editForm->isValid())
    {
        $em->persist($entity);
        $em->flush();

        $this->get('session')->getFlashBag()->add(
            'notice',
            'Edit success.'
        );
        return $this->redirect($this->generateUrl('ent_edit', array('comp' => $comp, 'id' => $id)));
    }
    return $this->render("MyBundle::edit.html.twig", array("entity"=> $entity, "comp" => $comp, "edit_form"=>$editForm->createView(), "path_form" => "ent_update"));
}

MyEntity.php

namespace My\Bundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use My\Bundle\Entity\Documents;

/**
 * MyEntity
 *
 * @ORM\Table()
 * @ORM\Entity
 * @ORM\HasLifecycleCallbacks
 */
class MyEntity
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @var string
     *
     * @ORM\Column(name="name", type="string", length=255)
     */
    private $name;

    /**
     * @ORM\ManyToOne(targetEntity="Comp", inversedBy="myentity")
     * @ORM\JoinColumn(name="comp_id", referencedColumnName="id")
     */
    protected $Comp;

    /**
     * @var string
     *
     * @ORM\Column(name="web", type="string", length=255, nullable=true)
     */
    private $web;

    /**
     * @ORM\ManyToOne(targetEntity="Documents", cascade={"persist", "remove"}, inversedBy="myentity")
     * @ORM\JoinColumn(name="logo", referencedColumnName="id", nullable=true)
     */
    private $logo = null;

    /**
     * @ORM\Column(type="datetime")
     *
     * @var \DateTime
     */
    private $updatedAt;

    /**
     * @ORM\ManyToOne(targetEntity="Documents", inversedBy="myentity")
     * @ORM\JoinColumn(name="images", referencedColumnName="id", nullable=true)
     */
    protected $images = null;

    public function __construct()
    {
    }

    /**
     * Get id
     *
     * @return integer 
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * Set name
     *
     * @param string $name
     * @return MyEntity
     */
    public function setName($name)
    {
        $this->name = $name;

        return $this;
    }

    /**
     * Get name
     *
     * @return string 
     */
    public function getName()
    {
        return $this->name;
    }

    /**
     * Set comp
     *
     * @param \My\Bundle\Entity\Comp $comp
     * @return MyEntity
     */
    public function setComp(\My\Bundle\Entity\Comp $comp = null)
    {
        $this->comp = $comp;

        return $this;
    }

    /**
     * Get comp
     *
     * @return \My\Bundle\Entity\Comp 
     */
    public function getCompeticion()
    {
        return $this->competicion;
    }

    /**
     * Set web
     *
     * @param string $web
     * @return MyEntity
     */
    public function setWeb($web)
    {
        $this->web = $web;

        return $this;
    }

    /**
     * Get web
     *
     * @return string 
     */
    public function getWeb()
    {
        return $this->web;
    }

    /**
     * Set images
     *
     * @param \My\Bundle\Entity\Documents $images
     * @return MyEntity
     */
    public function setImages(\My\Bundle\Entity\Documents $images = null)
    {
        $this->images = $images;

        return $this;
    }

    /**
     * Get images
     *
     * @return \My\Bundle\Entity\Documents 
     */
    public function getImages()
    {
        return $this->images;
    }

    /**
     * Set updatedAt
     *
     * @param \DateTime $updatedAt
     * @return MyEntity
     */
    public function setUpdatedAt($updatedAt)
    {
        $this->updatedAt = $updatedAt;

        return $this;
    }

    /**
     * Get updatedAt
     *
     * @return \DateTime 
     */
    public function getUpdatedAt()
    {
        return $this->updatedAt;
    }

    /**
     * Set logo
     *
     * @param \My\Bundle\Entity\Documents $logo
     * @return MyEntity
     */
    public function setLogo(\My\Bundle\Entity\Documents $logo = null)
    {
        $this->logo = $logo;

        return $this;
    }

    /**
     * Get logo
     *
     * @return \My\Bundle\Entity\Documents 
     */
    public function getLogo()
    {
        return $this->logo;
    }
}

Documents.php

namespace My\Bundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Assert;
use Symfony\Component\HttpFoundation\File\UploadedFile;
use Symfony\Component\HttpFoundation\File\File;

/**
 * Documents
 *
 * @ORM\Table()
 * @ORM\Entity
 * @ORM\HasLifecycleCallbacks
 */
class Documents
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @var string
     *
     * @ORM\Column(name="name", type="string", length=255)
     * @Assert\NotBlank
     */
    private $name;

    /**
     * @var string
     *
     * @ORM\Column(name="path", type="string", length=255)
     */
    private $path;

    public $file;

    /**
     * @ORM\OneToMany(targetEntity="MyEntity", mappedBy="images")
     */
    protected $myentity;

    public function __construct($path = "")
    {
        $this->myentity = new \Doctrine\Common\Collections\ArrayCollection();
        $this->setPath($path);
    }

    /**
     * @ORM\PrePersist()
     * @ORM\PreUpdate()
     */
    public function preUpload()
    {
        $this->tempFile = $this->getAbsolutePath();
        $this->oldFile = $this->getPath();
        if(null !== $this->file)
        {
            $filename = sha1(uniqid(mt_rand(), true));
            $this->name = $filename.'.'.$this->file->guessExtension();
        }
    }

    /**
     * @ORM\PostPersist()
     * @ORM\PostUpdate()
     */
    public function upload()
    {
        if (null === $this->file)
            return;
        $this->preUpload($this->file);
        if(null !== $this->file)
        {
            $this->file->move($this->getUploadRootDir().$this->path, $this->name);
            unset($this->file);

            //if($this->oldFile != null)
            //     unlink($this->tempFile);    
        }
    }

    /**
     *  @ORM\PostLoad()
     */
    public function postLoad()
    {
        $this->updatedAt = new \DateTime('now');
    }

    /**
     * @ORM\PreRemove()
     */
    public function preRemoveUpload()
    {
        $this->tempFile = $this->getAbsolutePath();
    }

    /**
     * @ORM\PostRemove()
     */
    public function removeUpload()
    {
        // if($file = $this->getAbsolutePath())
        //     unlink($file);
        if(file_exists($this->tempFile))
            unlink($this->tempFile);
    }

    public function getWebPath()
    {
        return null === $this->path
            ? null
            : $this->getUploadDir().'/'.$this->path;
    }

    public function getAssetPath()
    {
        return $this->getUploadDir().$this->path;
    }

    /**
     * Called after entity persistence
     *
     * @ORM\PostPersist()
     * @ORM\PostUpdate()
     */

    protected function getUploadRootDir()
    {
        // the absolute directory path where uploaded
        // documents should be saved
        return __DIR__.'/../../../../web/'.$this->getUploadDir();
    }

    public function getAbsolutePath()
    {
        return null === $this->path ? null : $this->getUploadRootDir().$this->path;
    }

    protected function getUploadDir()
    {
        // get rid of the __DIR__ so it doesn't screw up
        // when displaying uploaded doc/image in the view.
        return 'uploads/';
    }

    /**
     * Get id
     *
     * @return integer 
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * Set name
     *
     * @param string $name
     * @return Documents
     */
    public function setName($name)
    {
        $this->name = $name;

        return $this;
    }

    /**
     * Get name
     *
     * @return string 
     */
    public function getName()
    {
        return $this->name;
    }

    /**
     * Set path
     *
     * @param string $path
     * @return Documents
     */
    public function setPath($path)
    {
        $this->path = $path;

        return $this;
    }

    /**
     * Get path
     *
     * @return string 
     */
    public function getPath()
    {
        return $this->path;
    }

    /**
     * Add myentity
     *
     * @param \My\Bundle\Entity\MyEntity $myentity
     * @return Documents
     */
    public function addMyentity(\My\Bundle\Entity\MyEntity $myentity)
    {
        $this->myentity[] = $myentity;

        return $this;
    }

    /**
     * Remove myentity
     *
     * @param \My\Bundle\Entity\MyEntity $myentity
     */
    public function removeMyentity(\My\Bundle\Entity\MyEntity $myentity)
    {
        $this->myentity->removeElement($myentity);
    }

    /**
     * Get myentity
     *
     * @return \Doctrine\Common\Collections\Collection 
     */
    public function getMyentity()
    {
        return $this->myentity;
    }

    /**
     * @return File
     */
    public function getFile()
    {
        return $this->file;
    }
}

MyEntityType.php

namespace My\Bundle\Form;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Doctrine\ORM\EntityRepository;

class MyEntityType extends AbstractType
{
    public function __construct()
    {
    }
    /**
     * @param FormBuilderInterface $builder
     * @param array $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('comp', "entity", array(
                "label"=>"Comp", 
                "class" => "MyBundle:Comp",
                "property" => "nameEn",
                "query_builder" => function(EntityRepository $er) {
                    return $er->createQueryBuilder("c")
                        ->orderBy("c.nameEn", "ASC");
                }))
            ->add('name', 'text')
            ->add('country', "entity", array(
                "label"=>"Country", 
                "class" => "CountryBundle:Country",
                "property" => "englishName",
                "query_builder" => function(EntityRepository $er) {
                    return $er->createQueryBuilder("p")
                        ->orderBy("p.englishName", "ASC");
                }))
            ->add('web', 'text', array("required" => false, "render_optional_text" => false))
            ->add('logo', new DocumentsType('logo_entity'), array('horizontal' => false, 'label' => 'Logo'))
        ;
    }

    /**
     * @param OptionsResolverInterface $resolver
     */
    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'My\Bundle\Entity\MyEntity'
        ));
    }

    /**
     * @return string
     */
    public function getName()
    {
        return '';
    }
}

DocumentsType.php

namespace My\Bundle\Form;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;

class DocumentsType extends AbstractType
{
    /**
     * @param FormBuilderInterface $builder
     * @param array $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('file', 'file', array('required' => false, 'label' => false))
        ;
    }

    /**
     * @param OptionsResolverInterface $resolver
     */
    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'My\Bundle\Entity\Documents'
        ));
    }

    /**
     * @return string
     */
    public function getName()
    {
        return 'documents';
    }
}

edit.html.twig

{% extends 'MyBundle::base.html.twig' %}

{% block stylesheets %}
    {{ parent() }}
{% endblock %}
{% block javascripts %}
    {{parent()}}
{% endblock %}
{% block wrapper %}
    {{parent()}}
    <h3>Edit</h3>
    {% for flashMessage in app.session.flashbag.get('notice') %}
        {{ flashMessage }}
    {% endfor %}
    <form action="{{ path(path_form, { 'id': entity.id, 'comp' : comp }) }}" method="POST" {{ form_enctype(edit_form) }}>
        {{ form(edit_form) }}
    </form>
{% endblock %}

您好,認為問題出在您的控制器(MyEntityController.php)中
表單驗證之后,在updateTeamAction中,您將使用以下代碼:

    if($editForm->isValid())
{
    $em->persist($entity);
    $em->flush();

    $this->get('session')->getFlashBag()->add(
        'notice',
        'Edit success.'
    );
    return $this->redirect($this->generateUrl('ent_edit', array('comp' => $comp, 'id' => $id)));
}  

但是通過這種方式,您將保留$ entity,而不更新它。 您需要通過以下方式從表單中獲取實體:

    if($editForm->isValid())
{
    $em->persist($editForm->getData());
    $em->flush();

    $this->get('session')->getFlashBag()->add(
        'notice',
        'Edit success.'
    );
    return $this->redirect($this->generateUrl('ent_edit', array('comp' => $comp, 'id' => $id)));
}  

嘗試這種方式,並給我們您的反饋意見

好吧,我找到了(最后!)解決該問題的方法。

Controller的UpdateAction必須檢查是否向表單發送了新文件。 如果不是,則實體必須設置setLogo(在我的情況下)是數據庫中的實際徽標。 如果存在一個新文件並且有效,那么我將創建一個新的Documents對象,並使用已上傳的文件設置File。 最后,我從服務器上刪除了以前的徽標(這是可選的)。

代碼的updateAction:

MyEntityController.php

public function updateTeamAction(Request $request, $comp, $id) {
    $em = $this->getDoctrine()->getEntityManager();
    $entity = $em->getRepository('MyBundle:MyEntity')->find($id);
    if(!$entity)
        throw $this->createNotFoundException('Entity not found with id '. $id);

    $editForm = $this->createEditForm($entity);
    //Save original Logo
    $logoOriginal = $entity->getLogo();
    $editForm->handleRequest($request);
    if($editForm->isValid())
    {
        $files = $request->files;
        $uploadedFile = $files->get('logo')['file'];
        if(count($uploadedFile)==0)
            $entity->setLogo($logoOriginal);
        else
        {
            $documents = new Documents();
            $documents->setPath('logo');
            $documents->setFile($uploadedFile);

            $entity->setLogo($documents);
            if(!empty($logoOriginal))
            {
                $fs = new Filesystem();
                $fs->remove($documents->getAbsolutePath().'/'.$logoOriginal->getName());
            }
        }
        $em->persist($entity);
        $em->flush();
        $this->get('session')->getFlashBag()->add(
            'notice',
            'Edit success.'
        );
        return $this->redirect($this->generateUrl('ent_edit', array('comp' => $comp, 'id' => $id)));
    }
    return $this->render("MyBundle::edit.html.twig", array("entity"=> $entity, "comp" => $comp, "edit_form"=>$editForm->createView(), "path_form" => "ent_update"));
}

如果要刪除該步驟,請向控制器添加以下內容:

use Symfony\Component\Filesystem\Filesystem;

最后,我意識到我創建的新文件的名稱存在問題(與服務器文件夾中創建的數據庫中存儲的文件不同)。 在Documents實體中,我修改了upload()函數,刪除了這一行

$這個 - > preUpload($這個 - >文件);

因此,下一個upload()函數的最終代碼是:

Documents.php

/**
 * @ORM\PostPersist()
 * @ORM\PostUpdate()
 */
public function upload()
{
    if (null === $this->file)
        return;
    if(null !== $this->file)
    {
        $this->file->move($this->getUploadRootDir().$this->path, $this->name);
        unset($this->file);
        //if($this->oldFile != null)
        //     unlink($this->tempFile);    
    }
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM