具有属性的一对多对一表格与Symfony 3/Doctrine

Jul*_*0sS 4 php postgresql relational-database symfony doctrine-orm

这是问题所在:

我有一个有3个班级的模型

  • person_job
  • 工作

一个人可以有多个工作,任何工作人员关系都可以有"date_start"属性,"date_end"和"comment".所以我用一个包含这些属性的jointable(person_job)构建了这个模型,并在2个manyToOne属性上创建了一个名为person和job的关系(用doctrine注释生成)

人物属性如下:

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

/**
* @var string
* @ORM\Column(name="firstname",type="string",length=255,nullable=true)
*/
private $firstname;
/**
* @var bool
* @ORM\Column(name="active", type="boolean")
*/
private $active;
Run Code Online (Sandbox Code Playgroud)

作业属性如下所示:

/**
* @var string
* @ORM\Column(name="name",type="string",length=255,nullable=false)
*/
private $name;
Run Code Online (Sandbox Code Playgroud)

person_job看起来像这样:

/**
* @ORM\ManyToOne(targetEntity="...\Person")
* @ORM\JoinColumn(nullable=false)
*/
private $person;

/**
* @ORM\ManyToOne(targetEntity="...\Job")
* @ORM\JoinColumn(nullable=false)
*/
private $job;

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

/**
* @var \DateTime
* @ORM\Column(name="startdate",type="datetime")
*/
private $datestart;

/**
* @var \DateTime
* @ORM\Column(name="enddate",type="datetime")
*/
private $dateend;
Run Code Online (Sandbox Code Playgroud)

现在我想为我的"人"构建一个表单,我可以在列表中选择作业,并添加(如果需要)date_start,date_end或与此作业相关的注释.我的FormBuilder对于"Person"看起来像这样:

$builder
  ->add('name')
  ->add('firstname')
  ->add('jobs',Job::class,array('label'=>'Job'));
Run Code Online (Sandbox Code Playgroud)

这失败了.我的Person类没有"jobs"属性..所以,我怎样才能实现这样的东西呢?我是否必须添加一个具有oneToMany关系的jobs属性,并使用"mappedBy"声明?

学说中的这些关系仍然让我感到困惑,我是symfony的新手,我在互联网上看了一下却找不到合适的解决方案/例子......

感谢您的阅读/帮助

Lum*_*men 6

你遇到了Symfony形式最难的问题之一.幸运的是,有一些很好的文档.让我总结一下重要的步骤.

你是对的:如果你想从Person的角度来操纵这种关系,那么实体Person需要知道它与PersonJob的关系.所以你需要添加一个属性:

// src/AppBundle/Entity/Person.php
/**
 * @ORM\OneToMany(targetEntity="PersonJob", mappedBy="person")
 */
private $personJobs;

public function __construct()
{
    $this->personJobs = new \Doctrine\Common\Collections\ArrayCollection();
}
Run Code Online (Sandbox Code Playgroud)

然后你将拥有表单类型

// src/AppBundle/Form/PersonType.php
$builder
    ->add('name')
    ->add('firstname')
    ->add('personJobs', CollectionType::class, array(
        'entry_type'   => PersonJobType::class,
        'allow_add' => true,
    )
;
Run Code Online (Sandbox Code Playgroud)

请注意personJobs字段的类型.由于一个人可以拥有许多PersonJobs,因此您需要一个可以处理集合的表单类型.这是内置的目的CollectionType(查看其文档!).您还需要表单类型PersonJobType,以便CollectionType知道如何构建子表单:

class PersonJobType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('comment')
            ->add('datestart', DateTimeType::class)
            ->add('dateend', DateTimeType::class)
            ->add('job') // requires Job::__toString() to be defined!
        ;
    }

    public function configureOptions(OptionsResolver $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'AppBundle\Entity\PersonJob'
        ));
    }
}
Run Code Online (Sandbox Code Playgroud)

出于调试目的,将控制器更改为

 public function testAction()
 {
    $person = new Person();
    $form = $this->createForm(PersonType::class, $person);
    $form->add('submit', SubmitType::class);

    $form->handleRequest($request);

    if ($form->isSubmitted() && $form->isValid()) {
        print '<pre>';
        var_dump($form->getData());
        die();
    }

    return $this->render('default/index.html.twig', [
        'form' => $form->createView(),
    ]);
}
Run Code Online (Sandbox Code Playgroud)

现在继续复制和粘贴,从树枝和JavaScript代码,添加和删除项目(你必须做出细微的变化,如更换form.emailsform.personJobs).

表格

表格看起来像

只是带有"添加另一个PersonJob"链接的Person表单:

第一步

添加PersonJob:

添加PersonJob

添加anothing PersonJob:

添加另一个PersonJob

收到的数据

提交表单并查看输出var_dump:

object(AppBundle\Entity\Person)#247 (5) {
  ["id":"AppBundle\Entity\Person":private]=>
  NULL
  ["name":"AppBundle\Entity\Person":private]=>
  string(12) "Charles Dude"
  ["firstName":"AppBundle\Entity\Person":private]=>
  string(7) "Charles"
  ["active":"AppBundle\Entity\Person":private]=>
  bool(true)
  ["personJobs":"AppBundle\Entity\Person":private]=>
  object(Doctrine\Common\Collections\ArrayCollection)#248 (1) {
    ["elements":"Doctrine\Common\Collections\ArrayCollection":private]=>
    array(2) {
      [0]=>
      object(AppBundle\Entity\PersonJob)#962 (6) {
        ["id":"AppBundle\Entity\PersonJob":private]=>
        NULL
        ["comment":"AppBundle\Entity\PersonJob":private]=>
        string(19) "Something important"
        ["datestart":"AppBundle\Entity\PersonJob":private]=> 
        object(DateTime)#1088 (3) { … }
        ["dateend": …] => …
        ["person":"AppBundle\Entity\PersonJob":private]=>
        NULL
        ["job":"AppBundle\Entity\PersonJob":private]=>
        object(AppBundle\Entity\Job)#1171 (2) {
          ["id":"AppBundle\Entity\Job":private]=>
          int(2)
          ["name":"AppBundle\Entity\Job":private]=>
          string(5) "Job 2"
        }
      }
      [1]=> …
  }
}
Run Code Online (Sandbox Code Playgroud)

还有两件事要做:

  1. person嵌套PersonJob实体的属性正确设置为新的(但尚未保留)Person.

  2. PersonJob通过调用$em->persist(…)它们来告诉Doctrine有关新实体的信息.

相关文件: