mirror of
https://github.com/ppy/osu.git
synced 2024-12-14 14:03:21 +08:00
Integrate PeriodTracker changes
This commit is contained in:
parent
ee2ff77b89
commit
1d999bb634
@ -5,9 +5,9 @@ using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using osu.Game.Beatmaps;
|
||||
using osu.Game.Beatmaps.Timing;
|
||||
using osu.Game.Rulesets.Judgements;
|
||||
using osu.Game.Rulesets.Objects;
|
||||
using osu.Game.Utils;
|
||||
|
||||
namespace osu.Game.Rulesets.Scoring
|
||||
{
|
||||
@ -49,33 +49,7 @@ namespace osu.Game.Rulesets.Scoring
|
||||
private double targetMinimumHealth;
|
||||
private double drainRate = 1;
|
||||
|
||||
private readonly List<(double startTime, double endTime)> nonDrainSections = new List<(double, double)>();
|
||||
private int currentNonDrainSection;
|
||||
|
||||
private bool isInNonDrainSection
|
||||
{
|
||||
get
|
||||
{
|
||||
if (nonDrainSections.Count == 0)
|
||||
return false;
|
||||
|
||||
var time = Time.Current;
|
||||
|
||||
if (time > nonDrainSections[currentNonDrainSection].endTime)
|
||||
{
|
||||
while (time > nonDrainSections[currentNonDrainSection].endTime && currentNonDrainSection < nonDrainSections.Count - 1)
|
||||
currentNonDrainSection++;
|
||||
}
|
||||
else
|
||||
{
|
||||
while (time < nonDrainSections[currentNonDrainSection].startTime && currentNonDrainSection > 0)
|
||||
currentNonDrainSection--;
|
||||
}
|
||||
|
||||
var closestSection = nonDrainSections[currentNonDrainSection];
|
||||
return time >= closestSection.startTime && time <= closestSection.endTime;
|
||||
}
|
||||
}
|
||||
private PeriodTracker noDrainPeriodTracker;
|
||||
|
||||
/// <summary>
|
||||
/// Creates a new <see cref="DrainingHealthProcessor"/>.
|
||||
@ -90,7 +64,7 @@ namespace osu.Game.Rulesets.Scoring
|
||||
{
|
||||
base.Update();
|
||||
|
||||
if (isInNonDrainSection)
|
||||
if (noDrainPeriodTracker?.IsInAny(Time.Current) == true)
|
||||
return;
|
||||
|
||||
// When jumping in and out of gameplay time within a single frame, health should only be drained for the period within the gameplay time
|
||||
@ -102,23 +76,23 @@ namespace osu.Game.Rulesets.Scoring
|
||||
|
||||
public override void ApplyBeatmap(IBeatmap beatmap)
|
||||
{
|
||||
nonDrainSections.Clear();
|
||||
|
||||
this.beatmap = beatmap;
|
||||
|
||||
if (beatmap.HitObjects.Count > 0)
|
||||
gameplayEndTime = beatmap.HitObjects[^1].GetEndTime();
|
||||
|
||||
// Ranges between the end of last hit object before a break
|
||||
// and the start of first hit object after a break should
|
||||
// not allow HP draining. (with break periods in)
|
||||
foreach (BreakPeriod b in beatmap.Breaks)
|
||||
{
|
||||
var startTime = beatmap.HitObjects.LastOrDefault(h => h.GetEndTime() < b.StartTime)?.GetEndTime() ?? double.MinValue;
|
||||
var endTime = beatmap.HitObjects.FirstOrDefault(h => h.StartTime > b.EndTime)?.StartTime ?? double.MaxValue;
|
||||
|
||||
nonDrainSections.Add((startTime, endTime));
|
||||
}
|
||||
noDrainPeriodTracker = new PeriodTracker(beatmap.Breaks.Select(breakPeriod => new Period(
|
||||
beatmap.HitObjects
|
||||
.Select(hitObject => hitObject.GetEndTime())
|
||||
.Where(endTime => endTime < breakPeriod.StartTime)
|
||||
.DefaultIfEmpty(double.MinValue)
|
||||
.Last(),
|
||||
beatmap.HitObjects
|
||||
.Select(hitObject => hitObject.StartTime)
|
||||
.Where(startTime => startTime > breakPeriod.EndTime)
|
||||
.DefaultIfEmpty(double.MaxValue)
|
||||
.First()
|
||||
)));
|
||||
|
||||
targetMinimumHealth = BeatmapDifficulty.DifficultyRange(beatmap.BeatmapInfo.BaseDifficulty.DrainRate, min_health_target, mid_health_target, max_health_target);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user