在我们的Symfony2项目,我们有与嵌入表单形式非常复杂的结构....现在我们得把形式的输出在特定的顺序要求。
这里的问题是:我们使用form_widget(形式),现在我们正在寻找目标的解决方案(例如,通过注释)或formbuilder到特定的字段移动到表格末尾。 在symfony的1.4这是窗口小部件movefield()函数,我猜...
谢谢...
在我们的Symfony2项目,我们有与嵌入表单形式非常复杂的结构....现在我们得把形式的输出在特定的顺序要求。
这里的问题是:我们使用form_widget(形式),现在我们正在寻找目标的解决方案(例如,通过注释)或formbuilder到特定的字段移动到表格末尾。 在symfony的1.4这是窗口小部件movefield()函数,我猜...
谢谢...
您可以重新排序使用此包中的字段: https://github.com/egeloen/IvoryOrderedFormBundle
这允许你做这样的事情:
$builder
->add('g', 'text', array('position' => 'last'))
->add('a', 'text', array('position' => 'first'))
->add('c', 'text')
->add('f', 'text')
->add('e', 'text', array('position' => array('before' => 'f')))
->add('d', 'text', array('position' => array('after' => 'c')))
->add('b', 'text', array('position' => 'first'));
这是要在核心,但遭到了拒绝,拿出成捆。
有同样的问题,今天的表单元素排序。
结束了一个特点,这将覆盖finishView方法和一个FormView的子女财产重新排序项:
trait OrderedTrait
{
abstract function getFieldsOrder();
public function finishView(FormView $view, FormInterface $form, array $options)
{
/** @var FormView[] $fields */
$fields = [];
foreach ($this->getFieldsOrder() as $field) {
if ($view->offsetExists($field)) {
$fields[$field] = $view->offsetGet($field);
$view->offsetUnset($field);
}
}
$view->children = $fields + $view->children;
parent::finishView($view, $form, $options);
}
}
然后,在类型实现getFieldsOrder方法:
use OrderedTrait;
function getFieldsOrder()
{
return [
'first',
'second',
'next',
'etc.',
];
}
有没有必要“重新排序”字段。 所有你需要做的是调用form_label
和/或form_widget
单独为每个字段。 假设你用Twig
,你可以,例如,做:
<div>{{ form_label(form.firstName) }}</div>
<div>{{ form_widget(form.firstName) }}</div>
<div>{{ form_label(form.lastName) }}</div>
<div>{{ form_widget(form.lastName) }}</div>
这里是我想出了一个解决方案。
我创建了一个类,我的类型扩展。
namespace WF\CORE\CoreBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;
class WfBaseForm extends AbstractType
{
protected function useFields(FormBuilderInterface $builder, $fields)
{
foreach ($builder->all() as $field) {
if (!in_array($field->getName(), $fields))
$builder->remove($field->getName());
}
}
public function reorder(FormBuilderInterface $builder, $keys = array())
{
$fields = $builder->all();
$ordered_fields = array_merge(array_flip($keys), $fields);
$this->useFields($builder, array());
foreach($ordered_fields as $field)
$builder->add($field);
}
public function getName()
{
return 'base';
}
}
然后你就可以在你的继承类使用它。
class AddressType extends WfBaseForm
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
// I add as many fields as I need
$builder->add( '…');
}
此类扩展上面的一个
class ModifyAddressType extends BaseType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
parent::buildForm($builder, $options);
$builder->add('title', 'text', array('constraints' => array(new NotBlank())));
$this->reorder($builder, array('title', 'firstname', 'lastname'));
}
}
据我所知,您希望只使用在最终模板form_widget(形式)。
让我们假设我们有两个继承模型(MODELA,ModelB)和表单类型为他们(ModelAType,ModelBType)
class ModelA {
private $A;
private $B;
// Getters and setters
}
class ModelB extends ModelA {
private $C;
// Getters and setters
}
/**
* @DI\Service(id = "form.type.modelA")
* @DI\Tag("form.type", attributes={ "alias":"model_a_type" })
*/
class FormAType extends AbstractType {
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('A')
->add('B')
;
}
// getName and so on
}
/**
* @DI\Service(id = "form.type.modelA")
* @DI\Tag("form.type", attributes={ "alias":"model_b_type" })
*/
class FormAType extends AbstractType {
public function getParent() {
return "model_a_type";
}
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('C')
;
}
// getName and so on
}
如果渲染formB,你会得到A,B,C的顺序,但你要A,C,B。 为了实现这个目标,创建表单模板,并添加引用的应用程序。 配置文件:
#app/config/config.yml
twig:
....
form:
resources:
- YourAppBundle:Form:fields.html.twig
{# src/YourAppBundle/Resources/views/Form/fields.html.twig #}
{% block model_a_type_widget %}
{{ form_widget(form.A) }}
{{ form_widget(form.B) }}
{% endblock model_a_type_widget %}
{% block model_b_type_widget %}
{{ form_widget(form.A) }}
{{ form_widget(form.C) }}
{{ block('model_a_type_widget') }}
{% endblock model_b_type_widget %}
现在,当你渲染formB,你会看到希望的顺序,并保持代码的结构。 这是因为每个控件呈现只有一次,所以如果你调用父块之前渲染它,你会改变它们的顺序。
你可以这样做的权利在你的控制器,你渲染模板之前:
$form = ...
$view = $form->createView();
// Perform standard array operations to reorder: `$view->children`
// Here's a very simple example:
$firstField = $view->children['firstField'];
unset($view->children['firstField']);
$view->children['firstField'] = $firstField;
return array('form' => $view);
uasort
工程,以及,但usort
创建一个非关联数组这将打破形式呈现以后。
{{ form_start(form) }}
<div>{{ form_label(form.title) }}</div>
<div>{{ form_widget(form.title,{'id': 'blog_title'} )}}</div>
<div>{{ form_label(form.tag) }}</div>
<div>{{ form_widget(form.tag,{'id': 'blog_tag'} )}}</div>
<div>{{ form_label(form.category) }}</div>
<div>{{ form_widget(form.category,{'id': 'blog_category'} )}}</div>
<div>{{ form_label(form.image) }}</div>
<div>{{ form_widget(form.image,{'id': 'blog_image'} )}}</div>
<div>{{ form_label(form.body) }}</div>
<div>{{ form_widget(form.body,{'id': 'editor'} )}}</div>
<input type="submit" class="btn btn-success" value="Create" />
{{ form_end(form) }}
我有同样的问题,但解决它以不同的方式。 这里是我的解决方案,作为一个想法为别人谁正在寻找这个问题。
您可以添加在事件监听所有表单域,因为在这里你把所有的对象数据来决定字段顺序。 例如,你可以使用的方法从数据对象上的字段决定订购:
public function buildForm(FormBuilderInterface $builder, array $options)
{
// Don't use $builder->add(), or just for those fields which are always
// at the beginning of the form.
$builder->addEventListener(FormEvents::PRE_SET_DATA, function (FormEvent $event) {
$entry = $event->getData();
$form = $event->getForm();
// Now add all the fields in the order you need them to be, e.g by reading
// the needed order from the data entry.
if ($entry->isFieldAFirst()) {
$form->add(fieldA);
$form->add(fieldB);
} else {
$form->add(fieldB);
$form->add(fieldA);
}
}
}