前回で披露したコントローラー中でのルールの共有のメソッドrules()では、新規レコード作成と既存レコードの編集における入力項目の違いを、$actionの引数をswitchで分岐して対応しました。今回は、sometimesを使用してそれらの必要性をなくします。
sometimesは、入力項目があるときのみに、それ以降のルールを適応するルールです。tinkerを使ってテストしてみましょう。
通常は、
>>> $input = [];
=> []
>>> $rules = ['email' => 'required|email'];
=> [
     "email" => "required|email",
   ]
>>> validator($input, $rules)->errors()->all();
=> [
     "必ず指定してください",
   ]
とエラーとなるところ、sometimesを使用すると、
>>> $input = [];
=> []
>>> $rules = ['email' => 'sometimes|required|email'];
=> [
     "email" => "sometimes|required|email",
   ]
>>> validator($input, $rules)->errors()->all();
=> []
>>> 
とemailの項目が入力にないのにエラーとはなりません。
これを利用して前回のrules()の定義を書き変えてみます。
namespace App\Http\Controllers;
use Illuminate\Http\Request;
use Illuminate\Validation\Rule;
use App\User;
class UserController extends Controller
{
    public function rule(User $user = null)
   {
        // 共有するルール kana, mailcode, phone_with_dashはカスタムバリデーター
        $rules = [
            'name'        => 'required',
            'name_kana'   => 'required|kana',
            'mailcode'    => 'required|mailcode',
            'prefecture'  => 'required',
            'city'        => 'required',
            'address1'    => 'required',
            'phone'       => 'required|phone_with_dash',
            'password'    => 'sometimes|required|min:8|max:20|confirmed',
            'email'       => [
                    'required',
                    'email',
                    Rule::unique('users')->ignore(optional($user)->id), //重複がないかチェック
                ]
        ];
        return [
            // rules
            $rules,
            // messages
            [
                'password.min' => '8から20文字長でお願いします',
                'password.max' => '8から20文字長でお願いします'
            ]
            // attributes
        ];
    }
...
    public function store(Request $request)
    {
        $request->validate(...$this->rules());
        ...
    }
...
    public function update(Request $request, User $user)
    {
        $request->validate(...$this->rules($user));
        ...
    }
...
前回と変わったのは、関数の引数の$actionとその値によるswitchによる分岐がなくなっています。新規作成時しか必要でない、passwordの項目に、sometimesを使用して、その項目がないときはルールを適用しないようにしています。また、emailの重複のチェックにおいては、以下のようにヘルパーのoptional()を使用して、新規には存在しない$userの値でundefinedのエラーが出ないようにしています。
 Rule::unique('users')->ignore(optional($user)->id)
最後に、sometimes|requiredは以下のようにfilledに置き換えることも可能です。filledに関してはこちらも参考にしてください。
'password'    => 'filled|min:8|max:20|confirmed'
