首页 > 解决方案 > 我如何在 whereHas Laravel 8 中使用相关模型范围

问题描述

我有两个模型。TaskTaskCheck

TaskCheck我有

class TaskCheck extends Model
{
    public function task(): BelongsTo
    {
        return $this->belongsTo(Task::class);
    }

    public function owner(): BelongsTo
    {
        return $this->belongsTo(User::class, 'user_id', 'id');
    }

    public function scopeOwnedBy(Builder $query, int $userId): Builder
    {
        return $query->where('user_id', '=', $userId);
    }
}

Task我的模型中

class Task extends Model
{
    public function taskCheck(): HasMany
    {
        return $this->hasMany(TaskCheck::class)->with(['taskState', 'owner']);
    }
}

并想使用这样的东西:

public function scopeHasOwner(Builder $query, int $taskOwnerId): Builder
{
    return $query->whereHas('taskCheck', function ($q) use ($taskOwnerId) {
        $q->hasOwner($taskOwnerId);
    });
}

但是这会引发异常Call to undefined method App\Models\Task::hasOwner(),因为内部查询似乎不知道任务模型。

我知道我可以改用它并且它有效

public function scopeHasOwner(Builder $query, int $taskOwnerId): Builder
    {
        return $query->whereHas('taskCheck', function ($q) use ($taskOwnerId) {
            $q->where('user_id', '=', $taskOwnerId);
        });
    }

但我宁愿不在每个相关模型中重复 where 子句,因为在关系中存在更多相关模型,它们将使用类似的功能,而我只想将它放在一个地方。

标签: laravelscopeeloquent-relationship

解决方案


在您的TaskCheck模型中,您有ownedBy()范围,但您hasOwner()whereHas查询中调用了。

将您的查询更改为ownedBy()

$query->whereHas('taskCheck', function ($q) use ($taskOwnerId) {
    $q->ownedBY($taskOwnerId);
});

推荐阅读