Может ли инфраструктура Yii проверять массив полей в функции правила класса CFormModel

Я добавляю динамически текстовые поля к свойству в модели с именем names, но мне нравится устанавливать правила проверки для каждого имени в именах массивов в представлении. Можно ли это сделать, не используя правила проверки в модели CForm? Мне нравится подтверждать, что каждое имя будет необходимо после нажатия кнопки.

Это код представления с CActiveForm

<?php $form=$this->beginWidget('CActiveForm', array(
'id'=>'test',
'enableClientValidation'=>true,
'clientOptions'=>array(
    'validateOnSubmit'=>true,
),
)); ?>

<p class="note">Campos <span class="required">*</span> son obligatorios.</p>

<div class="row">
    <?php echo $form->labelEx($model,'Age'); ?>
    <?php echo $form->textField($model,'age'); ?>
    <?php echo $form->error($model,'age'); ?>
</div>

<div class="row">
    <?php echo $form->labelEx($model,'Name'); ?>
    <?php echo $form->passwordField($model,'names[0]'); ?>
    <?php echo $form->error($model,'names[0]'); ?>

</div>

<div class="row">
    <?php echo $form->labelEx($model,'Name'); ?>
    <?php echo $form->passwordField($model,'names[1]'); ?>
    <?php echo $form->error($model,'names[1]'); ?>

</div>

<div class="row buttons">
        <?php echo CHtml::submitButton('test'); ?>
    </div>

<?php $this->endWidget(); ?>

2 ответа

Вам нужно написать собственный валидатор, который предполагает, что проверяемый атрибут является массивом и применяет результат другого валидатора к каждому из его элементов.

Я использовал этот валидатор, который делает именно это как отправную точку; после небольшой очистки и максимально упрощения я достиг этого кода для проверки на стороне сервера:

protected function validateAttribute($object, $attribute)
{
    $value = $object->$attribute;

    if (!is_array($value) ) {
        $this->addError($object, $attribute, "TODO: error message");
        return;
    }
    else if ($this->isEmpty($value)) {
        if (!$this->allowEmpty) {
            $this->addError($object, $attribute, "TODO: error message");
        }

        return;
    }

    // $this->validator and $this->parameters are supposed to be
    // attributes of your custom validator class that you set from
    // inside rules().
    $validator = self::createValidator(
        $this->validator,
        $object,
        array($attribute),
        $this->parameters);

    $errors = array();

    // Iterate over $value, validating each item in turn.
    // Since $validator may be a filtering validator, we need to make
    // sure that any changes it makes to the validated item stick, so
    // we iterate by reference and do a bit of back and forth shuffling.
    foreach($value as $key => &$item) {
        $object->$attribute = $item; // the only way
        $validator->validateAttribute($object, $attribute);
        $item = $object->$attribute; // make any changes stick

        if ($object->hasErrors($attribute)) {
            $errors[$key] = $object->gerErrors($attribute);
            $object->clearErrors($attribute);
        }
    }

    unset($item); // always a good idea after foreach by reference
    $object->$attribute = $value; // undo the damage

    // And now decide what to do with $errors, which is per-item.
    // This should be good:
    foreach ($errors as $key => $error) {
         $object->addError("$attribute[$key]", $error);
    }
}

Хорошо, а как насчет проверки на стороне клиента? Кажется, что это должно быть возможно, но я не проверял это:

public function clientValidateAttribute($object,$attribute)
{
    // Since this is copy/pasted from above, it's an obvious candidate
    // for refactoring into a private method. I 'm keeping it simple.
    $validator = self::createValidator(
        $this->validator,
        $object,
        array($attribute),
        $this->parameters);

    $js = '';

    // No need for filtering support here (I think...)
    foreach($value as $key => $item) {
        $object->$attribute = $item;
        $js .= $validator->clientValidateAttribute($object, $attribute);
    }

    return $js;        
}

Сначала вам нужно указать отношение внутри модели.

'user' => array(self::BELONGS_TO, 'User', 'user_id'),

Затем в функции поиска используйте это отношение как:

$criteria->with=array('user');

И используйте это по вашему мнению, как:

array(
'name'=>'username_search',
'value'=>'$data->user->username',
'header'=>'Posted By',
),
Другие вопросы по тегам