首先,让我们看一下如何使用基本的查询构建器来实现这一点。然后,我们将讨论如何使用Eloquent模型来执行此查询:
function paginateDishesFromPoint(Point $point, $pageSize)
{
$distanceField = "ST_Distance_Sphere(locations.coordinates, "
. "ST_GeomFromText('{$point->toWKT()}') AS distance";
return DB::table('dishes')
->select('dishes.*', DB::raw($distanceField))
->join('dish_locations', 'dish_locations.dish_id', '=', 'dishes.id')
->join('locations', 'locations.id', '=', 'dish_locations.location_id')
->orderBy('distance')
->paginate($pageSize);
}
ST_Distance_Sphere()
函数用于计算可以根据其排序的距离。 Laravel 的
paginate()
方法使用通过请求 URL 传递的
page
参数自动进行分页。阅读
pagination docs 以获取更多信息。使用上述函数,我们可以按以下方式获取分页结果集:
$point = new Point($latitude, $longitude);
$sortedDishes = paginateDishesFromPoint($point, 15);
...其中Point
是我们使用的Grimzy\LaravelMysqlSpatial\Types\Point
类,来自该扩展包,而15
是每页结果数。
现在,让我们尝试使用Eloquent模型来实现。我们将使用本地作用域来封装创建执行排序的查询部分所需的逻辑:
class Dish extends Model
{
...
public function locations()
{
return $this->belongsToMany(App\Location::class);
}
public function scopeOrderByDistanceFrom($query, Point $point)
{
$relation = $this->locations();
$locationsTable = $relation->getRelated()->getTable();
$distanceField = "ST_Distance_Sphere($locationsTable.coordinates, "
. "ST_GeomFromText('{$point->toWKT()}') AS distance";
return $query
->select($this->getTable() . '.*', DB::raw($distanceField))
->join(
$relation->getTable(),
$relation->getQualifiedForeignKeyName(),
'=',
$relation->getQualifiedParentKeyName()
)
->join(
$locationsTable,
$relation->getRelated()->getQualifiedKeyName(),
'=',
$relation->getQualifiedRelatedKeyName()
)
->orderBy('distance');
}
}
这个实现使用模型上的元数据将表名和字段名添加到查询中,因此如果它们更改,我们无需更新此方法。现在,我们可以使用该模型获取有序集合:
$point = new Point($latitude, $longitude);
$sortedDishes = Dish::orderByDistanceFrom($point)->paginate($pageSize);
$sortedDishes
是 Laravel 的 LengthAwarePaginator
实例,它包装了模型的 Collection
。如果我们将结果传递给视图,在 Blade 模板中展示如下:
<ul>
@foreach($sortedDishes as $dish)
<li>{{ $dish->name }} is {{ $dish->distance }} meters away.</li>
@endforeach
</ul>
<a href="{{ $sortedDishes->nextPageUrl() }}">Load more...</a>
如上所示,分页器提供了一些
便捷方法,我们可以使用这些方法轻松地在分页结果之间进行移动。
另外,我们也可以使用AJAX请求来加载结果。只需确保在请求数据的
page
参数中传递
当前页面+1即可。