我正在实现一个自定义表单类型,它提供一个自动完成字段来选择一个位置(国家、城市或地点)。表单类型创建两个字段,一个用于自动完成搜索输入的文本字段,一个用于保存所选位置的所选 ID 的隐藏字段。
在文本字段中输入时,会进行服务器调用并通过 jquery 自动完成显示结果。如果选择了一个位置,则所选位置的 id 将写入隐藏字段,而位置的名称将显示在文本字段中。在服务器上,我使用客户端转换器来查找隐藏字段传递的 id 的实体。文本字段被忽略。
我的模型类定义了一个带有属性的位置字段,以写回使用 NotNull 验证约束注释的位置实体。
到目前为止一切正常,但如果我不选择位置,验证消息“此值不应为空”。显示两次。
该捆绑包是公开的,可以在我的 github 存储库中找到。相关的类是LocationFieldType和LocationDataTransformer以及表单主题。
现在了解我如何将表单类型集成到我的项目中。我添加了整个代码,对不起质量;)
在模型中,我将属性定义如下:
class JourneyCreate
{
/**
* @Assert\NotNull()
* @Assert\Choice(choices = {"offer", "request"})
*/
public $type;
/**
* @Assert\NotNull()
* @Assert\Date()
*/
public $date;
/**
* @Assert\NotNull()
* @Assert\Time()
*/
public $time;
/**
* @Assert\NotNull()
*
*/
public $start;
/**
* @Assert\NotNull()
*
*/
public $destination;
public function buildJourney(User $owner)
{
switch($this->type)
{
case 'offer':
$journey = new JourneyOffer();
break;
case 'request':
$journey = new JourneyRequest();
break;
default:
throw new \InvalidArgumentException('Invalid journey type');
}
$journey->setDate($this->date);
$journey->setTime($this->time);
$journey->addStation(new JourneyStation($this->start));
$journey->addStation(new JourneyStation($this->destination));
$journey->setOwner($owner);
return $journey;
}
}
在主表单中,我添加如下字段:
class JourneyCreateType extends BaseType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('type','choice', array(
'choices' => array(
'offer' => 'Driver',
'request' => 'Passanger',
),
'empty_value'=>'',
'multiple' => false,
'expanded' => true,
))
->add('date','date',array(
'widget' => 'single_text',
'format' => $this->getDateFormat(\IntlDateFormatter::TRADITIONAL),
))
->add('time','time',array(
'widget' => 'single_text',
))
->add('start','room13_geo_location')
->add('destination','room13_geo_location')
;
}
public function setDefaultOptions(OptionsResolverInterface $resolver)
{
$resolver->setDefaults(array(
'data_class' => 'Acme\DemoBundle\Form\Model\JourneyCreate',
));
}
public function getName()
{
return 'journey_create';
}
}
和控制器代码:
/**
* @Route("/create/{type}", defaults={"type" = null})
* @Template()
*/
public function createAction($type=null)
{
if($type !== null && !in_array($type,array('request','offer')))
{
throw new NotFoundHttpException();
}
$journeyCreate = new JourneyCreate();
$journeyCreate->type = $type;
$form = $this->createForm(new JourneyCreateType(),$journeyCreate);
if($this->isPost())
{
$form->bind($this->getRequest());
if($form->isValid())
{
$journeyCreate = $form->getData();
$journey = $journeyCreate->buildJourney($this->getCurrentUser());
$this->persistAndFlush($journey);
return $this->redirect($this->generateUrl('acme_demo_journey_edit',array('id'=>$journey->getId())));
}
}
return array(
'form' => $form->createView(),
);
}
最后是显示表单的模板代码:
{% block page_body %}
<form class="form-horizontal" action="{{ path('acme_demo_journey_create') }}" method="post" novalidate>
{{form_widget(form)}}
<div class="form-actions">
<button class="btn btn-primary" type="submit">{{'form.submit'|trans}}</button>
<a href="{{path("acme_demo_journey_index")}}" class="btn">{{'form.cancel'|trans}}</a>
</div>
</form>
{% endblock %}
我有这样的理论,这可能是因为我使用了两个表单字段,但不知道如何解决这个问题。欢迎任何关于如何更优雅地解决这个问题的建议。