koel/app/Services/PlaylistService.php
2024-02-02 17:26:07 +01:00

142 lines
5 KiB
PHP

<?php
namespace App\Services;
use App\Exceptions\OperationNotApplicableForSmartPlaylistException;
use App\Exceptions\PlaylistBothSongsAndRulesProvidedException;
use App\Facades\License;
use App\Models\Playlist;
use App\Models\PlaylistFolder as Folder;
use App\Models\Song;
use App\Models\User;
use App\Repositories\SongRepository;
use App\Values\SmartPlaylistRuleGroupCollection;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\DB;
use InvalidArgumentException;
use Webmozart\Assert\Assert;
class PlaylistService
{
public function __construct(private SongRepository $songRepository)
{
}
public function createPlaylist(
string $name,
User $user,
?Folder $folder = null,
array $songs = [],
?SmartPlaylistRuleGroupCollection $ruleGroups = null,
bool $ownSongsOnly = false
): Playlist {
if ($folder) {
Assert::true($folder->ownedBy($user), 'The playlist folder does not belong to the user');
}
throw_if($songs && $ruleGroups, new PlaylistBothSongsAndRulesProvidedException());
throw_if($ownSongsOnly && (!$ruleGroups || !License::isPlus()), new InvalidArgumentException(
'"Own songs only" option only works with smart playlists and Plus license.'
));
return DB::transaction(
static function () use ($name, $user, $songs, $folder, $ruleGroups, $ownSongsOnly): Playlist {
/** @var Playlist $playlist */
$playlist = $user->playlists()->create([
'name' => $name,
'rules' => $ruleGroups,
'own_songs_only' => $ownSongsOnly,
'folder_id' => $folder?->id,
]);
if (!$playlist->is_smart && $songs) {
$playlist->addSongs($songs, $user);
}
return $playlist;
}
);
}
public function updatePlaylist(
Playlist $playlist,
string $name,
?Folder $folder = null,
?SmartPlaylistRuleGroupCollection $ruleGroups = null,
bool $ownSongsOnly = false
): Playlist {
if ($folder) {
Assert::true($playlist->ownedBy($folder->user), 'The playlist folder does not belong to the user');
}
throw_if($ownSongsOnly && (!$playlist->is_smart || !License::isPlus()), new InvalidArgumentException(
'"Own songs only" option only works with smart playlists and Plus license.'
));
$playlist->update([
'name' => $name,
'rules' => $ruleGroups,
'folder_id' => $folder?->id,
'own_songs_only' => $ownSongsOnly,
]);
return $playlist;
}
/** @return Collection<Song>|array<array-key, Song> */
public function addSongsToPlaylist(Playlist $playlist, Collection|Song|array $songs, User $user): Collection
{
return DB::transaction(function () use ($playlist, $songs, $user) {
$songs = Collection::wrap($songs);
$playlist->addSongs($songs->filter(static fn ($song): bool => !$playlist->songs->contains($song)), $user);
// if the playlist is collaborative, make the songs public
if ($playlist->is_collaborative) {
$this->makePlaylistSongsPublic($playlist);
}
// we want a fresh copy of the songs with the possibly updated visibility
return $this->songRepository->getManyInCollaborativeContext(
ids: $songs->pluck('id')->all(),
scopedUser: $user
);
});
}
public function removeSongsFromPlaylist(Playlist $playlist, Collection|Song|array $songs): void
{
$playlist->removeSongs($songs);
}
public function makePlaylistSongsPublic(Playlist $playlist): void
{
$playlist->songs()->where('is_public', false)->update(['is_public' => true]);
}
public function moveSongsInPlaylist(Playlist $playlist, array $movingIds, string $target, string $type): void
{
Assert::oneOf($type, ['before', 'after']);
throw_if($playlist->is_smart, OperationNotApplicableForSmartPlaylistException::class);
DB::transaction(static function () use ($playlist, $movingIds, $target, $type): void {
$targetPosition = $playlist->songs()->wherePivot('song_id', $target)->value('position');
$insertPosition = $type === 'before' ? $targetPosition : $targetPosition + 1;
// create a "gap" for the moving songs by incrementing the position of the songs after the target
$playlist->songs()
->newPivotQuery()
->where('position', $type === 'before' ? '>=' : '>', $targetPosition)
->whereNotIn('song_id', $movingIds)
->increment('position', count($movingIds));
$values = [];
foreach ($movingIds as $id) {
$values[$id] = ['position' => $insertPosition++];
}
$playlist->songs()->syncWithoutDetaching($values);
});
}
}