14

我们如何验证作为数组的表单字段?看看下面的代码

用户电话型号:

 public static $rules= array(
    'phonenumber'=>'required|numeric',
    'isPrimary'=>'in:0,1'
)
...........

用户控制器:

$validation = UserPhone::validate(Input::only('phonenumber')));


    if($validation->passes())
      {
         $allInputs = Input::only('phonenumber','tid');
         $loopSize = sizeOf($allInputs);

         for($i=0;$i<$loopSize;$i++)
         {

         $phone = UserPhone::find($allInputs['tid'][$i]);
         $phone->phonenumber = $allInputs['phonenumber'][$i];
         $phone->save();

        }

     return Redirect::to('myprofile')->with('message','Update OK');

  }
  else
  {
     return Redirect::to('editPhone')->withErrors($validation);

  } 

}

$validation来自扩展 Eloquent 的 BaseModel 。

在我看来:

 <?php $counter=1; ?>
          @foreach($phones as $thephone)

           <section class="col col-12">
              <label class="label">Phone Number {{$counter++}}</label>
              <label class="input">
              <i class="icon-append icon-phone"></i>
                 {{Form::text('phonenumber[]',$thephone->phonenumber)}}
                 {{Form::hidden('tid[]',$thephone->id)}}
              </label>
            </section>
          @endforeach

一切正常,我在更新表单中获得了我想要的所有电话号码,但我无法更新模型,因为验证失败并显示消息“电话号码必须是数字”。

我知道验证数组表单字段没有简单的解决方案,我尝试扩展验证器类但没有成功。

如何验证此类字段?

4

5 回答 5

25

这是我使用的解决方案:

用法

只需通过添加前缀来转换您的常用规则each。例如:

'names' => 'required|array|each:exists,users,name'

请注意,该each规则假定您的字段是一个数组,因此请不要忘记使用array之前的规则,如此处所示。

错误信息

错误消息将由str_singular()您的字段的单数形式(使用 Laravel 的助手)自动计算。在前面的示例中,属性是name

嵌套数组

此方法开箱即用,适用于点表示法中任意深度的嵌套数组。例如,这有效:

'members.names' => 'required|array|each:exists,users,name'

同样,此处用于错误消息的属性将是name.

自定义规则

此方法支持您开箱即用的任何自定义规则。

执行

1.扩展验证器类

class ExtendedValidator extends Illuminate\Validation\Validator {

    public function validateEach($attribute, $value, $parameters)
    {
        // Transform the each rule
        // For example, `each:exists,users,name` becomes `exists:users,name`
        $ruleName = array_shift($parameters);
        $rule = $ruleName.(count($parameters) > 0 ? ':'.implode(',', $parameters) : '');

        foreach ($value as $arrayKey => $arrayValue)
        {
            $this->validate($attribute.'.'.$arrayKey, $rule);
        }

        // Always return true, since the errors occur for individual elements.
        return true;
    }

    protected function getAttribute($attribute)
    {
        // Get the second to last segment in singular form for arrays.
        // For example, `group.names.0` becomes `name`.
        if (str_contains($attribute, '.'))
        {
            $segments = explode('.', $attribute);

            $attribute = str_singular($segments[count($segments) - 2]);
        }

        return parent::getAttribute($attribute);
    }
}

2. 注册您的验证器扩展

在您通常的引导位置中的任何位置,添加以下代码:

Validator::resolver(function($translator, $data, $rules, $messages)
{
    return new ExtendedValidator($translator, $data, $rules, $messages);
});

就是这样!享受!

奖励:数组的大小规则

正如评论指出的那样,似乎没有简单的方法来验证数组大小。但是,Laravel 文档缺少大小规则:它没有提到它可以计算数组元素。这意味着您实际上可以使用sizemin和规则maxbetween计算数组元素。

于 2014-10-23T21:14:53.480 回答
9

最好扩展 Validator 类并重用现有的 Validator 函数:

Validator::resolver(function($translator, $data, $rules, $messages)
{

    return new Validation($translator, $data, $rules, $messages);

});

class Validation extends Illuminate\Validation\Validator {

    /**
     * Magically adds validation methods. Normally the Laravel Validation methods
     * only support single values to be validated like 'numeric', 'alpha', etc.
     * Here we copy those methods to work also for arrays, so we can validate
     * if a value is OR an array contains only 'numeric', 'alpha', etc. values.
     *
     * $rules = array(
     *     'row_id' => 'required|integerOrArray', // "row_id" must be an integer OR an array containing only integer values
     *     'type'   => 'inOrArray:foo,bar' // "type" must be 'foo' or 'bar' OR an array containing nothing but those values
     * );
     *
     * @param string $method Name of the validation to perform e.g. 'numeric', 'alpha', etc.
     * @param array $parameters Contains the value to be validated, as well as additional validation information e.g. min:?, max:?, etc.
     */
    public function __call($method, $parameters)
    {

        // Convert method name to its non-array counterpart (e.g. validateNumericArray converts to validateNumeric)
        if (substr($method, -7) === 'OrArray')
            $method = substr($method, 0, -7);

        // Call original method when we are dealing with a single value only, instead of an array
        if (! is_array($parameters[1]))
            return call_user_func_array(array($this, $method), $parameters);

        $success = true;
        foreach ($parameters[1] as $value) {
            $parameters[1] = $value;
            $success &= call_user_func_array(array($this, $method), $parameters);
        }

        return $success;

    }

    /**
     * All ...OrArray validation functions can use their non-array error message counterparts
     *
     * @param mixed $attribute The value under validation
     * @param string $rule Validation rule
     */
    protected function getMessage($attribute, $rule)
    {

        if (substr($rule, -7) === 'OrArray')
            $rule = substr($rule, 0, -7);

        return parent::getMessage($attribute, $rule);

    }
}
于 2013-08-10T15:47:09.273 回答
6

每个()

它不在文档中,但 4.2 分支可能在第 220 行有一个简单的解决方案。

就像sometimes($attribute, $rules, callable $callback)函数一样,现在有一个each($attribute, $rules)函数。

要使用它,代码将比sometimes()调用更简单:

$v->each('array_attribute',array('rule','anotherRule')); //$v is your validator

注意事项

  • sometimes()并且each()似乎不容易相互链接,因此,如果您想对数组值执行专门的条件规则,那么现在最好使用其他答案中的魔术解决方案。
  • each()只深一层,与其他解决方案没有什么不同。魔术解决方案的好处是,它们会根据需要通过调用基本规则来达到 0 或 1 级深度,所以我想如果你想深入 1 到 2 级,你可以简单地通过调用each()和来合并这两种方法从其他答案中传递一条神奇的规则。
  • each()只需要一个属性,而不是一个属性数组sometimes(),但是添加这个特性each()不会对each()函数产生巨大的变化——只需循环遍历$attributeandarray_merge() $dataarray_get()结果。如果有人认为它是可取的并且尚未完成,那么有人可以将它作为对 master 的拉取请求,我们可以看看它是否会成为未来的构建。
于 2014-07-27T21:24:22.910 回答
1

现在有数组验证规则,以防万一这对任何人都有帮助。这些似乎还没有写在文档中。

https://github.com/laravel/laravel/commit/6a2ad475cfb21d12936cbbb544d8a136fc73be97

于 2014-02-10T11:44:29.797 回答
1

这是 Ronald 代码的更新,因为我的自定义规则不适用于数组扩展。使用 Laravel 4.1 测试,默认规则,扩展规则,......</p>

public function __call($method, $parameters) {
    $isArrayRule = FALSE;
    if(substr($method, -5) === 'Array') {
        $method = substr($method, 0, -5);
        $isArrayRule = TRUE;
    }

    //
    $rule = snake_case(substr($method, 8));

    // Default or custom rule
    if(!$isArrayRule) {
        // And we have a default value (not an array)
        if(!is_array($parameters[1])) {
            // Try getting the custom validation rule
            if(isset($this->extensions[$rule])) {
                return $this->callExtension($rule, $parameters);
            }

            // None found
            throw new \BadMethodCallException("Method [$method] does not exist.");
        } // Array given for default rule; cannot be!
        else return FALSE;
    }

    // Array rules
    $success = TRUE;
    foreach($parameters[1] as $value) {
        $parameters[1] = $value;

        // Default rule exists, use it
        if(is_callable("parent::$method")) {
            $success &= call_user_func_array(array($this, $method), $parameters);
        } else {
            // Try a custom rule
            if(isset($this->extensions[$rule])) {
                $success &= $this->callExtension($rule, $parameters);
            }

            // No custom rule found
            throw new \BadMethodCallException("Method [$method] does not exist.");
        }
    }

    // Did any of them (array rules) fail?
    return $success;
}
于 2014-01-10T15:13:18.333 回答