2019-01-24 17:43:03 +09:00
|
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-05-25 00:37:53 +03:00
|
|
|
|
|
|
|
|
|
using System;
|
2018-06-01 23:36:25 +03:00
|
|
|
|
using osu.Framework.Allocation;
|
2018-05-25 00:37:53 +03:00
|
|
|
|
using osu.Framework.Audio.Track;
|
2018-06-01 23:36:25 +03:00
|
|
|
|
using osu.Framework.Graphics;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
using osu.Framework.Threading;
|
2018-05-25 00:37:53 +03:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Audio
|
|
|
|
|
{
|
2019-11-01 19:40:45 +09:00
|
|
|
|
[LongRunningLoad]
|
2018-06-21 16:19:07 +09:00
|
|
|
|
public abstract partial class PreviewTrack : Component
|
2018-05-25 00:37:53 +03:00
|
|
|
|
{
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Invoked when this <see cref="PreviewTrack"/> has stopped playing.
|
2019-11-06 15:58:47 +09:00
|
|
|
|
/// Not invoked in a thread-safe context.
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// </summary>
|
2022-07-31 22:00:14 +08:00
|
|
|
|
public event Action? Stopped;
|
2018-06-21 18:54:42 +09:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Invoked when this <see cref="PreviewTrack"/> has started playing.
|
2019-11-06 15:58:47 +09:00
|
|
|
|
/// Not invoked in a thread-safe context.
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// </summary>
|
2022-07-31 22:00:14 +08:00
|
|
|
|
public event Action? Started;
|
2018-05-25 00:37:53 +03:00
|
|
|
|
|
2022-07-31 22:01:30 +08:00
|
|
|
|
protected Track? Track { get; private set; }
|
2019-11-10 23:09:04 +03:00
|
|
|
|
|
2018-06-22 12:12:59 +09:00
|
|
|
|
private bool hasStarted;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
2018-05-25 00:37:53 +03:00
|
|
|
|
{
|
2019-11-10 23:09:04 +03:00
|
|
|
|
Track = GetTrack();
|
|
|
|
|
if (Track != null)
|
|
|
|
|
Track.Completed += Stop;
|
2018-05-25 00:37:53 +03:00
|
|
|
|
}
|
|
|
|
|
|
2018-06-21 16:19:07 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Length of the track.
|
|
|
|
|
/// </summary>
|
2019-11-10 23:09:04 +03:00
|
|
|
|
public double Length => Track?.Length ?? 0;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The current track time.
|
|
|
|
|
/// </summary>
|
2019-11-10 23:09:04 +03:00
|
|
|
|
public double CurrentTime => Track?.CurrentTime ?? 0;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the track is loaded.
|
|
|
|
|
/// </summary>
|
2019-11-10 23:09:04 +03:00
|
|
|
|
public bool TrackLoaded => Track?.IsLoaded ?? false;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2018-06-21 19:31:07 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the track is playing.
|
|
|
|
|
/// </summary>
|
2019-11-10 23:09:04 +03:00
|
|
|
|
public bool IsRunning => Track?.IsRunning ?? false;
|
2018-06-21 19:31:07 +09:00
|
|
|
|
|
2022-07-31 22:01:30 +08:00
|
|
|
|
private ScheduledDelegate? startDelegate;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Starts playing this <see cref="PreviewTrack"/>.
|
|
|
|
|
/// </summary>
|
2019-05-23 12:07:22 +09:00
|
|
|
|
/// <returns>Whether the track is started or already playing.</returns>
|
|
|
|
|
public bool Start()
|
2018-05-25 00:37:53 +03:00
|
|
|
|
{
|
2019-11-10 23:09:04 +03:00
|
|
|
|
if (Track == null)
|
2019-05-23 12:07:22 +09:00
|
|
|
|
return false;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2019-05-23 12:07:22 +09:00
|
|
|
|
startDelegate = Schedule(() =>
|
|
|
|
|
{
|
|
|
|
|
if (hasStarted)
|
|
|
|
|
return;
|
2019-02-27 21:07:17 +09:00
|
|
|
|
|
2019-05-23 12:07:22 +09:00
|
|
|
|
hasStarted = true;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2019-11-10 23:09:04 +03:00
|
|
|
|
Track.Restart();
|
2019-05-23 12:07:22 +09:00
|
|
|
|
Started?.Invoke();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2018-05-25 00:37:53 +03:00
|
|
|
|
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Stops playing this <see cref="PreviewTrack"/>.
|
|
|
|
|
/// </summary>
|
2018-06-21 16:19:07 +09:00
|
|
|
|
public void Stop()
|
2018-05-25 00:37:53 +03:00
|
|
|
|
{
|
2018-06-21 16:19:07 +09:00
|
|
|
|
startDelegate?.Cancel();
|
|
|
|
|
|
2019-11-10 23:09:04 +03:00
|
|
|
|
if (Track == null)
|
2018-06-02 22:06:45 +03:00
|
|
|
|
return;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2018-06-22 12:12:59 +09:00
|
|
|
|
if (!hasStarted)
|
2018-06-21 16:19:07 +09:00
|
|
|
|
return;
|
2019-02-27 21:07:17 +09:00
|
|
|
|
|
2018-06-22 12:12:59 +09:00
|
|
|
|
hasStarted = false;
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2019-11-10 23:09:04 +03:00
|
|
|
|
Track.Stop();
|
2019-11-06 15:58:47 +09:00
|
|
|
|
|
2023-02-13 22:39:34 -08:00
|
|
|
|
// Ensure the track is reset immediately on stopping, so the next time it is started it has a correct time value.
|
|
|
|
|
Track.Seek(0);
|
|
|
|
|
|
2018-05-25 00:37:53 +03:00
|
|
|
|
Stopped?.Invoke();
|
|
|
|
|
}
|
2018-06-21 16:19:07 +09:00
|
|
|
|
|
2018-06-21 18:54:42 +09:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves the audio track.
|
|
|
|
|
/// </summary>
|
2022-07-31 22:01:30 +08:00
|
|
|
|
protected abstract Track? GetTrack();
|
2022-08-05 14:15:01 +09:00
|
|
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
|
{
|
|
|
|
|
base.Dispose(isDisposing);
|
2022-12-05 14:45:29 +03:00
|
|
|
|
|
|
|
|
|
Stop();
|
2022-08-05 14:15:01 +09:00
|
|
|
|
Track?.Dispose();
|
|
|
|
|
}
|
2018-05-25 00:37:53 +03:00
|
|
|
|
}
|
|
|
|
|
}
|