// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System; using System.Collections.Generic; using osu.Framework; using osu.Framework.Allocation; using osu.Framework.Audio; using osu.Framework.Audio.Sample; using osu.Game.Beatmaps.Samples; using osu.Game.Modes.Judgements; using Container = osu.Framework.Graphics.Containers.Container; using osu.Game.Modes.Objects.Types; using OpenTK.Graphics; namespace osu.Game.Modes.Objects.Drawables { public abstract class DrawableHitObject : Container, IStateful where TJudgement : Judgement { public override bool HandleInput => Interactive; public bool Interactive = true; public TJudgement Judgement; protected abstract TJudgement CreateJudgement(); protected abstract void UpdateState(ArmedState state); private ArmedState state; public ArmedState State { get { return state; } set { if (state == value) return; state = value; if (!IsLoaded) return; UpdateState(state); if (State == ArmedState.Hit) PlaySample(); } } protected SampleChannel Sample; protected virtual void PlaySample() { Sample?.Play(); } protected override void LoadComplete() { base.LoadComplete(); //we may be setting a custom judgement in test cases or what not. if (Judgement == null) Judgement = CreateJudgement(); //force application of the state that was set before we loaded. UpdateState(State); } } public abstract class DrawableHitObject : DrawableHitObject where TObject : HitObject where TJudgement : Judgement { public event Action> OnJudgement; public TObject HitObject; /// /// The colour used for various elements of this DrawableHitObject. /// public Color4 AccentColour { get; protected set; } protected DrawableHitObject(TObject hitObject) { HitObject = hitObject; } /// /// Process a hit of this hitobject. Carries out judgement. /// /// Whether a hit was processed. protected bool UpdateJudgement(bool userTriggered) { IPartialJudgement partial = Judgement as IPartialJudgement; // Never re-process non-partial hits, or partial judgements that were previously judged as misses if (Judgement.Result != HitResult.None && (partial == null || Judgement.Result == HitResult.Miss)) return false; // Update the judgement state double endTime = (HitObject as IHasEndTime)?.EndTime ?? HitObject.StartTime; Judgement.TimeOffset = Time.Current - endTime; // Update the judgement state bool hadResult = Judgement.Result != HitResult.None; CheckJudgement(userTriggered); // Don't process judgements with no result if (Judgement.Result == HitResult.None) return false; // Don't process judgements that previously had results but the results were unchanged if (hadResult && partial?.Changed != true) return false; switch (Judgement.Result) { default: State = ArmedState.Hit; break; case HitResult.Miss: State = ArmedState.Miss; break; } OnJudgement?.Invoke(this); if (partial != null) partial.Changed = false; return true; } protected virtual void CheckJudgement(bool userTriggered) { if (NestedHitObjects != null) { foreach (var d in NestedHitObjects) d.CheckJudgement(userTriggered); } } protected override void UpdateAfterChildren() { base.UpdateAfterChildren(); UpdateJudgement(false); } [BackgroundDependencyLoader] private void load(AudioManager audio) { SampleType type = HitObject.Sample?.Type ?? SampleType.None; if (type == SampleType.None) type = SampleType.Normal; SampleSet sampleSet = HitObject.Sample?.Set ?? SampleSet.Normal; Sample = audio.Sample.Get($@"Gameplay/{sampleSet.ToString().ToLower()}-hit{type.ToString().ToLower()}"); } private List> nestedHitObjects; protected IEnumerable> NestedHitObjects => nestedHitObjects; protected void AddNested(DrawableHitObject h) { if (nestedHitObjects == null) nestedHitObjects = new List>(); h.OnJudgement += d => OnJudgement?.Invoke(d); nestedHitObjects.Add(h); } } }