187 lines
5.2 KiB
PHP
Executable File
187 lines
5.2 KiB
PHP
Executable File
<?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,
|
|
];
|
|
}
|
|
}
|