// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using osu.Framework.Bindables; using osu.Framework.MathUtils; using osu.Game.Beatmaps; using osu.Game.Rulesets.Judgements; namespace osu.Game.Rulesets.Scoring { public class HealthProcessor : JudgementProcessor { /// /// Invoked when the is in a failed state. /// Return true if the fail was permitted. /// public event Func Failed; /// /// Additional conditions on top of that cause a failing state. /// public event Func FailConditions; /// /// The current health. /// public readonly BindableDouble Health = new BindableDouble(1) { MinValue = 0, MaxValue = 1 }; /// /// Whether this ScoreProcessor has already triggered the failed state. /// public bool HasFailed { get; private set; } public HealthProcessor(IBeatmap beatmap) : base(beatmap) { } protected override void ApplyResultInternal(JudgementResult result) { result.HealthAtJudgement = Health.Value; result.FailedAtJudgement = HasFailed; if (HasFailed) return; Health.Value += HealthAdjustmentFactorFor(result) * result.Judgement.HealthIncreaseFor(result); if (!DefaultFailCondition && FailConditions?.Invoke(this, result) != true) return; if (Failed?.Invoke() != false) HasFailed = true; } protected override void RevertResultInternal(JudgementResult result) { Health.Value = result.HealthAtJudgement; // Todo: Revert HasFailed state with proper player support } /// /// An adjustment factor which is multiplied into the health increase provided by a . /// /// The for which the adjustment should apply. /// The adjustment factor. protected virtual double HealthAdjustmentFactorFor(JudgementResult result) => 1; /// /// The default conditions for failing. /// protected virtual bool DefaultFailCondition => Precision.AlmostBigger(Health.MinValue, Health.Value); protected override void Reset(bool storeResults) { base.Reset(storeResults); Health.Value = 1; HasFailed = false; } /// /// Checks if the score is in a failed state and notifies subscribers. /// /// This can only ever notify subscribers once. /// /// private void updateFailed(JudgementResult result) { if (HasFailed) return; if (!DefaultFailCondition && FailConditions?.Invoke(this, result) != true) return; if (Failed?.Invoke() != false) HasFailed = true; } } }