2019-01-24 08:43:03 +00: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-24 21:37:53 +00:00
|
|
|
|
|
|
|
|
|
using System;
|
2018-06-01 20:36:25 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
2018-05-24 21:37:53 +00:00
|
|
|
|
using osu.Framework.Audio.Track;
|
2018-06-01 20:36:25 +00:00
|
|
|
|
using osu.Framework.Graphics;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
using osu.Framework.Threading;
|
2018-05-24 21:37:53 +00:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Audio
|
|
|
|
|
{
|
2019-11-01 10:40:45 +00:00
|
|
|
|
[LongRunningLoad]
|
2018-06-21 07:19:07 +00:00
|
|
|
|
public abstract partial class PreviewTrack : Component
|
2018-05-24 21:37:53 +00:00
|
|
|
|
{
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Invoked when this <see cref="PreviewTrack"/> has stopped playing.
|
2019-11-06 06:58:47 +00:00
|
|
|
|
/// Not invoked in a thread-safe context.
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// </summary>
|
2022-07-31 14:00:14 +00:00
|
|
|
|
public event Action? Stopped;
|
2018-06-21 09:54:42 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Invoked when this <see cref="PreviewTrack"/> has started playing.
|
2019-11-06 06:58:47 +00:00
|
|
|
|
/// Not invoked in a thread-safe context.
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// </summary>
|
2022-07-31 14:00:14 +00:00
|
|
|
|
public event Action? Started;
|
2018-05-24 21:37:53 +00:00
|
|
|
|
|
2022-07-31 14:01:30 +00:00
|
|
|
|
protected Track? Track { get; private set; }
|
2019-11-10 20:09:04 +00:00
|
|
|
|
|
2018-06-22 03:12:59 +00:00
|
|
|
|
private bool hasStarted;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
2018-05-24 21:37:53 +00:00
|
|
|
|
{
|
2019-11-10 20:09:04 +00:00
|
|
|
|
Track = GetTrack();
|
|
|
|
|
if (Track != null)
|
|
|
|
|
Track.Completed += Stop;
|
2018-05-24 21:37:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-06-21 07:19:07 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Length of the track.
|
|
|
|
|
/// </summary>
|
2019-11-10 20:09:04 +00:00
|
|
|
|
public double Length => Track?.Length ?? 0;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The current track time.
|
|
|
|
|
/// </summary>
|
2019-11-10 20:09:04 +00:00
|
|
|
|
public double CurrentTime => Track?.CurrentTime ?? 0;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the track is loaded.
|
|
|
|
|
/// </summary>
|
2019-11-10 20:09:04 +00:00
|
|
|
|
public bool TrackLoaded => Track?.IsLoaded ?? false;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2018-06-21 10:31:07 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the track is playing.
|
|
|
|
|
/// </summary>
|
2019-11-10 20:09:04 +00:00
|
|
|
|
public bool IsRunning => Track?.IsRunning ?? false;
|
2018-06-21 10:31:07 +00:00
|
|
|
|
|
2022-07-31 14:01:30 +00:00
|
|
|
|
private ScheduledDelegate? startDelegate;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Starts playing this <see cref="PreviewTrack"/>.
|
|
|
|
|
/// </summary>
|
2019-05-23 03:07:22 +00:00
|
|
|
|
/// <returns>Whether the track is started or already playing.</returns>
|
|
|
|
|
public bool Start()
|
2018-05-24 21:37:53 +00:00
|
|
|
|
{
|
2019-11-10 20:09:04 +00:00
|
|
|
|
if (Track == null)
|
2019-05-23 03:07:22 +00:00
|
|
|
|
return false;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2019-05-23 03:07:22 +00:00
|
|
|
|
startDelegate = Schedule(() =>
|
|
|
|
|
{
|
|
|
|
|
if (hasStarted)
|
|
|
|
|
return;
|
2019-02-27 12:07:17 +00:00
|
|
|
|
|
2019-05-23 03:07:22 +00:00
|
|
|
|
hasStarted = true;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2019-11-10 20:09:04 +00:00
|
|
|
|
Track.Restart();
|
2019-05-23 03:07:22 +00:00
|
|
|
|
Started?.Invoke();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2018-05-24 21:37:53 +00:00
|
|
|
|
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Stops playing this <see cref="PreviewTrack"/>.
|
|
|
|
|
/// </summary>
|
2018-06-21 07:19:07 +00:00
|
|
|
|
public void Stop()
|
2018-05-24 21:37:53 +00:00
|
|
|
|
{
|
2018-06-21 07:19:07 +00:00
|
|
|
|
startDelegate?.Cancel();
|
|
|
|
|
|
2019-11-10 20:09:04 +00:00
|
|
|
|
if (Track == null)
|
2018-06-02 19:06:45 +00:00
|
|
|
|
return;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2018-06-22 03:12:59 +00:00
|
|
|
|
if (!hasStarted)
|
2018-06-21 07:19:07 +00:00
|
|
|
|
return;
|
2019-02-27 12:07:17 +00:00
|
|
|
|
|
2018-06-22 03:12:59 +00:00
|
|
|
|
hasStarted = false;
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2023-12-26 03:06:56 +00:00
|
|
|
|
// This pre-check is important, fixes a BASS deadlock in some scenarios.
|
2023-12-25 18:01:06 +00:00
|
|
|
|
if (!Track.HasCompleted)
|
|
|
|
|
{
|
|
|
|
|
Track.Stop();
|
2019-11-06 06:58:47 +00:00
|
|
|
|
|
2023-12-25 18:01:06 +00: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);
|
|
|
|
|
}
|
2023-02-14 06:39:34 +00:00
|
|
|
|
|
2018-05-24 21:37:53 +00:00
|
|
|
|
Stopped?.Invoke();
|
|
|
|
|
}
|
2018-06-21 07:19:07 +00:00
|
|
|
|
|
2018-06-21 09:54:42 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Retrieves the audio track.
|
|
|
|
|
/// </summary>
|
2022-07-31 14:01:30 +00:00
|
|
|
|
protected abstract Track? GetTrack();
|
2022-08-05 05:15:01 +00:00
|
|
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
|
{
|
|
|
|
|
base.Dispose(isDisposing);
|
2022-12-05 11:45:29 +00:00
|
|
|
|
|
|
|
|
|
Stop();
|
2022-08-05 05:15:01 +00:00
|
|
|
|
Track?.Dispose();
|
|
|
|
|
}
|
2018-05-24 21:37:53 +00:00
|
|
|
|
}
|
|
|
|
|
}
|