Compare commits

...

5 Commits

31 changed files with 582 additions and 186 deletions

View File

@@ -2,9 +2,10 @@
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using PlaylistShared.Api.Entities; using PlaylistShared.Api.Entities;
using PlaylistShared.Api.Extensions;
using PlaylistShared.Api.Services; using PlaylistShared.Api.Services;
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.DTO; using PlaylistShared.Shared.Yandex;
using System.Security.Claims; using System.Security.Claims;
namespace PlaylistShared.Api.Controllers; namespace PlaylistShared.Api.Controllers;
@@ -87,7 +88,13 @@ public class AudioController : ControllerBase
{ {
Title = track.Title, Title = track.Title,
CoverUri = track.CoverUri, CoverUri = track.CoverUri,
Artists = track.Artists.Select(t => t.Name).ToList(), Artists = track.Artists.Select(a => new YandexArtist
{
Id = a.Id,
Name = a.Name,
CoverUrl = a.Cover.GetUrl(),
Description = a.Description?.Text ?? string.Empty,
}).ToList(),
DurationMs = track.DurationMs, DurationMs = track.DurationMs,
})); }));
} }

View File

@@ -6,7 +6,7 @@ using PlaylistShared.Api.Extensions;
using PlaylistShared.Api.Services; using PlaylistShared.Api.Services;
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.Enums; using PlaylistShared.Shared.Enums;
using PlaylistShared.Shared.Playlist; using PlaylistShared.Shared.Yandex;
using PlaylistShared.Shared.SharedPlaylist; using PlaylistShared.Shared.SharedPlaylist;
using YandexMusic; using YandexMusic;
@@ -32,7 +32,7 @@ public class PlaylistsController : ControllerBase
} }
[HttpGet] [HttpGet]
public async Task<ActionResult<ApiResponse<List<YandexPlaylistInfo>>>> GetMyPlaylists() public async Task<ActionResult<ApiResponse<List<YandexPlaylistShare>>>> GetMyPlaylists()
{ {
var userId = User.GetUserId(); var userId = User.GetUserId();
var user = await _userManager.FindByIdAsync(userId.ToString()); var user = await _userManager.FindByIdAsync(userId.ToString());
@@ -52,7 +52,7 @@ public class PlaylistsController : ControllerBase
var sharedPlaylists = await _sharedService.GetAllByUserAsync(userId); var sharedPlaylists = await _sharedService.GetAllByUserAsync(userId);
var result = ownPlaylists.Select(p => new YandexPlaylistInfo var result = ownPlaylists.Select(p => new YandexPlaylistShare
{ {
Kind = p.Kind, Kind = p.Kind,
OwnerUid = p.Owner.Uid, OwnerUid = p.Owner.Uid,
@@ -63,7 +63,7 @@ public class PlaylistsController : ControllerBase
ShareToken = sharedPlaylists.FirstOrDefault(s => s.YandexPlaylistKind == p.Kind && s.YandexPlaylistOwnerUid == p.Owner.Uid)?.ShareToken, ShareToken = sharedPlaylists.FirstOrDefault(s => s.YandexPlaylistKind == p.Kind && s.YandexPlaylistOwnerUid == p.Owner.Uid)?.ShareToken,
}).ToList(); }).ToList();
return Ok(ApiResponse<List<YandexPlaylistInfo>>.Ok(result)); return Ok(ApiResponse<List<YandexPlaylistShare>>.Ok(result));
} }
[HttpPost("share")] [HttpPost("share")]

View File

@@ -5,8 +5,8 @@ using PlaylistShared.Api.Entities;
using PlaylistShared.Api.Extensions; using PlaylistShared.Api.Extensions;
using PlaylistShared.Api.Services; using PlaylistShared.Api.Services;
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.DTO;
using PlaylistShared.Shared.SharedPlaylist; using PlaylistShared.Shared.SharedPlaylist;
using PlaylistShared.Shared.Yandex;
using YandexMusic.API.Models.Playlist; using YandexMusic.API.Models.Playlist;
[ApiController] [ApiController]
@@ -168,13 +168,19 @@ public class SharedPlaylistController : ControllerBase
{ {
return new YandexPlaylistData return new YandexPlaylistData
{ {
Title = playlist.Title ?? "", Title = playlist.Title,
Description = playlist.Description ?? "", Description = playlist.Description,
Tracks = playlist.Tracks?.Select(t => new YandexTrack Tracks = playlist.Tracks.Select(t => new YandexTrack
{ {
TrackId = t.Track?.Id ?? "", TrackId = t.Track.Id,
Title = t.Track?.Title ?? "", Title = t.Track.Title,
Artists = t.Track?.Artists?.Select(a => a.Name).ToList() ?? new List<string>(), Artists = t.Track.Artists.Select(t => new YandexArtist()
{
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
DurationMs = (int)(t.Track?.DurationMs ?? 0), DurationMs = (int)(t.Track?.DurationMs ?? 0),
CoverUri = t.Track?.CoverUri ?? "" CoverUri = t.Track?.CoverUri ?? ""
}).ToList() ?? new List<YandexTrack>() }).ToList() ?? new List<YandexTrack>()

View File

@@ -5,8 +5,8 @@ using PlaylistShared.Api.Entities;
using PlaylistShared.Api.Extensions; using PlaylistShared.Api.Extensions;
using PlaylistShared.Api.Services; using PlaylistShared.Api.Services;
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.DTO;
using PlaylistShared.Shared.Enums; using PlaylistShared.Shared.Enums;
using PlaylistShared.Shared.Yandex;
namespace PlaylistShared.Api.Controllers; namespace PlaylistShared.Api.Controllers;
@@ -26,16 +26,16 @@ public class YandexSearchController : ControllerBase
_sharedPlaylistService = sharedPlaylistService; _sharedPlaylistService = sharedPlaylistService;
} }
[HttpGet("tracks")] [HttpGet("search")]
public async Task<ActionResult<ApiResponse<List<YandexTrack>>>> SearchQuery( public async Task<ActionResult<ApiResponse<YandexSearchResult>>> SearchQuery(
[FromQuery] string query, [FromQuery] string query,
[FromQuery] int limit = 20, [FromQuery] int limit = 20,
[FromQuery] TrackSearchType? searchType = TrackSearchType.All, [FromQuery] TrackSearchType searchType = TrackSearchType.All,
[FromQuery] bool byId = false, [FromQuery] bool byId = false,
[FromQuery] string? shared_id = null) [FromQuery] string? shared_id = null)
{ {
if (string.IsNullOrWhiteSpace(query)) if (string.IsNullOrWhiteSpace(query))
return BadRequest(ApiResponse<List<YandexTrack>>.Fail(new ErrorResponse return BadRequest(ApiResponse<YandexSearchResult>.Fail(new ErrorResponse
{ {
StatusCode = 400, StatusCode = 400,
Message = "Поисковый запрос не может быть пустым." Message = "Поисковый запрос не может быть пустым."
@@ -65,23 +65,23 @@ public class YandexSearchController : ControllerBase
var decryptedToken = _yandexService.DecryptToken(user.YandexAccessToken); var decryptedToken = _yandexService.DecryptToken(user.YandexAccessToken);
if (string.IsNullOrEmpty(decryptedToken)) if (string.IsNullOrEmpty(decryptedToken))
return BadRequest(ApiResponse<List<YandexTrack>>.Fail(new ErrorResponse return BadRequest(ApiResponse<YandexSearchResult>.Fail(new ErrorResponse
{ {
StatusCode = 400, StatusCode = 400,
Message = "Токен Яндекс.Музыки не установлен или недействителен." Message = "Токен Яндекс.Музыки не установлен или недействителен."
})); }));
List<YandexTrack>? results = null; YandexSearchResult? results = null;
if (byId) if (byId)
{ {
results = await _yandexService.SearchTracksByIdAsync(user, query, searchType.Value, limit); results = await _yandexService.SearchTracksByIdAsync(user, query, searchType, limit);
} }
else else
{ {
results = await _yandexService.SearchTracksAsync(user, query, searchType, limit); results = await _yandexService.SearchAsync(user, query, searchType, limit);
} }
return Ok(ApiResponse<List<YandexTrack>>.Ok(results)); return Ok(ApiResponse<YandexSearchResult>.Ok(results));
} }
} }

View File

@@ -4,14 +4,16 @@ namespace PlaylistShared.Api.Extensions;
public static class YCoverExtensions public static class YCoverExtensions
{ {
public static string GetUrl(this YCover cover, string size = "200x200") public static string GetUrl(this YCover cover)
{ {
switch (cover) switch (cover)
{ {
case YCoverImage img when !string.IsNullOrEmpty(img.Uri): case YCoverImage img when !string.IsNullOrEmpty(img.Uri):
return $"https://{img.Uri.Replace("%%", size)}"; return img.Uri;
case YCoverPic pic when !string.IsNullOrEmpty(pic.Uri): case YCoverPic pic when !string.IsNullOrEmpty(pic.Uri):
return $"https://{pic.Uri.Replace("%%", size)}"; return pic.Uri;
case YCoverMosaic mosaic when mosaic.ItemsUri.Any():
return mosaic.ItemsUri.First();
default: default:
return string.Empty; return string.Empty;
} }

View File

@@ -27,7 +27,7 @@
<PackageReference Include="Swashbuckle.AspNetCore" Version="10.1.7" /> <PackageReference Include="Swashbuckle.AspNetCore" Version="10.1.7" />
<PackageReference Include="Swashbuckle.AspNetCore.Swagger" Version="10.1.7" /> <PackageReference Include="Swashbuckle.AspNetCore.Swagger" Version="10.1.7" />
<PackageReference Include="Swashbuckle.AspNetCore.SwaggerUI" Version="10.1.7" /> <PackageReference Include="Swashbuckle.AspNetCore.SwaggerUI" Version="10.1.7" />
<PackageReference Include="YandexMusic" Version="0.0.7" /> <PackageReference Include="YandexMusic" Version="0.0.8" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>

View File

@@ -3,7 +3,6 @@ using PlaylistShared.Api.Data;
using PlaylistShared.Api.Entities; using PlaylistShared.Api.Entities;
using PlaylistShared.Shared.Auth; using PlaylistShared.Shared.Auth;
using PlaylistShared.Shared.Enums; using PlaylistShared.Shared.Enums;
using PlaylistShared.Shared.Playlist;
using PlaylistShared.Shared.SharedPlaylist; using PlaylistShared.Shared.SharedPlaylist;
namespace PlaylistShared.Api.Services; namespace PlaylistShared.Api.Services;

View File

@@ -1,7 +1,8 @@
using Microsoft.AspNetCore.DataProtection; using Microsoft.AspNetCore.DataProtection;
using PlaylistShared.Api.Entities; using PlaylistShared.Api.Entities;
using PlaylistShared.Shared.DTO; using PlaylistShared.Api.Extensions;
using PlaylistShared.Shared.Enums; using PlaylistShared.Shared.Enums;
using PlaylistShared.Shared.Yandex;
using YandexMusic; using YandexMusic;
using YandexMusic.API.Extensions.API; using YandexMusic.API.Extensions.API;
using YandexMusic.API.Models.Playlist; using YandexMusic.API.Models.Playlist;
@@ -108,7 +109,7 @@ public class YandexMusicService
} }
} }
public async Task<List<YandexTrack>> SearchTracksAsync( public async Task<YandexSearchResult> SearchAsync(
ApplicationUser user, ApplicationUser user,
string query, string query,
TrackSearchType? searchType = TrackSearchType.All, TrackSearchType? searchType = TrackSearchType.All,
@@ -116,7 +117,7 @@ public class YandexMusicService
) )
{ {
var client = await CreateClientAsync(user); var client = await CreateClientAsync(user);
if (client == null) return new List<YandexTrack>(); if (client == null) return new YandexSearchResult();
var ySerchType = searchType switch var ySerchType = searchType switch
{ {
@@ -128,59 +129,193 @@ public class YandexMusicService
}; };
var searchResult = await client.SearchAsync(query, ySerchType, page: 0, pageSize: limit); var searchResult = await client.SearchAsync(query, ySerchType, page: 0, pageSize: limit);
if (searchResult?.Tracks?.Results == null) return new List<YandexTrack>(); if (searchResult?.Tracks?.Results == null) return new YandexSearchResult();
return searchResult.Tracks.Results.Select(t => new YandexTrack return new YandexSearchResult
{
Tracks = searchResult.Tracks.Results.Select(t => new YandexTrack
{ {
TrackId = t.Id, TrackId = t.Id,
Title = t.Title, Title = t.Title,
Artists = t.Artists?.Select(a => a.Name).ToList() ?? new List<string>(), Artists = t.Artists.Select(t => new YandexArtist()
CoverUri = t.CoverUri ?? string.Empty, {
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
CoverUri = t.CoverUri,
DurationMs = t.DurationMs, DurationMs = t.DurationMs,
}).ToList(); }).ToList(),
Playlists = searchResult.Playlists?.Results.Select(p => new YandexPlaylist
{
Uuid = p.PlaylistUuid,
Kind = p.Kind,
OwnerUid = p.Owner?.Uid ?? string.Empty,
Title = p.Title,
Description = p.Description,
CoverUrl = string.IsNullOrEmpty(p.CoverUri) ? p.Cover.GetUrl() : p.CoverUri,
TrackCount = p.TrackCount,
}).ToList(),
Artists = searchResult.Artists?.Results.Select(a => new YandexArtist
{
Id = a.Id,
Name = a.Name,
CoverUrl = a.Cover.GetUrl(),
Description = a.Description?.Text ?? string.Empty,
}).ToList(),
Albums = searchResult.Albums?.Results.Select(a => new YandexAlbum
{
Id = a.Id,
Title = a.Title,
Artists = a.Artists.Select(t => new YandexArtist()
{
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
CoverUrl = string.IsNullOrEmpty(a.CoverUri) ? a.Cover.GetUrl() : a.CoverUri,
Description = a.Description,
}).ToList(),
};
} }
public async Task<List<YandexTrack>> SearchTracksByIdAsync( public async Task<YandexSearchResult> SearchTracksByIdAsync(
ApplicationUser user, ApplicationUser user,
string id, string id,
TrackSearchType searchType, TrackSearchType searchType,
int limit = 20 int limit = 20
) )
{ {
YandexSearchResult result = new();
var client = await CreateClientAsync(user); var client = await CreateClientAsync(user);
if (client == null) return new List<YandexTrack>(); if (client == null) return result;
var ySerchType = searchType switch if (searchType == TrackSearchType.All)
{ {
TrackSearchType.Artist => YandexMusic.API.Models.Common.YSearchType.Artist, throw new Exception("Для поиска по ID необходимо указать конкретный тип (трек, альбом, исполнитель или плейлист).");
TrackSearchType.Album => YandexMusic.API.Models.Common.YSearchType.Album,
TrackSearchType.Playlist => YandexMusic.API.Models.Common.YSearchType.Playlist,
TrackSearchType.Track => YandexMusic.API.Models.Common.YSearchType.Track,
_ => YandexMusic.API.Models.Common.YSearchType.All
};
IEnumerable<YTrack> searchResult = searchType switch
{
TrackSearchType.Playlist => (await client.GetPlaylistAsync(id)).Tracks.Select(t => t.Track),
TrackSearchType.Track => (await client.GetTracksAsync([id])),
TrackSearchType.Album => (await client.GetAlbumAsync(id)).Volumes.SelectMany(t => t),
TrackSearchType.Artist => (await client.GetArtistAsync(id)).Albums.SelectMany(t => t.Volumes.SelectMany(v => v)),
_ => new List<YTrack>()
};
if (searchType != TrackSearchType.Track)
{
searchResult = searchResult.Distinct();
if (limit > 0) searchResult = searchResult.Take(limit);
} }
return searchResult.Select(t => new YandexTrack else if (searchType == TrackSearchType.Track)
{
var track = await client.GetTrackAsync(id);
if (track != null)
{
result.Tracks = new List<YandexTrack>()
{
new()
{
TrackId = track.Id,
Title = track.Title,
Artists = track.Artists.Select(t => new YandexArtist()
{
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
CoverUri = track.CoverUri ?? string.Empty,
DurationMs = track.DurationMs,
}
};
}
}
else if (searchType == TrackSearchType.Album)
{
var album = await client.GetAlbumAsync(id);
result.Tracks = album?.Volumes.SelectMany(v => v).Select(t => new YandexTrack
{ {
TrackId = t.Id, TrackId = t.Id,
Title = t.Title, Title = t.Title,
Artists = t.Artists?.Select(a => a.Name).ToList() ?? new List<string>(), Artists = t.Artists.Select(t => new YandexArtist()
{
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
CoverUri = t.CoverUri ?? string.Empty,
DurationMs = t.DurationMs,
}).ToList();
}
else if (searchType == TrackSearchType.Artist)
{
var artist = await client.GetArtistAsync(id);
if (artist != null)
{
result.Albums = artist.Albums.Select(a => new YandexAlbum()
{
Id = a.Id,
Title = a.Title,
Artists = a.Artists.Select(t => new YandexArtist()
{
Id = t.Id,
Name = t.Name,
CoverUrl = t.Cover.GetUrl(),
Description = t.Description?.Text ?? string.Empty,
}).ToList(),
CoverUrl = string.IsNullOrEmpty(a.CoverUri) ? a.Cover.GetUrl() : a.CoverUri,
Description = a.Description,
}).ToList();
result.Playlists = artist.Playlists.Select(p => new YandexPlaylist
{
Uuid = p.PlaylistUuid,
Kind = p.Kind,
OwnerUid = p.Owner?.Uid ?? string.Empty,
Title = p.Title,
Description = p.Description,
CoverUrl = p.Cover.GetUrl(),
TrackCount = p.TrackCount,
}).ToList();
result.Tracks = artist.PopularTracks.Select(t => new YandexTrack
{
TrackId = t.Id,
Title = t.Title,
Artists = t.Artists.Select(a => new YandexArtist()
{
Id = a.Id,
Name = a.Name,
CoverUrl = a.Cover.GetUrl(),
Description = a.Description?.Text ?? string.Empty,
}).ToList(),
CoverUri = t.CoverUri ?? string.Empty, CoverUri = t.CoverUri ?? string.Empty,
DurationMs = t.DurationMs, DurationMs = t.DurationMs,
}).ToList(); }).ToList();
} }
} }
else if (searchType == TrackSearchType.Playlist)
{
var playlist = await client.GetPlaylistAsync(id);
result.Tracks = playlist?.Tracks.Select(p => new YandexTrack
{
TrackId = p.Track.Id,
CoverUri = p.Track.CoverUri,
Artists = p.Track.Artists.Select(a => new YandexArtist
{
Id = a.Id,
Name = a.Name,
CoverUrl = a.Cover.GetUrl(),
Description = a.Description?.Text ?? string.Empty,
}).ToList(),
Title = p.Track.Title,
DurationMs = p.Track.DurationMs,
}).ToList();
}
return result;
}
}

View File

@@ -1,5 +1,6 @@
@using Microsoft.AspNetCore.Components.Web @using Microsoft.AspNetCore.Components.Web
@using PlaylistShared.Shared.DTO @using PlaylistShared.Shared.DTO
@using PlaylistShared.Shared.Yandex
@inject IAudioPlayerService AudioPlayerService @inject IAudioPlayerService AudioPlayerService
<MudItem @onmouseenter="HandleMouseEnter" <MudItem @onmouseenter="HandleMouseEnter"
@@ -11,7 +12,7 @@
@if (CanPlay && (_isHovered || IsCurrentTrackPlaying)) @if (CanPlay && (_isHovered || IsCurrentTrackPlaying))
{ {
<MudItem class="play-overlay" <MudItem class="play-overlay"
style="position: absolute; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.6); display: flex; align-items: center; justify-content: center; border-radius: 4px;"> style="position: absolute; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.6); display: flex; align-items: center; justify-content: center; border-radius: 4px; transition: opacity 0.2s ease; cursor: pointer;">
<MudIconButton Icon="@(IsCurrentTrackPlaying? Icons.Material.Filled.Pause : Icons.Material.Filled.PlayArrow)" <MudIconButton Icon="@(IsCurrentTrackPlaying? Icons.Material.Filled.Pause : Icons.Material.Filled.PlayArrow)"
Color="Color.Inherit" Color="Color.Inherit"
Size="Size.Large" Size="Size.Large"

View File

@@ -1,6 +1,7 @@
@using PlaylistShared.Shared.DTO @using PlaylistShared.Shared.DTO
@using PlaylistShared.Pwa.Components.Common @using PlaylistShared.Pwa.Components.Common
@using PlaylistShared.Pwa.Extensions @using PlaylistShared.Pwa.Extensions
@using PlaylistShared.Shared.Yandex
<MudStack Row AlignItems="AlignItems.Center"> <MudStack Row AlignItems="AlignItems.Center">
<!-- Обложка с фиксированной шириной --> <!-- Обложка с фиксированной шириной -->
@@ -16,7 +17,7 @@
<MudItem> <MudItem>
<MudStack Spacing="0"> <MudStack Spacing="0">
<MudText Typo="Typo.body1" Color="Color.Secondary">@Track.Title</MudText> <MudText Typo="Typo.body1" Color="Color.Secondary">@Track.Title</MudText>
<MudText Typo="Typo.body2" >@string.Join(", ", Track.Artists)</MudText> <MudText Typo="Typo.body2" >@string.Join(", ", Track.Artists.Select(a => a.Name))</MudText>
</MudStack> </MudStack>
</MudItem> </MudItem>

View File

@@ -53,7 +53,7 @@
</MudText> </MudText>
<MudText Typo="Typo.body2" Style="font-weight: 600;"> <MudText Typo="Typo.body2" Style="font-weight: 600;">
@if (AudioPlayerService.CurrentTrack != null) @string.Join(", ", AudioPlayerService.CurrentTrack.Artists) @if (AudioPlayerService.CurrentTrack != null) @string.Join(", ", AudioPlayerService.CurrentTrack.Artists.Select(a => a.Name))
</MudText> </MudText>
</MudStack> </MudStack>

View File

@@ -2,7 +2,7 @@
<div class="track-progress-container @ColorClass" <div class="track-progress-container @ColorClass"
@onwheel="HandleWheel" @onwheel="HandleWheel"
style="--track-height: @(Height)px; height: @(Math.Max(Height, 24))px; --track-opacity: @(Opacity.ToString(System.Globalization.CultureInfo.InvariantCulture));"> style="--track-height: @(Height)px; height: @(Height)px; --track-opacity: @(Opacity.ToString(System.Globalization.CultureInfo.InvariantCulture));">
<div class="progress-base-track"> <div class="progress-base-track">
@if (Buffer) @if (Buffer)
@@ -24,6 +24,7 @@
max="@Max.ToString(System.Globalization.CultureInfo.InvariantCulture)" max="@Max.ToString(System.Globalization.CultureInfo.InvariantCulture)"
step="@Step.ToString(System.Globalization.CultureInfo.InvariantCulture)" step="@Step.ToString(System.Globalization.CultureInfo.InvariantCulture)"
value="@Value.ToString(System.Globalization.CultureInfo.InvariantCulture)" value="@Value.ToString(System.Globalization.CultureInfo.InvariantCulture)"
height="@Height"
@oninput="OnInput" @oninput="OnInput"
class="progress-input" /> class="progress-input" />
</div> </div>

View File

@@ -1,14 +1,16 @@
@using PlaylistShared.Pwa.Components.Common @using PlaylistShared.Pwa.Components.Common
@using PlaylistShared.Pwa.Components.SharedPlaylist.Cards
@using PlaylistShared.Shared.DTO @using PlaylistShared.Shared.DTO
@using PlaylistShared.Shared.Enums @using PlaylistShared.Shared.Enums
@using PlaylistShared.Shared.SharedPlaylist @using PlaylistShared.Shared.SharedPlaylist
@using PlaylistShared.Shared.Yandex
@inject HttpClient Http @inject HttpClient Http
@inject ISnackbar Snackbar @inject ISnackbar Snackbar
<MudStack Style="height: 100%; overflow: hidden;"> <MudStack Style="height: 100%; overflow: hidden;">
<MudItem> <MudItem>
<MudTextField @bind-Value="_searchQuery" <MudTextField @bind-Value="_searchQuery"
@bind-Value:after="SearchTracks" @bind-Value:after="OnSearchQueryChanged"
Variant="Variant.Outlined" Variant="Variant.Outlined"
FullWidth FullWidth
Label="Название или ссылка на трек Яндекс.Музыки" Label="Название или ссылка на трек Яндекс.Музыки"
@@ -17,25 +19,81 @@
<MudToggleGroup T="TrackSearchType" <MudToggleGroup T="TrackSearchType"
@bind-Value="_searchType" @bind-Value="_searchType"
@bind-Value:after="SearchTracks" @bind-Value:after="OnSearchTypeChanged"
Size="Size.Small" Size="Size.Small"
Color="Color.Primary" Color="Color.Primary"
Disabled="@(_isSearching)"> Disabled="@(_isSearching)">
<MudToggleItem Value="TrackSearchType.All" Text="Все" /> <MudToggleItem Value="TrackSearchType.All" Text="Все" />
<MudToggleItem Value="TrackSearchType.Track" Text="Трек" /> <MudToggleItem Value="TrackSearchType.Track" Text="Треки" />
<MudToggleItem Value="TrackSearchType.Album" Text="Альбом" /> <MudToggleItem Value="TrackSearchType.Album" Text="Альбомы" />
<MudToggleItem Value="TrackSearchType.Artist" Text="Исполнитель" /> <MudToggleItem Value="TrackSearchType.Playlist" Text="Плейлисты" />
<MudToggleItem Value="TrackSearchType.Artist" Text="Исполнители" />
</MudToggleGroup> </MudToggleGroup>
</MudItem> </MudItem>
<MudTable Items="@_searchResults" <MudItem Style="overflow: auto; flex-grow:1;">
Virtualize @if (_isSearching)
{
<MudProgressCircular Indeterminate Class="mx-auto my-8" />
}
else if (_searchResult != null)
{
<MudExpansionPanels>
@* Секция исполнителей *@
@if (_searchResult?.Artists != null)
{
<MudExpansionPanel Text="Исполнители" Expanded="true">
<MudGrid>
@foreach (var artist in _searchResult.Artists)
{
<MudItem xs="12" sm="6" md="3" lg="2">
<ArtistCard Item="artist" OnClick="() => SearchTracksByEntity(artist.Id, artist.Name, TrackSearchType.Artist)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция альбомов *@
@if (_searchResult?.Albums != null)
{
<MudExpansionPanel Text="Альбомы" Expanded="true">
<MudGrid>
@foreach (var album in _searchResult.Albums)
{
<MudItem xs="12" sm="6" md="3" lg="2">
<AlbumCard Item="album" OnClick="() => SearchTracksByEntity(album.Id, album.Title, TrackSearchType.Album)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция плейлистов *@
@if (_searchResult?.Playlists != null)
{
<MudExpansionPanel Text="Плейлисты" Expanded="true">
<MudGrid>
@foreach (var playlist in _searchResult.Playlists)
{
<MudItem xs="12" sm="6" md="3" lg="2">
<PlaylistCard Item="playlist" OnClick="() => SearchTracksByEntity(playlist.Uuid, playlist.Title, TrackSearchType.Playlist)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция треков *@
@if (_searchResult?.Tracks != null)
{
<MudExpansionPanel Text="Треки" Expanded="true">
<MudTable Items="@_searchResult.Tracks"
Hover Hover
Elevation="0" Elevation="0"
Class="d-flex flex-grow-1 flex-column" Class="d-flex flex-grow-1 flex-column"
Style="min-height: 0;" Style="min-height: 0;"
Breakpoint="Breakpoint.Sm" Breakpoint="Breakpoint.Sm">
Loading="@_isSearching">
<RowTemplate> <RowTemplate>
<MudTd Class="pa-1" Style="width: 100%;"> <MudTd Class="pa-1" Style="width: 100%;">
<TrackItem Track="@context" PlaylistShareToken="@ShareToken" /> <TrackItem Track="@context" PlaylistShareToken="@ShareToken" />
@@ -50,6 +108,11 @@
</MudTd> </MudTd>
</RowTemplate> </RowTemplate>
</MudTable> </MudTable>
</MudExpansionPanel>
}
</MudExpansionPanels>
}
</MudItem>
</MudStack> </MudStack>
@code { @code {
@@ -60,22 +123,32 @@
private string _searchQuery = ""; private string _searchQuery = "";
private bool _isSearching = false; private bool _isSearching = false;
private bool _isFirstSearch = true;
private TrackSearchType _searchType = TrackSearchType.All; private TrackSearchType _searchType = TrackSearchType.All;
private List<YandexTrack> _searchResults = new(); private YandexSearchResult? _searchResult = null;
private async Task SearchTracks() private async Task OnSearchQueryChanged()
{ {
if (string.IsNullOrWhiteSpace(_searchQuery)) await SearchTracks(byId: false);
}
private async Task OnSearchTypeChanged()
{ {
await SearchTracks(byId: false);
}
private async Task SearchTracks(bool byId = false, string? forcedQuery = null)
{
var query = forcedQuery ?? _searchQuery;
if (string.IsNullOrWhiteSpace(query))
{
_searchResult = null;
return; return;
} }
var query = _searchQuery.Trim();
var type = _searchType; var type = _searchType;
bool byId = false;
if (Uri.TryCreate(_searchQuery, UriKind.Absolute, out var uri) && uri.Host == "music.yandex.ru") // Распознавание ссылки Яндекс.Музыки
if (!byId && Uri.TryCreate(query, UriKind.Absolute, out var uri) && uri.Host == "music.yandex.ru")
{ {
try try
{ {
@@ -89,58 +162,61 @@
} }
} }
_isFirstSearch = false;
_isSearching = true; _isSearching = true;
_searchResult = null;
StateHasChanged();
try try
{ {
var url = $"/api/yandexsearch/tracks?query={Uri.EscapeDataString(query)}&searchType={Uri.EscapeDataString(type.ToString())}&limit=20"; var url = $"/api/yandexsearch/search?query={Uri.EscapeDataString(query)}&searchType={Uri.EscapeDataString(type.ToString())}&limit=20";
if (byId)
url += "&byId=true";
if (!string.IsNullOrEmpty(ShareToken)) if (!string.IsNullOrEmpty(ShareToken))
url += $"&shared_id={Uri.EscapeDataString(ShareToken)}"; url += $"&shared_id={Uri.EscapeDataString(ShareToken)}";
if (byId)
url += $"&byId={byId}";
var response = await Http.GetFromJsonAsync<ApiResponse<List<YandexTrack>>>(url); var response = await Http.GetFromJsonAsync<ApiResponse<YandexSearchResult>>(url);
if (response?.Success == true) if (response?.Success == true)
_searchResults = response.Data ?? new(); {
_searchResult = response.Data ?? new YandexSearchResult();
}
else else
{
Snackbar.Add(response?.Error?.Message ?? "Ошибка поиска", Severity.Error); Snackbar.Add(response?.Error?.Message ?? "Ошибка поиска", Severity.Error);
_searchResult = null;
}
} }
catch (Exception ex) catch (Exception ex)
{ {
Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error); Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error);
_searchResult = null;
} }
finally finally
{ {
_isSearching = false; _isSearching = false;
StateHasChanged(); StateHasChanged();
} }
} }
private async Task SearchTracksByQuery(string query) private async Task SearchTracksByEntity(string entityId, string title, TrackSearchType entityType)
{ {
if (string.IsNullOrWhiteSpace(query)) // Переключаем тип и ищем по ID
return; _searchType = entityType;
_searchQuery = title;
await SearchTracks(byId: true, forcedQuery: entityId);
} }
private async Task ToggleTrack(YandexTrack track) private async Task ToggleTrack(YandexTrack track)
{ {
if (ExistingTrackIds.Contains(track.TrackId)) if (ExistingTrackIds.Contains(track.TrackId))
{
await RemoveTrack(track); await RemoveTrack(track);
}
else else
{
await AddTrack(track); await AddTrack(track);
} }
}
private async Task RemoveTrack(YandexTrack track) private async Task RemoveTrack(YandexTrack track)
{ {
if (!ExistingTrackIds.Remove(track.TrackId)) return; if (!ExistingTrackIds.Remove(track.TrackId)) return;
try try
{ {
await RemoveTrackById(track.TrackId); await RemoveTrackById(track.TrackId);
@@ -164,7 +240,7 @@
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/remove-tracks", request); var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/remove-tracks", request);
if (response.IsSuccessStatusCode) if (response.IsSuccessStatusCode)
{ {
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился await OnTrackAdded.InvokeAsync();
} }
else else
{ {
@@ -201,7 +277,7 @@
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/add-tracks", request); var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/add-tracks", request);
if (response.IsSuccessStatusCode) if (response.IsSuccessStatusCode)
{ {
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился await OnTrackAdded.InvokeAsync();
} }
else else
{ {

View File

@@ -0,0 +1,34 @@
@using PlaylistShared.Shared.Yandex
<MudItem Class="d-flex flex-column align-center pa-2 cursor-pointer" @onclick="HandleClick">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Size="MudBlazor.Size.Large">
<MudImage Src="@Item.CoverUrl.FormatCoverUrl(Size, Size)" />
</MudAvatar>
}
else
{
<MudAvatar Size="MudBlazor.Size.Large" Variant="Variant.Filled">
<MudIcon Icon="@Icons.Material.Filled.AccountCircle" />
</MudAvatar>
}
<MudText Typo="Typo.body2" Align="Align.Center" Class="mt-2">@Item.Title</MudText>
<MudText Typo="Typo.caption" Align="Align.Center" Color="Color.Secondary">
@string.Join(", ", Item.Artists.Select(a => a.Name))
</MudText>
</MudItem>
@code {
[Parameter] public YandexAlbum Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
private async Task HandleClick()
{
if (OnClick.HasDelegate)
{
await OnClick.InvokeAsync();
}
}
}

View File

@@ -0,0 +1,31 @@
@using PlaylistShared.Shared.Yandex
<MudItem Class="d-flex flex-column align-center pa-2 cursor-pointer" @onclick="HandleClick">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Size="MudBlazor.Size.Large">
<MudImage Src="@Item.CoverUrl.FormatCoverUrl(Size, Size)" />
</MudAvatar>
}
else
{
<MudAvatar Size="MudBlazor.Size.Large" Variant="Variant.Filled">
<MudIcon Icon="@Icons.Material.Filled.Album" />
</MudAvatar>
}
<MudText Typo="Typo.body2" Align="Align.Center" Class="mt-2">@Item.Name</MudText>
</MudItem>
@code {
[Parameter] public YandexArtist Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
private async Task HandleClick()
{
if (OnClick.HasDelegate)
{
await OnClick.InvokeAsync();
}
}
}

View File

@@ -0,0 +1,31 @@
@using PlaylistShared.Shared.Yandex
<MudItem Class="d-flex flex-column align-center pa-2 cursor-pointer" @onclick="HandleClick">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Size="MudBlazor.Size.Large">
<MudImage Src="@Item.CoverUrl.FormatCoverUrl(Size, Size)" />
</MudAvatar>
}
else
{
<MudAvatar Size="MudBlazor.Size.Large" Variant="Variant.Filled">
<MudIcon Icon="@Icons.Material.Filled.PlaylistPlay" />
</MudAvatar>
}
<MudText Typo="Typo.body2" Align="Align.Center" Class="mt-2">@Item.Title</MudText>
</MudItem>
@code {
[Parameter] public YandexPlaylist Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
private async Task HandleClick()
{
if (OnClick.HasDelegate)
{
await OnClick.InvokeAsync();
}
}
}

View File

@@ -3,7 +3,8 @@
@attribute [Authorize] @attribute [Authorize]
@using PlaylistShared.Shared.DTO @using PlaylistShared.Shared.DTO
@using PlaylistShared.Shared.Playlist @using PlaylistShared.Shared.SharedPlaylist
@using PlaylistShared.Shared.Yandex
@inject HttpClient Http @inject HttpClient Http
@inject ISnackbar Snackbar @inject ISnackbar Snackbar
@inject NavigationManager Navigation @inject NavigationManager Navigation
@@ -70,11 +71,11 @@
</MudContainer> </MudContainer>
@code { @code {
private List<YandexPlaylistInfo> _playlists; private List<YandexPlaylistShare> _playlists;
private bool _loading = true; private bool _loading = true;
private bool _showOnlyShared = false; private bool _showOnlyShared = false;
private List<YandexPlaylistInfo> FilteredPlaylists => _showOnlyShared ? _playlists?.Where(p => p.IsShared).ToList() : _playlists; private List<YandexPlaylistShare> FilteredPlaylists => _showOnlyShared ? _playlists?.Where(p => p.IsShared).ToList() : _playlists;
protected override async Task OnInitializedAsync() protected override async Task OnInitializedAsync()
{ {
@@ -86,7 +87,7 @@
_loading = true; _loading = true;
try try
{ {
var response = await Http.GetFromJsonAsync<ApiResponse<List<YandexPlaylistInfo>>>("/api/playlists"); var response = await Http.GetFromJsonAsync<ApiResponse<List<YandexPlaylistShare>>>("/api/playlists");
if (response?.Success == true) if (response?.Success == true)
_playlists = response.Data; _playlists = response.Data;
else else
@@ -103,7 +104,7 @@
} }
} }
private async Task SharePlaylist(YandexPlaylistInfo playlist) private async Task SharePlaylist(YandexPlaylistShare playlist)
{ {
var request = new SharePlaylistRequest { Kind = playlist.Kind, OwnerUid = playlist.OwnerUid }; var request = new SharePlaylistRequest { Kind = playlist.Kind, OwnerUid = playlist.OwnerUid };
var response = await Http.PostAsJsonAsync("/api/playlists/share", request); var response = await Http.PostAsJsonAsync("/api/playlists/share", request);
@@ -118,7 +119,7 @@
} }
} }
private void GoToShared(YandexPlaylistInfo playlist) private void GoToShared(YandexPlaylistShare playlist)
{ {
if (!string.IsNullOrEmpty(playlist.ShareToken)) if (!string.IsNullOrEmpty(playlist.ShareToken))
Navigation.NavigateTo($"/shared/{playlist.ShareToken}"); Navigation.NavigateTo($"/shared/{playlist.ShareToken}");

View File

@@ -7,6 +7,7 @@
@using PlaylistShared.Shared.Enums @using PlaylistShared.Shared.Enums
@using PlaylistShared.Pwa.Services @using PlaylistShared.Pwa.Services
@using PlaylistShared.Shared.SharedPlaylist @using PlaylistShared.Shared.SharedPlaylist
@using PlaylistShared.Shared.Yandex
@inject HttpClient Http @inject HttpClient Http
@inject ISnackbar Snackbar @inject ISnackbar Snackbar
@inject NavigationManager Navigation @inject NavigationManager Navigation

View File

@@ -1,5 +1,6 @@
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.SharedPlaylist; using PlaylistShared.Shared.SharedPlaylist;
using PlaylistShared.Shared.Yandex;
using System.Net.Http.Json; using System.Net.Http.Json;
namespace PlaylistShared.Pwa.Services.Api; namespace PlaylistShared.Pwa.Services.Api;

View File

@@ -1,6 +1,6 @@
using MudBlazor; using MudBlazor;
using PlaylistShared.Shared; using PlaylistShared.Shared;
using PlaylistShared.Shared.DTO; using PlaylistShared.Shared.Yandex;
using System.Net.Http.Json; using System.Net.Http.Json;
namespace PlaylistShared.Pwa.Services; namespace PlaylistShared.Pwa.Services;

View File

@@ -1,4 +1,4 @@
using PlaylistShared.Shared.DTO; using PlaylistShared.Shared.Yandex;
namespace PlaylistShared.Pwa.Services; namespace PlaylistShared.Pwa.Services;

View File

@@ -109,41 +109,19 @@ code {
text-align: start; text-align: start;
} }
.track-cover-container { /* Горизонтальный скролинг */
border-radius: 4px; .horizontal-scroll {
overflow: hidden; overflow-x: auto;
transition: transform 0.2s ease; scroll-snap-type: x mandatory;
overflow-y: hidden; /* отключаем вертикальный скролл */
cursor: grab;
} }
.track-cover-container:hover { .horizontal-scroll:active {
transform: scale(1.05); cursor: grabbing;
}
.play-overlay {
transition: opacity 0.2s ease;
cursor: pointer;
}
/* Фиксированный плеер внизу */
.fixed-player {
position: sticky;
display: flex;
bottom: 0;
width: 100%;
right: 0;
justify-content: center;
background-color: var(--mud-palette-background);
box-shadow: 0 -2px 8px rgba(0, 0, 0, 0.1);
}
/* Отступ снизу, когда плеер виден */
.page-with-player {
padding-bottom: 80px; /* Высота плеера (подберите под свою тему) */
}
/* На мобильных устройствах можно уменьшить отступ */
@media (max-width: 600px) {
.page-with-player {
padding-bottom: 100px; /* если плеер выше на мобильных */
} }
/* Для WebKit (Chrome, Edge, Safari) можно включить горизонтальный скролл мышью */
.horizontal-scroll {
scrollbar-width: thin;
-webkit-overflow-scrolling: touch;
} }

View File

@@ -1,7 +1,7 @@
using PlaylistShared.Shared.Enums; using PlaylistShared.Shared.Enums;
using System.Text.Json.Serialization; using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Playlist; namespace PlaylistShared.Shared.SharedPlaylist;
/// <summary>Запрос на создание нового шеринг-плейлиста.</summary> /// <summary>Запрос на создание нового шеринг-плейлиста.</summary>
public class SharePlaylistDto public class SharePlaylistDto

View File

@@ -1,4 +1,4 @@
namespace PlaylistShared.Shared.Playlist; namespace PlaylistShared.Shared.SharedPlaylist;
public class SharePlaylistRequest public class SharePlaylistRequest
{ {

View File

@@ -0,0 +1,27 @@
using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Yandex;
/// <summary>Информация о альбоме из Яндекс.Музыки.</summary>
public class YandexAlbum
{
/// <summary>Идентификатор альбома (id).</summary>
[JsonPropertyName("id")]
public string Id { get; set; } = null!;
/// <summary>Наименование альбома.</summary>
[JsonPropertyName("title")]
public string Title { get; set; } = null!;
/// <summary>Исполнители альбома.</summary>
[JsonPropertyName("artists")]
public List<YandexArtist> Artists { get; set; } = null!;
/// <summary>Описание альбома.</summary>
[JsonPropertyName("description")]
public string? Description { get; set; }
/// <summary>URL обложки альбома.</summary>
[JsonPropertyName("coverUrl")]
public string? CoverUrl { get; set; }
}

View File

@@ -0,0 +1,23 @@
using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Yandex;
/// <summary>Информация о исполнителе из Яндекс.Музыки.</summary>
public class YandexArtist
{
/// <summary>Идентификатор исполнителя (id).</summary>
[JsonPropertyName("id")]
public string Id { get; set; } = null!;
/// <summary>Наименование исполнителя.</summary>
[JsonPropertyName("name")]
public string Name { get; set; } = null!;
/// <summary>Описание исполнителя.</summary>
[JsonPropertyName("description")]
public string? Description { get; set; }
/// <summary>URL исполнителя.</summary>
[JsonPropertyName("coverUrl")]
public string? CoverUrl { get; set; }
}

View File

@@ -1,10 +1,14 @@
using System.Text.Json.Serialization; using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Playlist; namespace PlaylistShared.Shared.Yandex;
/// <summary>Информация о плейлисте из Яндекс.Музыки (для импорта).</summary> /// <summary>Информация о плейлисте из Яндекс.Музыки.</summary>
public class YandexPlaylistInfo public class YandexPlaylist
{ {
/// <summary>Идентификатор плейлиста (uuid).</summary>
[JsonPropertyName("uuid")]
public string Uuid { get; set; } = null!;
/// <summary>Идентификатор плейлиста (kind).</summary> /// <summary>Идентификатор плейлиста (kind).</summary>
[JsonPropertyName("kind")] [JsonPropertyName("kind")]
public string Kind { get; set; } = null!; public string Kind { get; set; } = null!;
@@ -28,12 +32,4 @@ public class YandexPlaylistInfo
/// <summary>Кол-во треков.</summary> /// <summary>Кол-во треков.</summary>
[JsonPropertyName("trackCount")] [JsonPropertyName("trackCount")]
public int TrackCount { get; set; } public int TrackCount { get; set; }
/// <summary>Расшаренный</summary>
[JsonPropertyName("isShared")]
public bool IsShared { get; set; }
/// <summary>Расшаренная ссылка</summary>
[JsonPropertyName("shareToken")]
public string? ShareToken { get; set; }
} }

View File

@@ -1,6 +1,4 @@
using PlaylistShared.Shared.DTO; namespace PlaylistShared.Shared.Yandex;
namespace PlaylistShared.Shared.SharedPlaylist;
public class YandexPlaylistData public class YandexPlaylistData
{ {

View File

@@ -0,0 +1,16 @@
using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Yandex;
/// <summary>Информация о плейлисте из Яндекс.Музыки с пометкой о шаринге.</summary>
public class YandexPlaylistShare : YandexPlaylist
{
/// <summary>Расшаренный</summary>
[JsonPropertyName("isShared")]
public bool IsShared { get; set; }
/// <summary>Расшаренная ссылка</summary>
[JsonPropertyName("shareToken")]
public string? ShareToken { get; set; }
}

View File

@@ -0,0 +1,31 @@
using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.Yandex;
/// <summary>Информация о плейлисте из Яндекс.Музыки (для импорта).</summary>
public class YandexSearchResult
{
/// <summary>
/// Найденные треки.
/// </summary>
[JsonPropertyName("tracks")]
public List<YandexTrack>? Tracks { get; set; } = null;
/// <summary>
/// Найденные плейлисты.
/// </summary>
[JsonPropertyName("playlists")]
public List<YandexPlaylist>? Playlists { get; set; } = null;
/// <summary>
/// Найденные исполнители.
/// </summary>
[JsonPropertyName("artists")]
public List<YandexArtist>? Artists { get; set; } = null;
/// <summary>
/// Найденные альбомы.
/// </summary>
[JsonPropertyName("albumns")]
public List<YandexAlbum>? Albums { get; set; } = null;
}

View File

@@ -1,6 +1,6 @@
using System.Text.Json.Serialization; using System.Text.Json.Serialization;
namespace PlaylistShared.Shared.DTO; namespace PlaylistShared.Shared.Yandex;
/// <summary>Результат поиска трека в Яндекс.Музыке.</summary> /// <summary>Результат поиска трека в Яндекс.Музыке.</summary>
public class YandexTrack public class YandexTrack
@@ -12,7 +12,7 @@ public class YandexTrack
public string Title { get; set; } = string.Empty; public string Title { get; set; } = string.Empty;
[JsonPropertyName("artists")] [JsonPropertyName("artists")]
public List<string> Artists { get; set; } = new(); public List<YandexArtist> Artists { get; set; } = new();
[JsonPropertyName("coverUri")] [JsonPropertyName("coverUri")]
public string CoverUri { get; set; } = string.Empty; public string CoverUri { get; set; } = string.Empty;