osu/osu.Game/Audio/PreviewTrack.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

125 lines
3.2 KiB
C#
Raw Normal View History

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