我正在使用Laravel 5.6,并且试图过滤User
模型中的关系。用户可以参加课程,课程中包含要点。
用户可以通过参加课程获得这些积分。这是BelongsToMany
关系。
我试图在此User
模型中创建一个范围,该范围仅包含几年内的参加课程。
/**
* Retrieves the courses which the user has attended
*/
public function attendedCourses()
{
return $this->belongsToMany(Course::class, 'course_attendees');
}
/**
* Searches the user model
*
* @param \Illuminate\Database\Eloquent\Builder $builder
* @param array $years
*
* @return \Illuminate\Database\Eloquent\Builder
*/
public function scopeAttendedCoursesInYears(Builder $builder, array $years)
{
# Filter on the years
$callback = function($q) use ($years) {
foreach ($years as $year) {
$q->year($year);
}
};
return $builder->with(['attendedCourses' => $callback]);
}
在我的Course
模型中,我有一个范围可以过滤该课程所在的年份。
public function scopeYear(Builder $query, int $year)
{
return $query->whereYear('end_date', $year);
}
有了这个attendedCoursesInYears
范围,我希望我可以使用Course
模型上的其他范围,通过对课程分数求和来为每个用户计算分数。
public function scopeExternal(Builder $query, bool $flag = true)
{
$categoryIsExternal = function($query) use ($flag) {
$query->external($flag);
};
return $query->whereHas('category', $categoryIsExternal);
}
在我的CourseCategory
模态中,范围如下所示:
/**
* Scope a query to only include external categories.
*
* @param \Illuminate\Database\Eloquent\Builder $query
*
* @param bool $flag
*
* @return \Illuminate\Database\Eloquent\Builder
*/
public function scopeExternal(Builder $query, $flag = true)
{
return $query->where('type', '=', $flag ? 'Extern' : 'Intern');
}
为了计算这个,我尝试做这样的事情。
# Retrieve all the active users
$users = User::all()->attendedCoursesInYears($years);
$test = $users->find(123);
# Calculate the points
$test->attendedCourses()->external(false)->sum('points');
但这返回了所有课程的总和。
使用作用域是这里唯一的选择,正如我所看到的。我想使用这样的访问器从这些值创建自定义属性。这样一来,我可以轻松地对计算得出的值进行排序。
/**
* The users internal course points
*
* @param array $years The years to look for attended courses
*
* @return float
*/
public function getInternalPointsAttribute() : float
{
return $this->attendedCourses()->external(false)->sum('points');
}
这里唯一的问题是年份过滤器。我希望可以像第一个示例一样在调用访问器之前过滤User集合。
我在这里做什么错了?
我目前正在使用此替代方法。这似乎很糟糕,因为我要重复很多代码。
/**
* @param \Illuminate\Database\Eloquent\Builder $builder
*
* @param array $years
*
* @return \Illuminate\Database\Eloquent\Builder|\Illuminate\Database\Query\Builder
*/
public function scopeWithPoints(Builder $builder, array $years = [])
{
# Join all columns
$builder->join('user_roles', 'users.role_id', '=', 'user_roles.id')
->leftJoin('course_attendees', 'users.id', '=', 'course_attendees.user_id');
# Join the course table for the years
$builder->leftJoin('courses', function(JoinClause $join) use ($years) {
# Join the courses table with year filters
$join->on('course_attendees.course_id', '=', 'courses.id');
# Apply the filters if available
!empty($years) and $join->whereIn(DB::raw('YEAR(courses.end_date)'), $years);
});
# Select the columns
$builder->select('users.*')->groupBy('users.id');
# Sums
$internalPoints = 'SUM(courses.points_internal)';
$externalPoints = 'SUM(courses.points_external)';
# Select the points
$builder->selectRaw('COALESCE(' . $internalPoints . ', 0) as internal_points');
$builder->selectRaw('COALESCE(' . $externalPoints . ', 0) as external_points');
$builder->selectRaw('COALESCE(' . $internalPoints . ' + ' . $externalPoints . ', 0) as total_points');
# Sum up the course points
return $builder;
}
我的数据库结构的迁移可以在这里找到。
Schema::create('course_attendees', function(Blueprint $table)
{
$table->integer('id', true);
$table->integer('user_id')->index('course_attendees_users_id_fk');
$table->integer('course_id')->index('course_attendees_courses_id_fk');
$table->boolean('mijnafas');
});
Schema::create('courses', function(Blueprint $table)
{
$table->integer('id', true);
$table->string('title');
$table->string('subject');
$table->string('presenter');
$table->date('start_date')->nullable()->comment('Set to not null later');
$table->date('end_date')->nullable();
$table->decimal('points', 4)->nullable();
$table->string('location');
$table->timestamps();
});
Schema::create('users', function(Blueprint $table)
{
$table->integer('id', true);
$table->string('first_name')->nullable();
$table->string('last_name')->nullable();
$table->timestamps();
});
Schema::table('course_attendees', function(Blueprint $table)
{
$table->foreign('course_id', 'course_attendees_courses_id_fk')->references('id')->on('courses')->onUpdate('RESTRICT')->onDelete('RESTRICT');
$table->foreign('user_id', 'course_attendees_users_id_fk')->references('id')->on('users')->onUpdate('RESTRICT')->onDelete('RESTRICT');
});
我注意到,仅在调用$test->attendedCourses
时,它会检索已过滤的集合。问题在于,我无法对此应用范围。
问题
答案 0 :(得分:3)
问题在于,您的User类中的--f
方法是要确定在特定年份内参加过这些课程的用户的范围,而不是对该年份所进行过的课程的范围进行限制。
要执行所需的操作,可以在关系中添加一个参数以过滤数据透视表而不是用户表:
--force
然后您可以像这样实现结果:
scopeAttendedCoursesInYears