1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-14 19:22:54 +08:00

Remove HitJudgementResolver; reimplement in DrawableHitObject.

This commit is contained in:
Dean Herbert 2016-11-26 16:51:51 +09:00
parent ef0f975960
commit 396ca9fe91
14 changed files with 115 additions and 104 deletions

View File

@ -52,7 +52,6 @@ namespace osu.Desktop.VisualTests.Tests
Origin = Anchor.Centre, Origin = Anchor.Centre,
Depth = -i, Depth = -i,
State = ArmedState.Hit, State = ArmedState.Hit,
Judgement = new JudgementInfo { Result = HitResult.Hit300 }
}; };
approachContainer.Add(d.ApproachCircle.CreateProxy()); approachContainer.Add(d.ApproachCircle.CreateProxy());

View File

@ -2,6 +2,7 @@
//Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE //Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System; using System;
using System.ComponentModel;
using osu.Framework.Graphics; using osu.Framework.Graphics;
using osu.Framework.Graphics.Transformations; using osu.Framework.Graphics.Transformations;
using osu.Game.Modes.Objects.Drawables; using osu.Game.Modes.Objects.Drawables;
@ -10,7 +11,7 @@ using OpenTK;
namespace osu.Game.Modes.Osu.Objects.Drawables namespace osu.Game.Modes.Osu.Objects.Drawables
{ {
public class DrawableHitCircle : DrawableHitObject public class DrawableHitCircle : DrawableOsuHitObject
{ {
private OsuHitObject osuObject; private OsuHitObject osuObject;
@ -39,9 +40,12 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
circle = new CirclePiece circle = new CirclePiece
{ {
Colour = osuObject.Colour, Colour = osuObject.Colour,
Hit = () => Hit(new JudgementInfo { Hit = () =>
UserTriggered = true, {
}), ((PositionalJudgementInfo)Judgement).PositionOffset = Vector2.Zero; //todo: set to correct value
UpdateJudgement(true);
return true;
},
}, },
number = new NumberPiece(), number = new NumberPiece(),
ring = new RingPiece(), ring = new RingPiece(),
@ -68,6 +72,38 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
UpdateState(State); UpdateState(State);
} }
double hit50 = 150;
double hit100 = 80;
double hit300 = 30;
protected override void CheckJudgement(bool userTriggered)
{
if (!userTriggered)
{
if (Judgement.TimeOffset > hit50)
Judgement.Result = HitResult.Miss;
return;
}
double hitOffset = Math.Abs(Judgement.TimeOffset);
if (hitOffset < hit50)
{
Judgement.Result = HitResult.Hit;
OsuJudgementInfo osuInfo = Judgement as OsuJudgementInfo;
if (hitOffset < hit300)
osuInfo.Score = OsuScoreResult.Hit300;
else if (hitOffset < hit100)
osuInfo.Score = OsuScoreResult.Hit100;
else if (hitOffset < hit50)
osuInfo.Score = OsuScoreResult.Hit50;
}
else
Judgement.Result = HitResult.Miss;
}
protected override void UpdateState(ArmedState state) protected override void UpdateState(ArmedState state)
{ {
if (!IsLoaded) return; if (!IsLoaded) return;
@ -75,7 +111,7 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
Flush(true); //move to DrawableHitObject Flush(true); //move to DrawableHitObject
ApproachCircle.Flush(true); ApproachCircle.Flush(true);
double t = osuObject.EndTime + (Judgement?.TimeOffset ?? 0); double t = osuObject.EndTime + Judgement.TimeOffset;
Alpha = 0; Alpha = 0;
@ -121,7 +157,7 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
explosion?.Expire(); explosion?.Expire();
explosion = null; explosion = null;
Schedule(() => Add(explosion = new HitExplosion(HitResult.Miss))); Schedule(() => Add(explosion = new HitExplosion((OsuJudgementInfo)Judgement)));
FadeOut(800); FadeOut(800);
break; break;
@ -134,7 +170,7 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
explode.FadeIn(flash_in); explode.FadeIn(flash_in);
Schedule(() => Add(explosion = new HitExplosion(Judgement.Result))); Schedule(() => Add(explosion = new HitExplosion((OsuJudgementInfo)Judgement)));
Delay(flash_in, true); Delay(flash_in, true);

View File

@ -1,34 +1,33 @@
//Copyright (c) 2007-2016 ppy Pty Ltd <contact@ppy.sh>. using System;
//Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.ComponentModel; using System.ComponentModel;
using System.Diagnostics;
using System.Linq; using System.Linq;
using System.Text; using System.Text;
using System.Threading.Tasks; using System.Threading.Tasks;
using osu.Game.Modes.Objects; using osu.Game.Modes.Objects;
using osu.Game.Modes.Objects.Drawables; using osu.Game.Modes.Objects.Drawables;
using OpenTK;
namespace osu.Game.Modes namespace osu.Game.Modes.Osu.Objects.Drawables
{ {
public class HitJudgementResolver public class DrawableOsuHitObject : DrawableHitObject
{ {
public virtual void CheckJudgement(DrawableHitObject h, JudgementInfo info) public DrawableOsuHitObject(OsuHitObject hitObject)
: base(hitObject)
{ {
info.Result = HitResult.Hit300; }
public override JudgementInfo CreateJudgementInfo() => new OsuJudgementInfo();
protected override void UpdateState(ArmedState state)
{
throw new NotImplementedException();
} }
} }
public class JudgementInfo public class OsuJudgementInfo : PositionalJudgementInfo
{ {
public bool UserTriggered; public OsuScoreResult Score;
public ComboResult Combo; public ComboResult Combo;
public HitResult Result;
public double TimeOffset;
public Vector2 PositionOffset;
} }
public enum ComboResult public enum ComboResult
@ -41,9 +40,8 @@ namespace osu.Game.Modes
Perfect Perfect
} }
public enum HitResult public enum OsuScoreResult
{ {
Ignore,
[Description(@"Miss")] [Description(@"Miss")]
Miss, Miss,
[Description(@"50")] [Description(@"50")]

View File

@ -5,7 +5,7 @@ using OpenTK;
namespace osu.Game.Modes.Osu.Objects.Drawables namespace osu.Game.Modes.Osu.Objects.Drawables
{ {
class DrawableSlider : DrawableHitObject class DrawableSlider : DrawableOsuHitObject
{ {
public DrawableSlider(Slider h) : base(h) public DrawableSlider(Slider h) : base(h)
{ {
@ -18,7 +18,6 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
Add(new CirclePiece Add(new CirclePiece
{ {
Colour = h.Colour, Colour = h.Colour,
Hit = () => Hit(new JudgementInfo()),
Position = h.Curve.PositionAt(i) - h.Position //non-relative? Position = h.Curve.PositionAt(i) - h.Position //non-relative?
}); });
} }

View File

@ -3,6 +3,7 @@ using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Sprites; using osu.Framework.Graphics.Sprites;
using osu.Framework.Graphics.Transformations; using osu.Framework.Graphics.Transformations;
using osu.Game.Modes.Objects.Drawables;
using OpenTK; using OpenTK;
namespace osu.Game.Modes.Osu.Objects.Drawables namespace osu.Game.Modes.Osu.Objects.Drawables
@ -12,7 +13,7 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
private SpriteText line1; private SpriteText line1;
private SpriteText line2; private SpriteText line2;
public HitExplosion(HitResult hitResult, ComboResult comboResult = ComboResult.None) public HitExplosion(OsuJudgementInfo judgement)
{ {
AutoSizeAxes = Axes.Both; AutoSizeAxes = Axes.Both;
Anchor = Anchor.Centre; Anchor = Anchor.Centre;
@ -27,13 +28,13 @@ namespace osu.Game.Modes.Osu.Objects.Drawables
{ {
Anchor = Anchor.TopCentre, Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre, Origin = Anchor.TopCentre,
Text = hitResult.GetDescription(), Text = judgement.Score.GetDescription(),
Font = @"Venera", Font = @"Venera",
TextSize = 20, TextSize = 20,
}, },
line2 = new SpriteText line2 = new SpriteText
{ {
Text = comboResult.GetDescription(), Text = judgement.Combo.GetDescription(),
Font = @"Venera", Font = @"Venera",
TextSize = 14, TextSize = 14,
} }

View File

@ -49,8 +49,7 @@ namespace osu.Game.Modes.Osu.Objects.Drawables.Pieces
protected override bool OnMouseDown(InputState state, MouseDownEventArgs args) protected override bool OnMouseDown(InputState state, MouseDownEventArgs args)
{ {
Hit?.Invoke(); return Hit?.Invoke() ?? false;
return true;
} }
} }
} }

View File

@ -1,41 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using osu.Game.Modes.Objects.Drawables;
using osu.Game.Modes.Osu.Objects.Drawables;
namespace osu.Game.Modes.Osu
{
class OsuHitJudgementResolver : HitJudgementResolver
{
double hit50 = 150;
double hit100 = 80;
double hit300 = 30;
public override void CheckJudgement(DrawableHitObject h, JudgementInfo info)
{
DrawableHitCircle circle = h as DrawableHitCircle;
if (circle != null)
{
if (!info.UserTriggered)
{
if (info.TimeOffset > hit50)
info.Result = HitResult.Miss;
return;
}
double hitOffset = Math.Abs(info.TimeOffset);
if (hitOffset < hit300)
info.Result = HitResult.Hit300;
else if (hitOffset < hit100)
info.Result = HitResult.Hit100;
else if (hitOffset < hit50)
info.Result = HitResult.Hit50;
else
info.Result = HitResult.Miss;
}
}
}
}

View File

@ -13,8 +13,6 @@ namespace osu.Game.Modes.Osu
{ {
public override ScoreOverlay CreateScoreOverlay() => new OsuScoreOverlay(); public override ScoreOverlay CreateScoreOverlay() => new OsuScoreOverlay();
public override HitJudgementResolver CreateHitJudgement() => new OsuHitJudgementResolver();
public override HitRenderer CreateHitRendererWith(List<HitObject> objects) => new OsuHitRenderer { Objects = objects }; public override HitRenderer CreateHitRendererWith(List<HitObject> objects) => new OsuHitRenderer { Objects = objects };
public override HitObjectParser CreateHitObjectParser() => new OsuHitObjectParser(); public override HitObjectParser CreateHitObjectParser() => new OsuHitObjectParser();

View File

@ -41,6 +41,7 @@
<Reference Include="System.Xml" /> <Reference Include="System.Xml" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Compile Include="Objects\Drawables\DrawableOsuHitObject.cs" />
<Compile Include="Objects\Drawables\Pieces\ApproachCircle.cs" /> <Compile Include="Objects\Drawables\Pieces\ApproachCircle.cs" />
<Compile Include="Objects\Drawables\Pieces\CirclePiece.cs" /> <Compile Include="Objects\Drawables\Pieces\CirclePiece.cs" />
<Compile Include="Objects\Drawables\DrawableSlider.cs" /> <Compile Include="Objects\Drawables\DrawableSlider.cs" />
@ -52,7 +53,6 @@
<Compile Include="Objects\Drawables\Pieces\RingPiece.cs" /> <Compile Include="Objects\Drawables\Pieces\RingPiece.cs" />
<Compile Include="Objects\Drawables\Pieces\Triangles.cs" /> <Compile Include="Objects\Drawables\Pieces\Triangles.cs" />
<Compile Include="Objects\OsuHitObjectParser.cs" /> <Compile Include="Objects\OsuHitObjectParser.cs" />
<Compile Include="OsuHitJudgementResolver.cs" />
<Compile Include="UI\OsuComboCounter.cs" /> <Compile Include="UI\OsuComboCounter.cs" />
<Compile Include="UI\OsuHitRenderer.cs" /> <Compile Include="UI\OsuHitRenderer.cs" />
<Compile Include="UI\OsuPlayfield.cs" /> <Compile Include="UI\OsuPlayfield.cs" />

View File

@ -2,23 +2,26 @@
//Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE //Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System; using System;
using System.ComponentModel;
using System.Diagnostics; using System.Diagnostics;
using osu.Framework; using osu.Framework;
using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Containers;
using OpenTK;
using Container = osu.Framework.Graphics.Containers.Container;
namespace osu.Game.Modes.Objects.Drawables namespace osu.Game.Modes.Objects.Drawables
{ {
public abstract class DrawableHitObject : Container, IStateful<ArmedState> public abstract class DrawableHitObject : Container, IStateful<ArmedState>
{ {
//todo: move to a more central implementation. this logic should not be at a drawable level. //todo: move to a more central implementation. this logic should not be at a drawable level.
public Action<DrawableHitObject> OnHit; public Action<DrawableHitObject, JudgementInfo> OnHit;
public Action<DrawableHitObject> OnMiss; public Action<DrawableHitObject, JudgementInfo> OnMiss;
public Action<DrawableHitObject, JudgementInfo> CheckJudgement;
public Container<DrawableHitObject> ChildObjects; public Container<DrawableHitObject> ChildObjects;
public JudgementInfo Judgement; protected JudgementInfo Judgement;
public abstract JudgementInfo CreateJudgementInfo();
public HitObject HitObject; public HitObject HitObject;
@ -42,47 +45,55 @@ namespace osu.Game.Modes.Objects.Drawables
} }
} }
protected override void LoadComplete()
{
base.LoadComplete();
Judgement = CreateJudgementInfo();
}
/// <summary> /// <summary>
/// Process a hit of this hitobject. Carries out judgement. /// Process a hit of this hitobject. Carries out judgement.
/// </summary> /// </summary>
/// <param name="judgement">Preliminary judgement information provided by the hit source.</param> /// <param name="judgement">Preliminary judgement information provided by the hit source.</param>
/// <returns>Whether a hit was processed.</returns> /// <returns>Whether a hit was processed.</returns>
protected bool Hit(JudgementInfo judgement) protected bool UpdateJudgement(bool userTriggered)
{ {
if (State != ArmedState.Idle) if (Judgement.Result != null)
return false; return false;
judgement.TimeOffset = Time.Current - HitObject.EndTime; Judgement.TimeOffset = Time.Current - HitObject.EndTime;
CheckJudgement?.Invoke(this, judgement); CheckJudgement(userTriggered);
if (judgement.Result == HitResult.Ignore) if (Judgement.Result == null)
return false; return false;
Judgement = judgement; switch (Judgement.Result)
switch (judgement.Result)
{ {
default: default:
State = ArmedState.Hit; State = ArmedState.Hit;
OnHit?.Invoke(this); OnHit?.Invoke(this, Judgement);
break; break;
case HitResult.Miss: case HitResult.Miss:
State = ArmedState.Miss; State = ArmedState.Miss;
OnMiss?.Invoke(this); OnMiss?.Invoke(this, Judgement);
break; break;
} }
return true; return true;
} }
protected virtual void CheckJudgement(bool userTriggered)
{
}
protected override void Update() protected override void Update()
{ {
base.Update(); base.Update();
if (Time.Current >= HitObject.EndTime && Judgement == null) UpdateJudgement(false);
Hit(new JudgementInfo());
} }
protected abstract void UpdateState(ArmedState state); protected abstract void UpdateState(ArmedState state);
@ -94,4 +105,23 @@ namespace osu.Game.Modes.Objects.Drawables
Hit, Hit,
Miss Miss
} }
public class PositionalJudgementInfo : JudgementInfo
{
public Vector2 PositionOffset;
}
public class JudgementInfo
{
public HitResult? Result;
public double TimeOffset;
}
public enum HitResult
{
[Description(@"Miss")]
Miss,
[Description(@"Hit")]
Hit,
}
} }

View File

@ -22,8 +22,6 @@ namespace osu.Game.Modes
public abstract HitObjectParser CreateHitObjectParser(); public abstract HitObjectParser CreateHitObjectParser();
public virtual HitJudgementResolver CreateHitJudgement() => new HitJudgementResolver();
public static void Register(Ruleset ruleset) => availableRulesets.TryAdd(ruleset.PlayMode, ruleset.GetType()); public static void Register(Ruleset ruleset) => availableRulesets.TryAdd(ruleset.PlayMode, ruleset.GetType());
protected abstract PlayMode PlayMode { get; } protected abstract PlayMode PlayMode { get; }

View File

@ -75,12 +75,12 @@ namespace osu.Game.Modes.UI
} }
} }
private void onMiss(DrawableHitObject obj) private void onMiss(DrawableHitObject obj, JudgementInfo judgement)
{ {
OnMiss?.Invoke(obj.HitObject); OnMiss?.Invoke(obj.HitObject);
} }
private void onHit(DrawableHitObject obj) private void onHit(DrawableHitObject obj, JudgementInfo judgement)
{ {
OnHit?.Invoke(obj.HitObject); OnHit?.Invoke(obj.HitObject);
} }

View File

@ -86,17 +86,12 @@ namespace osu.Game.Screens.Play
var scoreOverlay = ruleset.CreateScoreOverlay(); var scoreOverlay = ruleset.CreateScoreOverlay();
var hitRenderer = ruleset.CreateHitRendererWith(beatmap.HitObjects); var hitRenderer = ruleset.CreateHitRendererWith(beatmap.HitObjects);
var hitJudgement = ruleset.CreateHitJudgement();
hitRenderer.OnHit += delegate (HitObject h) { scoreOverlay.OnHit(h); }; hitRenderer.OnHit += delegate (HitObject h) { scoreOverlay.OnHit(h); };
hitRenderer.OnMiss += delegate (HitObject h) { scoreOverlay.OnMiss(h); }; hitRenderer.OnMiss += delegate (HitObject h) { scoreOverlay.OnMiss(h); };
if (Autoplay) if (Autoplay)
hitRenderer.Schedule(() => hitRenderer.DrawableObjects.ForEach(h => h.State = ArmedState.Hit)); hitRenderer.Schedule(() => hitRenderer.DrawableObjects.ForEach(h => h.State = ArmedState.Hit));
//bind DrawableHitObjects to HitJudgement
hitRenderer.Schedule(() => hitRenderer.DrawableObjects.ForEach(h => h.CheckJudgement = hitJudgement.CheckJudgement));
Children = new Drawable[] Children = new Drawable[]
{ {
new PlayerInputManager(game.Host) new PlayerInputManager(game.Host)

View File

@ -64,7 +64,6 @@
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Compile Include="Beatmaps\Drawables\BeatmapBackgroundSprite.cs" /> <Compile Include="Beatmaps\Drawables\BeatmapBackgroundSprite.cs" />
<Compile Include="Modes\HitJudgementResolver.cs" />
<Compile Include="Modes\Objects\HitObjectParser.cs" /> <Compile Include="Modes\Objects\HitObjectParser.cs" />
<Compile Include="Overlays\DragBar.cs" /> <Compile Include="Overlays\DragBar.cs" />
<Compile Include="Overlays\MusicController.cs" /> <Compile Include="Overlays\MusicController.cs" />