Пример того, как добиться желаемого, есть в Документация Symfony на Как встроить коллекцию форм.
Для вашего конкретного случая использования вы захотите создать UserCartsForm
и отдельный CartsForm
.
В UserCart
добавьте поле carts
как CollectionType
. Затем Symfony обработает поле как серию форм.
src/AppBundle/Form/UserCart.php
namespace AppBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type as FormType;
class UserCartsForm extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder->add('carts', FormType\CollectionType::class, [
'label' => false,
'entry_type' => CartsForm::class,
'entry_options' => array('label' => false),
]);
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults([
'data_class' => User::class,
]);
}
}
Добавьте поля, которые вы хотите редактировать в своей форме, в CartsForm.
src/AppBundle/Form/CartsForm.php
namespace AppBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\Extension\Core\Type as FormType;
class CartsForm extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder->add('quantity', FormType\IntegerType::class, [
'label' => false
//...
]);
//...
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults([
'data_class' => Cart::class,
]);
}
}
В вашем контроллере укажите объект пользователя как ваши UserCartsForm
данные.
src/AppBundle/Controller/DefaultController.php
namespace AppBundle\Controller;
use AppBundle\Form\UserCartsForm;
class DefaultController extends Controller
{
/**
* @Route('/user/{id}/carts')
*/
public function userCartsAction(Request $request, User $user)
{
$form = $this->createForm(UserCartsForm::class, $user);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
//... process entity
//$this->getDoctrine()->getManager()->flush();
return $this->redirectToRoute('some_route');
}
return $this->render('user_carts_form.html.twig', [
'form' => $form
]);
}
}
Затем вы сможете легко получить данные из своего шаблона ветки, чтобы отобразить их так, как вам хочется.
app/Resources/views/user_carts_form.html.twig
{% form_start(form) %}
<table>
<thead>
<tr>
<td>Name</td>
<td>Quantity</td>
<td></td>
</tr>
</thead>
<tbody>
{% for cart in form.carts %}
{% set cartEntity = cart.vars.data %}
<tr>
<td>{{ cartEntity.product.name }}</td>
<td>{{ form_widget(cart.quantity) }}</td>
<td><a class="button" href="{{ path('remove_cart_action', { id: cartEntity.id }) }}">Delete <icon/></a></td>
<tr>
{% endfor %}
</tbody>
</table>
<button type="submit">Submit</button>
{% form_end(form) %}
Обновление ограничений объектов
По умолчанию Symfony будет использовать все ограничения (Default
), назначенные объекту, при проверке вашей формы, в результате чего $form->isValid()
вернет false.
https://symfony.com/doc/3.4/validation/groups.html
Если группы не указаны, будут применены все ограничения, принадлежащие группе Default
.
Чтобы решить эту проблему, используйте группы проверки, чтобы разделить ограничения объектов и объявить желаемые группы в соответствующих формах.
Пример:
src/AppBundle/Entity/User.php
namespace AppBundle\Entity;
use Symfony\Component\Validator\Constraints as Assert;
/**
* @ORM\Entity
*/
class User
{
/**
* @Assert\NotBlank(groups={"registration"})
*/
private $username;
//...
}
Затем, чтобы использовать группу(ы) проверки в нужной форме, в этом случае RegistrationForm
вы объявляете нужную группу в AbstractTye::configureOptions
как одну из OptionsResolver:$defaults
.
src/AppBundle/Form/RegistrationForm.php
namespace AppBundle\Form;
use Symfony\Component\Form\AbstractType;
class RegistrationForm extends AbstractType
{
//...
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults([
'data_class' => User::class,
'validation_groups' => ['registration']
]);
}
}
Теперь ограничение User::NotBlank
будет применяться только к RegistrationForm::isValid()
или любой другой форме, объявляющей группу проверки регистрации.
person
Will B.
schedule
25.04.2018