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-04-13 18:19:50 +09:00
using System ;
using System.Collections.Generic ;
2021-11-18 23:26:45 +09:00
using System.Diagnostics ;
2021-04-17 17:47:13 +02:00
using System.IO ;
2019-02-21 19:04:31 +09:00
using System.Linq ;
2018-09-06 12:51:23 +09:00
using System.Threading ;
2019-06-24 12:42:21 +09:00
using System.Threading.Tasks ;
2020-08-12 00:48:45 +09:00
using JetBrains.Annotations ;
2019-05-28 23:54:42 +09:00
using osu.Framework.Audio ;
2020-03-07 21:32:03 -08:00
using osu.Framework.Audio.Track ;
using osu.Framework.Graphics.Textures ;
using osu.Framework.Logging ;
2020-09-04 20:34:26 +09:00
using osu.Framework.Testing ;
2018-04-19 22:04:12 +09:00
using osu.Game.Rulesets ;
2020-03-07 21:32:03 -08:00
using osu.Game.Rulesets.Mods ;
2019-05-29 16:43:27 +09:00
using osu.Game.Rulesets.Objects.Types ;
2018-04-19 22:04:12 +09:00
using osu.Game.Rulesets.UI ;
2018-04-13 18:19:50 +09:00
using osu.Game.Skinning ;
2020-03-07 21:32:03 -08:00
using osu.Game.Storyboards ;
2018-04-13 18:19:50 +09:00
namespace osu.Game.Beatmaps
{
2020-09-04 20:34:26 +09:00
[ExcludeFromDynamicCompile]
2020-02-10 17:01:41 +09:00
public abstract class WorkingBeatmap : IWorkingBeatmap
2018-04-13 18:19:50 +09:00
{
public readonly BeatmapInfo BeatmapInfo ;
public readonly BeatmapSetInfo BeatmapSetInfo ;
public readonly BeatmapMetadata Metadata ;
2019-05-31 14:40:53 +09:00
protected AudioManager AudioManager { get ; }
protected WorkingBeatmap ( BeatmapInfo beatmapInfo , AudioManager audioManager )
2018-04-13 18:19:50 +09:00
{
2019-05-31 14:40:53 +09:00
AudioManager = audioManager ;
2018-04-13 18:19:50 +09:00
BeatmapInfo = beatmapInfo ;
BeatmapSetInfo = beatmapInfo . BeatmapSet ;
Metadata = beatmapInfo . Metadata ? ? BeatmapSetInfo ? . Metadata ? ? new BeatmapMetadata ( ) ;
2018-09-06 12:51:23 +09:00
background = new RecyclableLazy < Texture > ( GetBackground , BackgroundStillValid ) ;
waveform = new RecyclableLazy < Waveform > ( GetWaveform ) ;
storyboard = new RecyclableLazy < Storyboard > ( GetStoryboard ) ;
2019-08-28 19:57:17 +09:00
skin = new RecyclableLazy < ISkin > ( GetSkin ) ;
2018-04-13 18:19:50 +09:00
}
2020-02-09 21:34:56 +09:00
protected virtual Track GetVirtualTrack ( double emptyLength = 0 )
2019-05-29 16:43:27 +09:00
{
const double excess_length = 1000 ;
2020-09-03 19:20:42 +09:00
var lastObject = Beatmap ? . HitObjects . LastOrDefault ( ) ;
2019-05-29 16:43:27 +09:00
double length ;
switch ( lastObject )
{
case null :
2020-02-09 21:34:56 +09:00
length = emptyLength ;
2019-05-29 16:43:27 +09:00
break ;
2020-05-27 12:38:39 +09:00
case IHasDuration endTime :
2019-05-29 16:43:27 +09:00
length = endTime . EndTime + excess_length ;
break ;
default :
length = lastObject . StartTime + excess_length ;
break ;
}
return AudioManager . Tracks . GetVirtual ( length ) ;
}
2019-07-31 19:48:50 +09:00
/// <summary>
/// Creates a <see cref="IBeatmapConverter"/> to convert a <see cref="IBeatmap"/> for a specified <see cref="Ruleset"/>.
/// </summary>
/// <param name="beatmap">The <see cref="IBeatmap"/> to be converted.</param>
/// <param name="ruleset">The <see cref="Ruleset"/> for which <paramref name="beatmap"/> should be converted.</param>
/// <returns>The applicable <see cref="IBeatmapConverter"/>.</returns>
protected virtual IBeatmapConverter CreateBeatmapConverter ( IBeatmap beatmap , Ruleset ruleset ) = > ruleset . CreateBeatmapConverter ( beatmap ) ;
2021-11-21 11:30:45 +01:00
public IBeatmap GetPlayableBeatmap ( [ NotNull ] IRulesetInfo ruleset , IReadOnlyList < Mod > mods = null )
2018-04-19 22:04:12 +09:00
{
2021-11-20 17:23:55 +01:00
try
{
using ( var cancellationTokenSource = new CancellationTokenSource ( 10_000 ) )
{
// don't apply the default timeout when debugger is attached (may be breakpointing / debugging).
2021-11-21 11:30:45 +01:00
return GetPlayableBeatmap ( ruleset , mods ? ? Array . Empty < Mod > ( ) , Debugger . IsAttached ? new CancellationToken ( ) : cancellationTokenSource . Token ) ;
2021-11-20 17:23:55 +01:00
}
}
catch ( OperationCanceledException )
{
throw new BeatmapLoadTimeoutException ( BeatmapInfo ) ;
}
}
2019-12-12 15:58:11 +09:00
2021-11-20 17:23:55 +01:00
public virtual IBeatmap GetPlayableBeatmap ( [ NotNull ] IRulesetInfo ruleset , [ NotNull ] IReadOnlyList < Mod > mods , CancellationToken token )
{
2021-11-08 14:33:32 +09:00
var rulesetInstance = ruleset . CreateInstance ( ) ;
2018-04-19 22:04:12 +09:00
2021-11-16 14:43:13 +09:00
if ( rulesetInstance = = null )
throw new RulesetLoadException ( "Creating ruleset instance failed when attempting to create playable beatmap." ) ;
2018-04-19 22:04:12 +09:00
2021-11-17 22:00:09 +01:00
IBeatmapConverter converter = CreateBeatmapConverter ( Beatmap , rulesetInstance ) ;
2021-11-08 14:33:32 +09:00
// Check if the beatmap can be converted
if ( Beatmap . HitObjects . Count > 0 & & ! converter . CanConvert ( ) )
throw new BeatmapInvalidForRulesetException ( $"{nameof(Beatmaps.Beatmap)} can not be converted for the ruleset (ruleset: {ruleset.InstantiationInfo}, converter: {converter})." ) ;
2018-04-19 22:04:12 +09:00
2021-11-08 14:33:32 +09:00
// Apply conversion mods
foreach ( var mod in mods . OfType < IApplicableToBeatmapConverter > ( ) )
{
2021-11-20 17:23:55 +01:00
token . ThrowIfCancellationRequested ( ) ;
2021-11-08 14:33:32 +09:00
mod . ApplyToBeatmapConverter ( converter ) ;
}
2018-04-19 22:04:12 +09:00
2021-11-08 14:33:32 +09:00
// Convert
2021-11-17 10:48:33 +09:00
IBeatmap converted = converter . Convert ( token ) ;
2018-05-18 18:11:52 +09:00
2021-11-08 14:33:32 +09:00
// Apply conversion mods to the result
foreach ( var mod in mods . OfType < IApplicableAfterBeatmapConversion > ( ) )
{
2021-11-20 17:23:55 +01:00
token . ThrowIfCancellationRequested ( ) ;
2021-11-08 14:33:32 +09:00
mod . ApplyToBeatmap ( converted ) ;
}
2020-08-18 01:40:55 +09:00
2021-11-08 14:33:32 +09:00
// Apply difficulty mods
if ( mods . Any ( m = > m is IApplicableToDifficulty ) )
{
foreach ( var mod in mods . OfType < IApplicableToDifficulty > ( ) )
2020-03-13 13:52:40 +09:00
{
2021-11-20 17:23:55 +01:00
token . ThrowIfCancellationRequested ( ) ;
2021-11-08 14:33:32 +09:00
mod . ApplyToDifficulty ( converted . Difficulty ) ;
2020-03-13 13:52:40 +09:00
}
2021-11-08 14:33:32 +09:00
}
2018-06-29 12:45:48 +09:00
2021-11-08 14:33:32 +09:00
IBeatmapProcessor processor = rulesetInstance . CreateBeatmapProcessor ( converted ) ;
2018-04-19 22:04:12 +09:00
2021-11-08 14:33:32 +09:00
foreach ( var mod in mods . OfType < IApplicableToBeatmapProcessor > ( ) )
mod . ApplyToBeatmapProcessor ( processor ) ;
2021-06-23 14:08:24 +09:00
2021-11-08 14:33:32 +09:00
processor ? . PreProcess ( ) ;
2018-04-19 22:04:12 +09:00
2021-11-08 14:33:32 +09:00
// Compute default values for hitobjects, including creating nested hitobjects in-case they're needed
2021-11-20 17:23:55 +01:00
foreach ( var obj in converted . HitObjects )
2021-11-08 14:33:32 +09:00
{
2021-11-20 17:23:55 +01:00
token . ThrowIfCancellationRequested ( ) ;
obj . ApplyDefaults ( converted . ControlPointInfo , converted . Difficulty , token ) ;
2021-11-08 14:33:32 +09:00
}
2018-05-25 16:21:51 +09:00
2021-11-08 14:33:32 +09:00
foreach ( var mod in mods . OfType < IApplicableToHitObject > ( ) )
{
foreach ( var obj in converted . HitObjects )
2020-03-13 13:52:40 +09:00
{
2021-11-20 17:23:55 +01:00
token . ThrowIfCancellationRequested ( ) ;
2021-11-08 14:33:32 +09:00
mod . ApplyToHitObject ( obj ) ;
2020-03-13 13:52:40 +09:00
}
2021-11-08 14:33:32 +09:00
}
2020-03-13 13:52:40 +09:00
2021-11-08 14:33:32 +09:00
processor ? . PostProcess ( ) ;
2019-08-01 12:41:46 +09:00
2021-11-08 14:33:32 +09:00
foreach ( var mod in mods . OfType < IApplicableToBeatmap > ( ) )
{
2021-11-17 10:48:33 +09:00
token . ThrowIfCancellationRequested ( ) ;
2021-11-08 14:33:32 +09:00
mod . ApplyToBeatmap ( converted ) ;
2019-11-11 19:53:22 +08:00
}
2021-11-08 14:33:32 +09:00
return converted ;
2018-04-19 22:04:12 +09:00
}
2020-02-10 17:01:41 +09:00
private CancellationTokenSource loadCancellation = new CancellationTokenSource ( ) ;
2018-07-19 18:43:11 +09:00
2021-11-15 18:24:00 +09:00
public void BeginAsyncLoad ( ) = > loadBeatmapAsync ( ) ;
2019-06-24 17:10:50 +09:00
2020-02-10 17:01:41 +09:00
public void CancelAsyncLoad ( )
{
2021-10-14 16:22:43 +09:00
lock ( beatmapFetchLock )
{
loadCancellation ? . Cancel ( ) ;
loadCancellation = new CancellationTokenSource ( ) ;
2020-02-10 17:01:41 +09:00
2021-10-14 16:22:43 +09:00
if ( beatmapLoadTask ? . IsCompleted ! = true )
beatmapLoadTask = null ;
}
2020-02-10 17:01:41 +09:00
}
2021-10-14 15:39:08 +09:00
private readonly object beatmapFetchLock = new object ( ) ;
private Task < IBeatmap > loadBeatmapAsync ( )
2019-06-24 17:10:50 +09:00
{
2021-10-14 15:39:08 +09:00
lock ( beatmapFetchLock )
{
return beatmapLoadTask ? ? = Task . Factory . StartNew ( ( ) = >
{
// Todo: Handle cancellation during beatmap parsing
var b = GetBeatmap ( ) ? ? new Beatmap ( ) ;
2019-06-24 17:10:50 +09:00
2021-10-14 15:39:08 +09:00
// The original beatmap version needs to be preserved as the database doesn't contain it
BeatmapInfo . BeatmapVersion = b . BeatmapInfo . BeatmapVersion ;
2019-06-24 17:10:50 +09:00
2021-10-14 15:39:08 +09:00
// Use the database-backed info for more up-to-date values (beatmap id, ranked status, etc)
b . BeatmapInfo = BeatmapInfo ;
2019-06-24 17:10:50 +09:00
2021-10-14 15:39:08 +09:00
return b ;
} , loadCancellation . Token , TaskCreationOptions . LongRunning , TaskScheduler . Default ) ;
}
}
2020-02-10 17:01:41 +09:00
public override string ToString ( ) = > BeatmapInfo . ToString ( ) ;
2020-04-30 16:42:38 +09:00
public virtual bool BeatmapLoaded = > beatmapLoadTask ? . IsCompleted ? ? false ;
2019-06-24 17:10:50 +09:00
2021-11-15 18:24:00 +09:00
IBeatmapInfo IWorkingBeatmap . BeatmapInfo = > BeatmapInfo ;
2021-11-15 19:30:46 +09:00
IBeatmapMetadataInfo IWorkingBeatmap . Metadata = > Metadata ;
IBeatmapSetInfo IWorkingBeatmap . BeatmapSetInfo = > BeatmapSetInfo ;
2021-11-15 18:24:00 +09:00
2019-07-02 22:25:51 +09:00
public IBeatmap Beatmap
{
get
{
try
{
2020-02-10 17:01:41 +09:00
return loadBeatmapAsync ( ) . Result ;
2019-07-02 22:25:51 +09:00
}
2020-02-10 16:41:10 +09:00
catch ( AggregateException ae )
2019-07-02 22:25:51 +09:00
{
2020-02-10 17:25:11 +09:00
// This is the exception that is generally expected here, which occurs via natural cancellation of the asynchronous load
if ( ae . InnerExceptions . FirstOrDefault ( ) is TaskCanceledException )
return null ;
2020-02-10 16:41:10 +09:00
2020-02-10 17:25:11 +09:00
Logger . Error ( ae , "Beatmap failed to load" ) ;
return null ;
}
catch ( Exception e )
{
Logger . Error ( e , "Beatmap failed to load" ) ;
2019-07-02 22:25:51 +09:00
return null ;
}
}
}
2019-06-24 17:10:50 +09:00
2018-09-06 12:51:23 +09:00
protected abstract IBeatmap GetBeatmap ( ) ;
2019-06-24 17:10:50 +09:00
private Task < IBeatmap > beatmapLoadTask ;
2018-04-13 18:19:50 +09:00
2018-09-06 12:51:23 +09:00
public bool BackgroundLoaded = > background . IsResultAvailable ;
public Texture Background = > background . Value ;
2018-09-11 11:28:02 +09:00
protected virtual bool BackgroundStillValid ( Texture b ) = > b = = null | | b . Available ;
2018-09-06 12:51:23 +09:00
protected abstract Texture GetBackground ( ) ;
private readonly RecyclableLazy < Texture > background ;
2018-04-13 18:19:50 +09:00
2020-08-17 15:38:16 +09:00
private Track loadedTrack ;
2020-08-12 00:48:45 +09:00
[NotNull]
2020-08-17 15:38:16 +09:00
public Track LoadTrack ( ) = > loadedTrack = GetBeatmapTrack ( ) ? ? GetVirtualTrack ( 1000 ) ;
2021-02-18 14:55:44 +09:00
public void PrepareTrackForPreviewLooping ( )
{
Track . Looping = true ;
Track . RestartPoint = Metadata . PreviewTime ;
if ( Track . RestartPoint = = - 1 )
{
if ( ! Track . IsLoaded )
{
// force length to be populated (https://github.com/ppy/osu-framework/issues/4202)
Track . Seek ( Track . CurrentTime ) ;
}
Track . RestartPoint = 0.4f * Track . Length ;
}
}
2020-08-18 13:01:35 +09:00
/// <summary>
/// Transfer a valid audio track into this working beatmap. Used as an optimisation to avoid reload / track swap
/// across difficulties in the same beatmap set.
/// </summary>
/// <param name="track">The track to transfer.</param>
public void TransferTrack ( [ NotNull ] Track track ) = > loadedTrack = track ? ? throw new ArgumentNullException ( nameof ( track ) ) ;
2020-08-21 13:53:12 +09:00
/// <summary>
/// Whether this beatmap's track has been loaded via <see cref="LoadTrack"/>.
/// </summary>
2020-08-21 15:05:56 +09:00
public virtual bool TrackLoaded = > loadedTrack ! = null ;
2020-08-21 13:53:12 +09:00
2020-08-17 15:38:16 +09:00
/// <summary>
/// Get the loaded audio track instance. <see cref="LoadTrack"/> must have first been called.
/// This generally happens via MusicController when changing the global beatmap.
/// </summary>
public Track Track
{
get
{
2020-08-21 13:53:12 +09:00
if ( ! TrackLoaded )
2020-08-17 15:38:16 +09:00
throw new InvalidOperationException ( $"Cannot access {nameof(Track)} without first calling {nameof(LoadTrack)}." ) ;
return loadedTrack ;
}
}
2020-08-04 21:53:00 +09:00
2020-08-07 22:31:41 +09:00
protected abstract Track GetBeatmapTrack ( ) ;
2018-04-13 18:19:50 +09:00
public bool WaveformLoaded = > waveform . IsResultAvailable ;
2018-09-06 12:51:23 +09:00
public Waveform Waveform = > waveform . Value ;
2019-01-07 18:50:27 +09:00
protected virtual Waveform GetWaveform ( ) = > new Waveform ( null ) ;
2018-09-06 12:51:23 +09:00
private readonly RecyclableLazy < Waveform > waveform ;
2018-04-13 18:19:50 +09:00
public bool StoryboardLoaded = > storyboard . IsResultAvailable ;
2018-09-06 12:51:23 +09:00
public Storyboard Storyboard = > storyboard . Value ;
protected virtual Storyboard GetStoryboard ( ) = > new Storyboard { BeatmapInfo = BeatmapInfo } ;
private readonly RecyclableLazy < Storyboard > storyboard ;
2018-04-13 18:19:50 +09:00
public bool SkinLoaded = > skin . IsResultAvailable ;
2019-08-28 19:57:17 +09:00
public ISkin Skin = > skin . Value ;
2019-05-28 23:54:42 +09:00
2021-08-15 18:38:01 +02:00
/// <summary>
/// Creates a new skin instance for this beatmap.
/// </summary>
/// <remarks>
/// This should only be called externally in scenarios where it is explicitly desired to get a new instance of a skin
/// (e.g. for editing purposes, to avoid state pollution).
/// For standard reading purposes, <see cref="Skin"/> should always be used directly.
/// </remarks>
protected internal abstract ISkin GetSkin ( ) ;
2021-05-30 15:19:47 +09:00
2019-08-28 19:57:17 +09:00
private readonly RecyclableLazy < ISkin > skin ;
2018-04-13 18:19:50 +09:00
2021-04-17 17:47:13 +02:00
public abstract Stream GetStream ( string storagePath ) ;
2018-09-06 12:51:23 +09:00
public class RecyclableLazy < T >
2018-04-13 18:19:50 +09:00
{
2018-09-06 12:51:23 +09:00
private Lazy < T > lazy ;
2018-04-13 18:19:50 +09:00
private readonly Func < T > valueFactory ;
private readonly Func < T , bool > stillValidFunction ;
2018-09-06 13:27:53 +09:00
private readonly object fetchLock = new object ( ) ;
2018-04-13 18:19:50 +09:00
2018-09-06 12:51:23 +09:00
public RecyclableLazy ( Func < T > valueFactory , Func < T , bool > stillValidFunction = null )
2018-04-13 18:19:50 +09:00
{
this . valueFactory = valueFactory ;
this . stillValidFunction = stillValidFunction ;
recreate ( ) ;
}
public void Recycle ( )
{
if ( ! IsResultAvailable ) return ;
2018-09-06 12:51:23 +09:00
( lazy . Value as IDisposable ) ? . Dispose ( ) ;
2018-04-13 18:19:50 +09:00
recreate ( ) ;
}
2018-09-06 12:51:23 +09:00
public bool IsResultAvailable = > stillValid ;
2018-04-13 18:19:50 +09:00
2018-09-06 12:51:23 +09:00
public T Value
2018-04-13 18:19:50 +09:00
{
get
{
2018-09-06 13:27:53 +09:00
lock ( fetchLock )
{
if ( ! stillValid )
recreate ( ) ;
return lazy . Value ;
}
2018-04-13 18:19:50 +09:00
}
}
2018-09-06 12:51:23 +09:00
private bool stillValid = > lazy . IsValueCreated & & ( stillValidFunction ? . Invoke ( lazy . Value ) ? ? true ) ;
private void recreate ( ) = > lazy = new Lazy < T > ( valueFactory , LazyThreadSafetyMode . ExecutionAndPublication ) ;
2018-04-13 18:19:50 +09:00
}
2020-03-16 11:33:26 +09:00
private class BeatmapLoadTimeoutException : TimeoutException
{
public BeatmapLoadTimeoutException ( BeatmapInfo beatmapInfo )
: base ( $"Timed out while loading beatmap ({beatmapInfo})." )
{
}
}
2018-04-13 18:19:50 +09:00
}
}