Php 如何在Symfony2中创建一个实体的多行表单

Php 如何在Symfony2中创建一个实体的多行表单,php,forms,symfony,formbuilder,Php,Forms,Symfony,Formbuilder,首先,我已经阅读了这两方面的文档,并且。。。这个例子是关于一个实体(任务),它与另一个实体(标记)有一对多的关系,我理解它,但我不能使它适应我想要的 为了更简单,假设我有一个任务实体,这个任务实体与其他对象(如用户和项目)有一些关系(每个任务可以有一个用户和一个项目) 我想制作一个表单,在这个表单中有一个任务列表,每个任务在一个表的一行中显示信息,如任务.title,任务.startdate,任务.user.name,任务.user.company.name,任务.project.name,它有

首先,我已经阅读了这两方面的文档,并且。。。这个例子是关于一个实体(任务),它与另一个实体(标记)有一对多的关系,我理解它,但我不能使它适应我想要的

为了更简单,假设我有一个任务实体,这个任务实体与其他对象(如用户和项目)有一些关系(每个任务可以有一个用户和一个项目)

我想制作一个表单,在这个表单中有一个任务列表,每个任务在一个表的一行中显示信息,如任务.title任务.startdate任务.user.name任务.user.company.name任务.project.name,它有两个可编辑的字段,文本框“Description”复选框“active”。您可以编辑多个任务,并使用主表单表格底部的一个按钮提交表单,因此基本上您应该能够在一个事务中更新多个记录(而不是每行制作一个表单和一个提交按钮,因此每次提交一次记录更新)

我对这个复杂的设计有很多问题:

首先,我想按照示例在主窗体中嵌入一个窗体集合,因此我为我的任务创建了一个窗体类型,该类型应类似于每行一个窗体。我创建了以下文件:

任务的表单类型:

// src/Acme/TaskBundle/Form/Type/TaskType.php
namespace Acme\TaskBundle\Form\Type;

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

class TaskType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
         $builder->add('description', 'text', ['label' => false, 'required' => false, 'attr' => ['placeholder' => 'description']]);
         $builder->add('active', 'checkbox', ['label' => false, 'required' => false, 'data' => true]);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'Acme\TaskBundle\Entity\Task',
        ));
    }

    public function getName()
    {
        return 'taskType';
    }
}
// src/Acme/TaskBundle/Form/Type/SaveTasksType.php
namespace Acme\TaskBundle\Form\Type;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Acme\TaskBundle\Form\Type\TaskType.php;

class SaveTasksType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('tasksCollection', 'collection', ['type' => new TaskType()]);
        $builder->add('tasksSubmit', 'submit', ['label' => 'Save']);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults([
            'attr' => ['class' => 'form-horizontal'],
            'method' => 'POST'
        ]);
    }

    public function getName()
    {
        return 'saveTasksType';
    }
}
// src/Acme/TaskBundle/Controller/ManageTasksController.php
namespace Acme\TaskBundle\Controller;

use Acme\TaskBundle\Entity\Task;
use Acme\TaskBundle\Form\Type\SaveTaskType;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;

class ManageTasksController extends Controller
{
    public function showListAction(Request $request)
    {
        $repository = $this->getDoctrine()->getRepository('ExampleBundle:Task');
        $tasks = $repository->findAll();

        $taskSaveForm = $this->createForm(new SaveTasksType(['tasks' => $tasks]));

        return $this->render('AcmeTaskBundle:Task:list.html.twig', array(
            'taskSaveForm' => $taskSaveForm->createView(),
        ));
    }
}
主窗体的窗体类型:

// src/Acme/TaskBundle/Form/Type/TaskType.php
namespace Acme\TaskBundle\Form\Type;

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

class TaskType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
         $builder->add('description', 'text', ['label' => false, 'required' => false, 'attr' => ['placeholder' => 'description']]);
         $builder->add('active', 'checkbox', ['label' => false, 'required' => false, 'data' => true]);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'Acme\TaskBundle\Entity\Task',
        ));
    }

    public function getName()
    {
        return 'taskType';
    }
}
// src/Acme/TaskBundle/Form/Type/SaveTasksType.php
namespace Acme\TaskBundle\Form\Type;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Acme\TaskBundle\Form\Type\TaskType.php;

class SaveTasksType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('tasksCollection', 'collection', ['type' => new TaskType()]);
        $builder->add('tasksSubmit', 'submit', ['label' => 'Save']);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults([
            'attr' => ['class' => 'form-horizontal'],
            'method' => 'POST'
        ]);
    }

    public function getName()
    {
        return 'saveTasksType';
    }
}
// src/Acme/TaskBundle/Controller/ManageTasksController.php
namespace Acme\TaskBundle\Controller;

use Acme\TaskBundle\Entity\Task;
use Acme\TaskBundle\Form\Type\SaveTaskType;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;

class ManageTasksController extends Controller
{
    public function showListAction(Request $request)
    {
        $repository = $this->getDoctrine()->getRepository('ExampleBundle:Task');
        $tasks = $repository->findAll();

        $taskSaveForm = $this->createForm(new SaveTasksType(['tasks' => $tasks]));

        return $this->render('AcmeTaskBundle:Task:list.html.twig', array(
            'taskSaveForm' => $taskSaveForm->createView(),
        ));
    }
}
任务表单控制器:

// src/Acme/TaskBundle/Form/Type/TaskType.php
namespace Acme\TaskBundle\Form\Type;

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

class TaskType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
         $builder->add('description', 'text', ['label' => false, 'required' => false, 'attr' => ['placeholder' => 'description']]);
         $builder->add('active', 'checkbox', ['label' => false, 'required' => false, 'data' => true]);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'Acme\TaskBundle\Entity\Task',
        ));
    }

    public function getName()
    {
        return 'taskType';
    }
}
// src/Acme/TaskBundle/Form/Type/SaveTasksType.php
namespace Acme\TaskBundle\Form\Type;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Acme\TaskBundle\Form\Type\TaskType.php;

class SaveTasksType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('tasksCollection', 'collection', ['type' => new TaskType()]);
        $builder->add('tasksSubmit', 'submit', ['label' => 'Save']);
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults([
            'attr' => ['class' => 'form-horizontal'],
            'method' => 'POST'
        ]);
    }

    public function getName()
    {
        return 'saveTasksType';
    }
}
// src/Acme/TaskBundle/Controller/ManageTasksController.php
namespace Acme\TaskBundle\Controller;

use Acme\TaskBundle\Entity\Task;
use Acme\TaskBundle\Form\Type\SaveTaskType;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;

class ManageTasksController extends Controller
{
    public function showListAction(Request $request)
    {
        $repository = $this->getDoctrine()->getRepository('ExampleBundle:Task');
        $tasks = $repository->findAll();

        $taskSaveForm = $this->createForm(new SaveTasksType(['tasks' => $tasks]));

        return $this->render('AcmeTaskBundle:Task:list.html.twig', array(
            'taskSaveForm' => $taskSaveForm->createView(),
        ));
    }
}
任务表单细枝模板(仅相关部分):

所以,我使用guide查看它是否有帮助,它有助于在查询结果中生成任务对象,我可以提取它并将其发送到表单,但表单的其余部分似乎不知道其余的对象


好吧,也许我选择了错误的方法,我不确定!如果你对我应该做什么有任何建议,请在这里写一个便条……我已经为此挣扎了一个多星期了!提前感谢你的时间!即使你没有写任何便条,我也感谢你花时间阅读我很长的问题!谢谢!)下面是一个可能解决方案的开始。下面的示例采用单个实体技能,并在单个页面上展示所有技能。我不知道的是,这种技术是否可以用于持久化子对象。我希望可以循环返回的数据,并根据需要持久化

下面的代码将生成一个页面,其中包含所有可能技能的列表,以及一个用于声明每个技能为已启用或已启用的复选框

在控制器中:

    $skills = $em->getRepository("TruckeeMatchingBundle:Skill")->getSkills();
    $formSkills = $this->createForm(new SkillsType(), array('skills' => $skills));
    ...
        if ($request->getMethod() == 'POST') {
            $formSkills->handleRequest($request);
            foreach ($skills as $existingSkill) {
                $em->persist($existingSkill);
            }
        }
    ...
    return ['formSkills' => $formSkills->createView(),...]
在模板中:

{% for skill in formSkills.skills %}
    {{ skill.vars.value.skill }}
            <input type="hidden" name="skills[skills][{{ loop.index0 }}][skill]" value="{{ skill.vars.value.skill }}">
            <input type="checkbox" name="skills[skills][{{ loop.index0 }}][enabled]" 
            {%if skill.vars.value.enabled %}checked="checked"{%endif%}
{% endfor %}
{%用于formSkills.skills%}
{{skill.vars.value.skill}

我使用了不同的策略。我的树枝文件与Monica的问题类似。但有一些非常有用的区别。下面是您的代码:

     {{ form_start(form) }}  
  {% for docente in docentes %}  
      Id: <input type="integer"  name="{{ docente.id }}" required="required"   style="width:30px" value="{{ docente.id }}" readonly> 
      Apellido: <input type="text"  name="{{ docente.apellido }}" required="required"  style="width: 80px" value="{{ docente.apellido }}" readonly>
       Nombres: <input type="text"  name="{{ docente.nombres }}" required="required"  style="width: 80px" value="{{ docente.nombres }}" readonly>
     Discrecional:  <input type="checkbox"  name="D{{ docente.id }}"  value="{{ docente.discrecional }}"  {% if docente.discrecional==1 %}checked{% endif %}> <br> 
      <br>
  {% endfor %}        
       <input type="submit" value="Grabar" />      
 {{ form_end(form) }} 
但寄存器的更新是通过使用update的查询在我的存储库文件中完成的工作,而不是在控制器文件中完成。为了避免不必要的查询和服务器过载,我只更新以前更改过的记录。在本例中,控制器中的以下行检查是否有更改在记录中(在我的例子中,我只是编辑一个名为“discrecional”的字段。如果该字段已更改,则调用查询并更新记录):

我的完整控制器文件如下:

public function discrecionalAction(Request $request)
{
    $em = $this->getDoctrine()->getManager();     
    $defaultData= $em->getRepository('BackendBundle:Docentes')->buscarDocentesActivos2();


// construimos un  formulario "vacío" sin campos definido
            $form = $this->createFormBuilder($defaultData);
        $form = $form->getForm();

            $form->handleRequest($request);    


            if ($form->isSubmitted() && $form->isValid()) {


            $i=0;
                foreach ($defaultData as $value) {
                     $data2= array('id' =>$request->request->get($defaultData[$i]['id']), 
                  'discrecional' =>$request->request->get('D'.$defaultData[$i]['id'])); 

                    if (($request->request->get('D'.$defaultData[$i]['id'])== '0' and $defaultData[$i]['discrecional']=='0') or
                        ($request->request->get('D'.$defaultData[$i]['id'])== NULL and $defaultData[$i]['discrecional']=='1'))    
                    {
                     $em->getRepository('BackendBundle:Docentes')->findDocenteFiltId2($data2);
                    }
                    $i=$i+1; 
                }

    return $this->redirectToRoute('docentes_discrecional');

               }        


      return $this->render('docentes/discrecional.html.twig', array(
                 'docentes' =>$defaultData,
        'form' => $form->createView() ));
}
我的第一个完整存储库查询如下:

public function buscarDocentesActivos2()
{


        $fields = array('d.id', 'd.apellido', 'd.nombres', 'd.discrecional');
$query = $this->getEntityManager()->createQueryBuilder();
        $query
            ->select($fields)
            ->from('BackendBundle:Docentes', 'd')
           ->where('d.activo=true')
           ->orderBy('d.apellido, d.nombres');     

        $consulta = $query->getQuery()->getResult();

        return $consulta;
    }
我使用更新功能完成的最终存储库查询如下:

public function findDocenteFiltId2($filtro)
{

 if (is_null($filtro['discrecional'])){ 
     $discrec= '0';
 };

 if ($filtro['discrecional']=='0'){ 
     $discrec= '1';
 };

 $em = $this->getEntityManager();

    $consulta = $em->createQuery('
        UPDATE BackendBundle:Docentes d
        SET d.discrecional = :disc
        WHERE d.id = :idver
    ');

   $consulta->setParameters(array(

    'idver' => $filtro['id'],
    'disc'  => $discrec,     
        ));
      return $consulta->getArrayResult();

}   

很抱歉,我没有返回给您!我的任务顺序已更改,因此我正在处理其他事项…我将返回给您,并让您知道它是否有效!我从您的答案中得到了一个想法,希望它能起作用!感谢您花时间发送答案…:)很抱歉检查您的答案,耽搁了很长时间!您的答案有帮助我太激动了,我要接受你的回答!:)我也会在以后更新我自己的问题,最终版本!不用担心延迟。很高兴它有帮助。