表单提交后,子表单值在接收端显示为空
Subform values are showing up as null on the receiving end after form submit
我构建了一个提交良好的表单,但是当我在控制器中查看它们时,子表单值在接收端最终都为 null。
这是我的 UserProfileType
表格,基于 User
class。具体来说,我们正在查看的子表单是 subscriptionTier1
、subscriptionTier1
和 subscriptionTier1
:
class UserProfileType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('firstName', TextType::class)
->add('lastName', TextType::class)
->add('email', EmailType::class)
// etc... I'll keep out the unimportant fields
// here are the subforms whose values show up as null on the back end
->add('subscriptionTier1', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('subscriptionTier2', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('subscriptionTier3', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('save', SubmitType::class, [
'attr' => ['class' => 'save'],
])
;
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => User::class,
'mode' => null
)
);
}
}
这是我的 UserSubscriptionTierType
表单类型 class 的样子:
class UserSubscriptionTierType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('name', TextType::class, [
'attr' => [
'maxlength' => 25
]
])
->add('price', ChoiceType::class, [
'choices' => [
' per month' => 10,
' per month' => 20,
' per month' => 30
]
])
->add('description', TextareaType::class)
->add('messaging', CheckboxType::class, [
'required' => false
])
;
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => UserSubscriptionTier::class,
));
}
}
如下图所示,subform subscriptionTier1的name、price、description值都是在前端设置的:
下面是 subscriptionTier1
子表单的 twig 代码:
<div class="row justify-content-center w-100 ml-0 mr-0 mt-3 tier tier-1" >
<div class="col-lg-6 mt-1">
<div class="form-group form-control-lg w-100">
<label for="firstname" class="mb-2">Tier 1 Name</label>
{{ form_widget(form.subscriptionTier1.name, { 'attr': {'class': 'form-control'}}) }}
</div>
</div>
<div class="col-lg-6 mt-1">
<div class="form-group form-control-lg w-100">
<label for="tier1price" class="mb-2">Tier 1 Price</label>
{{ form_widget(form.subscriptionTier1.price, { 'attr': {'class': 'form-control'}}) }}
</div>
</div>
<div class="col-lg-12 mt-3">
<div class="form-group form-control-lg w-100">
<label for="bio" class="mb-2">Tier 1 Description
<sup class="text-danger">*</sup>
</label>
{{ form_widget(form.subscriptionTier1.description, { 'attr': {'class': 'form-control border-box', 'rows':'8'}}) }}
</div>
</div>
<div class="col-lg-12">
<div class="form-group form-control-lg w-100">
<div class="form-check">
<label class="form-check-label">
{{ form_widget(form.subscriptionTier1.messaging, { 'attr': {'class': 'form-control form-check-input'}}) }}
<span class="form-check-sign"></span>
Enable Messaging
</label>
</div>
</div>
</div>
</div>
这里是接收端的代码,控制器:
public function saveProfileAction(Request $request)
{
$user = $this->getUser();
$form = $this->createForm(UserProfileType::class, $user);
$form->handleRequest($request);
dump($form->get('subscriptionTier1')->getData());
所以在调试时,如果我只转储第一种形式subscriptionTier1
,你可以看到这些值都是null
。
ProfileController.php on line 269:
Form {#2235 ▼
-config: FormBuilder {#2236 ▶}
-parent: Form {#2112 ▶}
-children: OrderedHashMap {#2237 ▶}
-errors: []
-submitted: true
-clickedButton: null
-modelData: UserSubscriptionTier {#2280 ▼
-id: null
-posts: ArrayCollection {#2323 ▶}
-subscriptions: null
-name: null // Don't understand why this is null
-price: null // Don't understand why this is null
-description: null // Don't understand why this is null
-tierNumber: null
-versionNumber: null
-messaging: false
-user: null
+"subsciptions": ArrayCollection {#2089 ▶}
}
-normData: UserSubscriptionTier {#2280 ▶}
-viewData: UserSubscriptionTier {#2280 ▶}
-extraData: []
-transformationFailure: null
-defaultDataSet: true
-lockSetData: false
}
有人知道为什么这些值没有传递到后端(或被清零)吗?
由于UserProfileType
表单类型的data_class
选项设置为User
class,模型数据将是User
[=的一个实例27=] 并且由于 User
class 没有 subscriptionTier1
等字段,因此这些不会出现在您的模型数据中。
相反,您可以像这样在控制器中访问表单中未映射的字段:
$subscriptionTier1 = $form->get('subscriptionTier1')->getData();
文档here
编辑:您只能访问 subscriptionTier1 等的值,只有在处理完表单并且提交表单后,否则它将为空:
public function saveProfileAction(Request $request)
{
$user = $this->getUser();
$form = $this->createForm(UserProfileType::class, $user);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
$subscriptionTier1 = $form->get('subscriptionTier1')->getData();
dump($subscriptionTier1);
}
我不知道这是否只是一种解决方法,或者这实际上是最佳实践方法,但我是这样实现的:
我有以下子表单标签:
{{ form_start(form.subscriptionTier1) }}
etc...
{{ form_end(form.subscriptionTier1) }}
这将嵌套表单标签。显然你不能像这样嵌套表格。
因此取出所有 subscriptionTier1
、subscriptionTier1
、subscriptionTier1
的 {{ form_start(form.subscriptionTier1) }}
、{{ form_end(form.subscriptionTier1) }}
,然后检查请求中的表单值对象有效。
我构建了一个提交良好的表单,但是当我在控制器中查看它们时,子表单值在接收端最终都为 null。
这是我的 UserProfileType
表格,基于 User
class。具体来说,我们正在查看的子表单是 subscriptionTier1
、subscriptionTier1
和 subscriptionTier1
:
class UserProfileType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('firstName', TextType::class)
->add('lastName', TextType::class)
->add('email', EmailType::class)
// etc... I'll keep out the unimportant fields
// here are the subforms whose values show up as null on the back end
->add('subscriptionTier1', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('subscriptionTier2', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('subscriptionTier3', UserSubscriptionTierType::class, [
'required' => false,
'mapped' => false
])
->add('save', SubmitType::class, [
'attr' => ['class' => 'save'],
])
;
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => User::class,
'mode' => null
)
);
}
}
这是我的 UserSubscriptionTierType
表单类型 class 的样子:
class UserSubscriptionTierType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('name', TextType::class, [
'attr' => [
'maxlength' => 25
]
])
->add('price', ChoiceType::class, [
'choices' => [
' per month' => 10,
' per month' => 20,
' per month' => 30
]
])
->add('description', TextareaType::class)
->add('messaging', CheckboxType::class, [
'required' => false
])
;
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => UserSubscriptionTier::class,
));
}
}
如下图所示,subform subscriptionTier1的name、price、description值都是在前端设置的:
下面是 subscriptionTier1
子表单的 twig 代码:
<div class="row justify-content-center w-100 ml-0 mr-0 mt-3 tier tier-1" >
<div class="col-lg-6 mt-1">
<div class="form-group form-control-lg w-100">
<label for="firstname" class="mb-2">Tier 1 Name</label>
{{ form_widget(form.subscriptionTier1.name, { 'attr': {'class': 'form-control'}}) }}
</div>
</div>
<div class="col-lg-6 mt-1">
<div class="form-group form-control-lg w-100">
<label for="tier1price" class="mb-2">Tier 1 Price</label>
{{ form_widget(form.subscriptionTier1.price, { 'attr': {'class': 'form-control'}}) }}
</div>
</div>
<div class="col-lg-12 mt-3">
<div class="form-group form-control-lg w-100">
<label for="bio" class="mb-2">Tier 1 Description
<sup class="text-danger">*</sup>
</label>
{{ form_widget(form.subscriptionTier1.description, { 'attr': {'class': 'form-control border-box', 'rows':'8'}}) }}
</div>
</div>
<div class="col-lg-12">
<div class="form-group form-control-lg w-100">
<div class="form-check">
<label class="form-check-label">
{{ form_widget(form.subscriptionTier1.messaging, { 'attr': {'class': 'form-control form-check-input'}}) }}
<span class="form-check-sign"></span>
Enable Messaging
</label>
</div>
</div>
</div>
</div>
这里是接收端的代码,控制器:
public function saveProfileAction(Request $request)
{
$user = $this->getUser();
$form = $this->createForm(UserProfileType::class, $user);
$form->handleRequest($request);
dump($form->get('subscriptionTier1')->getData());
所以在调试时,如果我只转储第一种形式subscriptionTier1
,你可以看到这些值都是null
。
ProfileController.php on line 269:
Form {#2235 ▼
-config: FormBuilder {#2236 ▶}
-parent: Form {#2112 ▶}
-children: OrderedHashMap {#2237 ▶}
-errors: []
-submitted: true
-clickedButton: null
-modelData: UserSubscriptionTier {#2280 ▼
-id: null
-posts: ArrayCollection {#2323 ▶}
-subscriptions: null
-name: null // Don't understand why this is null
-price: null // Don't understand why this is null
-description: null // Don't understand why this is null
-tierNumber: null
-versionNumber: null
-messaging: false
-user: null
+"subsciptions": ArrayCollection {#2089 ▶}
}
-normData: UserSubscriptionTier {#2280 ▶}
-viewData: UserSubscriptionTier {#2280 ▶}
-extraData: []
-transformationFailure: null
-defaultDataSet: true
-lockSetData: false
}
有人知道为什么这些值没有传递到后端(或被清零)吗?
由于UserProfileType
表单类型的data_class
选项设置为User
class,模型数据将是User
[=的一个实例27=] 并且由于 User
class 没有 subscriptionTier1
等字段,因此这些不会出现在您的模型数据中。
相反,您可以像这样在控制器中访问表单中未映射的字段:
$subscriptionTier1 = $form->get('subscriptionTier1')->getData();
文档here
编辑:您只能访问 subscriptionTier1 等的值,只有在处理完表单并且提交表单后,否则它将为空:
public function saveProfileAction(Request $request)
{
$user = $this->getUser();
$form = $this->createForm(UserProfileType::class, $user);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
$subscriptionTier1 = $form->get('subscriptionTier1')->getData();
dump($subscriptionTier1);
}
我不知道这是否只是一种解决方法,或者这实际上是最佳实践方法,但我是这样实现的:
我有以下子表单标签:
{{ form_start(form.subscriptionTier1) }}
etc...
{{ form_end(form.subscriptionTier1) }}
这将嵌套表单标签。显然你不能像这样嵌套表格。
因此取出所有 subscriptionTier1
、subscriptionTier1
、subscriptionTier1
的 {{ form_start(form.subscriptionTier1) }}
、{{ form_end(form.subscriptionTier1) }}
,然后检查请求中的表单值对象有效。