koel/app/Services/InteractionService.php
2024-07-06 17:45:11 +02:00

89 lines
2.7 KiB
PHP

<?php
namespace App\Services;
use App\Events\MultipleSongsLiked;
use App\Events\MultipleSongsUnliked;
use App\Events\SongLikeToggled;
use App\Models\Interaction;
use App\Models\Song as Playable;
use App\Models\User;
use Illuminate\Support\Collection;
class InteractionService
{
public function increasePlayCount(Playable $playable, User $user): Interaction
{
return tap(Interaction::query()->firstOrCreate([
'song_id' => $playable->id,
'user_id' => $user->id,
]), static function (Interaction $interaction): void {
if (!$interaction->exists) {
$interaction->liked = false;
}
$interaction->last_played_at = now();
++$interaction->play_count;
$interaction->save();
});
}
/**
* Like or unlike a song/episode as a user.
*
* @return Interaction The affected Interaction object
*/
public function toggleLike(Playable $playable, User $user): Interaction
{
return tap(Interaction::query()->firstOrCreate([
'song_id' => $playable->id,
'user_id' => $user->id,
]), static function (Interaction $interaction): void {
$interaction->liked = !$interaction->liked;
$interaction->save();
event(new SongLikeToggled($interaction));
});
}
/**
* Like several songs/episodes at once as a user.
*
* @param Collection<array-key, Playable> $playables
*
* @return Collection<array-key, Interaction> The array of Interaction objects
*/
public function likeMany(Collection $playables, User $user): Collection
{
$interactions = $playables->map(static function (Playable $playable) use ($user): Interaction {
return tap(Interaction::query()->firstOrCreate([
'song_id' => $playable->id,
'user_id' => $user->id,
]), static function (Interaction $interaction): void {
$interaction->play_count ??= 0;
$interaction->liked = true;
$interaction->save();
});
});
event(new MultipleSongsLiked($playables, $user));
return $interactions;
}
/**
* Unlike several songs/episodes at once.
*
* @param array<array-key, Playable>|Collection $playables
*/
public function unlikeMany(Collection $playables, User $user): void
{
Interaction::query()
->whereIn('song_id', $playables->pluck('id')->all())
->where('user_id', $user->id)
->update(['liked' => false]);
event(new MultipleSongsUnliked($playables, $user));
}
}