Updated playlist model class. Added playlist tests.
This commit is contained in:
@@ -5,35 +5,37 @@ namespace Harmonia.Core.Models;
|
|||||||
|
|
||||||
public class Playlist(string name)
|
public class Playlist(string name)
|
||||||
{
|
{
|
||||||
public string UID { get; set; } = Guid.NewGuid().ToString();
|
private readonly List<PlaylistSong> _songs = [];
|
||||||
|
|
||||||
|
public string UID { get; init; } = Guid.NewGuid().ToString();
|
||||||
public string Name { get; set; } = name;
|
public string Name { get; set; } = name;
|
||||||
public List<PlaylistSong> Songs { get; set; } = [];
|
public IReadOnlyList<PlaylistSong> Songs => _songs;
|
||||||
public List<GroupOption> GroupOptions { get; set; } = [];
|
public List<GroupOption> GroupOptions { get; set; } = [];
|
||||||
public List<SortOption> SortOptions { get; set; } = [];
|
public List<SortOption> SortOptions { get; set; } = [];
|
||||||
public bool IsLocked { get; set; }
|
public bool IsLocked { get; set; }
|
||||||
|
|
||||||
public event EventHandler<PlaylistUpdatedEventArgs>? PlaylistUpdated;
|
public event EventHandler<PlaylistUpdatedEventArgs>? PlaylistUpdated;
|
||||||
|
|
||||||
public void AddSong(Song song, int? index)
|
public void AddSong(Song song, int? index = null)
|
||||||
{
|
{
|
||||||
AddSongs([song], index);
|
AddSongs([song], index);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void AddSongs(Song[] songs, int? index)
|
public void AddSongs(Song[] songs, int? index = null)
|
||||||
{
|
{
|
||||||
PlaylistSong[] playlistSongs = songs.Select(song => new PlaylistSong(song)).ToArray();
|
PlaylistSong[] playlistSongs = songs.Select(song => new PlaylistSong(song)).ToArray();
|
||||||
|
|
||||||
AddSongs(playlistSongs, index);
|
AddSongs(playlistSongs, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void AddSongs(PlaylistSong[] playlistSongs, int? index)
|
private void AddSongs(PlaylistSong[] playlistSongs, int? index = null)
|
||||||
{
|
{
|
||||||
if (IsLocked)
|
if (IsLocked)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
int insertIndex = index ?? Songs.Count;
|
int insertIndex = index ?? Songs.Count;
|
||||||
|
|
||||||
Songs.InsertRange(insertIndex, playlistSongs);
|
_songs.InsertRange(insertIndex, playlistSongs);
|
||||||
|
|
||||||
PlaylistUpdatedEventArgs eventArgs = new()
|
PlaylistUpdatedEventArgs eventArgs = new()
|
||||||
{
|
{
|
||||||
@@ -48,7 +50,7 @@ public class Playlist(string name)
|
|||||||
|
|
||||||
public void MoveSong(PlaylistSong playlistSong, int newIndex)
|
public void MoveSong(PlaylistSong playlistSong, int newIndex)
|
||||||
{
|
{
|
||||||
int currentIndex = Songs.IndexOf(playlistSong);
|
int currentIndex = _songs.IndexOf(playlistSong);
|
||||||
|
|
||||||
MoveSong(currentIndex, newIndex);
|
MoveSong(currentIndex, newIndex);
|
||||||
}
|
}
|
||||||
@@ -63,8 +65,8 @@ public class Playlist(string name)
|
|||||||
|
|
||||||
PlaylistSong playlistSong = Songs[oldIndex];
|
PlaylistSong playlistSong = Songs[oldIndex];
|
||||||
|
|
||||||
Songs.Remove(playlistSong);
|
_songs.Remove(playlistSong);
|
||||||
Songs.Insert(newIndex, playlistSong);
|
_songs.Insert(newIndex, playlistSong);
|
||||||
|
|
||||||
PlaylistUpdatedEventArgs eventArgs = new()
|
PlaylistUpdatedEventArgs eventArgs = new()
|
||||||
{
|
{
|
||||||
@@ -81,8 +83,8 @@ public class Playlist(string name)
|
|||||||
public void SortSongs(PlaylistSong[] playlistSongs, SortOption[] sortOptions)
|
public void SortSongs(PlaylistSong[] playlistSongs, SortOption[] sortOptions)
|
||||||
{
|
{
|
||||||
Dictionary<int, PlaylistSong> oldPlaylistSongs = playlistSongs
|
Dictionary<int, PlaylistSong> oldPlaylistSongs = playlistSongs
|
||||||
.OrderBy(Songs.IndexOf)
|
.OrderBy(_songs.IndexOf)
|
||||||
.ToDictionary(Songs.IndexOf, playlistSong => playlistSong);
|
.ToDictionary(_songs.IndexOf, playlistSong => playlistSong);
|
||||||
|
|
||||||
Song[] songs = playlistSongs.Select(playlistSong => playlistSong.Song).ToArray();
|
Song[] songs = playlistSongs.Select(playlistSong => playlistSong.Song).ToArray();
|
||||||
Song[] sortedSongs = [.. songs.SortBy(sortOptions)];
|
Song[] sortedSongs = [.. songs.SortBy(sortOptions)];
|
||||||
@@ -98,8 +100,8 @@ public class Playlist(string name)
|
|||||||
if (newPlaylistSong == playlistSong)
|
if (newPlaylistSong == playlistSong)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
Songs.Remove(playlistSong);
|
_songs.RemoveAt(index);
|
||||||
Songs.Insert(index, newPlaylistSong);
|
_songs.Insert(index, newPlaylistSong);
|
||||||
}
|
}
|
||||||
|
|
||||||
PlaylistUpdatedEventArgs eventArgs = new()
|
PlaylistUpdatedEventArgs eventArgs = new()
|
||||||
@@ -120,22 +122,9 @@ public class Playlist(string name)
|
|||||||
|
|
||||||
public void RemoveSongs(int index, int count)
|
public void RemoveSongs(int index, int count)
|
||||||
{
|
{
|
||||||
if (IsLocked)
|
PlaylistSong[] playlistSongs = [.. _songs.GetRange(index, count)];
|
||||||
return;
|
|
||||||
|
|
||||||
PlaylistSong[] playlistSongs = [.. Songs.GetRange(index, count)];
|
RemoveSongs(playlistSongs);
|
||||||
|
|
||||||
Songs.RemoveRange(index, count);
|
|
||||||
|
|
||||||
PlaylistUpdatedEventArgs eventArgs = new()
|
|
||||||
{
|
|
||||||
Action = PlaylistUpdateAction.Remove,
|
|
||||||
Index = index,
|
|
||||||
Count = count,
|
|
||||||
Songs = playlistSongs
|
|
||||||
};
|
|
||||||
|
|
||||||
PlaylistUpdated?.Invoke(this, eventArgs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void RemoveSong(PlaylistSong playlistSong)
|
public void RemoveSong(PlaylistSong playlistSong)
|
||||||
@@ -145,11 +134,14 @@ public class Playlist(string name)
|
|||||||
|
|
||||||
public void RemoveSongs(PlaylistSong[] playlistSongs)
|
public void RemoveSongs(PlaylistSong[] playlistSongs)
|
||||||
{
|
{
|
||||||
|
if (IsLocked)
|
||||||
|
return;
|
||||||
|
|
||||||
List<PlaylistSong> removedSongs = [];
|
List<PlaylistSong> removedSongs = [];
|
||||||
|
|
||||||
foreach (PlaylistSong playlistSong in playlistSongs)
|
foreach (PlaylistSong playlistSong in playlistSongs)
|
||||||
{
|
{
|
||||||
if (Songs.Remove(playlistSong))
|
if (_songs.Remove(playlistSong))
|
||||||
{
|
{
|
||||||
removedSongs.Add(playlistSong);
|
removedSongs.Add(playlistSong);
|
||||||
}
|
}
|
||||||
|
|||||||
33
Harmonia.Tests/Harmonia.Tests.csproj
Normal file
33
Harmonia.Tests/Harmonia.Tests.csproj
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net9.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<IsPackable>false</IsPackable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="coverlet.collector" Version="6.0.4">
|
||||||
|
<PrivateAssets>all</PrivateAssets>
|
||||||
|
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||||
|
</PackageReference>
|
||||||
|
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0" />
|
||||||
|
<PackageReference Include="NSubstitute" Version="5.3.0" />
|
||||||
|
<PackageReference Include="Shouldly" Version="4.3.0" />
|
||||||
|
<PackageReference Include="xunit" Version="2.9.3" />
|
||||||
|
<PackageReference Include="xunit.runner.visualstudio" Version="3.0.2">
|
||||||
|
<PrivateAssets>all</PrivateAssets>
|
||||||
|
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||||
|
</PackageReference>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Harmonia.Core\Harmonia.Core.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Using Include="Xunit" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
126
Harmonia.Tests/PlaylistTests.cs
Normal file
126
Harmonia.Tests/PlaylistTests.cs
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
using Harmonia.Core.Models;
|
||||||
|
using Shouldly;
|
||||||
|
using System.ComponentModel;
|
||||||
|
|
||||||
|
namespace Harmonia.Tests;
|
||||||
|
|
||||||
|
public class PlaylistTests
|
||||||
|
{
|
||||||
|
[Fact]
|
||||||
|
public void Add_Songs()
|
||||||
|
{
|
||||||
|
Playlist playlist = new("Test");
|
||||||
|
|
||||||
|
Song[] songs =
|
||||||
|
[
|
||||||
|
new Song() { FileName = "Song1.mp3" },
|
||||||
|
new Song() { FileName = "Song2.mp3" },
|
||||||
|
new Song() { FileName = "Song3.mp3" }
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.AddSongs(songs);
|
||||||
|
|
||||||
|
playlist.Songs.Count.ShouldBe(3);
|
||||||
|
|
||||||
|
Song song = new() { FileName = "Song4.mp3" };
|
||||||
|
playlist.AddSong(song);
|
||||||
|
|
||||||
|
playlist.Songs.Count.ShouldBe(4);
|
||||||
|
playlist.Songs[3].Song.FileName.ShouldBe("Song4.mp3");
|
||||||
|
|
||||||
|
Song song2 = new() { FileName = "Song5.mp3" };
|
||||||
|
playlist.AddSong(song2, 1);
|
||||||
|
|
||||||
|
playlist.Songs.Count.ShouldBe(5);
|
||||||
|
playlist.Songs[1].Song.FileName.ShouldBe("Song5.mp3");
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Sort_Songs()
|
||||||
|
{
|
||||||
|
Playlist playlist = new("Test");
|
||||||
|
|
||||||
|
Song[] songs =
|
||||||
|
[
|
||||||
|
new Song() { FileName = "Song1.mp3", Album = "Album 2", TrackNumber = 2 },
|
||||||
|
new Song() { FileName = "Song2.mp3", Album = "Album 1", TrackNumber = 2 },
|
||||||
|
new Song() { FileName = "Song3.mp3", Album = "Album 2", TrackNumber = 1 },
|
||||||
|
new Song() { FileName = "Song4.mp3", Album = "Album 1", TrackNumber = 1 },
|
||||||
|
new Song() { FileName = "Song5.mp3", Album = "Album 2", TrackNumber = 3 },
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.AddSongs(songs);
|
||||||
|
|
||||||
|
SortOption[] sortOptions =
|
||||||
|
[
|
||||||
|
new("Album", ListSortDirection.Ascending),
|
||||||
|
new("TrackNumber", ListSortDirection.Ascending)
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.SortSongs([.. playlist.Songs], sortOptions);
|
||||||
|
|
||||||
|
string[] expectedSortedFileNames =
|
||||||
|
[
|
||||||
|
"Song4.mp3",
|
||||||
|
"Song2.mp3",
|
||||||
|
"Song3.mp3",
|
||||||
|
"Song1.mp3",
|
||||||
|
"Song5.mp3"
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.Songs.Select(x => x.Song.FileName).ToArray()
|
||||||
|
.ShouldBeEquivalentTo(expectedSortedFileNames);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Remove_Songs()
|
||||||
|
{
|
||||||
|
Playlist playlist = new("Test");
|
||||||
|
|
||||||
|
Song[] songs =
|
||||||
|
[
|
||||||
|
new Song() { FileName = "Song1.mp3" },
|
||||||
|
new Song() { FileName = "Song2.mp3" },
|
||||||
|
new Song() { FileName = "Song3.mp3" }
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.AddSongs(songs);
|
||||||
|
|
||||||
|
playlist.RemoveSong(1);
|
||||||
|
|
||||||
|
string[] expectedFileNames =
|
||||||
|
[
|
||||||
|
"Song1.mp3",
|
||||||
|
"Song3.mp3"
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.Songs.Select(x => x.Song.FileName).ToArray()
|
||||||
|
.ShouldBeEquivalentTo(expectedFileNames);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void Lock_Playlist()
|
||||||
|
{
|
||||||
|
Playlist playlist = new("Test");
|
||||||
|
|
||||||
|
Song[] songs =
|
||||||
|
[
|
||||||
|
new Song() { FileName = "Song1.mp3" },
|
||||||
|
new Song() { FileName = "Song2.mp3" },
|
||||||
|
new Song() { FileName = "Song3.mp3" }
|
||||||
|
];
|
||||||
|
|
||||||
|
playlist.AddSongs(songs);
|
||||||
|
|
||||||
|
playlist.IsLocked = true;
|
||||||
|
|
||||||
|
Song song = new() { FileName = "Song4.mp3" };
|
||||||
|
playlist.AddSong(song);
|
||||||
|
|
||||||
|
playlist.Songs.Count.ShouldBe(3);
|
||||||
|
|
||||||
|
playlist.RemoveSong(0);
|
||||||
|
|
||||||
|
playlist.Songs.Count.ShouldBe(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
|
|
||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
# Visual Studio Version 17
|
# Visual Studio Version 17
|
||||||
VisualStudioVersion = 17.12.35728.132 d17.12
|
VisualStudioVersion = 17.12.35728.132
|
||||||
MinimumVisualStudioVersion = 10.0.40219.1
|
MinimumVisualStudioVersion = 10.0.40219.1
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Harmonia.Core", "Harmonia.Core\Harmonia.Core.csproj", "{BC40B066-CD33-4A9E-A470-A0F9216D1822}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Harmonia.Core", "Harmonia.Core\Harmonia.Core.csproj", "{BC40B066-CD33-4A9E-A470-A0F9216D1822}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Harmonia.Tests", "Harmonia.Tests\Harmonia.Tests.csproj", "{F17DDB67-F609-4316-B0AB-2235BE69135B}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -15,6 +17,10 @@ Global
|
|||||||
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Release|Any CPU.Build.0 = Release|Any CPU
|
{BC40B066-CD33-4A9E-A470-A0F9216D1822}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{F17DDB67-F609-4316-B0AB-2235BE69135B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{F17DDB67-F609-4316-B0AB-2235BE69135B}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{F17DDB67-F609-4316-B0AB-2235BE69135B}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{F17DDB67-F609-4316-B0AB-2235BE69135B}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
|
|||||||
Reference in New Issue
Block a user