186
app/Models/Person.php
Executable file
186
app/Models/Person.php
Executable file
@@ -0,0 +1,186 @@
|
||||
<?php
|
||||
|
||||
namespace App\Models;
|
||||
|
||||
use App\Actions\People\StorePersonData;
|
||||
use App\Actions\Titles\HandlesEncodedTmdbId;
|
||||
use App\Actions\Titles\InsertsTmdbTitleOrPerson;
|
||||
use App\Services\Data\Tmdb\TmdbApi;
|
||||
use Carbon\Carbon;
|
||||
use Common\Core\BaseModel;
|
||||
use Illuminate\Database\Eloquent\Builder;
|
||||
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
||||
use Illuminate\Support\Arr;
|
||||
use Laravel\Scout\Searchable;
|
||||
|
||||
class Person extends BaseModel
|
||||
{
|
||||
use Searchable, HandlesEncodedTmdbId, InsertsTmdbTitleOrPerson;
|
||||
|
||||
public const MODEL_TYPE = 'person';
|
||||
|
||||
protected $guarded = ['id', 'relation_data', 'model_type'];
|
||||
protected $appends = ['model_type'];
|
||||
|
||||
protected $casts = [
|
||||
'id' => 'integer',
|
||||
'tmdb_id' => 'integer',
|
||||
'allow_update' => 'boolean',
|
||||
'fully_synced' => 'boolean',
|
||||
'adult' => 'boolean',
|
||||
'birth_date' => 'date',
|
||||
'death_date' => 'date',
|
||||
];
|
||||
|
||||
protected static function booted()
|
||||
{
|
||||
static::addGlobalScope('adult', function (Builder $builder) {
|
||||
if (!config('tmdb.includeAdult')) {
|
||||
$builder->where('adult', false);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public function scopeOrderByBirthDate(Builder $query, string $direction)
|
||||
{
|
||||
$query->orderByRaw(
|
||||
"CASE WHEN birth_date IS NULL THEN 1 ELSE 0 END, birth_date $direction",
|
||||
);
|
||||
}
|
||||
|
||||
public static function firstOrCreateFromEncodedTmdbId(
|
||||
string $encodedId,
|
||||
): static {
|
||||
[$tmdbId] = static::decodeTmdbIdOrFail($encodedId);
|
||||
return static::withoutGlobalScope('adult')->firstOrCreate([
|
||||
'tmdb_id' => $tmdbId,
|
||||
]);
|
||||
}
|
||||
|
||||
public function maybeUpdateFromExternal(array $options = []): static|null
|
||||
{
|
||||
$tmdbImportingIsEnabled =
|
||||
settings('content.people_provider') === 'tmdb' ||
|
||||
Arr::get($options, 'forceAutomation');
|
||||
|
||||
if (
|
||||
$tmdbImportingIsEnabled &&
|
||||
$this->needsUpdating($options['ignoreLastUpdate'] ?? false)
|
||||
) {
|
||||
$data = app(TmdbApi::class)->getPerson($this);
|
||||
if (!$data) {
|
||||
return null;
|
||||
}
|
||||
app(StorePersonData::class)->execute($this, $data);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function needsUpdating($force = false): bool
|
||||
{
|
||||
if (!$this->exists || !$this->tmdb_id) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($force) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// sync every week
|
||||
return $this->allow_update &&
|
||||
(!$this->updated_at ||
|
||||
$this->updated_at->lessThan(Carbon::now()->subWeek()));
|
||||
}
|
||||
|
||||
public static function getModelTypeAttribute(): string
|
||||
{
|
||||
return self::MODEL_TYPE;
|
||||
}
|
||||
|
||||
public function scopeCompact(Builder $query): Builder
|
||||
{
|
||||
return $query->select(['titles.id', 'titles.name', 'titles.poster']);
|
||||
}
|
||||
|
||||
public function credits(): BelongsToMany
|
||||
{
|
||||
return $this->morphedByMany(Title::class, 'creditable')
|
||||
->select(
|
||||
'titles.id',
|
||||
'is_series',
|
||||
'poster',
|
||||
'backdrop',
|
||||
'popularity',
|
||||
'name',
|
||||
'release_date',
|
||||
'tmdb_vote_average',
|
||||
'local_vote_average',
|
||||
)
|
||||
->withPivot(['id', 'job', 'department', 'order', 'character'])
|
||||
->orderBy('titles.release_date', 'desc');
|
||||
}
|
||||
|
||||
public function episodeCredits(int $tileId = null): BelongsToMany
|
||||
{
|
||||
$query = $this->morphedByMany(Episode::class, 'creditable');
|
||||
if ($tileId) {
|
||||
$query->where('episodes.title_id', $tileId);
|
||||
}
|
||||
$query
|
||||
->select(
|
||||
'episodes.id',
|
||||
'episodes.title_id',
|
||||
'name',
|
||||
'release_date',
|
||||
'season_number',
|
||||
'episode_number',
|
||||
)
|
||||
->withPivot(['job', 'department', 'order', 'character'])
|
||||
->orderBy('episodes.season_number', 'desc')
|
||||
->orderBy('episodes.episode_number', 'desc');
|
||||
return $query;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param int|null $tileId
|
||||
* @return BelongsToMany
|
||||
*/
|
||||
public function seasonCredits($tileId = null)
|
||||
{
|
||||
$query = $this->morphedByMany(Season::class, 'creditable');
|
||||
if ($tileId) {
|
||||
$query->where('seasons.title_id', $tileId);
|
||||
}
|
||||
$query
|
||||
->select('seasons.id', 'seasons.title_id')
|
||||
->withPivot(['job', 'department', 'order', 'character'])
|
||||
->orderBy('seasons.number', 'desc');
|
||||
return $query;
|
||||
}
|
||||
|
||||
public function toSearchableArray(): array
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'created_at' => $this->created_at->timestamp ?? '_null',
|
||||
'updated_at' => $this->updated_at->timestamp ?? '_null',
|
||||
];
|
||||
}
|
||||
|
||||
public static function filterableFields(): array
|
||||
{
|
||||
return ['id', 'created_at', 'updated_at'];
|
||||
}
|
||||
|
||||
public function toNormalizedArray(): array
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'image' => $this->poster,
|
||||
'model_type' => self::MODEL_TYPE,
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user