forge/app/Models/User.php

302 lines
7.6 KiB
PHP
Raw Normal View History

2024-05-13 18:55:34 -04:00
<?php
2025-01-30 00:23:55 -05:00
declare(strict_types=1);
2024-05-13 18:55:34 -04:00
namespace App\Models;
use App\Http\Filters\V1\QueryFilter;
use App\Notifications\ResetPassword;
use App\Notifications\VerifyEmail;
use App\Traits\HasCoverPhoto;
2025-01-30 15:44:05 -05:00
use Carbon\Carbon;
use Illuminate\Contracts\Auth\MustVerifyEmail;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Casts\Attribute;
2025-01-30 15:44:05 -05:00
use Illuminate\Database\Eloquent\Collection;
2024-05-13 18:55:34 -04:00
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
use Illuminate\Database\Eloquent\Relations\HasMany;
2024-05-13 18:55:34 -04:00
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Illuminate\Support\Str;
2024-05-13 18:55:34 -04:00
use Laravel\Fortify\TwoFactorAuthenticatable;
use Laravel\Jetstream\HasProfilePhoto;
use Laravel\Sanctum\HasApiTokens;
2024-06-02 22:03:59 -04:00
use Laravel\Scout\Searchable;
2024-06-16 21:40:00 -04:00
use Mchev\Banhammer\Traits\Bannable;
2025-01-30 00:23:55 -05:00
use SensitiveParameter;
2024-05-13 18:55:34 -04:00
2025-01-30 15:44:05 -05:00
/**
* @property int $id
* @property int|null $hub_id
* @property int|null $discord_id
* @property string $name
* @property string $email
* @property Carbon|null $email_verified_at
* @property string|null $password
* @property string $about
* @property int|null $user_role_id
* @property string|null $remember_token
* @property string|null $profile_photo_path
* @property string|null $cover_photo_path
* @property Carbon $created_at
* @property Carbon $updated_at
* @property-read string $profile_photo_url
* @property-read UserRole|null $role
* @property-read Collection<int, Mod> $mods
* @property-read Collection<int, User> $followers
* @property-read Collection<int, User> $following
* @property-read Collection<int, OAuthConnection> $oAuthConnections
*/
class User extends Authenticatable implements MustVerifyEmail
2024-05-13 18:55:34 -04:00
{
2024-06-16 21:40:00 -04:00
use Bannable;
2024-05-13 18:55:34 -04:00
use HasApiTokens;
use HasCoverPhoto;
2024-05-13 18:55:34 -04:00
use HasFactory;
use HasProfilePhoto;
use Notifiable;
2024-06-02 22:03:59 -04:00
use Searchable;
2024-06-03 02:04:49 +00:00
use TwoFactorAuthenticatable;
2024-05-13 18:55:34 -04:00
protected $hidden = [
'password',
'remember_token',
'two_factor_recovery_codes',
'two_factor_secret',
];
protected $appends = [
'profile_photo_url',
];
/**
* Get the storage path for profile photos.
*/
public static function profilePhotoStoragePath(): string
{
return 'profile-photos';
}
2024-07-20 19:52:36 -04:00
/**
* The relationship between a user and their mods.
2024-09-12 13:19:52 -04:00
*
2025-01-30 15:44:05 -05:00
* @return BelongsToMany<Mod, $this>
2024-07-20 19:52:36 -04:00
*/
public function mods(): BelongsToMany
2024-05-15 00:31:24 -04:00
{
return $this->belongsToMany(Mod::class);
2024-05-15 00:31:24 -04:00
}
2024-06-02 22:03:59 -04:00
2024-08-28 16:59:28 -04:00
/**
* The relationship between a user and users that follow them.
*
2025-01-30 15:44:05 -05:00
* @return BelongsToMany<User, $this>
2024-08-28 16:59:28 -04:00
*/
public function followers(): BelongsToMany
2024-08-28 16:59:28 -04:00
{
return $this->belongsToMany(User::class, 'user_follows', 'following_id', 'follower_id')
->withTimestamps();
2024-08-28 16:59:28 -04:00
}
/**
* Follow another user.
2024-08-28 16:59:28 -04:00
*/
2024-08-29 21:50:59 -04:00
public function follow(User|int $user): void
{
$userId = $user instanceof User ? $user->id : $user;
2024-09-01 23:42:44 -04:00
if ($this->id === $userId) {
// Don't allow following yourself.
2024-09-01 23:42:44 -04:00
return;
}
$this->following()->syncWithoutDetaching([$userId]);
2024-08-29 21:50:59 -04:00
}
/**
* The relationship between a user and users they follow.
*
2025-01-30 15:44:05 -05:00
* @return BelongsToMany<User, $this>
*/
public function following(): BelongsToMany
{
return $this->belongsToMany(User::class, 'user_follows', 'follower_id', 'following_id')
->withTimestamps();
}
/**
* Unfollow another user.
*/
2024-08-29 21:50:59 -04:00
public function unfollow(User|int $user): void
{
$userId = $user instanceof User ? $user->id : $user;
2024-09-01 23:42:44 -04:00
if ($this->isFollowing($userId)) {
$this->following()->detach($userId);
}
2024-08-29 21:50:59 -04:00
}
/**
* Check if the user is following another user.
*/
public function isFollowing(User|int $user): bool
{
$userId = $user instanceof User ? $user->id : $user;
return $this->following()->where('following_id', $userId)->exists();
}
2024-07-20 19:52:36 -04:00
/**
* The data that is searchable by Scout.
*/
2024-06-02 22:03:59 -04:00
public function toSearchableArray(): array
{
return [
'id' => (int) $this->id,
'name' => $this->name,
];
}
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
{
$this->load(['bans']);
return $this->isNotBanned();
2024-06-02 22:03:59 -04:00
}
2024-06-16 21:40:00 -04:00
2024-07-20 19:52:36 -04:00
/**
* Check if the user has the role of a moderator.
*/
public function isMod(): bool
{
return Str::lower($this->role?->name) === 'moderator';
}
2024-07-20 19:52:36 -04:00
/**
* Check if the user has the role of an administrator.
*/
public function isAdmin(): bool
{
return Str::lower($this->role?->name) === 'administrator';
}
/**
* Overwritten to instead use the queued version of the VerifyEmail notification.
*/
public function sendEmailVerificationNotification(): void
{
$this->notify(new VerifyEmail);
}
/**
* Overwritten to instead use the queued version of the ResetPassword notification.
*/
2025-01-30 00:23:55 -05:00
public function sendPasswordResetNotification(#[SensitiveParameter] $token): void
{
$this->notify(new ResetPassword($token));
}
/**
* Get the relative URL to the user's profile page.
*/
public function profileUrl(): string
{
return route('user.show', [
'user' => $this->id,
'username' => $this->slug(),
]);
}
2024-07-20 19:52:36 -04:00
/**
* Get the slug of the user's name.
*/
public function slug(): string
{
return Str::lower(Str::slug($this->name));
}
/**
* Assign a role to the user.
*/
2025-01-30 00:23:55 -05:00
public function assignRole(UserRole $userRole): bool
{
2025-01-30 00:23:55 -05:00
$this->role()->associate($userRole);
return $this->save();
}
/**
* The relationship between a user and their role.
2024-09-12 13:19:52 -04:00
*
2025-01-30 15:44:05 -05:00
* @return BelongsTo<UserRole, $this>
*/
public function role(): BelongsTo
{
return $this->belongsTo(UserRole::class, 'user_role_id');
}
/**
* Scope a query by applying QueryFilter filters.
*/
2025-01-30 00:23:55 -05:00
public function scopeFilter(Builder $builder, QueryFilter $queryFilter): Builder
{
2025-01-30 00:23:55 -05:00
return $queryFilter->apply($builder);
}
/**
* The relationship between a user and their OAuth providers.
2025-01-30 15:44:05 -05:00
*
* @return HasMany<OAuthConnection, $this>
*/
public function oAuthConnections(): HasMany
{
return $this->hasMany(OAuthConnection::class);
}
/**
* Handle the about default value if empty. Thanks, MySQL!
*/
protected function about(): Attribute
{
return Attribute::make(
set: function ($value) {
// MySQL will not allow you to set a default value of an empty string for a (LONG)TEXT column. *le sigh*
// NULL is the default. If NULL is saved, we'll swap it out for an empty string.
if (is_null($value)) {
return '';
}
return $value;
},
);
}
/**
* Get the disk that profile photos should be stored on.
*/
protected function profilePhotoDisk(): string
{
return config('filesystems.asset_upload', 'public');
}
2024-07-20 19:52:36 -04:00
/**
* The attributes that should be cast to native types.
*/
2024-06-16 21:40:00 -04:00
protected function casts(): array
{
return [
'id' => 'integer',
'hub_id' => 'integer',
2024-06-16 21:40:00 -04:00
'email_verified_at' => 'datetime',
'password' => 'hashed',
'created_at' => 'datetime',
'updated_at' => 'datetime',
2024-06-16 21:40:00 -04:00
];
}
2024-05-13 18:55:34 -04:00
}