gpt4 book ai didi

php - Symfony 表单中的 Fieldset 实现

转载 作者:搜寻专家 更新时间:2023-10-31 20:55:19 25 4
gpt4 key购买 nike

在我用 Symfony 编写的项目中,表单中经常会有字段集,所以我想创建一种机制,以便我可以按字段集对字段进行分组,并且仍然使用我的表单的 __toString() 方法。在 this page ,我阅读了有关 sfWidgetFormSchema 的信息,以及如何将其视为可以嵌套字段的小部件。所以这就是我所做的:我创建了嵌套字段:

   $this->setWidgets(array(
'customer' => new sfWidgetFormSchema(array(
'customer_name' => new sfWidgetFormInputText(),
'customer_email' => new sfWidgetFormInputText(array())
)),
'library' => new sfWidgetFormSchema(array(
'library_name' => new sfWidgetFormInputText(),
'library_address' => new sfWidgetFormInputText(),
'library_city' => new sfWidgetFormInputText(),
'library_postcode' => new sfWidgetFormInputText(),
'library_website' => new sfWidgetFormInputText()
)),
'message' => new sfWidgetFormTextarea(array(),array( "cols" => 50, "rows" => 10 )),
));

然后我创建了一个 fieldsetFormSchemaFormatter 类,它基本上将字段包装在标签中,并将其与 sfWidgetFormSchema 字段相关联:

foreach (array('customer', 'library') as $fieldset)
{
$this->widgetSchema[$fieldset]->addFormFormatter('tableless',
new tableLessFormSchemaFormatter($this->widgetSchema['customer']));
$this->widgetSchema[$fieldset]->setFormFormatterName('tableless');
$this->widgetSchema[$fieldset]->setNameFormat('%s');
}
$this->widgetSchema->addFormFormatter('fieldset',
new FieldsetFormSchemaFormatter($this->widgetSchema,
'TableLessFormSchemaFormatter'));
$this->widgetSchema->setFormFormatterName('fieldset');

它运行良好,我得到了我的字段集表单。我遇到的问题是验证,我在这个问题中较早链接的页面上根本没有描述。错误消息出现在所有字段的表单顶部,但“消息”字段除外,该字段后面紧跟着一条错误消息。我不认为我将能够在行之后立即显示错误消息并且仍然使用 echo $form 构造而不编写难看的代码,所以我想我将采用另一种实现方式。我认为 sfWidgetFormSchema 小部件旨在构建相互依赖的字段,这些字段将具有全局验证规则。

您将如何实现此字段集功能?

最佳答案

这是我经过一番研究后得出的结论,它似乎工作正常,但渲染时不再使用位置机制。我想知道这是否有问题。

<?php
class UcWidgetFormSchema extends sfWidgetFormSchema
{
/**
* An associative array with all the fieldsets
* <code>
* array(
* "fieldset1" => array("fieldName1", "fieldName2"),
* "fieldset2" => array("fieldName3", "fieldName4"),
* )
* </code>
*
* @var array
*/
private $fieldsets;

/**
* A fieldset-compatible constructor.
*
* @param mixed $fields Initial fields. Values can be given this way:
* <code>
* array(
* "fieldset1" => array(
* "field1" => $widget1,
* "field2" => $widget2
* )
* "fieldset1" => array(
* "field3" => $widget3,
* "field4" => $widget4,
* "field5" => $widget5
* )
* "message" => $widget6
* )
* </code>
* @param array $options An array of options
* @param array $attributes An array of default HTML attributes
* @param array $labels An array of HTML labels
* @param array $helps An array of help texts
*/
public function __construct($fields = null, $options = array(),
$attributes = array(), $labels = array(), $helps = array())
{
$this->addOption('name_format', '%s');
$this->addOption('form_formatter', null);

parent::__construct($options, $attributes);

if (is_array($fields))
{
$fieldsets = array();
foreach ($fields as $name => $value)
{
if (is_array($value))
{
$fieldsets[$name] = array_keys($value);
foreach ($value as $valueName=> $valueWidget)
{
$this[$valueName] = $valueWidget;
}
}
else
{
$this[$name] = $value;
}
}
$this->setFieldsets($fieldsets);
}
else if (null !== $fields)
{
throw new InvalidArgumentException('sfWidgetFormSchema constructor takes an array of sfWidget objects.');
}

$this->setLabels($labels);
$this->helps = $helps;
}

/**
* Setter for the fieldsets
*
* @param array $fieldsets an associative array
*
* @return null
*/
public function setFieldsets(array $fieldsets)
{
$fieldNames = array();
foreach ($fieldsets as $fieldset => $fieldsetFieldNames)
{
$fieldNames = array_merge($fieldNames, $fieldsetFieldNames);
}
$availableFieldsNames = array_keys($this->getFields());
if ($diff = array_diff(array_unique($fieldNames), $fieldNames))
{
throw new InvalidArgumentException(
'A field can only be used once in all fieldset. These do not: ' .
implode(', ', $diff));
}

if ($diff = array_diff($fieldNames, $availableFieldsNames))
{
throw new InvalidArgumentException(
'Widget schema does not include the following field(s): ' .
implode(', ', $diff));
}
$this->fieldsets = $fieldsets;
}

public function render($name, $values = array(), $attributes = array(), $errors = array())
{
if(!$this->getFormFormatter() instanceof FieldsettedFormFormatterInterface )
{
throw new LogicException('The formatter you are using must implement FieldsettedFormFormatterInterface');
}

if (null === $values)
{
$values = array();
}

if (!is_array($values) && !$values instanceof ArrayAccess)
{
throw new InvalidArgumentException('You must pass an array of values to render a widget schema');
}

$formFormat = $this->getFormFormatter();


$groups = array();
$hiddenRows = array();
$errorRows = array();
$lonelyFields = $this->getPositions();
$lonelyRows = array();

// render each field
foreach ($this->fieldsets as $fieldset => $fieldNames)
{
$rows = array();
foreach ($fieldNames as $name)
{
$lonelyFields = array_diff($lonelyFields, array($name));
$widget = $this[$name];
$value = isset($values[$name]) ? $values[$name] : null;
$error = isset($errors[$name]) ? $errors[$name] : array();
$widgetAttributes = isset($attributes[$name]) ? $attributes[$name] : array();

if ($widget instanceof sfWidgetForm && $widget->isHidden())
{
$hiddenRows[] = $this->renderField($name, $value, $widgetAttributes);
}
else
{
$field = $this->renderField($name, $value, $widgetAttributes, $error);

// don't add a label tag and errors if we embed a form schema
$label = $widget instanceof sfWidgetFormSchema ?
$this->getFormFormatter()->generateLabelName($name) :
$this->getFormFormatter()->generateLabel($name);
$error = $widget instanceof sfWidgetFormSchema ? array() : $error;

$rows[] = $formFormat->formatRow($label, $field, $error,
$this->getHelp($name));
}
$groups[$fieldset] = $rows;
}
}

foreach ($lonelyFields as $name)
{
$widget = $this[$name];
$value = isset($values[$name]) ? $values[$name] : null;
$error = isset($errors[$name]) ? $errors[$name] : array();
$widgetAttributes = isset($attributes[$name]) ? $attributes[$name] : array();

if ($widget instanceof sfWidgetForm && $widget->isHidden())
{
$hiddenRows[] = $this->renderField($name, $value, $widgetAttributes);
}
else
{
$field = $this->renderField($name, $value, $widgetAttributes, $error);

// don't add a label tag and errors if we embed a form schema
$label = $widget instanceof sfWidgetFormSchema ?
$this->getFormFormatter()->generateLabelName($name) :
$this->getFormFormatter()->generateLabel($name);
$error = $widget instanceof sfWidgetFormSchema ? array() : $error;

$lonelyRows[] = strtr($formFormat
->formatRow($label, $field, $error, $this->getHelp($name)),
array('%hidden_fields%' => ''));
}
}

$html = '';

if ($groups)
{
// insert hidden fields in the last row
$i = 0;
$maxGroup = count($groups);
foreach ($groups as $fieldset => $group)
{

for ($j = 0, $max = count($group); $j < $max; $j++)
{
$group[$j] = strtr($group[$j], array('%hidden_fields%' =>
(($i == $maxGroup -1) && $j == $max - 1) ?
implode("\n", $hiddenRows) : ''));
}

$html .= $this->getFormFormatter()
->formatFieldSet($fieldset, implode('', $group));
$i++;
}
}
else
{
// only hidden fields
$lonelyRows[] = implode("\n", $hiddenRows);
}
$html .= implode('', $lonelyRows);

return $this->getFormFormatter()
->formatErrorRow($this->getGlobalErrors($errors)) . $html;
}
}

如果你想使用它,这里是你的格式化程序必须实现的接口(interface):

interface FieldsettedFormFormatterInterface
{
/**
* This method will be used to render a fieldset
*
* @param string $name the name of the widget
* @param string $widgets the widgets html
*
* @return string the html for the fieldset
*/
public function formatFieldset($name, $widgets);
}

关于php - Symfony 表单中的 Fieldset 实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3306952/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com