forge/app/Models/Mod.php

245 lines
6.7 KiB
PHP
Raw Normal View History

2024-05-15 00:31:24 -04:00
<?php
namespace App\Models;
use App\Http\Filters\V1\QueryFilter;
use App\Models\Scopes\DisabledScope;
2024-07-26 09:35:09 -04:00
use App\Models\Scopes\PublishedScope;
2024-09-12 13:19:52 -04:00
use Database\Factories\ModFactory;
use Illuminate\Database\Eloquent\Builder;
2024-05-15 00:31:24 -04:00
use Illuminate\Database\Eloquent\Casts\Attribute;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
2024-05-15 00:31:24 -04:00
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Relations\HasOne;
2024-05-15 00:31:24 -04:00
use Illuminate\Database\Eloquent\SoftDeletes;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\Storage;
2024-05-15 00:31:24 -04:00
use Illuminate\Support\Str;
2024-06-02 22:03:59 -04:00
use Laravel\Scout\Searchable;
2024-05-15 00:31:24 -04:00
class Mod extends Model
{
2024-09-12 13:19:52 -04:00
/** @use HasFactory<ModFactory> */
use HasFactory;
use Searchable;
use SoftDeletes;
2024-05-15 00:31:24 -04:00
2024-07-20 19:52:36 -04:00
/**
* Post boot method to configure the model.
*/
protected static function booted(): void
{
static::addGlobalScope(new DisabledScope);
2024-07-26 09:35:09 -04:00
static::addGlobalScope(new PublishedScope);
}
/**
* Calculate the total number of downloads for the mod.
*/
public function calculateDownloads(): void
{
$this->downloads = $this->versions->sum('downloads');
$this->saveQuietly();
}
/**
2024-07-20 19:52:36 -04:00
* The relationship between a mod and its users.
2024-09-12 13:19:52 -04:00
*
* @return BelongsToMany<User>
*/
public function users(): BelongsToMany
2024-05-15 00:31:24 -04:00
{
return $this->belongsToMany(User::class);
2024-05-15 00:31:24 -04:00
}
2024-07-20 19:52:36 -04:00
/**
* The relationship between a mod and its license.
2024-09-12 13:19:52 -04:00
*
* @return BelongsTo<License, Mod>
2024-07-20 19:52:36 -04:00
*/
2024-05-15 00:31:24 -04:00
public function license(): BelongsTo
{
return $this->belongsTo(License::class);
}
2024-07-20 19:52:36 -04:00
/**
* The relationship between a mod and its versions.
2024-09-12 13:19:52 -04:00
*
* @return HasMany<ModVersion>
2024-07-20 19:52:36 -04:00
*/
public function versions(): HasMany
2024-05-15 00:31:24 -04:00
{
return $this->hasMany(ModVersion::class)
->whereHas('latestSptVersion')
->orderByDesc('version')
->chaperone();
2024-05-15 00:31:24 -04:00
}
2024-07-20 19:52:36 -04:00
/**
* The relationship between a mod and its last updated version.
2024-09-12 13:19:52 -04:00
*
* @return HasOne<ModVersion>
2024-07-20 19:52:36 -04:00
*/
public function lastUpdatedVersion(): HasOne
2024-05-17 23:54:03 -04:00
{
return $this->hasOne(ModVersion::class)
->whereHas('latestSptVersion')
->orderByDesc('updated_at')
->chaperone();
2024-05-15 00:31:24 -04:00
}
2024-06-02 22:03:59 -04:00
/**
2024-07-20 19:52:36 -04:00
* The data that is searchable by Scout.
2024-09-12 13:19:52 -04:00
*
* @return array<string, mixed>
*/
2024-06-02 22:03:59 -04:00
public function toSearchableArray(): array
{
return [
'id' => $this->id,
2024-06-02 22:03:59 -04:00
'name' => $this->name,
'slug' => $this->slug,
'description' => $this->description,
'thumbnail' => $this->thumbnail,
'featured' => $this->featured,
'created_at' => strtotime($this->created_at),
'updated_at' => strtotime($this->updated_at),
2024-07-26 09:34:57 -04:00
'published_at' => strtotime($this->published_at),
2024-09-12 13:19:52 -04:00
'latestVersion' => $this->latestVersion()->first()->latestSptVersion()->first()->version_formatted,
'latestVersionColorClass' => $this->latestVersion()->first()->latestSptVersion()->first()->color_class,
2024-06-02 22:03:59 -04:00
];
}
/**
* The relationship to the latest mod version, dictated by the mod version number.
2024-09-12 13:19:52 -04:00
*
* @return HasOne<ModVersion>
*/
public function latestVersion(): HasOne
{
return $this->hasOne(ModVersion::class)
->whereHas('sptVersions')
->orderByDesc('version')
2024-07-20 19:52:36 -04:00
->orderByDesc('updated_at')
->take(1)
->chaperone();
}
/**
2024-07-20 19:52:36 -04:00
* Determine if the model instance should be searchable.
*/
2024-06-02 22:03:59 -04:00
public function shouldBeSearchable(): bool
{
// Ensure the mod is not disabled.
if ($this->disabled) {
return false;
}
// Ensure the mod has a publish date.
if (is_null($this->published_at)) {
return false;
}
// Fetch the latest version instance.
2024-09-12 13:19:52 -04:00
$latestVersion = $this->latestVersion()->first();
// Ensure the mod has a latest version.
if (is_null($latestVersion)) {
return false;
}
// Ensure the latest version has a latest SPT version.
if ($latestVersion->latestSptVersion()->doesntExist()) {
return false;
}
// Ensure the latest SPT version is within the last three minor versions.
$activeSptVersions = Cache::remember('active-spt-versions', 60 * 60, function () {
return SptVersion::getVersionsForLastThreeMinors();
});
if (! in_array($latestVersion->latestSptVersion()->first()->version, $activeSptVersions->pluck('version')->toArray())) {
return false;
}
// All conditions are met; the mod should be searchable.
return true;
2024-06-02 22:03:59 -04:00
}
/**
2024-07-20 19:52:36 -04:00
* Build the URL to the mod's thumbnail.
2024-09-12 13:19:52 -04:00
*
* @return Attribute<string, never>
*/
public function thumbnailUrl(): Attribute
{
return Attribute::get(function (): string {
return $this->thumbnail
? Storage::disk($this->thumbnailDisk())->url($this->thumbnail)
: '';
});
}
/**
2024-07-20 19:52:36 -04:00
* Get the disk where the thumbnail is stored based on the current environment.
*/
protected function thumbnailDisk(): string
{
return match (config('app.env')) {
'production' => 'r2', // Cloudflare R2 Storage
default => 'public', // Local
};
}
/**
* Scope a query by applying QueryFilter filters.
2024-09-12 13:19:52 -04:00
*
* @param Builder<Mod> $builder
* @param QueryFilter<Mod> $filters
* @return Builder<Mod>
*/
public function scopeFilter(Builder $builder, QueryFilter $filters): Builder
{
return $filters->apply($builder);
}
/**
* Build the URL to the mod's detail page.
*/
public function detailUrl(): string
{
return route('mod.show', [$this->id, $this->slug]);
}
2024-07-20 19:52:36 -04:00
/**
* The attributes that should be cast to native types.
2024-09-12 13:19:52 -04:00
*
* @return array<string, string>
2024-07-20 19:52:36 -04:00
*/
protected function casts(): array
{
return [
'featured' => 'boolean',
'contains_ai_content' => 'boolean',
'contains_ads' => 'boolean',
'disabled' => 'boolean',
];
}
/**
2024-07-20 19:52:36 -04:00
* Mutate the slug attribute to always be lower case on get and slugified on set.
2024-09-12 13:19:52 -04:00
*
* @return Attribute<string, string>
*/
protected function slug(): Attribute
{
return Attribute::make(
get: fn (string $value) => Str::lower($value),
set: fn (string $value) => Str::slug($value),
);
}
2024-05-15 00:31:24 -04:00
}