Доработан поиск и добавление треков
This commit is contained in:
@@ -74,7 +74,7 @@ public class AudioController : ControllerBase
|
|||||||
|
|
||||||
[HttpGet("track-info/{trackId}")]
|
[HttpGet("track-info/{trackId}")]
|
||||||
[AllowAnonymous]
|
[AllowAnonymous]
|
||||||
public async Task<ActionResult<ApiResponse<TrackInfoDto>>> GetTrackInfo(string trackId, [FromQuery] string? access_token = null, [FromQuery] string? shared_id = null)
|
public async Task<ActionResult<ApiResponse<YandexTrack>>> GetTrackInfo(string trackId, [FromQuery] string? access_token = null, [FromQuery] string? shared_id = null)
|
||||||
{
|
{
|
||||||
var user = await GetUserFromToken(access_token);
|
var user = await GetUserFromToken(access_token);
|
||||||
if (user == null) user = await GetUserFromSharedPlaylistId(shared_id);
|
if (user == null) user = await GetUserFromSharedPlaylistId(shared_id);
|
||||||
@@ -83,10 +83,12 @@ public class AudioController : ControllerBase
|
|||||||
var track = await _yandexService.GetYTrackAsync(user, trackId);
|
var track = await _yandexService.GetYTrackAsync(user, trackId);
|
||||||
if (track == null) return NotFound();
|
if (track == null) return NotFound();
|
||||||
|
|
||||||
return Ok(ApiResponse<TrackInfoDto>.Ok(new TrackInfoDto
|
return Ok(ApiResponse<YandexTrack>.Ok(new YandexTrack
|
||||||
{
|
{
|
||||||
Title = track.Title,
|
Title = track.Title,
|
||||||
CoverUri = track.CoverUri,
|
CoverUri = track.CoverUri,
|
||||||
|
Artists = track.Artists.Select(t => t.Name).ToList(),
|
||||||
|
DurationMs = track.DurationMs,
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -189,7 +189,7 @@ public class SharedPlaylistController : ControllerBase
|
|||||||
Description = playlist.Description ?? "",
|
Description = playlist.Description ?? "",
|
||||||
Tracks = playlist.Tracks?.Select(t => new YandexTrack
|
Tracks = playlist.Tracks?.Select(t => new YandexTrack
|
||||||
{
|
{
|
||||||
Id = 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(a => a.Name).ToList() ?? new List<string>(),
|
||||||
DurationMs = (int)(t.Track?.DurationMs ?? 0),
|
DurationMs = (int)(t.Track?.DurationMs ?? 0),
|
||||||
|
|||||||
@@ -26,13 +26,13 @@ public class YandexSearchController : ControllerBase
|
|||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("tracks")]
|
[HttpGet("tracks")]
|
||||||
public async Task<ActionResult<ApiResponse<List<YandexTrackSearchResult>>>> SearchTracks(
|
public async Task<ActionResult<ApiResponse<List<YandexTrack>>>> SearchTracks(
|
||||||
[FromQuery] string query,
|
[FromQuery] string query,
|
||||||
[FromQuery] int limit = 20,
|
[FromQuery] int limit = 20,
|
||||||
[FromQuery] string? shared_id = null)
|
[FromQuery] string? shared_id = null)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(query))
|
if (string.IsNullOrWhiteSpace(query))
|
||||||
return BadRequest(ApiResponse<List<YandexTrackSearchResult>>.Fail(new ErrorResponse
|
return BadRequest(ApiResponse<List<YandexTrack>>.Fail(new ErrorResponse
|
||||||
{
|
{
|
||||||
StatusCode = 400,
|
StatusCode = 400,
|
||||||
Message = "Поисковый запрос не может быть пустым."
|
Message = "Поисковый запрос не может быть пустым."
|
||||||
@@ -62,13 +62,13 @@ 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<YandexTrackSearchResult>>.Fail(new ErrorResponse
|
return BadRequest(ApiResponse<List<YandexTrack>>.Fail(new ErrorResponse
|
||||||
{
|
{
|
||||||
StatusCode = 400,
|
StatusCode = 400,
|
||||||
Message = "Токен Яндекс.Музыки не установлен или недействителен."
|
Message = "Токен Яндекс.Музыки не установлен или недействителен."
|
||||||
}));
|
}));
|
||||||
|
|
||||||
var results = await _yandexService.SearchTracksAsync(user, query, limit);
|
var results = await _yandexService.SearchTracksAsync(user, query, limit);
|
||||||
return Ok(ApiResponse<List<YandexTrackSearchResult>>.Ok(results));
|
return Ok(ApiResponse<List<YandexTrack>>.Ok(results));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -10,24 +10,24 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.DataProtection.EntityFrameworkCore" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.DataProtection.EntityFrameworkCore" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Identity.UI" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.Identity.UI" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="10.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="10.0.6" />
|
||||||
<PackageReference Include="AspNet.Security.OAuth.Yandex" Version="10.0.0" />
|
<PackageReference Include="AspNet.Security.OAuth.Yandex" Version="10.0.0" />
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="10.0.5">
|
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="10.0.6">
|
||||||
<PrivateAssets>all</PrivateAssets>
|
<PrivateAssets>all</PrivateAssets>
|
||||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="10.0.5" />
|
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.Extensions.Caching.SqlServer" Version="10.0.5" />
|
<PackageReference Include="Microsoft.Extensions.Caching.SqlServer" Version="10.0.6" />
|
||||||
<PackageReference Include="Microsoft.VisualStudio.Azure.Containers.Tools.Targets" Version="1.23.0" />
|
<PackageReference Include="Microsoft.VisualStudio.Azure.Containers.Tools.Targets" Version="1.23.0" />
|
||||||
<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.6" />
|
<PackageReference Include="YandexMusic" Version="0.0.7" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|||||||
@@ -55,12 +55,16 @@ public class YandexMusicService
|
|||||||
{
|
{
|
||||||
var client = await CreateClientAsync(user);
|
var client = await CreateClientAsync(user);
|
||||||
if (client == null) return null;
|
if (client == null) return null;
|
||||||
// Получаем треки по ID
|
|
||||||
var tracks = await client.GetTracksAsync(trackIds);
|
|
||||||
if (tracks == null || !tracks.Any()) return null;
|
|
||||||
var playlist = await client.GetPlaylistAsync(ownerUid, kind);
|
var playlist = await client.GetPlaylistAsync(ownerUid, kind);
|
||||||
if (playlist == null) return null;
|
if (playlist == null) return null;
|
||||||
return await playlist.InsertTracksAsync(tracks.ToArray());
|
|
||||||
|
var tracks = await client.GetTracksAsync(trackIds);
|
||||||
|
if (tracks == null || !tracks.Any()) return null;
|
||||||
|
|
||||||
|
var insertedTracks = tracks.Where(t => !playlist.Tracks.Any(p => p.Track.Id == t.Id)).ToArray();
|
||||||
|
|
||||||
|
return await playlist.InsertTracksAsync(insertedTracks);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<YPlaylist?> RemoveTracksAsync(ApplicationUser user, string ownerUid, string kind, IEnumerable<string> trackIds)
|
public async Task<YPlaylist?> RemoveTracksAsync(ApplicationUser user, string ownerUid, string kind, IEnumerable<string> trackIds)
|
||||||
@@ -103,15 +107,15 @@ public class YandexMusicService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<List<YandexTrackSearchResult>> SearchTracksAsync(ApplicationUser user, string query, int limit = 20)
|
public async Task<List<YandexTrack>> SearchTracksAsync(ApplicationUser user, string query, int limit = 20)
|
||||||
{
|
{
|
||||||
var client = await CreateClientAsync(user);
|
var client = await CreateClientAsync(user);
|
||||||
if (client == null) return new List<YandexTrackSearchResult>();
|
if (client == null) return new List<YandexTrack>();
|
||||||
|
|
||||||
var searchResult = await client.SearchAsync(query, YandexMusic.API.Models.Common.YSearchType.Track, page: 0, pageSize: limit);
|
var searchResult = await client.SearchAsync(query, YandexMusic.API.Models.Common.YSearchType.Track, page: 0, pageSize: limit);
|
||||||
if (searchResult?.Tracks?.Results == null) return new List<YandexTrackSearchResult>();
|
if (searchResult?.Tracks?.Results == null) return new List<YandexTrack>();
|
||||||
|
|
||||||
return searchResult.Tracks.Results.Select(t => new YandexTrackSearchResult
|
return searchResult.Tracks.Results.Select(t => new YandexTrack
|
||||||
{
|
{
|
||||||
TrackId = t.Id,
|
TrackId = t.Id,
|
||||||
Title = t.Title,
|
Title = t.Title,
|
||||||
|
|||||||
@@ -4,14 +4,12 @@
|
|||||||
@inject ISnackbar Snackbar
|
@inject ISnackbar Snackbar
|
||||||
|
|
||||||
<MudPaper Class="pa-4" Elevation="0" Style="background-color: rgba(0,0,0,0.05); border-radius: 8px;">
|
<MudPaper Class="pa-4" Elevation="0" Style="background-color: rgba(0,0,0,0.05); border-radius: 8px;">
|
||||||
<MudText Typo="Typo.h6" GutterBottom>Поиск трека</MudText>
|
|
||||||
|
|
||||||
<div style="display: flex; gap: 8px; margin-bottom: 16px;">
|
<div style="display: flex; gap: 8px; margin-bottom: 16px;">
|
||||||
<MudTextField @bind-Value="_searchQuery"
|
<MudTextField @bind-Value="_searchQuery"
|
||||||
Label="Название трека или исполнитель"
|
Label="Название трека или исполнитель"
|
||||||
Variant="Variant.Outlined"
|
Variant="Variant.Outlined"
|
||||||
FullWidth="true"
|
FullWidth="true"
|
||||||
OnKeyDown="@(async (e) => { if (e.Key == "Enter") await SearchTracks(); })"
|
OnKeyUp="@(async (e) => { if (e.Key == "Enter") await SearchTracks(); })"
|
||||||
Placeholder="Например: Bohemian Rhapsody" />
|
Placeholder="Например: Bohemian Rhapsody" />
|
||||||
<MudButton Variant="Variant.Filled"
|
<MudButton Variant="Variant.Filled"
|
||||||
Color="Color.Primary"
|
Color="Color.Primary"
|
||||||
@@ -44,7 +42,7 @@
|
|||||||
<MudText Typo="Typo.body2" Color="Color.Secondary">@string.Join(", ", track.Artists)</MudText>
|
<MudText Typo="Typo.body2" Color="Color.Secondary">@string.Join(", ", track.Artists)</MudText>
|
||||||
</div>
|
</div>
|
||||||
<div style="flex-shrink: 0;">
|
<div style="flex-shrink: 0;">
|
||||||
<MudText Typo="Typo.body2">@FormatDuration(track.DurationMs)</MudText>
|
<MudText Typo="Typo.body2">@track.DurationMs.FormatDuration()</MudText>
|
||||||
</div>
|
</div>
|
||||||
<div style="flex-shrink: 0;">
|
<div style="flex-shrink: 0;">
|
||||||
<MudIconButton Icon="@Icons.Material.Filled.AddCircle"
|
<MudIconButton Icon="@Icons.Material.Filled.AddCircle"
|
||||||
@@ -57,25 +55,29 @@
|
|||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
else if (!string.IsNullOrEmpty(_searchQuery) && !_isSearching)
|
else if (!_isFirstSearch)
|
||||||
{
|
{
|
||||||
<MudAlert Severity="Severity.Info">Ничего не найдено. Попробуйте изменить запрос.</MudAlert>
|
<MudAlert Severity="Severity.Info">Ничего не найдено. Попробуйте изменить запрос.</MudAlert>
|
||||||
}
|
}
|
||||||
</MudPaper>
|
</MudPaper>
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
[Parameter] public EventCallback<string> OnAddTrack { get; set; }
|
[Parameter] public EventCallback<string> OnAddTrack { get; set; }
|
||||||
[Parameter] public string ShareToken { get; set; } = string.Empty;
|
[Parameter] public string ShareToken { get; set; } = string.Empty;
|
||||||
|
|
||||||
private string _searchQuery = "";
|
private List<YandexTrack> _searchResults = new();
|
||||||
private List<YandexTrackSearchResult> _searchResults = new();
|
|
||||||
private bool _isSearching;
|
private bool _isSearching;
|
||||||
|
private bool _isFirstSearch = true;
|
||||||
private HashSet<string> _addingTrackIds = new();
|
private HashSet<string> _addingTrackIds = new();
|
||||||
|
private string _searchQuery = string.Empty;
|
||||||
|
|
||||||
private async Task SearchTracks()
|
private async Task SearchTracks()
|
||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(_searchQuery)) return;
|
|
||||||
|
|
||||||
|
if (string.IsNullOrWhiteSpace(_searchQuery))
|
||||||
|
return;
|
||||||
|
|
||||||
|
_isFirstSearch = false;
|
||||||
_isSearching = true;
|
_isSearching = true;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -83,7 +85,7 @@
|
|||||||
if (!string.IsNullOrEmpty(ShareToken))
|
if (!string.IsNullOrEmpty(ShareToken))
|
||||||
url += $"&shared_id={Uri.EscapeDataString(ShareToken)}";
|
url += $"&shared_id={Uri.EscapeDataString(ShareToken)}";
|
||||||
|
|
||||||
var response = await Http.GetFromJsonAsync<ApiResponse<List<YandexTrackSearchResult>>>(url);
|
var response = await Http.GetFromJsonAsync<ApiResponse<List<YandexTrack>>>(url);
|
||||||
if (response?.Success == true)
|
if (response?.Success == true)
|
||||||
_searchResults = response.Data ?? new();
|
_searchResults = response.Data ?? new();
|
||||||
else
|
else
|
||||||
@@ -100,7 +102,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task AddTrack(YandexTrackSearchResult track)
|
private async Task AddTrack(YandexTrack track)
|
||||||
{
|
{
|
||||||
if (_addingTrackIds.Contains(track.TrackId)) return;
|
if (_addingTrackIds.Contains(track.TrackId)) return;
|
||||||
_addingTrackIds.Add(track.TrackId);
|
_addingTrackIds.Add(track.TrackId);
|
||||||
@@ -119,12 +121,4 @@
|
|||||||
StateHasChanged();
|
StateHasChanged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private string FormatDuration(long ms)
|
|
||||||
{
|
|
||||||
var seconds = ms / 1000;
|
|
||||||
var mins = seconds / 60;
|
|
||||||
var secs = seconds % 60;
|
|
||||||
return $"{mins}:{secs:D2}";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -9,7 +9,8 @@
|
|||||||
<MudItem xs="12" sm="10">
|
<MudItem xs="12" sm="10">
|
||||||
<MudTextField @bind-Value="_trackLink" Label="Ссылка на трек Яндекс.Музыки"
|
<MudTextField @bind-Value="_trackLink" Label="Ссылка на трек Яндекс.Музыки"
|
||||||
Variant="Variant.Outlined" FullWidth="true"
|
Variant="Variant.Outlined" FullWidth="true"
|
||||||
Placeholder="https://music.yandex.ru/album/2488464/track/21696942" />
|
Placeholder="https://music.yandex.ru/album/2488464/track/21696942"
|
||||||
|
OnKeyUp="@(async (e) => { if (e.Key == "Enter") await AddTrackByLink(); })" />
|
||||||
</MudItem>
|
</MudItem>
|
||||||
<MudItem xs="12" sm="2">
|
<MudItem xs="12" sm="2">
|
||||||
<MudButton Variant="Variant.Filled" Color="Color.Primary" OnClick="AddTrackByLink"
|
<MudButton Variant="Variant.Filled" Color="Color.Primary" OnClick="AddTrackByLink"
|
||||||
@@ -25,9 +26,6 @@
|
|||||||
</MudButton>
|
</MudButton>
|
||||||
</MudItem>
|
</MudItem>
|
||||||
</MudGrid>
|
</MudGrid>
|
||||||
<MudText Typo="Typo.body2" Class="mt-2" Color="Color.Secondary">
|
|
||||||
Поддерживаются ссылки вида: https://music.yandex.ru/album/12345/track/67890
|
|
||||||
</MudText>
|
|
||||||
</MudTabPanel>
|
</MudTabPanel>
|
||||||
<MudTabPanel Text="Поиск" Style="padding: 16px;">
|
<MudTabPanel Text="Поиск" Style="padding: 16px;">
|
||||||
<AddTrackBySearch OnAddTrack="AddTrackById"
|
<AddTrackBySearch OnAddTrack="AddTrackById"
|
||||||
|
|||||||
@@ -25,7 +25,7 @@
|
|||||||
@if (CanPlay)
|
@if (CanPlay)
|
||||||
{
|
{
|
||||||
<TrackCoverWithPlay CoverUrl="@context.CoverUri"
|
<TrackCoverWithPlay CoverUrl="@context.CoverUri"
|
||||||
TrackId="@context.Id"
|
TrackId="@context.TrackId"
|
||||||
Width="50" Height="50"/>
|
Width="50" Height="50"/>
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -35,13 +35,13 @@
|
|||||||
}
|
}
|
||||||
</MudTd>
|
</MudTd>
|
||||||
<MudTd DataLabel="Название">
|
<MudTd DataLabel="Название">
|
||||||
<MudLink Href="@($"https://music.yandex.ru/track/{context.Id}")" Target="_blank" Underline="Underline.Hover">
|
<MudLink Href="@($"https://music.yandex.ru/track/{context.TrackId}")" Target="_blank" Underline="Underline.Hover">
|
||||||
@context.Title
|
@context.Title
|
||||||
<MudIcon Icon="@Icons.Material.Filled.OpenInNew" Size="Size.Small" Class="ml-1" />
|
<MudIcon Icon="@Icons.Material.Filled.OpenInNew" Size="Size.Small" Class="ml-1" />
|
||||||
</MudLink>
|
</MudLink>
|
||||||
</MudTd>
|
</MudTd>
|
||||||
<MudTd DataLabel="Исполнитель">@string.Join(", ", context.Artists)</MudTd>
|
<MudTd DataLabel="Исполнитель">@string.Join(", ", context.Artists)</MudTd>
|
||||||
<MudTd DataLabel="Длительность">@FormatDuration(context.DurationMs)</MudTd>
|
<MudTd DataLabel="Длительность">@context.DurationMs.FormatDuration()</MudTd>
|
||||||
@if (CanRemove)
|
@if (CanRemove)
|
||||||
{
|
{
|
||||||
<MudTd DataLabel="">
|
<MudTd DataLabel="">
|
||||||
@@ -82,7 +82,7 @@
|
|||||||
{
|
{
|
||||||
_tracks = response.Data.Tracks.Select((t, idx) => new TrackDisplay
|
_tracks = response.Data.Tracks.Select((t, idx) => new TrackDisplay
|
||||||
{
|
{
|
||||||
Id = t.Id,
|
TrackId = t.TrackId,
|
||||||
Title = t.Title,
|
Title = t.Title,
|
||||||
Artists = t.Artists,
|
Artists = t.Artists,
|
||||||
DurationMs = t.DurationMs,
|
DurationMs = t.DurationMs,
|
||||||
@@ -117,7 +117,7 @@
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var request = new RemoveTracksRequest { TrackIds = new List<string> { track.Id } };
|
var request = new RemoveTracksRequest { TrackIds = new List<string> { track.TrackId } };
|
||||||
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)
|
||||||
{
|
{
|
||||||
@@ -141,14 +141,6 @@
|
|||||||
await OnPlayTrack.InvokeAsync(trackId);
|
await OnPlayTrack.InvokeAsync(trackId);
|
||||||
}
|
}
|
||||||
|
|
||||||
private string FormatDuration(long ms)
|
|
||||||
{
|
|
||||||
var seconds = ms / 1000;
|
|
||||||
var mins = seconds / 60;
|
|
||||||
var secs = seconds % 60;
|
|
||||||
return $"{mins}:{secs:D2}";
|
|
||||||
}
|
|
||||||
|
|
||||||
private class TrackDisplay : YandexTrack
|
private class TrackDisplay : YandexTrack
|
||||||
{
|
{
|
||||||
public int Index { get; set; }
|
public int Index { get; set; }
|
||||||
|
|||||||
15
PlaylistShared.Pwa/Extensions/LongExtensions.cs
Normal file
15
PlaylistShared.Pwa/Extensions/LongExtensions.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
namespace PlaylistShared.Pwa.Extensions;
|
||||||
|
|
||||||
|
public static class LongExtensions
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Преобразует миллисекунды в формат Минуты:Секунды
|
||||||
|
/// </summary>
|
||||||
|
public static string FormatDuration(this long ms)
|
||||||
|
{
|
||||||
|
var seconds = ms / 1000;
|
||||||
|
var mins = seconds / 60;
|
||||||
|
var secs = seconds % 60;
|
||||||
|
return $"{mins}:{secs:D2}";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -9,7 +9,6 @@
|
|||||||
<MudLayout>
|
<MudLayout>
|
||||||
<MudAppBar Elevation="1">
|
<MudAppBar Elevation="1">
|
||||||
<MudIconButton Icon="@Icons.Material.Filled.Menu" Color="Color.Inherit" Edge="Edge.Start" OnClick="@DrawerToggle" />
|
<MudIconButton Icon="@Icons.Material.Filled.Menu" Color="Color.Inherit" Edge="Edge.Start" OnClick="@DrawerToggle" />
|
||||||
<MudText Typo="Typo.h6" Class="ml-2">Playlist share</MudText>
|
|
||||||
<MudSpacer />
|
<MudSpacer />
|
||||||
<LoginDisplay />
|
<LoginDisplay />
|
||||||
<MudIconButton Icon="@(DarkLightModeButtonIcon)" Color="Color.Inherit" OnClick="@DarkModeToggle" Class="ml-2" />
|
<MudIconButton Icon="@(DarkLightModeButtonIcon)" Color="Color.Inherit" OnClick="@DarkModeToggle" Class="ml-2" />
|
||||||
|
|||||||
@@ -25,7 +25,7 @@
|
|||||||
|
|
||||||
<!-- Локальная форма входа -->
|
<!-- Локальная форма входа -->
|
||||||
<MudTextField @bind-Value="_loginModel.Username" Label="Имя пользователя" Variant="Variant.Outlined" FullWidth="true" Class="mb-3" />
|
<MudTextField @bind-Value="_loginModel.Username" Label="Имя пользователя" Variant="Variant.Outlined" FullWidth="true" Class="mb-3" />
|
||||||
<MudTextField @bind-Value="_loginModel.Password" Label="Пароль" Variant="Variant.Outlined" FullWidth="true" InputType="InputType.Password" @onkeypress="@(async (e) => { if (e.Key == "Enter") await LocalLogin(); })" />
|
<MudTextField @bind-Value="_loginModel.Password" Label="Пароль" Variant="Variant.Outlined" FullWidth="true" InputType="InputType.Password" OnKeyUp="@(async (e) => { if (e.Key == "Enter") await LocalLogin(); })" />
|
||||||
|
|
||||||
<MudButton Variant="Variant.Outlined" Color="Color.Secondary" OnClick="LocalLogin" FullWidth="true" Class="mt-4">
|
<MudButton Variant="Variant.Outlined" Color="Color.Secondary" OnClick="LocalLogin" FullWidth="true" Class="mt-4">
|
||||||
Войти (локально)
|
Войти (локально)
|
||||||
@@ -48,6 +48,12 @@
|
|||||||
|
|
||||||
private async Task LocalLogin()
|
private async Task LocalLogin()
|
||||||
{
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(_loginModel.Username) || string.IsNullOrWhiteSpace(_loginModel.Password))
|
||||||
|
{
|
||||||
|
Snackbar.Add("Пожалуйста, заполните все поля", Severity.Warning);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var response = await Http.PostAsJsonAsync("/api/account/login", _loginModel);
|
var response = await Http.PostAsJsonAsync("/api/account/login", _loginModel);
|
||||||
if (response.IsSuccessStatusCode)
|
if (response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -176,7 +176,7 @@ public class AudioPlayerService : IAudioPlayerService
|
|||||||
else if (!string.IsNullOrEmpty(sharedPlaylistId))
|
else if (!string.IsNullOrEmpty(sharedPlaylistId))
|
||||||
url += $"?shared_id={sharedPlaylistId}";
|
url += $"?shared_id={sharedPlaylistId}";
|
||||||
|
|
||||||
var response = await _http.GetFromJsonAsync<ApiResponse<TrackInfoDto>>(url);
|
var response = await _http.GetFromJsonAsync<ApiResponse<YandexTrack>>(url);
|
||||||
if (response?.Success == true)
|
if (response?.Success == true)
|
||||||
{
|
{
|
||||||
return (response.Data.Title, response.Data.CoverUri);
|
return (response.Data.Title, response.Data.CoverUri);
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
namespace PlaylistShared.Shared.DTO;
|
|
||||||
|
|
||||||
public class TrackInfoDto
|
|
||||||
{
|
|
||||||
public string Title { get; set; }
|
|
||||||
public string CoverUri { get; set; }
|
|
||||||
}
|
|
||||||
@@ -1,10 +1,22 @@
|
|||||||
namespace PlaylistShared.Shared.DTO;
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
|
namespace PlaylistShared.Shared.DTO;
|
||||||
|
|
||||||
|
/// <summary>Результат поиска трека в Яндекс.Музыке.</summary>
|
||||||
public class YandexTrack
|
public class YandexTrack
|
||||||
{
|
{
|
||||||
public string Id { get; set; } = "";
|
[JsonPropertyName("trackId")]
|
||||||
public string Title { get; set; } = "";
|
public string TrackId { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[JsonPropertyName("title")]
|
||||||
|
public string Title { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[JsonPropertyName("artists")]
|
||||||
public List<string> Artists { get; set; } = new();
|
public List<string> Artists { get; set; } = new();
|
||||||
|
|
||||||
|
[JsonPropertyName("coverUri")]
|
||||||
|
public string CoverUri { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[JsonPropertyName("durationMs")]
|
||||||
public long DurationMs { get; set; }
|
public long DurationMs { get; set; }
|
||||||
public string CoverUri { get; set; } = "";
|
|
||||||
}
|
}
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
using System.Text.Json.Serialization;
|
|
||||||
|
|
||||||
namespace PlaylistShared.Shared.DTO;
|
|
||||||
|
|
||||||
/// <summary>Результат поиска трека в Яндекс.Музыке.</summary>
|
|
||||||
public class YandexTrackSearchResult
|
|
||||||
{
|
|
||||||
[JsonPropertyName("trackId")]
|
|
||||||
public string TrackId { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
[JsonPropertyName("title")]
|
|
||||||
public string Title { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
[JsonPropertyName("artists")]
|
|
||||||
public List<string> Artists { get; set; } = new();
|
|
||||||
|
|
||||||
[JsonPropertyName("coverUri")]
|
|
||||||
public string CoverUri { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
[JsonPropertyName("durationMs")]
|
|
||||||
public long DurationMs { get; set; }
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user