> */ public function rules(): array { return [ // ID of the organization member that the time entry should belong to 'member_id' => [ 'string', ExistsEloquent::make(Member::class, null, function (Builder $builder): Builder { /** @var Builder $builder */ return $builder->whereBelongsTo($this->organization, 'organization'); })->uuid(), ], // ID of the project that the time entry should belong to 'project_id' => [ 'nullable', 'string', 'required_with:task_id', ExistsEloquent::make(Project::class, null, function (Builder $builder): Builder { /** @var Builder $builder */ return $builder->whereBelongsTo($this->organization, 'organization'); })->uuid(), ], // ID of the task that the time entry should belong to 'task_id' => [ 'nullable', 'string', ExistsEloquent::make(Task::class, null, function (Builder $builder): Builder { /** @var Builder $builder */ return $builder->whereBelongsTo($this->organization, 'organization'); })->uuid(), ExistsEloquent::make(Task::class, null, function (Builder $builder): Builder { /** @var Builder $builder */ return $builder->whereBelongsTo($this->organization, 'organization') ->where('project_id', $this->input('project_id')); })->uuid()->withMessage(__('validation.task_belongs_to_project')), ], // Start of time entry (ISO 8601 format, UTC timezone) 'start' => [ 'date_format:Y-m-d\TH:i:s\Z', ], // End of time entry (ISO 8601 format, UTC timezone) 'end' => [ 'nullable', 'date_format:Y-m-d\TH:i:s\Z', 'after_or_equal:start', ], // Whether time entry is billable 'billable' => [ 'boolean', ], // Description of time entry 'description' => [ 'nullable', 'string', 'max:500', ], // List of tag IDs 'tags' => [ 'nullable', 'array', ], 'tags.*' => [ 'string', ExistsEloquent::make(Tag::class, null, function (Builder $builder): Builder { /** @var Builder $builder */ return $builder->whereBelongsTo($this->organization, 'organization'); })->uuid(), ], ]; } }