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

Mon*_*ica 7 php forms formbuilder symfony

首先,我已经阅读了集合字段类型如何嵌入表单集合的文档......示例是关于一个与另一个实体(标签)具有一对多关系的实体(任务),我理解它,但我不能适应我想要的!

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

我想创建一个表单,在此表单中包含一个任务列表,每个任务在一行表中,显示task.title,task.startdate,task.user.name,task.user.company.name,task等信息. .project.name,它有2个字段可编辑,文本框"描述"复选框"活动".您可以使用主窗体中表格底部的一个按钮编辑多个任务并提交表单,因此基本上您应该能够在一个事务中更新多个记录(而不是每行创建一个表单和一个提交按钮,因此每次提交一次记录更新).

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

首先,我想按照示例在主窗体中嵌入一组表单,所以我为我的Task创建了一个Form Type,它应该像每行一个表单一样.我做了这些文件:

任务的表单类型:

// 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';
    }
}
Run Code Online (Sandbox Code Playgroud)

主表格的表格类型:

// 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';
    }
}
Run Code Online (Sandbox Code Playgroud)

任务表单控制器:

// 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(),
        ));
    }
}
Run Code Online (Sandbox Code Playgroud)

任务表格树枝模板(只是相关部分):

<div class="innerAll">
    {{ form_start(taskSaveForm) }}
    {{ form_errors(taskSaveForm) }}

    <table class="table table-bordered table-striped table-primary list-table">
        <thead>
            <tr>
                <th>Task ID</th>
                <th>Title</th>
                <th>Start Date</th>
                <th>User</th>
                <th>Company</th>
                <th>Project</th>
                <th>Description</th>
                <th>Active</th>
            </tr>
        </thead>
        <tbody>
            {% for task in taskSaveForm.tasksCollection %}

                <tr>
                    <td>{{ task.id }}</td>
                    <td><a href="https://localhost/taskid={{ task.id }}">{{ task.title }}</a></td>
                    <td>{{ task.startDate }}</td>
                    <td>{{ task.userName }}</td>
                    <td>{{ task.companyName }}</td>
                    <td>{{ task.projectName }}</td>
                    <td>{{ form_widget(task.description) }}</td>
                    <td>{{ form_widget(task.active) }}</td>
                    <td></td>
                </tr>
            {% endfor %}
        </tbody>
    </table>
    <div>{{ form_row(taskSaveForm.tasksSubmit) }}</div>
    {{ form_end(taskSaveForm) }}
</div>
Run Code Online (Sandbox Code Playgroud)

但是这里有一个问题,当我从查询构建器获得结果时,它是包含其中对象的数组的混乱,我得到一个错误

表单的视图数据应该是Acme\TaskBundle\Entity\Task类的实例,但它是一个(n)数组.您可以通过将"data_class"选项设置为null或通过添加将(n)数组转换为Acme\TaskBundle\Entity\Task实例的视图转换器来避免此错误.

这是查询:

createQueryBuilder()
    ->select(
            "
            task.id,
            task.title,
            task.startDate,
            task.description,
            user.name as userName,
            company.name as companyName,
            project.name as projectName,
            "
    )
        ->from('Acme\TaskBundle\Entity\Task', 'task')
        ->innerJoin('task.project', 'project')
        ->innerJoin('task.user', 'user')
        ->innerJoin('Acme\TaskBundle\Entity\Company', 'company', 'with', 'store.company = company')
        ->where('task.active = :isActive')->setParameter('isActive', true);
Run Code Online (Sandbox Code Playgroud)

Soooo,我使用Partial Objects指南来查看它是否可以帮助,它有助于在查询结果中创建任务对象,我可以提取它并将其发送到表单,但似乎其余的表单似乎没有意识到其余的对象...

好吧,也许我选择了错误的方法,我不确定!如果你对我该怎么做有任何建议,请在这里写一下...我已经挣扎了一个多星期了!在此先感谢您的时间!即使你没有任何说明,我也很感激你花时间阅读我很长的问题!谢谢!:)

geo*_*eoB 5

这是一个可能的解决方案的开始。下面的示例采用单个实体技能并在单个页面上显示所有这些。我不知道的是这种技术是否可以用于持久化子对象。我希望人们可以遍历返回的数据并根据需要保留。

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

在控制器中:

    $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(),...]
Run Code Online (Sandbox Code Playgroud)

在模板中:

{% 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 %}
Run Code Online (Sandbox Code Playgroud)