У меня есть форма ProductType
, которая содержит Collection
из ProductImageType
(для целей загрузки). Через прототип я определяю вид и добавляю новые элементы с помощью Javascript.
Проблема заключается в том, что при нажатии Add
вид прототипа обнаруживается и отображается, но виджеты форм в ProductImageType
не отображаются вообще.
Вот соответствующая часть моего кода.
Что происходит, код внутри product/prototype.images.html.twig
отображается, за исключением поля формы FileType, если я добавлю другие поля в форму (поле сопоставления egnot только для целей тестирования) и попытаюсь отобразить их в этом представлении они не отображаются.
Очень похожий код работал в других проектах.
ProductType
class ProductType extends AbstractType
{
/**
* {@inheritdoc}
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder->add('category' , null, ['label' => "Category", 'attr' => ['placeholder' => "Seleccione una categoría"]])
->add('name' , null, ['label' => "Title"])
->add('description', null, ['label' => "Description", 'attr' => ['class' => 'product-description-edit-field']])
->add('images' , CollectionType::class, [
'label' => 'false',
'entry_type' => ProductImageType::class,
'allow_add' => true,
'allow_delete' => true,
// 'delete_empty' => true,
'prototype' => true,
'prototype_data' => New ProductImage
])
;
}/**
ProductImageType
class ProductImageType extends AbstractType
{
/**
* {@inheritdoc}
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('image', FileType::class, ['label' => "Upload a picture",
// 'data_class' => null,
'attr' => ['class' => ''],
]);
}
edit.html.twig
{# ... #}
{{ form_row(edit_form.images) }}
<ul id="edit-image-list" class="edit-image-list" data-prototype="{% filter escape %}
{{ include('product/prototype.images.html.twig', { 'form': edit_form.images.vars.prototype }) }}
{% endfilter %}">
{# iterate over each existing image and render its only field: name #}
{% for image in edit_form.images %}
<li>{{ form_row(image.image) }}</li>
{% endfor %}
</ul>
<div class="button-wrapper left">
<a href="javascript:void(0)" class="btn btn-sm btn-outline-success" onclick="addImageUploader()">
Add image
</a>
</div>
{# ... #}
<script type="text/javascript">
function addImageUploader()
{
// Get the data-prototype explained earlier
var prototype = $('#edit-image-list').data('prototype');
// get the new index
var index = $('#edit-image-list li').length;
// Replace '__name__' in the prototype's HTML to instead be a number based on how many items we have
var newForm = prototype.replace(/__name__/g, index);
$('#edit-image-list').append(newForm);
}
</script>
продукт / prototype.images.html.twig
<li>
<div class="row">
<div class="col-md-8">
{{ form_row(form.image, {'attr': {'class': 'form-control'}}) }}
</div>
<div class="col-md-4">
<a href="https://via.placeholder.com/150"><img src="https://via.placeholder.com/150"></a>
</div>
</div>
</li>
Товарный объект
/**
* @ORM\Entity
*/
class Product
{
/**
* @ORM\OneToMany(targetEntity="ProductImage", mappedBy="product", cascade={"persist", "remove"})
*/
private $images;
public function __construct()
{
$this->images = new \Doctrine\Common\Collections\ArrayCollection();
}
ProductImage Entity
/**
* @ORM\Entity
*/
class ProductImage
{
/**
* @ORM\Column(type="string", length=255, nullable=true)
* @Assert\Image()
*/
private $image;