2021-05-20 19:28:18 +00:00
|
|
|
#nullable disable
|
|
|
|
|
2020-05-29 09:28:19 +00:00
|
|
|
#pragma warning disable CS1591
|
|
|
|
|
2014-08-02 02:34:45 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2019-02-28 22:22:57 +00:00
|
|
|
using System.Globalization;
|
2014-08-02 02:34:45 +00:00
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
2020-05-20 17:07:53 +00:00
|
|
|
using Jellyfin.Data.Entities;
|
2023-05-13 18:44:31 +00:00
|
|
|
using Jellyfin.Data.Enums;
|
2024-01-17 15:51:39 +00:00
|
|
|
using Jellyfin.Extensions;
|
2017-05-21 07:25:49 +00:00
|
|
|
using MediaBrowser.Controller.Dto;
|
2019-01-13 19:22:00 +00:00
|
|
|
using MediaBrowser.Controller.Entities;
|
|
|
|
using MediaBrowser.Controller.Entities.Audio;
|
2020-03-13 22:14:25 +00:00
|
|
|
using MediaBrowser.Controller.Extensions;
|
2019-01-13 19:22:00 +00:00
|
|
|
using MediaBrowser.Controller.Library;
|
|
|
|
using MediaBrowser.Controller.Playlists;
|
|
|
|
using MediaBrowser.Controller.Providers;
|
|
|
|
using MediaBrowser.Model.Entities;
|
2016-10-25 19:02:04 +00:00
|
|
|
using MediaBrowser.Model.IO;
|
2019-01-13 19:22:00 +00:00
|
|
|
using MediaBrowser.Model.Playlists;
|
2020-03-13 22:14:25 +00:00
|
|
|
using Microsoft.Extensions.Configuration;
|
2019-01-13 19:22:00 +00:00
|
|
|
using Microsoft.Extensions.Logging;
|
2018-09-12 17:26:21 +00:00
|
|
|
using PlaylistsNET.Content;
|
|
|
|
using PlaylistsNET.Models;
|
2020-05-20 17:07:53 +00:00
|
|
|
using Genre = MediaBrowser.Controller.Entities.Genre;
|
|
|
|
using MusicAlbum = MediaBrowser.Controller.Entities.Audio.MusicAlbum;
|
2014-08-02 02:34:45 +00:00
|
|
|
|
2016-11-02 21:05:17 +00:00
|
|
|
namespace Emby.Server.Implementations.Playlists
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
|
|
|
public class PlaylistManager : IPlaylistManager
|
|
|
|
{
|
|
|
|
private readonly ILibraryManager _libraryManager;
|
|
|
|
private readonly IFileSystem _fileSystem;
|
|
|
|
private readonly ILibraryMonitor _iLibraryMonitor;
|
2020-06-06 00:15:56 +00:00
|
|
|
private readonly ILogger<PlaylistManager> _logger;
|
2014-08-02 02:34:45 +00:00
|
|
|
private readonly IUserManager _userManager;
|
2015-07-24 15:20:11 +00:00
|
|
|
private readonly IProviderManager _providerManager;
|
2020-03-13 22:14:25 +00:00
|
|
|
private readonly IConfiguration _appConfig;
|
2014-08-02 02:34:45 +00:00
|
|
|
|
2019-01-17 22:55:05 +00:00
|
|
|
public PlaylistManager(
|
|
|
|
ILibraryManager libraryManager,
|
|
|
|
IFileSystem fileSystem,
|
|
|
|
ILibraryMonitor iLibraryMonitor,
|
2020-03-03 22:07:10 +00:00
|
|
|
ILogger<PlaylistManager> logger,
|
2019-01-17 22:55:05 +00:00
|
|
|
IUserManager userManager,
|
2020-03-13 22:14:25 +00:00
|
|
|
IProviderManager providerManager,
|
|
|
|
IConfiguration appConfig)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
|
|
|
_libraryManager = libraryManager;
|
|
|
|
_fileSystem = fileSystem;
|
|
|
|
_iLibraryMonitor = iLibraryMonitor;
|
2020-03-03 22:07:10 +00:00
|
|
|
_logger = logger;
|
2014-08-02 02:34:45 +00:00
|
|
|
_userManager = userManager;
|
2015-07-24 15:20:11 +00:00
|
|
|
_providerManager = providerManager;
|
2020-03-13 22:14:25 +00:00
|
|
|
_appConfig = appConfig;
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
public Playlist GetPlaylistForUser(Guid playlistId, Guid userId)
|
2024-03-26 14:29:48 +00:00
|
|
|
{
|
|
|
|
return GetPlaylists(userId).Where(p => p.Id.Equals(playlistId)).FirstOrDefault();
|
|
|
|
}
|
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
public IEnumerable<Playlist> GetPlaylists(Guid userId)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2014-09-14 15:10:51 +00:00
|
|
|
var user = _userManager.GetUserById(userId);
|
2024-05-06 01:22:21 +00:00
|
|
|
return _libraryManager.GetItemList(new InternalItemsQuery
|
|
|
|
{
|
|
|
|
IncludeItemTypes = [BaseItemKind.Playlist],
|
|
|
|
Recursive = true,
|
|
|
|
DtoOptions = new DtoOptions(false)
|
|
|
|
})
|
|
|
|
.Cast<Playlist>()
|
|
|
|
.Where(p => p.IsVisible(user));
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
public async Task<PlaylistCreationResult> CreatePlaylist(PlaylistCreationRequest request)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
var name = request.Name;
|
2019-12-14 03:51:27 +00:00
|
|
|
var folderName = _fileSystem.GetValidFilename(name);
|
2024-04-01 18:43:05 +00:00
|
|
|
var parentFolder = GetPlaylistsFolder(request.UserId);
|
2022-12-05 14:00:20 +00:00
|
|
|
if (parentFolder is null)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2021-12-24 21:18:24 +00:00
|
|
|
throw new ArgumentException(nameof(parentFolder));
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
if (request.MediaType is null || request.MediaType == MediaType.Unknown)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
foreach (var itemId in request.ItemIdList)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-03-26 15:13:07 +00:00
|
|
|
var item = _libraryManager.GetItemById(itemId) ?? throw new ArgumentException("No item exists with the supplied Id");
|
2023-10-14 17:01:03 +00:00
|
|
|
if (item.MediaType != MediaType.Unknown)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
request.MediaType = item.MediaType;
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
else if (item is MusicArtist || item is MusicAlbum || item is MusicGenre)
|
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
request.MediaType = MediaType.Audio;
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
else if (item is Genre)
|
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
request.MediaType = MediaType.Video;
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-09-10 20:37:53 +00:00
|
|
|
if (item is Folder folder)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
request.MediaType = folder.GetRecursiveChildren(i => !i.IsFolder && i.SupportsAddingToPlaylist)
|
2014-08-03 02:16:37 +00:00
|
|
|
.Select(i => i.MediaType)
|
2023-05-13 18:44:31 +00:00
|
|
|
.FirstOrDefault(i => i != MediaType.Unknown);
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
if (request.MediaType is not null && request.MediaType != MediaType.Unknown)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
if (request.MediaType is null || request.MediaType == MediaType.Unknown)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
request.MediaType = MediaType.Audio;
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
var user = _userManager.GetUserById(request.UserId);
|
2014-08-02 02:34:45 +00:00
|
|
|
var path = Path.Combine(parentFolder.Path, folderName);
|
2014-08-03 02:16:37 +00:00
|
|
|
path = GetTargetPath(path);
|
2014-08-02 02:34:45 +00:00
|
|
|
|
|
|
|
_iLibraryMonitor.ReportFileSystemChangeBeginning(path);
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2019-01-26 21:08:04 +00:00
|
|
|
Directory.CreateDirectory(path);
|
2014-08-03 02:16:37 +00:00
|
|
|
var playlist = new Playlist
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
|
|
|
Name = name,
|
2018-09-12 17:26:21 +00:00
|
|
|
Path = path,
|
2024-04-01 18:43:05 +00:00
|
|
|
OwnerUserId = request.UserId,
|
|
|
|
Shares = request.Users ?? [],
|
|
|
|
OpenAccess = request.Public ?? false
|
2014-08-02 02:34:45 +00:00
|
|
|
};
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
playlist.SetMediaType(request.MediaType);
|
2021-07-10 16:09:02 +00:00
|
|
|
parentFolder.AddChild(playlist);
|
2014-08-02 02:34:45 +00:00
|
|
|
|
2019-09-10 20:37:53 +00:00
|
|
|
await playlist.RefreshMetadata(new MetadataRefreshOptions(new DirectoryService(_fileSystem)) { ForceSave = true }, CancellationToken.None)
|
2014-08-02 02:34:45 +00:00
|
|
|
.ConfigureAwait(false);
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
if (request.ItemIdList.Count > 0)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2024-04-01 18:43:05 +00:00
|
|
|
await AddToPlaylistInternal(playlist.Id, request.ItemIdList, user, new DtoOptions(false)
|
2017-05-21 07:25:49 +00:00
|
|
|
{
|
|
|
|
EnableImages = true
|
2020-08-21 20:19:16 +00:00
|
|
|
}).ConfigureAwait(false);
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2020-04-05 16:10:56 +00:00
|
|
|
return new PlaylistCreationResult(playlist.Id.ToString("N", CultureInfo.InvariantCulture));
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
finally
|
|
|
|
{
|
|
|
|
// Refresh handled internally
|
|
|
|
_iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-06 01:22:21 +00:00
|
|
|
private List<Playlist> GetUserPlaylists(Guid userId)
|
|
|
|
{
|
|
|
|
var user = _userManager.GetUserById(userId);
|
|
|
|
|
|
|
|
return GetPlaylistsFolder(userId).GetChildren(user, true).OfType<Playlist>().ToList();
|
|
|
|
}
|
|
|
|
|
2024-03-26 14:29:48 +00:00
|
|
|
private static string GetTargetPath(string path)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2019-01-26 21:59:53 +00:00
|
|
|
while (Directory.Exists(path))
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
|
|
|
path += "1";
|
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2024-03-26 22:45:14 +00:00
|
|
|
private IReadOnlyList<BaseItem> GetPlaylistItems(IEnumerable<Guid> itemIds, MediaType playlistMediaType, User user, DtoOptions options)
|
2014-08-03 02:16:37 +00:00
|
|
|
{
|
2024-03-26 22:45:14 +00:00
|
|
|
var items = itemIds.Select(_libraryManager.GetItemById).Where(i => i is not null);
|
2014-08-03 02:16:37 +00:00
|
|
|
|
2017-05-21 07:25:49 +00:00
|
|
|
return Playlist.GetPlaylistItems(playlistMediaType, items, user, options);
|
2014-08-22 02:24:38 +00:00
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
public Task AddItemToPlaylistAsync(Guid playlistId, IReadOnlyCollection<Guid> itemIds, Guid userId)
|
2014-08-22 02:24:38 +00:00
|
|
|
{
|
2024-01-17 15:51:39 +00:00
|
|
|
var user = userId.IsEmpty() ? null : _userManager.GetUserById(userId);
|
2014-08-22 02:24:38 +00:00
|
|
|
|
2020-08-21 20:01:19 +00:00
|
|
|
return AddToPlaylistInternal(playlistId, itemIds, user, new DtoOptions(false)
|
2017-05-21 07:25:49 +00:00
|
|
|
{
|
|
|
|
EnableImages = true
|
|
|
|
});
|
2014-08-03 02:16:37 +00:00
|
|
|
}
|
|
|
|
|
2020-11-17 03:29:46 +00:00
|
|
|
private async Task AddToPlaylistInternal(Guid playlistId, IReadOnlyCollection<Guid> newItemIds, User user, DtoOptions options)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2020-03-02 20:12:35 +00:00
|
|
|
// Retrieve the existing playlist
|
|
|
|
var playlist = _libraryManager.GetItemById(playlistId) as Playlist
|
|
|
|
?? throw new ArgumentException("No Playlist exists with Id " + playlistId);
|
2014-08-02 02:34:45 +00:00
|
|
|
|
2020-03-02 20:12:35 +00:00
|
|
|
// Retrieve all the items to be added to the playlist
|
2020-03-13 22:14:25 +00:00
|
|
|
var newItems = GetPlaylistItems(newItemIds, playlist.MediaType, user, options)
|
|
|
|
.Where(i => i.SupportsAddingToPlaylist);
|
|
|
|
|
|
|
|
// Filter out duplicate items, if necessary
|
|
|
|
if (!_appConfig.DoPlaylistsAllowDuplicates())
|
|
|
|
{
|
|
|
|
var existingIds = playlist.LinkedChildren.Select(c => c.ItemId).ToHashSet();
|
|
|
|
newItems = newItems
|
|
|
|
.Where(i => !existingIds.Contains(i.Id))
|
|
|
|
.Distinct();
|
|
|
|
}
|
2014-08-02 02:34:45 +00:00
|
|
|
|
2020-03-13 22:14:25 +00:00
|
|
|
// Create a list of the new linked children to add to the playlist
|
|
|
|
var childrenToAdd = newItems
|
2021-04-17 10:19:09 +00:00
|
|
|
.Select(LinkedChild.Create)
|
2020-03-03 17:18:00 +00:00
|
|
|
.ToList();
|
2020-03-02 20:12:35 +00:00
|
|
|
|
|
|
|
// Log duplicates that have been ignored, if any
|
2020-03-13 22:14:25 +00:00
|
|
|
int numDuplicates = newItemIds.Count - childrenToAdd.Count;
|
2020-03-03 16:47:16 +00:00
|
|
|
if (numDuplicates > 0)
|
2014-08-07 02:51:09 +00:00
|
|
|
{
|
2020-03-02 20:12:35 +00:00
|
|
|
_logger.LogWarning("Ignored adding {DuplicateCount} duplicate items to playlist {PlaylistName}.", numDuplicates, playlist.Name);
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2020-03-03 16:48:11 +00:00
|
|
|
// Do nothing else if there are no items to add to the playlist
|
2020-03-13 22:14:25 +00:00
|
|
|
if (childrenToAdd.Count == 0)
|
2020-03-03 16:48:11 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-03-02 20:12:35 +00:00
|
|
|
// Update the playlist in the repository
|
2024-05-06 01:22:21 +00:00
|
|
|
playlist.LinkedChildren = [.. playlist.LinkedChildren, .. childrenToAdd];
|
2015-07-24 15:20:11 +00:00
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2020-03-02 20:12:35 +00:00
|
|
|
// Refresh playlist metadata
|
2019-09-10 20:37:53 +00:00
|
|
|
_providerManager.QueueRefresh(
|
|
|
|
playlist.Id,
|
2020-03-03 13:40:07 +00:00
|
|
|
new MetadataRefreshOptions(new DirectoryService(_fileSystem))
|
|
|
|
{
|
|
|
|
ForceSave = true
|
|
|
|
},
|
2019-09-10 20:37:53 +00:00
|
|
|
RefreshPriority.High);
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
public async Task RemoveItemFromPlaylistAsync(string playlistId, IEnumerable<string> entryIds)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2021-08-28 22:32:50 +00:00
|
|
|
if (_libraryManager.GetItemById(playlistId) is not Playlist playlist)
|
2014-08-06 04:18:13 +00:00
|
|
|
{
|
|
|
|
throw new ArgumentException("No Playlist exists with the supplied Id");
|
|
|
|
}
|
|
|
|
|
2014-08-11 23:41:11 +00:00
|
|
|
var children = playlist.GetManageableItems().ToList();
|
2014-08-06 04:18:13 +00:00
|
|
|
|
|
|
|
var idList = entryIds.ToList();
|
|
|
|
|
2014-08-11 23:41:11 +00:00
|
|
|
var removals = children.Where(i => idList.Contains(i.Item1.Id));
|
2014-08-06 04:18:13 +00:00
|
|
|
|
|
|
|
playlist.LinkedChildren = children.Except(removals)
|
2014-08-11 23:41:11 +00:00
|
|
|
.Select(i => i.Item1)
|
2017-08-10 18:01:31 +00:00
|
|
|
.ToArray();
|
2014-08-06 04:18:13 +00:00
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2019-12-14 03:51:27 +00:00
|
|
|
_providerManager.QueueRefresh(
|
|
|
|
playlist.Id,
|
|
|
|
new MetadataRefreshOptions(new DirectoryService(_fileSystem))
|
|
|
|
{
|
|
|
|
ForceSave = true
|
|
|
|
},
|
|
|
|
RefreshPriority.High);
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
|
2020-08-21 20:01:19 +00:00
|
|
|
public async Task MoveItemAsync(string playlistId, string entryId, int newIndex)
|
2015-10-15 05:48:03 +00:00
|
|
|
{
|
2021-08-28 22:32:50 +00:00
|
|
|
if (_libraryManager.GetItemById(playlistId) is not Playlist playlist)
|
2015-10-15 05:48:03 +00:00
|
|
|
{
|
|
|
|
throw new ArgumentException("No Playlist exists with the supplied Id");
|
|
|
|
}
|
|
|
|
|
|
|
|
var children = playlist.GetManageableItems().ToList();
|
|
|
|
|
|
|
|
var oldIndex = children.FindIndex(i => string.Equals(entryId, i.Item1.Id, StringComparison.OrdinalIgnoreCase));
|
|
|
|
|
2015-10-15 15:51:00 +00:00
|
|
|
if (oldIndex == newIndex)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-10-15 05:48:03 +00:00
|
|
|
var item = playlist.LinkedChildren[oldIndex];
|
|
|
|
|
2017-08-10 18:01:31 +00:00
|
|
|
var newList = playlist.LinkedChildren.ToList();
|
2016-07-16 18:02:39 +00:00
|
|
|
|
2017-08-10 18:01:31 +00:00
|
|
|
newList.Remove(item);
|
|
|
|
|
|
|
|
if (newIndex >= newList.Count)
|
2016-07-16 18:02:39 +00:00
|
|
|
{
|
2017-08-10 18:01:31 +00:00
|
|
|
newList.Add(item);
|
2016-07-16 18:02:39 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-08-10 18:01:31 +00:00
|
|
|
newList.Insert(newIndex, item);
|
2016-07-16 18:02:39 +00:00
|
|
|
}
|
2015-10-15 05:48:03 +00:00
|
|
|
|
2024-03-26 14:29:48 +00:00
|
|
|
playlist.LinkedChildren = [.. newList];
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
2023-05-15 12:45:33 +00:00
|
|
|
/// <inheritdoc />
|
|
|
|
public void SavePlaylistFile(Playlist item)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2019-12-14 03:51:27 +00:00
|
|
|
// this is probably best done as a metadata provider
|
|
|
|
// saving a file over itself will require some work to prevent this from happening when not needed
|
2018-09-12 17:26:21 +00:00
|
|
|
var playlistPath = item.Path;
|
2023-10-05 22:40:09 +00:00
|
|
|
var extension = Path.GetExtension(playlistPath.AsSpan());
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2023-10-05 22:40:09 +00:00
|
|
|
if (extension.Equals(".wpl", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
|
|
|
var playlist = new WplPlaylist();
|
|
|
|
foreach (var child in item.GetLinkedChildren())
|
|
|
|
{
|
|
|
|
var entry = new WplPlaylistEntry()
|
|
|
|
{
|
|
|
|
Path = NormalizeItemPath(playlistPath, child.Path),
|
|
|
|
TrackTitle = child.Name,
|
|
|
|
AlbumTitle = child.Album
|
|
|
|
};
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasAlbumArtist hasAlbumArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasArtist hasArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.TrackArtist = hasArtist.Artists.Count > 0 ? hasArtist.Artists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (child.RunTimeTicks.HasValue)
|
|
|
|
{
|
|
|
|
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
|
|
|
|
}
|
2019-12-14 03:51:27 +00:00
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
playlist.PlaylistEntries.Add(entry);
|
|
|
|
}
|
|
|
|
|
2019-01-26 22:09:07 +00:00
|
|
|
string text = new WplContent().ToText(playlist);
|
|
|
|
File.WriteAllText(playlistPath, text);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
2023-10-05 22:40:09 +00:00
|
|
|
else if (extension.Equals(".zpl", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
|
|
|
var playlist = new ZplPlaylist();
|
|
|
|
foreach (var child in item.GetLinkedChildren())
|
|
|
|
{
|
|
|
|
var entry = new ZplPlaylistEntry()
|
|
|
|
{
|
|
|
|
Path = NormalizeItemPath(playlistPath, child.Path),
|
|
|
|
TrackTitle = child.Name,
|
|
|
|
AlbumTitle = child.Album
|
|
|
|
};
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasAlbumArtist hasAlbumArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasArtist hasArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.TrackArtist = hasArtist.Artists.Count > 0 ? hasArtist.Artists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (child.RunTimeTicks.HasValue)
|
|
|
|
{
|
|
|
|
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
|
|
|
|
}
|
2020-06-15 21:43:52 +00:00
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
playlist.PlaylistEntries.Add(entry);
|
|
|
|
}
|
|
|
|
|
2019-01-26 22:09:07 +00:00
|
|
|
string text = new ZplContent().ToText(playlist);
|
|
|
|
File.WriteAllText(playlistPath, text);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
2023-10-05 22:40:09 +00:00
|
|
|
else if (extension.Equals(".m3u", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
var playlist = new M3uPlaylist
|
|
|
|
{
|
|
|
|
IsExtended = true
|
|
|
|
};
|
2018-09-12 17:26:21 +00:00
|
|
|
foreach (var child in item.GetLinkedChildren())
|
|
|
|
{
|
|
|
|
var entry = new M3uPlaylistEntry()
|
|
|
|
{
|
|
|
|
Path = NormalizeItemPath(playlistPath, child.Path),
|
|
|
|
Title = child.Name,
|
|
|
|
Album = child.Album
|
|
|
|
};
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasAlbumArtist hasAlbumArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (child.RunTimeTicks.HasValue)
|
|
|
|
{
|
|
|
|
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
|
|
|
|
}
|
2019-12-14 03:51:27 +00:00
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
playlist.PlaylistEntries.Add(entry);
|
|
|
|
}
|
|
|
|
|
2019-01-26 22:09:07 +00:00
|
|
|
string text = new M3uContent().ToText(playlist);
|
|
|
|
File.WriteAllText(playlistPath, text);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
2023-10-05 22:40:09 +00:00
|
|
|
else if (extension.Equals(".m3u8", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2024-03-26 14:29:48 +00:00
|
|
|
var playlist = new M3uPlaylist
|
|
|
|
{
|
|
|
|
IsExtended = true
|
|
|
|
};
|
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
foreach (var child in item.GetLinkedChildren())
|
|
|
|
{
|
|
|
|
var entry = new M3uPlaylistEntry()
|
|
|
|
{
|
|
|
|
Path = NormalizeItemPath(playlistPath, child.Path),
|
|
|
|
Title = child.Name,
|
|
|
|
Album = child.Album
|
|
|
|
};
|
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
if (child is IHasAlbumArtist hasAlbumArtist)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
entry.AlbumArtist = hasAlbumArtist.AlbumArtists.Count > 0 ? hasAlbumArtist.AlbumArtists[0] : null;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (child.RunTimeTicks.HasValue)
|
|
|
|
{
|
|
|
|
entry.Duration = TimeSpan.FromTicks(child.RunTimeTicks.Value);
|
|
|
|
}
|
2019-12-14 03:51:27 +00:00
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
playlist.PlaylistEntries.Add(entry);
|
|
|
|
}
|
|
|
|
|
2020-06-15 14:34:24 +00:00
|
|
|
string text = new M3uContent().ToText(playlist);
|
2019-01-26 22:09:07 +00:00
|
|
|
File.WriteAllText(playlistPath, text);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
2023-10-05 22:40:09 +00:00
|
|
|
else if (extension.Equals(".pls", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
|
|
|
var playlist = new PlsPlaylist();
|
|
|
|
foreach (var child in item.GetLinkedChildren())
|
|
|
|
{
|
|
|
|
var entry = new PlsPlaylistEntry()
|
|
|
|
{
|
|
|
|
Path = NormalizeItemPath(playlistPath, child.Path),
|
|
|
|
Title = child.Name
|
|
|
|
};
|
|
|
|
|
|
|
|
if (child.RunTimeTicks.HasValue)
|
|
|
|
{
|
|
|
|
entry.Length = TimeSpan.FromTicks(child.RunTimeTicks.Value);
|
|
|
|
}
|
2019-12-14 03:51:27 +00:00
|
|
|
|
2018-09-12 17:26:21 +00:00
|
|
|
playlist.PlaylistEntries.Add(entry);
|
|
|
|
}
|
|
|
|
|
2019-01-26 22:09:07 +00:00
|
|
|
string text = new PlsContent().ToText(playlist);
|
|
|
|
File.WriteAllText(playlistPath, text);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-26 14:29:48 +00:00
|
|
|
private static string NormalizeItemPath(string playlistPath, string itemPath)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2019-01-26 20:47:11 +00:00
|
|
|
return MakeRelativePath(Path.GetDirectoryName(playlistPath), itemPath);
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
2019-01-06 20:50:43 +00:00
|
|
|
private static string MakeRelativePath(string folderPath, string fileAbsolutePath)
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2022-10-13 17:08:00 +00:00
|
|
|
ArgumentException.ThrowIfNullOrEmpty(folderPath);
|
|
|
|
ArgumentException.ThrowIfNullOrEmpty(fileAbsolutePath);
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2019-12-14 03:51:27 +00:00
|
|
|
if (!folderPath.EndsWith(Path.DirectorySeparatorChar))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
folderPath += Path.DirectorySeparatorChar;
|
2018-09-12 17:26:21 +00:00
|
|
|
}
|
|
|
|
|
2019-01-13 20:37:13 +00:00
|
|
|
var folderUri = new Uri(folderPath);
|
|
|
|
var fileAbsoluteUri = new Uri(fileAbsolutePath);
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2019-12-14 03:51:27 +00:00
|
|
|
// path can't be made relative
|
|
|
|
if (folderUri.Scheme != fileAbsoluteUri.Scheme)
|
|
|
|
{
|
|
|
|
return fileAbsolutePath;
|
|
|
|
}
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2019-01-13 20:37:13 +00:00
|
|
|
var relativeUri = folderUri.MakeRelativeUri(fileAbsoluteUri);
|
2019-01-06 20:50:43 +00:00
|
|
|
string relativePath = Uri.UnescapeDataString(relativeUri.ToString());
|
2018-09-12 17:26:21 +00:00
|
|
|
|
2021-11-15 14:57:07 +00:00
|
|
|
if (fileAbsoluteUri.Scheme.Equals("file", StringComparison.OrdinalIgnoreCase))
|
2018-09-12 17:26:21 +00:00
|
|
|
{
|
|
|
|
relativePath = relativePath.Replace(Path.AltDirectorySeparatorChar, Path.DirectorySeparatorChar);
|
|
|
|
}
|
|
|
|
|
|
|
|
return relativePath;
|
|
|
|
}
|
|
|
|
|
2024-05-06 01:22:21 +00:00
|
|
|
/// <inheritdoc />
|
2023-09-16 13:08:45 +00:00
|
|
|
public Folder GetPlaylistsFolder()
|
|
|
|
{
|
|
|
|
return GetPlaylistsFolder(Guid.Empty);
|
|
|
|
}
|
|
|
|
|
2024-05-06 01:22:21 +00:00
|
|
|
/// <inheritdoc />
|
2018-09-12 17:26:21 +00:00
|
|
|
public Folder GetPlaylistsFolder(Guid userId)
|
2014-08-02 02:34:45 +00:00
|
|
|
{
|
2020-07-24 14:37:54 +00:00
|
|
|
const string TypeName = "PlaylistsFolder";
|
2016-11-02 21:05:17 +00:00
|
|
|
|
2020-07-24 14:37:54 +00:00
|
|
|
return _libraryManager.RootFolder.Children.OfType<Folder>().FirstOrDefault(i => string.Equals(i.GetType().Name, TypeName, StringComparison.Ordinal)) ??
|
|
|
|
_libraryManager.GetUserRootFolder().Children.OfType<Folder>().FirstOrDefault(i => string.Equals(i.GetType().Name, TypeName, StringComparison.Ordinal));
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
2023-03-10 16:46:59 +00:00
|
|
|
|
2023-03-25 17:41:09 +00:00
|
|
|
/// <inheritdoc />
|
2023-03-12 18:42:18 +00:00
|
|
|
public async Task RemovePlaylistsAsync(Guid userId)
|
2023-03-10 16:46:59 +00:00
|
|
|
{
|
2024-05-06 01:22:21 +00:00
|
|
|
var playlists = GetUserPlaylists(userId);
|
2023-03-10 16:46:59 +00:00
|
|
|
foreach (var playlist in playlists)
|
|
|
|
{
|
|
|
|
// Update owner if shared
|
2024-04-30 19:32:59 +00:00
|
|
|
var rankedShares = playlist.Shares.OrderByDescending(x => x.CanEdit).ToList();
|
|
|
|
if (rankedShares.Count > 0)
|
2023-03-10 16:46:59 +00:00
|
|
|
{
|
2024-04-01 17:59:48 +00:00
|
|
|
playlist.OwnerUserId = rankedShares[0].UserId;
|
2023-03-10 16:46:59 +00:00
|
|
|
playlist.Shares = rankedShares.Skip(1).ToArray();
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2023-03-10 16:46:59 +00:00
|
|
|
}
|
2023-05-12 13:11:59 +00:00
|
|
|
else if (!playlist.OpenAccess)
|
2023-03-10 16:46:59 +00:00
|
|
|
{
|
|
|
|
// Remove playlist if not shared
|
|
|
|
_libraryManager.DeleteItem(
|
|
|
|
playlist,
|
|
|
|
new DeleteOptions
|
|
|
|
{
|
|
|
|
DeleteFileLocation = false,
|
|
|
|
DeleteFromExternalProvider = false
|
|
|
|
},
|
|
|
|
playlist.GetParent(),
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-26 14:29:48 +00:00
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
public async Task UpdatePlaylist(PlaylistUpdateRequest request)
|
2024-03-26 14:29:48 +00:00
|
|
|
{
|
2024-04-02 06:08:36 +00:00
|
|
|
var playlist = GetPlaylistForUser(request.Id, request.UserId);
|
2024-04-01 18:43:05 +00:00
|
|
|
|
|
|
|
if (request.Ids is not null)
|
|
|
|
{
|
|
|
|
playlist.LinkedChildren = [];
|
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
|
|
|
|
|
|
|
var user = _userManager.GetUserById(request.UserId);
|
|
|
|
await AddToPlaylistInternal(request.Id, request.Ids, user, new DtoOptions(false)
|
2024-05-06 01:22:21 +00:00
|
|
|
{
|
|
|
|
EnableImages = true
|
|
|
|
}).ConfigureAwait(false);
|
2024-04-01 18:43:05 +00:00
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
playlist = GetPlaylistForUser(request.Id, request.UserId);
|
2024-04-01 18:43:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (request.Name is not null)
|
|
|
|
{
|
|
|
|
playlist.Name = request.Name;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request.Users is not null)
|
|
|
|
{
|
|
|
|
playlist.Shares = request.Users;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request.Public is not null)
|
|
|
|
{
|
|
|
|
playlist.OpenAccess = request.Public.Value;
|
|
|
|
}
|
2024-03-26 14:29:48 +00:00
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2024-03-26 14:29:48 +00:00
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
public async Task AddUserToShares(PlaylistUserUpdateRequest request)
|
2024-03-26 14:29:48 +00:00
|
|
|
{
|
2024-04-02 06:08:36 +00:00
|
|
|
var userId = request.UserId;
|
|
|
|
var playlist = GetPlaylistForUser(request.Id, userId);
|
2024-03-26 14:29:48 +00:00
|
|
|
var shares = playlist.Shares.ToList();
|
2024-04-02 06:08:36 +00:00
|
|
|
var existingUserShare = shares.FirstOrDefault(s => s.UserId.Equals(userId));
|
2024-03-26 14:29:48 +00:00
|
|
|
if (existingUserShare is not null)
|
|
|
|
{
|
|
|
|
shares.Remove(existingUserShare);
|
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
shares.Add(new PlaylistUserPermissions(userId, request.CanEdit ?? false));
|
2024-03-26 14:29:48 +00:00
|
|
|
playlist.Shares = shares;
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2024-03-26 14:29:48 +00:00
|
|
|
}
|
|
|
|
|
2024-04-02 06:08:36 +00:00
|
|
|
public async Task RemoveUserFromShares(Guid playlistId, Guid userId, PlaylistUserPermissions share)
|
2024-03-26 14:29:48 +00:00
|
|
|
{
|
2024-04-02 06:08:36 +00:00
|
|
|
var playlist = GetPlaylistForUser(playlistId, userId);
|
2024-03-26 14:29:48 +00:00
|
|
|
var shares = playlist.Shares.ToList();
|
|
|
|
shares.Remove(share);
|
|
|
|
playlist.Shares = shares;
|
2024-04-01 18:43:05 +00:00
|
|
|
await UpdatePlaylistInternal(playlist).ConfigureAwait(false);
|
2024-03-26 14:29:48 +00:00
|
|
|
}
|
|
|
|
|
2024-04-01 18:43:05 +00:00
|
|
|
private async Task UpdatePlaylistInternal(Playlist playlist)
|
2024-03-26 14:29:48 +00:00
|
|
|
{
|
|
|
|
await playlist.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
|
|
|
|
|
|
|
|
if (playlist.IsFile)
|
|
|
|
{
|
|
|
|
SavePlaylistFile(playlist);
|
|
|
|
}
|
|
|
|
}
|
2014-08-02 02:34:45 +00:00
|
|
|
}
|
|
|
|
}
|