11

我有一个查询范围方法,可以过滤模型作者对 Foo 模型的结果。复杂之处在于作者没有直接关系。

Foo 属于 Bar,Bar 属于 User。如果 Foo 属于用户,我可以这样做:

public function scopeAuthorOnly($query, User $user)
{
    return $query->whereuser_id($user->id);
}

这显然不起作用,因为 Foo 模型没有 user_id 列,而是有一个 bar_id 列。但是,我不确定如何以过滤 user_id 的方式构建查询。

任何指针?

4

3 回答 3

20

您可以使用whereHas()in 范围来查询关系。

假设您users()在此模型中有一个关系函数。然后它会是这样的:

public function scopeAuthorOnly($query, User $user)
{
    return $query->whereHas('users', function($q) use($user){
        $q->where('id', $user->id);
    });
}

更新:您还可以嵌套whereHas语句:如果当前Foo有一个名为 的关系bar,并且bar有一个名为 的关系users,它会是这样的:

public function scopeAuthorOnly($query, User $user)
{
    return $query->whereHas('bar', function($q) use($user){
        return $q->whereHas('users', function($q2) use($user){
            $q2->where('id', $user->id);
        });
    });
}

更多信息:这里

于 2014-06-11T10:42:14.743 回答
8

您也可以使用关系来执行此操作(无需使用手动连接):

public function scopeAuthorOnly($query, User $user)
{
    $query->whereHas(array('Bar' => function($query) use($user){
                $query->where('user_id', '=', $user->id);
            }));
}

编辑:现在使用whereHas而不是with(仅限 Laravel >= 4.1)。感谢阿尔达的更正。

编辑:在 4.2 中 whereHas 的语法发生了变化,因此关系名称是参数 1,而闭包是参数 2(而不是作为数组传入):

public function scopeAuthorOnly($query, User $user)
{
    $query->whereHas('Bar', function($query) use($user){
                $query->where('user_id', '=', $user->id);
            });
}
于 2014-05-13T15:10:54.673 回答
6

找到了解决方案:

public function scopeAuthorOnly($query, User $user)
{
    $query
        ->join('bars', 'foos.bar_id', '=', 'bars.id')
        ->join('users', 'bars.user_id', '=', 'users.id')
        ->where('users.id', '=', $user->id);

    return $query;
}
于 2013-09-03T14:18:01.527 回答