Доработан плеер

This commit is contained in:
FrigaT
2026-04-16 04:11:04 +03:00
parent 3e18537a0e
commit 35140b71b7
4 changed files with 181 additions and 200 deletions

View File

@@ -10,15 +10,18 @@ public class AudioPlayerService : IAudioPlayerService
private readonly TokenStorage _tokenStorage;
private readonly ISnackbar _snackbar;
private readonly HttpClient _http;
private readonly PlayerStorage _playerStorage;
private string? _currentTrackId;
private string? _currentTrackTitle;
private string? _currentTrackCoverUrl;
private bool _isPlaying;
private double _currentVolume = 70;
private double _currentVolume = 50;
private double _currentProgress;
private string _currentTime = "0:00";
private string _totalTime = "0:00";
private double _currentTime = 0;
private double _totalTime = 0;
private string _currentTimeString = "0:00";
private string _totalTimeString = "0:00";
public string? CurrentTrackId => _currentTrackId;
public string? CurrentTrackTitle => _currentTrackTitle;
@@ -34,21 +37,42 @@ public class AudioPlayerService : IAudioPlayerService
}
}
public double CurrentProgress => _currentProgress;
public string CurrentTime => _currentTime;
public string TotalTime => _totalTime;
public double CurrentTime => _currentTime;
public double TotalTime => _totalTime;
public string CurrentTimeString => _currentTimeString;
public string TotalTimeString => _totalTimeString;
public event Action? OnStateChanged;
public AudioPlayerService(TokenStorage tokenStorage, ISnackbar snackbar, HttpClient httpClient)
public AudioPlayerService(TokenStorage tokenStorage, ISnackbar snackbar, HttpClient httpClient, PlayerStorage playerStorage)
{
_tokenStorage = tokenStorage;
_snackbar = snackbar;
_http = httpClient;
_playerStorage = playerStorage;
_ = LoadVolume();
}
private async Task LoadVolume()
{
var savedVolume = await _playerStorage.GetVolumeAsync();
if (savedVolume != null)
{
_currentVolume = savedVolume.Value;
}
}
// Внешние команды (вызываются из компонентов)
public async Task LoadAndPlayAsync(string trackId, string? accessToken = null, string? playlistShareToken = null, string? title = null, string? coverUrl = null)
{
if (_currentTrackId == trackId)
{
await PlayAsync();
return;
}
// Если accessToken не передан, пытаемся получить его из хранилища
if (string.IsNullOrWhiteSpace(accessToken))
{
@@ -101,19 +125,10 @@ public class AudioPlayerService : IAudioPlayerService
OnPauseRequested?.Invoke();
}
public async Task StopAsync()
{
_isPlaying = false;
_currentTrackId = null;
_currentProgress = 0;
_currentTime = "0:00";
OnStateChanged?.Invoke();
OnStopRequested?.Invoke();
}
public async Task SeekToAsync(double percent)
{
OnSeekRequested?.Invoke(percent);
var newTime = (percent / 100) * _totalTime;
OnSeekRequested?.Invoke(newTime);
}
public async Task SetVolumeAsync(double volume)
@@ -121,13 +136,13 @@ public class AudioPlayerService : IAudioPlayerService
_currentVolume = volume;
OnStateChanged?.Invoke();
OnVolumeChangeRequested?.Invoke(volume);
await _playerStorage.SetVolumeAsync(volume);
}
// События для связи с реальным AudioPlayer компонентом
public event Func<string, string?, string?, Task>? OnLoadAndPlayRequested;
public event Func<Task>? OnPlayRequested;
public event Func<Task>? OnPauseRequested;
public event Func<Task>? OnStopRequested;
public event Func<double, Task>? OnSeekRequested;
public event Func<double, Task>? OnVolumeChangeRequested;
@@ -144,11 +159,14 @@ public class AudioPlayerService : IAudioPlayerService
OnStateChanged?.Invoke();
}
public void UpdateProgress(double progress, string currentTime, string totalTime)
public void UpdateProgress(double currentTime, double totalTime)
{
var progress = (currentTime / totalTime) * 100;
_currentProgress = progress;
_currentTime = currentTime;
_currentTimeString = FormatDuration(currentTime);
_totalTime = totalTime;
_totalTimeString = FormatDuration(totalTime);
OnStateChanged?.Invoke();
}
@@ -157,7 +175,10 @@ public class AudioPlayerService : IAudioPlayerService
_isPlaying = false;
_currentTrackId = null;
_currentProgress = 0;
_currentTime = "0:00";
_currentTime = 0;
_currentTimeString = "0:00";
_totalTime = 0;
_currentTimeString = "0:00";
OnStateChanged?.Invoke();
}
@@ -183,4 +204,11 @@ public class AudioPlayerService : IAudioPlayerService
}
return null;
}
private string FormatDuration(double seconds)
{
var mins = (int)(seconds / 60);
var secs = (int)(seconds % 60);
return $"{mins}:{secs:D2}";
}
}

View File

@@ -19,11 +19,17 @@ public interface IAudioPlayerService
/// <summary>Прогресс воспроизведения в процентах (0100).</summary>
double CurrentProgress { get; }
/// <summary>Текущее время в секундах.</summary>
double CurrentTime { get; }
/// <summary>Общая длительность в секундах</summary>
double TotalTime { get; }
/// <summary>Отформатированное текущее время (мм:сс).</summary>
string CurrentTime { get; }
string CurrentTimeString { get; }
/// <summary>Отформатированная общая длительность (мм:сс).</summary>
string TotalTime { get; }
string TotalTimeString { get; }
/// <summary>Отформатированное название текущего трека.</summary>
string? CurrentTrackTitle { get; }
@@ -47,9 +53,6 @@ public interface IAudioPlayerService
/// <summary>Поставить на паузу.</summary>
Task PauseAsync();
/// <summary>Остановить воспроизведение и выгрузить трек.</summary>
Task StopAsync();
/// <summary>Перемотать на указанный процент (0100).</summary>
Task SeekToAsync(double percent);
@@ -76,10 +79,7 @@ public interface IAudioPlayerService
/// <summary>Запрос на паузу.</summary>
event Func<Task>? OnPauseRequested;
/// <summary>Запрос на остановку и выгрузку трека.</summary>
event Func<Task>? OnStopRequested;
/// <summary>Запрос на перемотку (процент 0100).</summary>
/// <summary>Запрос на перемотку (секунды).</summary>
event Func<double, Task>? OnSeekRequested;
/// <summary>Запрос на изменение громкости (0100).</summary>
@@ -94,7 +94,7 @@ public interface IAudioPlayerService
void SetCurrentTrack(string? trackId);
/// <summary>Обновить прогресс и отображаемое время.</summary>
void UpdateProgress(double progress, string currentTime, string totalTime);
void UpdateProgress(double currentTime, double totalTime);
/// <summary>Уведомить об окончании трека.</summary>
void NotifyTrackEnded();