Доработка компонентка добавления треков

This commit is contained in:
FrigaT
2026-04-16 17:40:33 +03:00
parent 68d7c7fc12
commit 5a8ae3d680
11 changed files with 222 additions and 62 deletions

View File

@@ -82,6 +82,6 @@ public class YandexSearchController : ControllerBase
results = await _yandexService.SearchAsync(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

@@ -158,7 +158,7 @@ public class YandexMusicService
Id = a.Id, Id = a.Id,
Name = a.Name, Name = a.Name,
CoverUrl = a.Cover.GetUrl(), CoverUrl = a.Cover.GetUrl(),
Description = a.Description.Text, Description = a.Description?.Text ?? string.Empty,
}).ToList(), }).ToList(),
Albums = searchResult.Albums?.Results.Select(a => new YandexAlbum Albums = searchResult.Albums?.Results.Select(a => new YandexAlbum
@@ -170,7 +170,7 @@ public class YandexMusicService
Id = t.Id, Id = t.Id,
Name = t.Name, Name = t.Name,
CoverUrl = t.Cover.GetUrl(), CoverUrl = t.Cover.GetUrl(),
Description = t.Description.Text, Description = t.Description?.Text ?? string.Empty,
}).ToList(), }).ToList(),
CoverUrl = string.IsNullOrEmpty(a.CoverUri) ? a.Cover.GetUrl() : a.CoverUri, CoverUrl = string.IsNullOrEmpty(a.CoverUri) ? a.Cover.GetUrl() : a.CoverUri,
Description = a.Description, Description = a.Description,

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"

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">
<!-- Обложка с фиксированной шириной --> <!-- Обложка с фиксированной шириной -->

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,4 +1,5 @@
@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
@@ -9,7 +10,7 @@
<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="Название или ссылка на трек Яндекс.Музыки"
@@ -18,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 (ShouldShowSection(TrackSearchType.Artist))
{
<MudExpansionPanel Text="Исполнители" Expanded="true">
<MudGrid>
@foreach (var artist in _searchResult.Artists!)
{
<MudItem xs="12" sm="6" md="4" lg="3">
<ArtistCard Item="artist" OnClick="() => SearchTracksByEntity(artist.Id, TrackSearchType.Artist)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция альбомов *@
@if (ShouldShowSection(TrackSearchType.Album))
{
<MudExpansionPanel Text="Альбомы" Expanded="true">
<MudGrid>
@foreach (var album in _searchResult.Albums!)
{
<MudItem xs="12" sm="6" md="4" lg="3">
<AddTrackSection Item="album" OnClick="() => SearchTracksByEntity(album.Id, TrackSearchType.Album)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция плейлистов *@
@if (ShouldShowSection(TrackSearchType.Playlist))
{
<MudExpansionPanel Text="Плейлисты" Expanded="true">
<MudGrid>
@foreach (var playlist in _searchResult.Playlists!)
{
<MudItem xs="12" sm="6" md="4" lg="3">
<PlaylistCard Item="playlist" OnClick="() => SearchTracksByEntity(playlist.Kind, TrackSearchType.Playlist)" />
</MudItem>
}
</MudGrid>
</MudExpansionPanel>
}
@* Секция треков *@
@if (ShouldShowSection(TrackSearchType.Track))
{
<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" />
@@ -51,6 +108,11 @@
</MudTd> </MudTd>
</RowTemplate> </RowTemplate>
</MudTable> </MudTable>
</MudExpansionPanel>
}
</MudExpansionPanels>
}
</MudItem>
</MudStack> </MudStack>
@code { @code {
@@ -63,20 +125,44 @@
private bool _isSearching = false; private bool _isSearching = false;
private bool _isFirstSearch = true; 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 bool ShouldShowSection(TrackSearchType sectionType)
{ {
if (string.IsNullOrWhiteSpace(_searchQuery)) return sectionType switch
{ {
TrackSearchType.Track => _searchResult?.Tracks?.Any() == true,
TrackSearchType.Album => _searchResult?.Albums?.Any() == true,
TrackSearchType.Playlist => _searchResult?.Playlists?.Any() == true,
TrackSearchType.Artist => _searchResult?.Artists?.Any() == true,
_ => false
};
}
private async Task OnSearchQueryChanged()
{
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;
_isFirstSearch = true;
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
{ {
@@ -92,56 +178,59 @@
_isFirstSearch = false; _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, TrackSearchType entityType)
{ {
if (string.IsNullOrWhiteSpace(query)) // Переключаем тип на треки и ищем по ID
return; _searchType = TrackSearchType.Track;
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);
@@ -165,7 +254,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
{ {
@@ -202,7 +291,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,24 @@
@using PlaylistShared.Shared.Yandex
<MudPaper Class="d-flex flex-column align-center pa-2 cursor-pointer" Elevation="0" OnClick="OnClick.InvokeAsync">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Image="@Item.CoverUrl.FormatCoverUrl(Size, Size)" Size="MudBlazor.Size.Large" />
}
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>
</MudPaper>
@code {
[Parameter] public YandexAlbum Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
}

View File

@@ -0,0 +1,21 @@
@using PlaylistShared.Shared.Yandex
<MudPaper Class="d-flex flex-column align-center pa-2 cursor-pointer" Elevation="0" OnClick="OnClick.InvokeAsync">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Image="@Item.CoverUrl.FormatCoverUrl(Size, Size)" Size="MudBlazor.Size.Large" />
}
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>
</MudPaper>
@code {
[Parameter] public YandexArtist Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
}

View File

@@ -0,0 +1,21 @@
@using PlaylistShared.Shared.Yandex
<MudPaper Class="d-flex flex-column align-center pa-2 cursor-pointer" Elevation="0" OnClick="OnClick.InvokeAsync">
@if (!string.IsNullOrEmpty(Item.CoverUrl))
{
<MudAvatar Image="@Item.CoverUrl.FormatCoverUrl(Size, Size)" Size="MudBlazor.Size.Large" />
}
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>
</MudPaper>
@code {
[Parameter] public YandexPlaylist Item { get; set; } = null!;
[Parameter] public EventCallback OnClick { get; set; }
[Parameter] public int Size { get; set; } = 50;
}

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