<?php
namespace App\Services;
use App\Enums\Placement;
use App\Exceptions\OperationNotApplicableForSmartPlaylistException;
use App\Models\Playlist;
use App\Models\Song as Playable;
use App\Models\User;
use App\Repositories\SongRepository;
use App\Values\Playlist\PlaylistCreateData;
use App\Values\Playlist\PlaylistUpdateData;
use Illuminate\Database\Eloquent\Collection as EloquentCollection;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\DB;
class PlaylistService
{
public function __construct(
private readonly ImageStorage $imageStorage,
private readonly SongRepository $songRepository,
) {
}
public function createPlaylist(PlaylistCreateData $data, User $user): Playlist
{
$cover = rescue_if($data->cover, function () use ($data) {
return $this->imageStorage->storeImage($data->cover);
});
return DB::transaction(
static function () use ($data, $cover, $user): Playlist {
/** @var Playlist $playlist */
$playlist = Playlist::query()->create([
'name' => $data->name,
'description' => $data->description,
'rules' => $data->ruleGroups,
'cover' => $cover,
]);
$user->ownedPlaylists()->attach($playlist, [
'role' => 'owner',
]);
$playlist->folders()->attach($data->folderId);
if (!$playlist->is_smart && $data->playableIds) {
$playlist->addPlayables($data->playableIds, $user);
}
return $playlist;
}
);
}
public function updatePlaylist(Playlist $playlist, PlaylistUpdateData $dto): Playlist
{
$data = [
'name' => $dto->name,
'description' => $dto->description,
'rules' => $dto->ruleGroups,
];
if (is_string($dto->cover)) {
$data['cover'] = rescue_if($dto->cover, fn () => $this->imageStorage->storeImage($dto->cover), '');
}
$playlist->update($data);
if ($dto->folderId) {
$playlist->folders()->syncWithoutDetaching([$dto->folderId]);
}
return $playlist->refresh();
}
public function addPlayablesToPlaylist(
Playlist $playlist,
Collection|Playable|array $playables,
User $user
): EloquentCollection {
return DB::transaction(function () use ($playlist, $playables, $user) {
$playables = Collection::wrap($playables);
$playlist->addPlayables(
$playables->filter(static fn ($song): bool => !$playlist->playables->contains($song)),
$user
);
if ($playlist->is_collaborative) {
$this->makePlaylistContentPublic($playlist);
}
return $this->songRepository->getManyInCollaborativeContext(
ids: $playables->pluck('id')->all(),
scopedUser: $user
);
});
}
public function removePlayablesFromPlaylist(Playlist $playlist, Collection|Playable|array $playables): void
{
$playlist->removePlayables($playables);
}
public function makePlaylistContentPublic(Playlist $playlist): void
{
$playlist->playables()->where('is_public', false)->update(['is_public' => true]);
}
public function movePlayablesInPlaylist(
Playlist $playlist,
array $movingIds,
string $target,
Placement $placement,
): void {
throw_if($playlist->is_smart, OperationNotApplicableForSmartPlaylistException::class);
DB::transaction(static function () use ($playlist, $movingIds, $target, $placement): void {
$targetPosition = $playlist->playables()->wherePivot('song_id', $target)->value('position');
$insertPosition = $placement === Placement::BEFORE ? $targetPosition : $targetPosition + 1;
$playlist->playables()
->newPivotQuery()
->where('position', $placement === Placement::BEFORE ? '>=' : '>', $targetPosition)
->whereNotIn('song_id', $movingIds)
->increment('position', count($movingIds));
$values = [];
foreach ($movingIds as $id) {
$values[$id] = ['position' => $insertPosition++];
}
$playlist->playables()->syncWithoutDetaching($values);
});
}
}