Compare commits

..

4 Commits

Author SHA1 Message Date
FrigaT
cbb0cb8c8e fix добавление трека 2026-04-15 19:06:24 +03:00
FrigaT
50ed75b413 flex table track 2026-04-15 18:49:18 +03:00
FrigaT
e82dcdeaa4 flex 2026-04-15 18:07:16 +03:00
FrigaT
629f14cbb9 Доработана передача списка выбранных треков 2026-04-15 16:57:11 +03:00
7 changed files with 180 additions and 244 deletions

View File

@@ -7,7 +7,7 @@
<MudImage Src="@CoverUrl.FormatCoverUrl(Width, Height)" Height="@Height" Width="@Width" Class="rounded" Style="display: block;" /> <MudImage Src="@CoverUrl.FormatCoverUrl(Width, Height)" Height="@Height" Width="@Width" Class="rounded" Style="display: block;" />
@if (_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;">
@@ -26,6 +26,7 @@
[Parameter] public int Height { get; set; } = 50; [Parameter] public int Height { get; set; } = 50;
[Parameter] public int Width { get; set; } = 50; [Parameter] public int Width { get; set; } = 50;
[Parameter] public string PlaylistShareToken { get; set; } = string.Empty; [Parameter] public string PlaylistShareToken { get; set; } = string.Empty;
[Parameter] public bool CanPlay { get; set; } = false;
private bool IsCurrentTrackPlaying => AudioPlayerService.IsPlaying && AudioPlayerService.CurrentTrackId == TrackId; private bool IsCurrentTrackPlaying => AudioPlayerService.IsPlaying && AudioPlayerService.CurrentTrackId == TrackId;

View File

@@ -9,6 +9,7 @@
TrackId="@Track.TrackId" TrackId="@Track.TrackId"
TrackTitle="@Track.Title" TrackTitle="@Track.Title"
PlaylistShareToken="@PlaylistShareToken" PlaylistShareToken="@PlaylistShareToken"
CanPlay="@CanPlay"
Width="40" Height="40" /> Width="40" Height="40" />
</MudItem> </MudItem>
@@ -31,4 +32,5 @@
@code { @code {
[Parameter] public YandexTrack Track { get; set; } = null!; [Parameter] public YandexTrack Track { get; set; } = null!;
[Parameter] public string PlaylistShareToken { get; set; } = string.Empty; [Parameter] public string PlaylistShareToken { get; set; } = string.Empty;
[Parameter] public bool CanPlay { get; set; } = true;
} }

View File

@@ -5,46 +5,48 @@
@inject HttpClient Http @inject HttpClient Http
@inject ISnackbar Snackbar @inject ISnackbar Snackbar
<MudStack AlignItems="AlignItems.Stretch"> <MudStack Style="height: 100%; overflow: hidden;">
<MudTextField @bind-Value="_searchQuery" <MudItem>
@bind-Value:after="SearchTracks" <MudTextField @bind-Value="_searchQuery"
Variant="Variant.Outlined" @bind-Value:after="SearchTracks"
FullWidth Variant="Variant.Outlined"
Label="Название или ссылка на трек Яндекс.Музыки" FullWidth
Disabled="@_isSearching" Label="Название или ссылка на трек Яндекс.Музыки"
Adornment="Adornment.End" AdornmentIcon="@Icons.Material.Filled.Search" AdornmentColor="Color.Secondary" Disabled="@_isSearching"
/> Adornment="Adornment.End" AdornmentIcon="@Icons.Material.Filled.Search" AdornmentColor="Color.Secondary" />
<MudToggleGroup T="TrackSearchType" <MudToggleGroup T="TrackSearchType"
@bind-Value="_searchType" @bind-Value="_searchType"
@bind-Value:after="SearchTracks" @bind-Value:after="SearchTracks"
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.Artist" Text="Исполнитель" /> </MudToggleGroup>
</MudToggleGroup> </MudItem>
<MudTable Items="@_searchResults" <MudTable Items="@_searchResults"
Virtualize="@true" Virtualize
Height="400px" Hover
Hover="true" Elevation="0"
Breakpoint="Breakpoint.Sm" Class="d-flex flex-grow-1 flex-column"
Loading="@_isSearching"> Style="min-height: 0;"
Breakpoint="Breakpoint.Sm"
Loading="@_isSearching">
<RowTemplate> <RowTemplate>
<MudTd Style="width: 100%;"> <MudTd Class="pa-1" Style="width: 100%;">
<TrackItem Track="@context" PlaylistShareToken="@ShareToken" /> <TrackItem Track="@context" PlaylistShareToken="@ShareToken" />
</MudTd> </MudTd>
<MudTd> <MudTd Class="pa-1">
<MudToggleIconButton Toggled="_addingTrackIds.Contains(context.TrackId)" <MudToggleIconButton Toggled="@ExistingTrackIds.Contains(context.TrackId)"
Icon="@Icons.Material.Filled.AddCircle" Icon="@Icons.Material.Filled.AddCircle"
Color="@Color.Primary" Color="@Color.Primary"
ToggledIcon="@Icons.Material.Filled.RemoveCircle" ToggledIcon="@Icons.Material.Filled.RemoveCircle"
ToggledColor="@Color.Error" ToggledColor="@Color.Error"
ToggledChanged="() => ToggleTrack(context)" /> ToggledChanged="() => ToggleTrack(context)" />
</MudTd> </MudTd>
</RowTemplate> </RowTemplate>
</MudTable> </MudTable>
@@ -54,13 +56,13 @@
[Parameter] public string ShareToken { get; set; } = string.Empty; [Parameter] public string ShareToken { get; set; } = string.Empty;
[Parameter] public EventCallback OnTrackAdded { get; set; } [Parameter] public EventCallback OnTrackAdded { get; set; }
[Parameter] public EventCallback OnTrackRemoved { get; set; } [Parameter] public EventCallback OnTrackRemoved { get; set; }
[Parameter] public HashSet<string> ExistingTrackIds { get; set; } = new();
private string _searchQuery = ""; private string _searchQuery = "";
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 List<YandexTrack> _searchResults = new();
private HashSet<string> _addingTrackIds = new();
private async Task SearchTracks() private async Task SearchTracks()
{ {
@@ -124,7 +126,7 @@
private async Task ToggleTrack(YandexTrack track) private async Task ToggleTrack(YandexTrack track)
{ {
if (_addingTrackIds.Contains(track.TrackId)) if (ExistingTrackIds.Contains(track.TrackId))
{ {
await RemoveTrack(track); await RemoveTrack(track);
} }
@@ -136,67 +138,19 @@
private async Task RemoveTrack(YandexTrack track) private async Task RemoveTrack(YandexTrack track)
{ {
if (!_addingTrackIds.Remove(track.TrackId)) return; if (!ExistingTrackIds.Remove(track.TrackId)) return;
try try
{ {
await RemoveTrackById(track.TrackId); await RemoveTrackById(track.TrackId);
await OnTrackRemoved.InvokeAsync(); await OnTrackRemoved.InvokeAsync();
Snackbar.Add($"Трек \"{track.Title}\" удален", Severity.Success); Snackbar.Add($"Трек \"{track.Title}\" удален", Severity.Success, c => c.SnackbarVariant = Variant.Outlined);
} }
catch (Exception ex) catch (Exception ex)
{ {
Snackbar.Add($"Ошибка добавления: {ex.Message}", Severity.Error); Snackbar.Add($"{ex.Message}", Severity.Error);
_addingTrackIds.Add(track.TrackId); ExistingTrackIds.Add(track.TrackId);
}
finally
{
StateHasChanged();
}
}
private async Task AddTrack(YandexTrack track)
{
if (_addingTrackIds.Contains(track.TrackId)) return;
_addingTrackIds.Add(track.TrackId);
try
{
await AddTrackById(track.TrackId);
await OnTrackAdded.InvokeAsync();
Snackbar.Add($"Трек \"{track.Title}\" добавлен", Severity.Success);
}
catch (Exception ex)
{
Snackbar.Add($"Ошибка добавления: {ex.Message}", Severity.Error);
_addingTrackIds.Remove(track.TrackId);
}
finally
{
StateHasChanged();
}
}
private async Task AddTrackById(string trackId)
{
try
{
var request = new UpdateTrackListRequest { TrackIds = new List<string> { trackId } };
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/add-tracks", request);
if (response.IsSuccessStatusCode)
{
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
Snackbar.Add(error?.Error?.Message ?? "Ошибка добавления трека", Severity.Error);
}
}
catch (Exception ex)
{
Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error);
} }
finally finally
{ {
@@ -206,23 +160,34 @@
private async Task RemoveTrackById(string trackId) private async Task RemoveTrackById(string trackId)
{ {
var request = new UpdateTrackListRequest { TrackIds = new List<string> { trackId } };
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/remove-tracks", request);
if (response.IsSuccessStatusCode)
{
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
throw new Exception(error?.Error?.Message ?? "Ошибка удаления трека");
}
}
private async Task AddTrack(YandexTrack track)
{
if (ExistingTrackIds.Contains(track.TrackId)) return;
ExistingTrackIds.Add(track.TrackId);
try try
{ {
var request = new UpdateTrackListRequest { TrackIds = new List<string> { trackId } }; await AddTrackById(track.TrackId);
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/remove-tracks", request); await OnTrackAdded.InvokeAsync();
if (response.IsSuccessStatusCode) Snackbar.Add($"Трек \"{track.Title}\" добавлен", Severity.Success, c => c.SnackbarVariant = Variant.Outlined);
{
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
Snackbar.Add(error?.Error?.Message ?? "Ошибка удаления трека", Severity.Error);
}
} }
catch (Exception ex) catch (Exception ex)
{ {
Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error); Snackbar.Add($"{ex.Message}", Severity.Error);
ExistingTrackIds.Remove(track.TrackId);
} }
finally finally
{ {
@@ -230,6 +195,21 @@
} }
} }
private async Task AddTrackById(string trackId)
{
var request = new UpdateTrackListRequest { TrackIds = new List<string> { trackId } };
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/add-tracks", request);
if (response.IsSuccessStatusCode)
{
await OnTrackAdded.InvokeAsync(); // уведомляем родителя, что список треков изменился
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
throw new Exception(error?.Error?.Message ?? "Ошибка добавления трека");
}
}
private static (TrackSearchType Type, string Id) ParseYandexMusicUrl(Uri uri) private static (TrackSearchType Type, string Id) ParseYandexMusicUrl(Uri uri)
{ {
var segments = uri.AbsolutePath.Split('/', StringSplitOptions.RemoveEmptyEntries); var segments = uri.AbsolutePath.Split('/', StringSplitOptions.RemoveEmptyEntries);

View File

@@ -40,7 +40,6 @@
Size="Size.Medium" /> Size="Size.Medium" />
} }
</MudStack> </MudStack>
<MudText Typo="Typo.body2" Color="Color.Secondary">Владелец: @Playlist?.Creator?.UserName</MudText>
</MudStack> </MudStack>
</MudStack> </MudStack>

View File

@@ -1,115 +0,0 @@
@using PlaylistShared.Pwa.Components.Common
@using PlaylistShared.Shared.DTO
@using PlaylistShared.Shared.SharedPlaylist
@inject HttpClient Http
@inject ISnackbar Snackbar
@inject IDialogService DialogService
<MudTable Items="@_tracks" Hover="true" Breakpoint="Breakpoint.Sm" Loading="@_loading">
<RowTemplate>
<MudTd Style="width: 100%;">
<TrackItem Track="@context" PlaylistShareToken="@ShareToken" />
</MudTd>
@if (CanRemove)
{
<MudTd>
<MudIconButton Icon="@Icons.Material.Filled.Delete" Color="Color.Error" OnClick="() => RemoveTrack(context)" />
</MudTd>
}
</RowTemplate>
</MudTable>
@code {
[Parameter] public string ShareToken { get; set; } = string.Empty;
[Parameter] public bool CanPlay { get; set; }
[Parameter] public bool CanRemove { get; set; }
[Parameter] public bool IsPlaying { get; set; }
[Parameter] public EventCallback<string> OnPlayTrack { get; set; }
public async Task Reload()
{
await LoadTracks();
}
private List<TrackDisplay> _tracks = new();
private bool _loading = true;
protected override async Task OnInitializedAsync()
{
await LoadTracks();
}
private async Task LoadTracks()
{
_loading = true;
try
{
var response = await Http.GetFromJsonAsync<ApiResponse<YandexPlaylistData>>($"/api/sharedplaylist/{ShareToken}/tracks");
if (response?.Success == true && response.Data != null)
{
_tracks = response.Data.Tracks.Select((t, idx) => new TrackDisplay
{
TrackId = t.TrackId,
Title = t.Title,
Artists = t.Artists,
DurationMs = t.DurationMs,
CoverUri = t.CoverUri,
Index = idx + 1
}).ToList();
}
else
{
Snackbar.Add(response?.Error?.Message ?? "Не удалось загрузить треки", Severity.Error);
}
}
catch (Exception ex)
{
Snackbar.Add($"Ошибка загрузки треков: {ex.Message}", Severity.Error);
}
finally
{
_loading = false;
StateHasChanged();
}
}
private async Task RemoveTrack(TrackDisplay track)
{
var confirmed = await DialogService.ShowMessageBoxAsync(
"Подтверждение удаления",
$"Вы уверены, что хотите удалить трек \"{track.Title}\"?",
yesText: "Удалить", cancelText: "Отмена");
if (confirmed != true) return;
try
{
var request = new UpdateTrackListRequest { TrackIds = new List<string> { track.TrackId } };
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{ShareToken}/remove-tracks", request);
if (response.IsSuccessStatusCode)
{
Snackbar.Add("Трек удалён", Severity.Success);
await LoadTracks();
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
Snackbar.Add(error?.Error?.Message ?? "Ошибка удаления", Severity.Error);
}
}
catch (Exception ex)
{
Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error);
}
}
private async Task PlayTrack(string trackId)
{
await OnPlayTrack.InvokeAsync(trackId);
}
private class TrackDisplay : YandexTrack
{
public int Index { get; set; }
}
}

View File

@@ -21,13 +21,14 @@
<NavMenu /> <NavMenu />
</MudDrawer> </MudDrawer>
<MudMainContent Class="pt-16 pa-4" Style="display: flex; flex-direction: column; min-height: 100vh;"> <MudMainContent Class="pt-16 d-flex flex-column" Style="height: 100vh;">
<div style="flex: 1;"> <MudItem Class="flex-grow-1 overflow-y-auto">
@Body @Body
</div> </MudItem>
<div style="width: 100%; margin-top: 16px;">
<MudItem>
<AudioPlayer /> <AudioPlayer />
</div> </MudItem>
</MudMainContent> </MudMainContent>
</MudLayout> </MudLayout>

View File

@@ -1,4 +1,5 @@
@page "/shared/{token}" @page "/shared/{token}"
@using PlaylistShared.Pwa.Components.Common
@using PlaylistShared.Pwa.Components.SharedPlaylist @using PlaylistShared.Pwa.Components.SharedPlaylist
@using PlaylistShared.Shared.DTO @using PlaylistShared.Shared.DTO
@using PlaylistShared.Shared.Enums @using PlaylistShared.Shared.Enums
@@ -10,7 +11,7 @@
@inject AuthenticationStateProvider AuthProvider @inject AuthenticationStateProvider AuthProvider
@inject IDialogService DialogService @inject IDialogService DialogService
<MudContainer MaxWidth="MaxWidth.ExtraLarge" Class="mt-8"> <MudContainer MaxWidth="MaxWidth.ExtraLarge" Class="pa-4" Style="height: 100%;">
@if (_loading) @if (_loading)
{ {
<MudProgressCircular Indeterminate /> <MudProgressCircular Indeterminate />
@@ -21,37 +22,55 @@
} }
else else
{ {
<MudSplitPanel> <MudSplitPanel Class="flex-grow-1" Style="height: 100%;">
<FirstPanel> <FirstPanel>
<MudCard> <MudCard Class="d-flex flex-column" Style="height: 100%;">
<!-- Заголовок с обложкой -->
<MudCardHeader> <MudCardHeader>
<CardHeaderContent> <CardHeaderContent>
<PlaylistHeader Playlist="@_playlist" /> <PlaylistHeader Playlist="@_playlist" />
</CardHeaderContent> </CardHeaderContent>
</MudCardHeader> </MudCardHeader>
<MudCardContent> <MudCardContent Class="flex-grow-1 d-flex flex-column" Style="overflow: hidden;">
<MudIconButton Icon="@Icons.Material.Filled.Refresh" OnClick="LoadTracks" Disabled="_tracksLoading" Size="Size.Medium" /> <MudItem>
<TracksTable @ref="_tracksTableRef" <MudIconButton Icon="@Icons.Material.Filled.Refresh" OnClick="LoadTracks" Disabled="@_tracksLoading" Size="Size.Medium" />
ShareToken="@Token" </MudItem>
CanPlay="@_canPlay"
CanRemove="@_canRemove" <MudTable Items="@_tracks"
/> Virtualize
Hover
Elevation="0"
Class="d-flex flex-grow-1 flex-column"
Style="min-height: 0;"
Breakpoint="Breakpoint.Sm"
Loading="@_tracksLoading">
<RowTemplate>
<MudTd Class="pa-1" Style="width: 100%;">
<TrackItem Track="@context" PlaylistShareToken="@Token" CanPlay="@_canPlay" />
</MudTd>
@if (_canRemove)
{
<MudTd Class="pa-1">
<MudIconButton Icon="@Icons.Material.Filled.Delete" Color="Color.Error" OnClick="() => RemoveTrack(context)" />
</MudTd>
}
</RowTemplate>
</MudTable>
</MudCardContent> </MudCardContent>
</MudCard> </MudCard>
</FirstPanel> </FirstPanel>
<SecondPanel> <SecondPanel>
@if (_canAdd) @if (_canAdd)
{ {
<MudCard> <MudCard Class="d-flex flex-column" Style="height: 100%;">
<MudCardHeader> <MudCardHeader>
<CardHeaderContent> <CardHeaderContent>
<MudText Typo="Typo.h5" Color="Color.Primary">Добавление треков</MudText> <MudText Typo="Typo.h5" Color="Color.Primary">Добавление треков</MudText>
</CardHeaderContent> </CardHeaderContent>
</MudCardHeader> </MudCardHeader>
<MudCardContent>
<AddTrackSection ShareToken="@Token" OnTrackAdded="LoadTracks" OnTrackRemoved="LoadTracks" /> <MudCardContent Class="flex-grow-1 d-flex flex-column" Style="overflow: hidden;">
<AddTrackSection ShareToken="@Token" OnTrackAdded="LoadTracks" OnTrackRemoved="LoadTracks" ExistingTrackIds="_existingTrackIds" />
</MudCardContent> </MudCardContent>
</MudCard> </MudCard>
} }
@@ -64,7 +83,9 @@
@code { @code {
[Parameter] public string Token { get; set; } [Parameter] public string Token { get; set; }
private TracksTable? _tracksTableRef; private HashSet<string> _existingTrackIds = new();
private bool _firstLoadExistingTrackIds;
private List<YandexTrack> _tracks = new();
private SharedPlaylistDto? _playlist; private SharedPlaylistDto? _playlist;
private bool _loading = true; private bool _loading = true;
@@ -90,6 +111,7 @@
_isAuthenticated = authState.User.Identity?.IsAuthenticated == true; _isAuthenticated = authState.User.Identity?.IsAuthenticated == true;
_currentUserId = authState.User.FindFirst(System.Security.Claims.ClaimTypes.NameIdentifier)?.Value; _currentUserId = authState.User.FindFirst(System.Security.Claims.ClaimTypes.NameIdentifier)?.Value;
await LoadPlaylist(); await LoadPlaylist();
await LoadTracks();
} }
private async Task ConfigurePermissions() private async Task ConfigurePermissions()
@@ -141,7 +163,6 @@
_playlist = response.Data; _playlist = response.Data;
await ConfigurePermissions(); await ConfigurePermissions();
//await LoadTracks();
} }
else else
{ {
@@ -162,14 +183,61 @@
private async Task LoadTracks() private async Task LoadTracks()
{ {
if (_playlist == null) return; if (_playlist == null) return;
if (_tracksTableRef == null) return;
_tracksLoading = true; _tracksLoading = true;
StateHasChanged();
await _tracksTableRef.Reload(); try
{
var response = await Http.GetFromJsonAsync<ApiResponse<YandexPlaylistData>>($"/api/sharedplaylist/{Token}/tracks");
if (response?.Success == true && response.Data != null)
{
_tracks = response.Data.Tracks;
_existingTrackIds = _tracks.Select(t => t.TrackId).ToHashSet();
}
else
{
Snackbar.Add(response?.Error?.Message ?? "Не удалось загрузить треки", Severity.Error);
}
_tracksLoading = false; }
StateHasChanged(); catch (Exception ex)
{
Snackbar.Add($"Ошибка загрузки треков: {ex.Message}", Severity.Error);
}
finally
{
_tracksLoading = false;
StateHasChanged();
}
}
private async Task RemoveTrack(YandexTrack track)
{
var confirmed = await DialogService.ShowMessageBoxAsync(
"Подтверждение удаления",
$"Вы уверены, что хотите удалить трек \"{track.Title}\"?",
yesText: "Удалить", cancelText: "Отмена");
if (confirmed != true) return;
try
{
var request = new UpdateTrackListRequest { TrackIds = new List<string> { track.TrackId } };
var response = await Http.PostAsJsonAsync($"/api/sharedplaylist/{Token}/remove-tracks", request);
if (response.IsSuccessStatusCode)
{
Snackbar.Add("Трек удалён", Severity.Success);
await LoadTracks();
}
else
{
var error = await response.Content.ReadFromJsonAsync<ApiResponse<object>>();
Snackbar.Add(error?.Error?.Message ?? "Ошибка удаления", Severity.Error);
}
}
catch (Exception ex)
{
Snackbar.Add($"Ошибка: {ex.Message}", Severity.Error);
}
} }
} }