Я использую Symfony 3.4.
У меня есть страны и вопросы сущностей.Затем у меня есть объект ответов, который связан как с странами, так и с вопросами.
В вопросе я определяю ожидаемый тип ответа (булево, строковое и т. Д.).
Для того, чтобы захватитьпоэтому в объекте ответов у меня есть несколько полей, соответствующих нескольким типам полей, которые я определил.
Я хотел бы отобразить одну форму для данной страны, чтобы редактировать ответы, соответствующие различным вопросам.
Для этого я использую форму collectionType из страны.
Но у меня есть две проблемы:
- как отобразить сами вопросы во встроенных формах
- как в каждом ответе я могу скрыть поля, которые не являются обязательными (не того типа, который соответствует вопросу).
То, что кажется очень простым при обработке одного вопроса, кажется неясным, когда приходится обрабатывать все вопросы/ ответы для данной страны.
Вот мой код:
организация страны
<?php
namespace cwt\psmdbBundle\Entity;
use APY\DataGridBundle\Grid\Mapping as GRID;
use Doctrine\ORM\Mapping as ORM;
/**
* countries
*
* @ORM\Table()
* @ORM\Entity(repositoryClass="cwt\psmdbBundle\Entity\Repository\countriesRepository")
*/
class countries
{
/**
* @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=100)
*/
private $name;
/**
* @ORM\OneToMany(targetEntity="cwt\psmdbBundle\Entity\CcdbServicesAnswers", mappedBy="country", cascade={"persist", "remove"})
*/
private $ccdbServicesAnswers;
организация вопросов:
<?php
namespace cwt\psmdbBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use APY\DataGridBundle\Grid\Mapping as GRID;
/**
* CcdbServicesQuestions
*
* @ORM\Table(name="ccdb_services_questions")
* @ORM\Entity(repositoryClass="cwt\psmdbBundle\Repository\CcdbServicesQuestionsRepository")
*/
class CcdbServicesQuestions
{
/**
* @var int
*
* @ORM\Column(name="id", type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @var string
*
* @ORM\Column(name="question", type="string", length=255)
*/
private $question;
/**
* @ORM\ManyToOne(targetEntity="cwt\psmdbBundle\Entity\CcdbServicesCategories")
* @ORM\JoinColumn(nullable=false)
* @GRID\Column(field="category.name", title="Category")
*/
private $category;
/**
* @ORM\ManyToOne(targetEntity="cwt\psmdbBundle\Entity\FieldTypes")
* @ORM\JoinColumn(nullable=false)
* @GRID\Column(field="fieldType.name", title="Field Type")
*/
private $fieldType;
/**
* @ORM\OneToMany(targetEntity="cwt\psmdbBundle\Entity\CcdbServicesAnswers", mappedBy="question", cascade={"persist", "remove"})
*/
private $ccdbServicesAnswers;
answers entity:
<?php
namespace cwt\psmdbBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use Gedmo\Mapping\Annotation as Gedmo;
/**
* CcdbServicesAnswers
*
* @ORM\Table(name="ccdb_services_answers")
* @ORM\Entity(repositoryClass="cwt\psmdbBundle\Repository\CcdbServicesAnswersRepository")
*/
class CcdbServicesAnswers
{
/**
* @var int
*
* @ORM\Column(name="id", type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @var string|null
*
* @ORM\Column(name="stringField", type="string", length=255, nullable=true)
*/
private $stringField;
/**
* @var string|null
*
* @ORM\Column(name="textField", type="text", nullable=true)
*/
private $textField;
/**
* @var bool|null
*
* @ORM\Column(name="booleanField", type="boolean", nullable=true)
*/
private $booleanField;
/**
* @var int|null
*
* @ORM\Column(name="integerField", type="integer", nullable=true)
*/
private $integerField;
/**
* @var float|null
*
* @ORM\Column(name="floatField", type="float", nullable=true)
*/
private $floatField;
/**
* @var float|null
*
* @ORM\Column(name="percentageField", type="float", nullable=true)
*/
private $percentageField;
/**
* @var string|null
*
* @ORM\Column(name="comment", type="string", length=255, nullable=true)
*/
private $comment;
/**
* @ORM\ManyToOne(targetEntity="countries", inversedBy="ccdbServicesAnswers")
* @ORM\JoinColumn(nullable=false)
*/
private $country;
/**
* @ORM\ManyToOne(targetEntity="cwt\psmdbBundle\Entity\CcdbServicesQuestions", inversedBy="ccdbServicesAnswers")
* @ORM\JoinColumn(nullable=false)
*/
private $question;
Вот форма моей страны:
<?php
namespace cwt\psmdbBundle\Form;
use cwt\psmdbBundle\Entity\countries;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolver;
use Symfony\Component\Form\Extension\Core\Type\CollectionType;
class countriesCCDBServicesAnswersType extends AbstractType
{
/**
* @param FormBuilderInterface $builder
* @param array $options
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('ccdbServicesAnswers', CollectionType::class, array(
'entry_type' => CcdbServicesAnswersType::class,
'entry_options' => array('label' => false),
))
;
}
/**
* @param OptionsResolver $resolver
*/
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => 'cwt\psmdbBundle\Entity\countries'
));
}
/**
* @return string
*/
public function getBlockPrefix()
{
return 'cwt_psmdbbundle_countries';
}
}
Вот форма моей Ответы:
<?php
namespace cwt\psmdbBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolver;
//use Symfony\Component\OptionsResolver\OptionsResolverInterface;
use Symfony\Component\Form\Extension\Core\Type\TextType;
class CcdbServicesAnswersType extends AbstractType
{
/**
* @param FormBuilderInterface $builder
* @param array $options
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('stringField')
->add('textField')
->add('booleanField')
->add('integerField')
->add('floatField')
->add('percentageField')
->add('comment')
// ->add('country')
// ->add('question')
;
}
/**
* @param OptionsResolver $resolver
*/
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => 'cwt\psmdbBundle\Entity\CcdbServicesAnswers'
));
}
/**
* @return string
*/
public function getName()
{
return 'cwt_psmdbbundle_ccdbservicesanswers';
}
}
На данный момент довольно простой контроллер (япытался получить доступ к полям формы, но я не знаю, что оттуда делать):
/**
* Show answers for a given country.
* @Route("/countries/{countryID}/edit", name="editCountryCcdbServicesAnswers", methods={"GET"})
* @Security("has_role('ROLE_USER')")
*
*/
public function editCountriesCcdbServicesAnswers($countryID)
{
// // Get categories
$em = $this->getDoctrine()->getManager();
$categories = $em->getRepository('psmdbBundle:CcdbServicesCategories')->findAll();
// $questions = $em->getRepository('psmdbBundle:CcdbServicesQuestions')->findAll();
// Get all answers for this country
$country = $em->getRepository('psmdbBundle:countries')->find($countryID);
$form = $this->createCountryEditForm($country);
foreach ($form as $field) {
$fieldClass = get_class($field);
if(get_class($field)=='Symfony\Component\Form\Form') {
foreach ($field as $answerForm) {
foreach ($answerForm as $answerField) {
}
}
}
}
return $this->render('psmdbBundle:ccdb_services_answers:edit_country.html.twig', array(
'categories' => $categories,
'form' => $form->createView(),
'entity' => $country,
// 'delete_form' => $deleteForm->createView(),
'show' => 'ccdbservicesanswers_show',
'cancel' => 'ccdbservicesanswers_show',
'title' => 'CCDB Services',
));
}
А вот шаблон (довольно бесполезный, поскольку он отображает все поля для каждого ответа и не отображает вопросы...):
{% extends 'psmdbBundle:templates:edit.html.twig' %}
{% block form_body -%}
<div class="col-lg-5">
{{ form(form) }}
</div>
{% endblock form_body %}
Одним из решений, о котором я думаю, было бы добавить поля вопросов в форму ответов, чтобы иметь возможность использовать их в теме формы.Но я понятия не имею, является ли это наилучшим подходом, и меня беспокоит производительность.
Другим решением было бы использование вызовов ajax в представлении шоу для создания формы на лету после нажатия кнопки «изменить».'по конкретному вопросу, но я не уверен в опыте пользователя!
Для информации существует более 100 вопросов.