1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-28 06:42:54 +08:00

Rewrite object lookup to use previous entry regardless

This changes the fallback logic to always prefer the previous resolved
lifetime entry rather than fallback to the first entry ever. I think
this is more correct in all cases.

Also rewrites the inline comments to hopefully be easier to parse.
This commit is contained in:
Dean Herbert 2021-08-25 14:57:41 +09:00
parent 4a294d4de4
commit 681215e5b5

View File

@ -16,15 +16,15 @@ namespace osu.Game.Rulesets.UI
/// </summary> /// </summary>
public class GameplaySampleTriggerSource : CompositeDrawable public class GameplaySampleTriggerSource : CompositeDrawable
{ {
private readonly HitObjectContainer hitObjectContainer;
private int nextHitSoundIndex;
/// <summary> /// <summary>
/// The number of concurrent samples allowed to be played concurrently so that it feels better when spam-pressing a key. /// The number of concurrent samples allowed to be played concurrently so that it feels better when spam-pressing a key.
/// </summary> /// </summary>
private const int max_concurrent_hitsounds = OsuGameBase.SAMPLE_CONCURRENCY; private const int max_concurrent_hitsounds = OsuGameBase.SAMPLE_CONCURRENCY;
private readonly HitObjectContainer hitObjectContainer;
private int nextHitSoundIndex;
private readonly Container<SkinnableSound> hitSounds; private readonly Container<SkinnableSound> hitSounds;
[Resolved] [Resolved]
@ -44,32 +44,38 @@ namespace osu.Game.Rulesets.UI
}; };
} }
private HitObject fallbackObject; private HitObjectLifetimeEntry fallbackObject;
/// <summary> /// <summary>
/// Play the most appropriate hit sound for the current point in time. /// Play the most appropriate hit sound for the current point in time.
/// </summary> /// </summary>
public void Play() public void Play()
{ {
// The most optimal lookup case we have is when an object is alive. There are usually very few alive objects so there's no drawbacks in attempting this lookup each time.
var nextObject = hitObjectContainer.AliveObjects.FirstOrDefault(h => h.HitObject.StartTime > Time.Current)?.HitObject; var nextObject = hitObjectContainer.AliveObjects.FirstOrDefault(h => h.HitObject.StartTime > Time.Current)?.HitObject;
// In the case a next object isn't available in drawable form, we need to do a somewhat expensive traversal to get a valid sound to play.
if (nextObject == null) if (nextObject == null)
{ {
if (fallbackObject == null || fallbackObject.StartTime < Time.Current) // This lookup can be skipped if the last entry is still valid (in the future and not yet hit).
if (fallbackObject == null || fallbackObject.HitObject.StartTime < Time.Current || fallbackObject.Result.IsHit)
{ {
// in the case a next object isn't available in drawable form, we need to do a somewhat expensive traversal to get a valid sound to play. // We need to use lifetime entries to find the next object (we can't just use `hitObjectContainer.Objects` due to pooling - it may even be empty).
// note that we don't want to cache the object if it is an alive object, as once it is hit we don't want to continue playing its sound. // If required, we can make this lookup more efficient by adding support to get next-future-entry in LifetimeEntryManager.
// check whether we can use the previous computed sample. var lookup = hitObjectContainer.Entries
.Where(e => e.Result?.HasResult != true && e.HitObject.StartTime > Time.Current)
.OrderBy(e => e.HitObject.StartTime)
.FirstOrDefault();
// fallback to non-alive objects to find next off-screen object // If the lookup failed, use the previously resolved lookup (we still want to play a sound, and it is still likely the most valid result).
// TODO: make lookup more efficient? if (lookup != null)
fallbackObject = hitObjectContainer.Entries fallbackObject = lookup;
.Where(e => e.Result?.HasResult != true && e.HitObject.StartTime > Time.Current)?
.OrderBy(e => e.HitObject.StartTime) // If we still can't find anything, just play whatever we can to get a sound out.
.FirstOrDefault()?.HitObject ?? hitObjectContainer.Entries.FirstOrDefault()?.HitObject; fallbackObject ??= hitObjectContainer.Entries.FirstOrDefault();
} }
nextObject = fallbackObject; nextObject = fallbackObject?.HitObject;
} }
if (nextObject != null) if (nextObject != null)