jonnybarnes.uk/app/Models/Place.php
2020-02-22 11:06:43 +00:00

224 lines
6.4 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Models;
use Cviebrock\EloquentSluggable\Sluggable;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\{Builder, Model};
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Str;
use Phaza\LaravelPostgis\Eloquent\PostgisTrait;
use Phaza\LaravelPostgis\Geometries\Point;
/**
* App\Models\Place
*
* @property int $id
* @property string $name
* @property string $slug
* @property string|null $description
* @property string $location
* @property string|null $polygon
* @property \Illuminate\Support\Carbon|null $created_at
* @property \Illuminate\Support\Carbon|null $updated_at
* @property string|null $icon
* @property string|null $foursquare
* @property mixed|null $external_urls
* @property-read float $latitude
* @property-read float $longitude
* @property-read string $longurl
* @property-read string $shorturl
* @property-read string $uri
* @property-read \Illuminate\Database\Eloquent\Collection|\App\Models\Note[] $notes
* @property-read int|null $notes_count
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place findSimilarSlugs($attribute, $config, $slug)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place near(\Phaza\LaravelPostgis\Geometries\Point $point, $distance = 1000)
* @method static \Phaza\LaravelPostgis\Eloquent\Builder|\App\Models\Place newModelQuery()
* @method static \Phaza\LaravelPostgis\Eloquent\Builder|\App\Models\Place newQuery()
* @method static \Phaza\LaravelPostgis\Eloquent\Builder|\App\Models\Place query()
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereCreatedAt($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereDescription($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereExternalURL($url)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereExternalUrls($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereFoursquare($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereIcon($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereId($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereLocation($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereName($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place wherePolygon($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereSlug($value)
* @method static \Illuminate\Database\Eloquent\Builder|\App\Models\Place whereUpdatedAt($value)
* @mixin \Eloquent
*/
class Place extends Model
{
use Sluggable;
use PostgisTrait;
/**
* The attributes that are mass assignable.
*
* @var array
*/
protected $fillable = ['name', 'slug'];
/**
* The attributes that are Postgis geometry objects.
*
* @var array
*/
protected $postgisFields = [
'location',
'polygon',
];
/**
* Return the sluggable configuration array for this model.
*
* @return array
*/
public function sluggable(): array
{
return [
'slug' => [
'source' => 'name',
'onUpdate' => true,
],
];
}
/**
* Define the relationship with Notes.
*
* @return HasMany
*/
public function notes()
{
return $this->hasMany('App\Models\Note');
}
/**
* Select places near a given location.
*
* @param Builder $query
* @param Point $point
* @param int $distance
* @return Builder
*/
public function scopeNear(Builder $query, Point $point, int $distance = 1000): Builder
{
$field = DB::raw(
sprintf(
"ST_Distance(%s.location, ST_GeogFromText('%s'))",
$this->getTable(),
$point->toWKT()
)
);
return $query->where($field, '<=', $distance)->orderBy($field);
}
/**
* Select places based on a URL.
*
* @param Builder $query
* @param string $url
* @return Builder
*/
public function scopeWhereExternalURL(Builder $query, string $url): Builder
{
return $query->where('external_urls', '@>', json_encode([
$this->getType($url) => $url,
]));
}
/**
* Get the latitude from the `location` property.
*
* @return float
*/
public function getLatitudeAttribute(): float
{
return $this->location->getLat();
}
/**
* Get the longitude from the `location` property.
*
* @return float
*/
public function getLongitudeAttribute(): float
{
return $this->location->getLng();
}
/**
* The Long URL for a place.
*
* @return string
*/
public function getLongurlAttribute(): string
{
return config('app.url') . '/places/' . $this->slug;
}
/**
* The Short URL for a place.
*
* @return string
*/
public function getShorturlAttribute(): string
{
return config('app.shorturl') . '/places/' . $this->slug;
}
/**
* This method is an alternative for `longurl`.
*
* @return string
*/
public function getUriAttribute(): string
{
return $this->longurl;
}
/**
* Dealing with a jsonb column, so we check input first.
*
* @param string|null $url
*/
public function setExternalUrlsAttribute(?string $url)
{
if ($url === null) {
return;
}
$type = $this->getType($url);
$already = [];
if (array_key_exists('external_urls', $this->attributes)) {
$already = json_decode($this->attributes['external_urls'], true);
}
$already[$type] = $url;
$this->attributes['external_urls'] = json_encode($already);
}
/**
* Given a URL, see if it is one of our known types.
*
* @param string $url
* @return string
*/
private function getType(string $url): string
{
$host = parse_url($url, PHP_URL_HOST);
if (Str::endsWith($host, 'foursquare.com') === true) {
return 'foursquare';
}
if (Str::endsWith($host, 'openstreetmap.org') === true) {
return 'osm';
}
return 'default';
}
}