From 481a16b491ce9b451fccfe7b58578c562b1b35d2 Mon Sep 17 00:00:00 2001
From: Aergwyn <aergwyn@t-online.de>
Date: Mon, 1 Jan 2018 11:55:24 +0100
Subject: [PATCH 1/7] extended hitobject tests

---
 .../Objects/Drawables/DrawableHitCircle.cs    |   2 +-
 .../Objects/Drawables/DrawableSpinner.cs      |  22 +--
 .../Tests/TestCaseHitCircle.cs                |  54 ++++----
 .../Tests/TestCaseHitCircleHidden.cs          |  26 ++++
 osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs | 130 ++++++++++--------
 .../Tests/TestCaseSliderHidden.cs             |  32 +++++
 .../Tests/TestCaseSpinner.cs                  |  50 +++++--
 .../Tests/TestCaseSpinnerHidden.cs            |  26 ++++
 .../osu.Game.Rulesets.Osu.csproj              |   3 +
 9 files changed, 231 insertions(+), 114 deletions(-)
 create mode 100644 osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
 create mode 100644 osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
 create mode 100644 osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs

diff --git a/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableHitCircle.cs b/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableHitCircle.cs
index 6220bbd120..72ca9b37a8 100644
--- a/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableHitCircle.cs
+++ b/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableHitCircle.cs
@@ -26,7 +26,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
             Origin = Anchor.Centre;
 
             Position = HitObject.StackedPosition;
-            Scale = new Vector2(HitObject.Scale);
+            Scale = new Vector2(h.Scale);
 
             Children = new Drawable[]
             {
diff --git a/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs b/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs
index 5351ad50c4..bbe6b3a0a0 100644
--- a/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs
+++ b/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs
@@ -19,7 +19,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
 {
     public class DrawableSpinner : DrawableOsuHitObject
     {
-        private readonly Spinner spinner;
+        protected readonly Spinner Spinner;
 
         public readonly SpinnerDisc Disc;
         public readonly SpinnerTicks Ticks;
@@ -50,7 +50,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
             // we are slightly bigger than our parent, to clip the top and bottom of the circle
             Height = 1.3f;
 
-            spinner = s;
+            Spinner = s;
 
             Children = new Drawable[]
             {
@@ -91,7 +91,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
                             Anchor = Anchor.Centre,
                             Origin = Anchor.Centre,
                         },
-                        Disc = new SpinnerDisc(spinner)
+                        Disc = new SpinnerDisc(Spinner)
                         {
                             Scale = Vector2.Zero,
                             Anchor = Anchor.Centre,
@@ -115,7 +115,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
             };
         }
 
-        public float Progress => MathHelper.Clamp(Disc.RotationAbsolute / 360 / spinner.SpinsRequired, 0, 1);
+        public float Progress => MathHelper.Clamp(Disc.RotationAbsolute / 360 / Spinner.SpinsRequired, 0, 1);
 
         protected override void CheckForJudgements(bool userTriggered, double timeOffset)
         {
@@ -136,7 +136,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
                 glow.FadeColour(completeColour, duration);
             }
 
-            if (!userTriggered && Time.Current >= spinner.EndTime)
+            if (!userTriggered && Time.Current >= Spinner.EndTime)
             {
                 if (Progress >= 1)
                     AddJudgement(new OsuJudgement { Result = HitResult.Great });
@@ -144,7 +144,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
                     AddJudgement(new OsuJudgement { Result = HitResult.Good });
                 else if (Progress > .75)
                     AddJudgement(new OsuJudgement { Result = HitResult.Meh });
-                else if (Time.Current >= spinner.EndTime)
+                else if (Time.Current >= Spinner.EndTime)
                     AddJudgement(new OsuJudgement { Result = HitResult.Miss });
             }
         }
@@ -180,7 +180,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
             Ticks.Rotation = Disc.Rotation;
             spmCounter.SetRotation(Disc.RotationAbsolute);
 
-            float relativeCircleScale = spinner.Scale * circle.DrawHeight / mainContainer.DrawHeight;
+            float relativeCircleScale = Spinner.Scale * circle.DrawHeight / mainContainer.DrawHeight;
             Disc.ScaleTo(relativeCircleScale + (1 - relativeCircleScale) * Progress, 200, Easing.OutQuint);
 
             symbol.RotateTo(Disc.Rotation / 2, 500, Easing.OutQuint);
@@ -190,22 +190,22 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables
         {
             base.UpdatePreemptState();
 
-            circleContainer.ScaleTo(spinner.Scale * 0.3f);
-            circleContainer.ScaleTo(spinner.Scale, TIME_PREEMPT / 1.4f, Easing.OutQuint);
+            circleContainer.ScaleTo(Spinner.Scale * 0.3f);
+            circleContainer.ScaleTo(Spinner.Scale, TIME_PREEMPT / 1.4f, Easing.OutQuint);
 
             Disc.RotateTo(-720);
             symbol.RotateTo(-720);
 
             mainContainer
                 .ScaleTo(0)
-                .ScaleTo(spinner.Scale * circle.DrawHeight / DrawHeight * 1.4f, TIME_PREEMPT - 150, Easing.OutQuint)
+                .ScaleTo(Spinner.Scale * circle.DrawHeight / DrawHeight * 1.4f, TIME_PREEMPT - 150, Easing.OutQuint)
                 .Then()
                 .ScaleTo(1, 500, Easing.OutQuint);
         }
 
         protected override void UpdateCurrentState(ArmedState state)
         {
-            var sequence = this.Delay(spinner.Duration).FadeOut(160);
+            var sequence = this.Delay(Spinner.Duration).FadeOut(160);
 
             switch (state)
             {
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
index cdce19ad21..0f03f7ed28 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
@@ -11,11 +11,13 @@ using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 using osu.Game.Tests.Visual;
 using OpenTK;
-using osu.Game.Rulesets.Osu.Mods;
 using OpenTK.Graphics;
 using osu.Game.Rulesets.Osu.Judgements;
 using System.Collections.Generic;
 using System;
+using osu.Game.Rulesets.Mods;
+using System.Linq;
+using osu.Game.Rulesets.Scoring;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
@@ -25,32 +27,34 @@ namespace osu.Game.Rulesets.Osu.Tests
         public override IReadOnlyList<Type> RequiredTypes => new[]
         {
             typeof(HitCircle),
-            typeof(OsuModHidden),
             typeof(DrawableHitCircle)
         };
 
         private readonly Container content;
         protected override Container<Drawable> Content => content;
 
-        private bool auto;
-        private bool hidden;
         private int depthIndex;
-        private int circleSize;
-        private float circleScale = 1;
+        protected readonly List<Mod> Mods = new List<Mod>();
 
         public TestCaseHitCircle()
         {
             base.Content.Add(content = new OsuInputManager(new RulesetInfo { ID = 0 }));
 
-            AddStep("Single", () => testSingle());
-            AddStep("Stream", testStream);
-            AddToggleStep("Auto", v => auto = v);
-            AddToggleStep("Hidden", v => hidden = v);
-            AddSliderStep("CircleSize", 0, 10, 0, s => circleSize = s);
-            AddSliderStep("CircleScale", 0.5f, 2, 1, s => circleScale = s);
+            AddStep("Miss Big Single", () => testSingle(2));
+            AddStep("Miss Medium Single", () => testSingle(5));
+            AddStep("Miss Small Single", () => testSingle(7));
+            AddStep("Hit Big Single", () => testSingle(2, true));
+            AddStep("Hit Medium Single", () => testSingle(5, true));
+            AddStep("Hit Small Single", () => testSingle(7, true));
+            AddStep("Miss Big Stream", () => testStream(2));
+            AddStep("Miss Medium Stream", () => testStream(5));
+            AddStep("Miss Small Stream", () => testStream(7));
+            AddStep("Hit Big Stream", () => testStream(2, true));
+            AddStep("Hit Medium Stream", () => testStream(5, true));
+            AddStep("Hit Small Stream", () => testStream(7, true));
         }
 
-        private void testSingle(double timeOffset = 0, Vector2? positionOffset = null)
+        private void testSingle(float circleSize, bool auto = false, double timeOffset = 0, Vector2? positionOffset = null)
         {
             positionOffset = positionOffset ?? Vector2.Zero;
 
@@ -66,27 +70,23 @@ namespace osu.Game.Rulesets.Osu.Tests
             var drawable = new TestDrawableHitCircle(circle, auto)
             {
                 Anchor = Anchor.Centre,
-                Scale = new Vector2(circleScale),
                 Depth = depthIndex++
             };
 
-            if (auto)
-                drawable.State.Value = ArmedState.Hit;
-
-            if (hidden)
-                new OsuModHidden().ApplyToDrawableHitObjects(new [] { drawable });
+            foreach (var mod in Mods.OfType<IApplicableToDrawableHitObjects>())
+                mod.ApplyToDrawableHitObjects(new[] { drawable });
 
             Add(drawable);
         }
 
-        private void testStream()
+        private void testStream(float circleSize, bool auto = false)
         {
-            Vector2 pos = Vector2.Zero;
+            Vector2 pos = new Vector2(-250, 0);
 
             for (int i = 0; i <= 1000; i += 100)
             {
-                testSingle(i, pos);
-                pos += new Vector2(10);
+                testSingle(circleSize, auto, i, pos);
+                pos.X += 50;
             }
         }
 
@@ -103,13 +103,15 @@ namespace osu.Game.Rulesets.Osu.Tests
             {
                 if (auto && !userTriggered && timeOffset > 0)
                 {
-                    // pretend we really hit it
+                    // force success
                     AddJudgement(new OsuJudgement
                     {
-                        Result = HitObject.ScoreResultForOffset(timeOffset)
+                        Result = HitResult.Great
                     });
+                    State.Value = ArmedState.Hit;
                 }
-                base.CheckForJudgements(userTriggered, timeOffset);
+                else
+                    base.CheckForJudgements(userTriggered, timeOffset);
             }
         }
     }
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
new file mode 100644
index 0000000000..4ba9413dde
--- /dev/null
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
@@ -0,0 +1,26 @@
+// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
+// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
+
+using System;
+using System.Collections.Generic;
+using osu.Game.Rulesets.Osu.Mods;
+using osu.Game.Rulesets.Osu.Objects;
+using osu.Game.Rulesets.Osu.Objects.Drawables;
+
+namespace osu.Game.Rulesets.Osu.Tests
+{
+    public class TestCaseHitCircleHidden : TestCaseHitCircle
+    {
+        public override IReadOnlyList<Type> RequiredTypes => new[]
+        {
+            typeof(HitCircle),
+            typeof(OsuModHidden),
+            typeof(DrawableHitCircle)
+        };
+
+        public TestCaseHitCircleHidden()
+        {
+            Mods.Add(new OsuModHidden());
+        }
+    }
+}
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
index 5b6b357351..50fb7b701c 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
@@ -13,8 +13,9 @@ using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 using osu.Game.Tests.Visual;
 using OpenTK;
-using osu.Game.Rulesets.Osu.Mods;
 using OpenTK.Graphics;
+using osu.Game.Rulesets.Mods;
+using System.Linq;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
@@ -25,7 +26,8 @@ namespace osu.Game.Rulesets.Osu.Tests
         {
             typeof(Slider),
             typeof(HitCircle),
-            typeof(OsuModHidden),
+            typeof(SliderTick),
+            typeof(RepeatPoint),
             typeof(DrawableSlider),
             typeof(DrawableHitCircle),
             typeof(DrawableSliderTick),
@@ -35,58 +37,84 @@ namespace osu.Game.Rulesets.Osu.Tests
         private readonly Container content;
         protected override Container<Drawable> Content => content;
 
-        private bool hidden;
-        private int repeats;
         private int depthIndex;
-        private int circleSize;
-        private float circleScale = 1;
-        private double speedMultiplier = 2;
-        private double sliderMultiplier = 2;
+        protected readonly List<Mod> Mods = new List<Mod>();
 
         public TestCaseSlider()
         {
             base.Content.Add(content = new OsuInputManager(new RulesetInfo { ID = 0 }));
 
-            AddStep("Single", () => testSingle());
-            AddStep("Stream", testStream);
-            AddStep("Repeated", () => testRepeated(repeats));
-            AddToggleStep("Hidden", v => hidden = v);
-            AddSliderStep("Repeats", 1, 10, 1, s => repeats = s);
-            AddSliderStep("CircleSize", 0, 10, 0, s => circleSize = s);
-            AddSliderStep("CircleScale", 0.5f, 2, 1, s => circleScale = s);
-            AddSliderStep("SpeedMultiplier", 0.1, 10, 2, s => speedMultiplier = s);
-            AddSliderStep("SliderMultiplier", 0.1, 10, 2, s => sliderMultiplier = s);
+            AddStep("Big Single", () => testSimpleBig());
+            AddStep("Medium Single", () => testSimpleMedium());
+            AddStep("Small Single", () => testSimpleSmall());
+            AddStep("Big 1 Repeat", () => testSimpleBig(1));
+            AddStep("Medium 1 Repeat", () => testSimpleMedium(1));
+            AddStep("Small 1 Repeat", () => testSimpleSmall(1));
+            AddStep("Big 2 Repeats", () => testSimpleBig(2));
+            AddStep("Medium 2 Repeats", () => testSimpleMedium(2));
+            AddStep("Small 2 Repeats", () => testSimpleSmall(2));
+
+            AddStep("Slow Slider", testSlowSpeed); // slow long sliders take ages already so no repeat steps
+            AddStep("Slow Short Slider", () => testShortSlowSpeed());
+            AddStep("Slow Short Slider 1 Repeats", () => testShortSlowSpeed(1));
+            AddStep("Slow Short Slider 2 Repeats", () => testShortSlowSpeed(2));
+
+            AddStep("Fast Slider", () => testHighSpeed());
+            AddStep("Fast Slider 1 Repeat", () => testHighSpeed(1));
+            AddStep("Fast Slider 2 Repeats", () => testHighSpeed(2));
+            AddStep("Fast Short Slider", () => testHighSpeed());
+            AddStep("Fast Short Slider 1 Repeat", () => testHighSpeed(1));
+            AddStep("Fast Short Slider 2 Repeats", () => testHighSpeed(2));
+
+            AddStep("Perfect Curve", testCurve);
+            // TODO more curve types?
         }
 
-        private void testSingle(double timeOffset = 0, Vector2? positionOffset = null)
+        private void testSimpleBig(int repeats = 0) => createSlider(2, repeats: repeats);
+
+        private void testSimpleMedium(int repeats = 0) => createSlider(5, repeats: repeats);
+
+        private void testSimpleSmall(int repeats = 0) => createSlider(7, repeats: repeats);
+
+        private void testSlowSpeed() => createSlider(speedMultiplier: 0.5);
+
+        private void testShortSlowSpeed(int repeats = 0) => createSlider(distance: 100, repeats: repeats, speedMultiplier: 0.5);
+
+        private void testHighSpeed(int repeats = 0) => createSlider(repeats: repeats, speedMultiplier: 15);
+
+        private void testShortHighSpeed(int repeats = 0) => createSlider(distance: 100, repeats: repeats, speedMultiplier: 15);
+
+        private void createSlider(float circleSize = 2, float distance = 800, int repeats = 0, double speedMultiplier = 2)
         {
-            positionOffset = positionOffset ?? Vector2.Zero;
+            repeats++; // The first run through the slider is considered a repeat
+
+            var repeatSamples = new List<List<SampleInfo>>();
+            if (repeats > 1)
+            {
+                for (int i = 0; i < repeats; i++)
+                    repeatSamples.Add(new List<SampleInfo>());
+            }
 
             var slider = new Slider
             {
-                StartTime = Time.Current + 1000 + timeOffset,
-                Position = new Vector2(-200, 0) + positionOffset.Value,
+                StartTime = Time.Current + 1000,
+                Position = new Vector2(-(distance / 2), 0),
                 ComboColour = Color4.LightSeaGreen,
                 ControlPoints = new List<Vector2>
                 {
-                    new Vector2(-200, 0) + positionOffset.Value,
-                    new Vector2(400, 0) + positionOffset.Value,
+                    new Vector2(-(distance / 2), 0),
+                    new Vector2(distance / 2, 0),
                 },
-                Distance = 400
+                Distance = distance,
+                RepeatCount = repeats,
+                RepeatSamples = repeatSamples
             };
 
-            addSlider(slider);
+            addSlider(slider, circleSize, speedMultiplier);
         }
 
-        private void testRepeated(int repeats)
+        private void testCurve()
         {
-            // The first run through the slider is considered a repeat
-            repeats++;
-
-            var repeatSamples = new List<List<SampleInfo>>();
-            for (int i = 0; i < repeats; i++)
-                repeatSamples.Add(new List<SampleInfo>());
-
             var slider = new Slider
             {
                 StartTime = Time.Current + 1000,
@@ -95,52 +123,32 @@ namespace osu.Game.Rulesets.Osu.Tests
                 ControlPoints = new List<Vector2>
                 {
                     new Vector2(-200, 0),
-                    new Vector2(400, 0),
+                    new Vector2(0, 200),
+                    new Vector2(200, 0)
                 },
-                Distance = 400,
-                RepeatCount = repeats,
-                RepeatSamples = repeatSamples
+                Distance = 600
             };
 
-            addSlider(slider);
+            addSlider(slider, 2, 3);
         }
 
-        private void testStream()
-        {
-            Vector2 pos = Vector2.Zero;
-
-            for (int i = 0; i <= 1000; i += 100)
-            {
-                testSingle(i, pos);
-                pos += new Vector2(10);
-            }
-        }
-
-        private void addSlider(Slider slider)
+        private void addSlider(Slider slider, float circleSize, double speedMultiplier)
         {
             var cpi = new ControlPointInfo();
             cpi.DifficultyPoints.Add(new DifficultyControlPoint { SpeedMultiplier = speedMultiplier });
 
-            var difficulty = new BeatmapDifficulty
-            {
-                SliderMultiplier = (float)sliderMultiplier,
-                CircleSize = circleSize
-            };
-
-            slider.ApplyDefaults(cpi, difficulty);
+            slider.ApplyDefaults(cpi, new BeatmapDifficulty { CircleSize = circleSize });
 
             var drawable = new DrawableSlider(slider)
             {
                 Anchor = Anchor.Centre,
-                Scale = new Vector2(circleScale),
                 Depth = depthIndex++
             };
 
-            if (hidden)
-                new OsuModHidden().ApplyToDrawableHitObjects(new [] { drawable });
+            foreach (var mod in Mods.OfType<IApplicableToDrawableHitObjects>())
+                mod.ApplyToDrawableHitObjects(new[] { drawable });
 
             Add(drawable);
         }
     }
-
 }
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
new file mode 100644
index 0000000000..e87f236b61
--- /dev/null
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
@@ -0,0 +1,32 @@
+// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
+// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
+
+using System;
+using System.Collections.Generic;
+using osu.Game.Rulesets.Osu.Mods;
+using osu.Game.Rulesets.Osu.Objects;
+using osu.Game.Rulesets.Osu.Objects.Drawables;
+
+namespace osu.Game.Rulesets.Osu.Tests
+{
+    public class TestCaseSliderHidden : TestCaseSlider
+    {
+        public override IReadOnlyList<Type> RequiredTypes => new[]
+        {
+            typeof(Slider),
+            typeof(HitCircle),
+            typeof(SliderTick),
+            typeof(RepeatPoint),
+            typeof(OsuModHidden),
+            typeof(DrawableSlider),
+            typeof(DrawableHitCircle),
+            typeof(DrawableSliderTick),
+            typeof(DrawableRepeatPoint)
+        };
+
+        public TestCaseSliderHidden()
+        {
+            Mods.Add(new OsuModHidden());
+        }
+    }
+}
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
index c4ee56455a..8cdb050b9d 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
@@ -3,13 +3,13 @@
 
 using System;
 using System.Collections.Generic;
+using System.Linq;
 using NUnit.Framework;
-using OpenTK;
 using osu.Framework.Graphics;
 using osu.Framework.Graphics.Containers;
 using osu.Game.Beatmaps;
 using osu.Game.Beatmaps.ControlPoints;
-using osu.Game.Rulesets.Osu.Mods;
+using osu.Game.Rulesets.Mods;
 using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 using osu.Game.Tests.Visual;
@@ -22,45 +22,65 @@ namespace osu.Game.Rulesets.Osu.Tests
         public override IReadOnlyList<Type> RequiredTypes => new[]
 {
             typeof(Spinner),
-            typeof(OsuModHidden),
             typeof(DrawableSpinner)
         };
 
         private readonly Container content;
         protected override Container<Drawable> Content => content;
 
-        private bool hidden;
         private int depthIndex;
-        private int circleSize;
-        private float circleScale = 1;
+        protected readonly List<Mod> Mods = new List<Mod>();
 
         public TestCaseSpinner()
         {
             base.Content.Add(content = new OsuInputManager(new RulesetInfo { ID = 0 }));
 
-            AddStep("Single", testSingle);
-            AddToggleStep("Hidden", v => hidden = v);
-            AddSliderStep("CircleSize", 0, 10, 0, s => circleSize = s);
-            AddSliderStep("CircleScale", 0.5f, 2, 1, s => circleScale = s);
+            AddStep("Miss Big", () => testSingle(2));
+            AddStep("Miss Medium", () => testSingle(5));
+            AddStep("Miss Small", () => testSingle(7));
+            AddStep("Hit Big", () => testSingle(2, true));
+            AddStep("Hit Medium", () => testSingle(5, true));
+            AddStep("Hit Small", () => testSingle(7, true));
         }
 
-        private void testSingle()
+        private void testSingle(float circleSize, bool auto = false)
         {
             var spinner = new Spinner { StartTime = Time.Current + 1000, EndTime = Time.Current + 4000 };
 
             spinner.ApplyDefaults(new ControlPointInfo(), new BeatmapDifficulty { CircleSize = circleSize });
 
-            var drawable = new DrawableSpinner(spinner)
+            var drawable = new TestDrawableSpinner(spinner, auto)
             {
                 Anchor = Anchor.Centre,
-                Scale = new Vector2(circleScale),
                 Depth = depthIndex++
             };
 
-            if (hidden)
-                new OsuModHidden().ApplyToDrawableHitObjects(new [] { drawable });
+            foreach (var mod in Mods.OfType<IApplicableToDrawableHitObjects>())
+                mod.ApplyToDrawableHitObjects(new[] { drawable });
 
             Add(drawable);
         }
+
+        private class TestDrawableSpinner : DrawableSpinner
+        {
+            private bool auto;
+
+            public TestDrawableSpinner(Spinner s, bool auto) : base(s)
+            {
+                this.auto = auto;
+            }
+
+            protected override void CheckForJudgements(bool userTriggered, double timeOffset)
+            {
+                if (auto && !userTriggered && Time.Current > Spinner.StartTime + Spinner.Duration / 2 && Progress < 1)
+                {
+                    // force completion only once to not break human interaction
+                    Disc.RotationAbsolute = Spinner.SpinsRequired * 360;
+                    auto = false;
+                }
+
+                base.CheckForJudgements(userTriggered, timeOffset);
+            }
+        }
     }
 }
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
new file mode 100644
index 0000000000..fab40f8dae
--- /dev/null
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
@@ -0,0 +1,26 @@
+// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
+// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
+
+using System;
+using System.Collections.Generic;
+using osu.Game.Rulesets.Osu.Mods;
+using osu.Game.Rulesets.Osu.Objects;
+using osu.Game.Rulesets.Osu.Objects.Drawables;
+
+namespace osu.Game.Rulesets.Osu.Tests
+{
+    public class TestCaseSpinnerHidden : TestCaseSpinner
+    {
+        public override IReadOnlyList<Type> RequiredTypes => new[]
+        {
+            typeof(Spinner),
+            typeof(OsuModHidden),
+            typeof(DrawableSpinner)
+        };
+
+        public TestCaseSpinnerHidden()
+        {
+            Mods.Add(new OsuModHidden());
+        }
+    }
+}
diff --git a/osu.Game.Rulesets.Osu/osu.Game.Rulesets.Osu.csproj b/osu.Game.Rulesets.Osu/osu.Game.Rulesets.Osu.csproj
index 785c3e17fb..05dec5a20d 100644
--- a/osu.Game.Rulesets.Osu/osu.Game.Rulesets.Osu.csproj
+++ b/osu.Game.Rulesets.Osu/osu.Game.Rulesets.Osu.csproj
@@ -88,9 +88,12 @@
     <Compile Include="OsuInputManager.cs" />
     <Compile Include="Replays\OsuReplayInputHandler.cs" />
     <Compile Include="Tests\TestCaseHitCircle.cs" />
+    <Compile Include="Tests\TestCaseHitCircleHidden.cs" />
     <Compile Include="Tests\TestCasePerformancePoints.cs" />
     <Compile Include="Tests\TestCaseSlider.cs" />
+    <Compile Include="Tests\TestCaseSliderHidden.cs" />
     <Compile Include="Tests\TestCaseSpinner.cs" />
+    <Compile Include="Tests\TestCaseSpinnerHidden.cs" />
     <Compile Include="UI\Cursor\CursorTrail.cs" />
     <Compile Include="UI\Cursor\GameplayCursor.cs" />
     <Compile Include="UI\OsuSettings.cs" />

From 80be40ed34ed4b7507466912d65b21a872f8eb6a Mon Sep 17 00:00:00 2001
From: Aergwyn <aergwyn@t-online.de>
Date: Mon, 1 Jan 2018 12:08:44 +0100
Subject: [PATCH 2/7] ignore new test classes

---
 osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs | 2 ++
 osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs    | 2 ++
 osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs   | 2 ++
 3 files changed, 6 insertions(+)

diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
index 4ba9413dde..01dfdd9550 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
@@ -3,12 +3,14 @@
 
 using System;
 using System.Collections.Generic;
+using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
 using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
+    [Ignore("getting CI working")]
     public class TestCaseHitCircleHidden : TestCaseHitCircle
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
index e87f236b61..be984dd743 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
@@ -3,12 +3,14 @@
 
 using System;
 using System.Collections.Generic;
+using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
 using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
+    [Ignore("getting CI working")]
     public class TestCaseSliderHidden : TestCaseSlider
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
index fab40f8dae..2ac08c2377 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
@@ -3,12 +3,14 @@
 
 using System;
 using System.Collections.Generic;
+using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
 using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
+    [Ignore("getting CI working")]
     public class TestCaseSpinnerHidden : TestCaseSpinner
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]

From 737a53d8265540d1208634d116f2af07756369bd Mon Sep 17 00:00:00 2001
From: Aergwyn <aergwyn@t-online.de>
Date: Tue, 2 Jan 2018 17:04:00 +0100
Subject: [PATCH 3/7] clean up RequiredTypes

---
 osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs |  1 -
 .../Tests/TestCaseHitCircleHidden.cs             | 10 ++--------
 osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs    | 12 +++++-------
 .../Tests/TestCaseSliderHidden.cs                | 16 ++--------------
 osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs   |  6 ++++--
 .../Tests/TestCaseSpinnerHidden.cs               | 10 ++--------
 6 files changed, 15 insertions(+), 40 deletions(-)

diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
index 0f03f7ed28..f307ff7c70 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircle.cs
@@ -26,7 +26,6 @@ namespace osu.Game.Rulesets.Osu.Tests
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]
         {
-            typeof(HitCircle),
             typeof(DrawableHitCircle)
         };
 
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
index 01dfdd9550..7cc0c343a2 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseHitCircleHidden.cs
@@ -3,22 +3,16 @@
 
 using System;
 using System.Collections.Generic;
+using System.Linq;
 using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
-using osu.Game.Rulesets.Osu.Objects;
-using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
     [Ignore("getting CI working")]
     public class TestCaseHitCircleHidden : TestCaseHitCircle
     {
-        public override IReadOnlyList<Type> RequiredTypes => new[]
-        {
-            typeof(HitCircle),
-            typeof(OsuModHidden),
-            typeof(DrawableHitCircle)
-        };
+        public override IReadOnlyList<Type> RequiredTypes => base.RequiredTypes.Concat(new[] { typeof(OsuModHidden) }).ToList();
 
         public TestCaseHitCircleHidden()
         {
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
index 50fb7b701c..baf75d5bd4 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
@@ -16,6 +16,7 @@ using OpenTK;
 using OpenTK.Graphics;
 using osu.Game.Rulesets.Mods;
 using System.Linq;
+using osu.Game.Rulesets.Osu.Objects.Drawables.Pieces;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
@@ -24,14 +25,11 @@ namespace osu.Game.Rulesets.Osu.Tests
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]
         {
-            typeof(Slider),
-            typeof(HitCircle),
-            typeof(SliderTick),
-            typeof(RepeatPoint),
+            typeof(SliderBall),
+            typeof(SliderBody),
             typeof(DrawableSlider),
-            typeof(DrawableHitCircle),
-            typeof(DrawableSliderTick),
-            typeof(DrawableRepeatPoint)
+            typeof(DrawableRepeatPoint),
+            typeof(DrawableOsuHitObject)
         };
 
         private readonly Container content;
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
index be984dd743..016909ad73 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSliderHidden.cs
@@ -3,28 +3,16 @@
 
 using System;
 using System.Collections.Generic;
+using System.Linq;
 using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
-using osu.Game.Rulesets.Osu.Objects;
-using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
     [Ignore("getting CI working")]
     public class TestCaseSliderHidden : TestCaseSlider
     {
-        public override IReadOnlyList<Type> RequiredTypes => new[]
-        {
-            typeof(Slider),
-            typeof(HitCircle),
-            typeof(SliderTick),
-            typeof(RepeatPoint),
-            typeof(OsuModHidden),
-            typeof(DrawableSlider),
-            typeof(DrawableHitCircle),
-            typeof(DrawableSliderTick),
-            typeof(DrawableRepeatPoint)
-        };
+        public override IReadOnlyList<Type> RequiredTypes => base.RequiredTypes.Concat(new[] { typeof(OsuModHidden) }).ToList();
 
         public TestCaseSliderHidden()
         {
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
index 8cdb050b9d..752574018c 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinner.cs
@@ -12,6 +12,7 @@ using osu.Game.Beatmaps.ControlPoints;
 using osu.Game.Rulesets.Mods;
 using osu.Game.Rulesets.Osu.Objects;
 using osu.Game.Rulesets.Osu.Objects.Drawables;
+using osu.Game.Rulesets.Osu.Objects.Drawables.Pieces;
 using osu.Game.Tests.Visual;
 
 namespace osu.Game.Rulesets.Osu.Tests
@@ -21,8 +22,9 @@ namespace osu.Game.Rulesets.Osu.Tests
     {
         public override IReadOnlyList<Type> RequiredTypes => new[]
 {
-            typeof(Spinner),
-            typeof(DrawableSpinner)
+            typeof(SpinnerDisc),
+            typeof(DrawableSpinner),
+            typeof(DrawableOsuHitObject)
         };
 
         private readonly Container content;
diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
index 2ac08c2377..9ef94b308f 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSpinnerHidden.cs
@@ -3,22 +3,16 @@
 
 using System;
 using System.Collections.Generic;
+using System.Linq;
 using NUnit.Framework;
 using osu.Game.Rulesets.Osu.Mods;
-using osu.Game.Rulesets.Osu.Objects;
-using osu.Game.Rulesets.Osu.Objects.Drawables;
 
 namespace osu.Game.Rulesets.Osu.Tests
 {
     [Ignore("getting CI working")]
     public class TestCaseSpinnerHidden : TestCaseSpinner
     {
-        public override IReadOnlyList<Type> RequiredTypes => new[]
-        {
-            typeof(Spinner),
-            typeof(OsuModHidden),
-            typeof(DrawableSpinner)
-        };
+        public override IReadOnlyList<Type> RequiredTypes => base.RequiredTypes.Concat(new[] { typeof(OsuModHidden) }).ToList();
 
         public TestCaseSpinnerHidden()
         {

From 9ddbed6729b99d735b9fa794614dc695bef66388 Mon Sep 17 00:00:00 2001
From: Aergwyn <aergwyn@t-online.de>
Date: Tue, 2 Jan 2018 17:10:05 +0100
Subject: [PATCH 4/7] crop slider length to not go out of bounds on small
 screens/ratios

+ use correct methods for short and fast sliders, ooops
---
 osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs | 8 ++++----
 1 file changed, 4 insertions(+), 4 deletions(-)

diff --git a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
index baf75d5bd4..1238572484 100644
--- a/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
+++ b/osu.Game.Rulesets.Osu/Tests/TestCaseSlider.cs
@@ -60,9 +60,9 @@ namespace osu.Game.Rulesets.Osu.Tests
             AddStep("Fast Slider", () => testHighSpeed());
             AddStep("Fast Slider 1 Repeat", () => testHighSpeed(1));
             AddStep("Fast Slider 2 Repeats", () => testHighSpeed(2));
-            AddStep("Fast Short Slider", () => testHighSpeed());
-            AddStep("Fast Short Slider 1 Repeat", () => testHighSpeed(1));
-            AddStep("Fast Short Slider 2 Repeats", () => testHighSpeed(2));
+            AddStep("Fast Short Slider", () => testShortHighSpeed());
+            AddStep("Fast Short Slider 1 Repeat", () => testShortHighSpeed(1));
+            AddStep("Fast Short Slider 2 Repeats", () => testShortHighSpeed(2));
 
             AddStep("Perfect Curve", testCurve);
             // TODO more curve types?
@@ -82,7 +82,7 @@ namespace osu.Game.Rulesets.Osu.Tests
 
         private void testShortHighSpeed(int repeats = 0) => createSlider(distance: 100, repeats: repeats, speedMultiplier: 15);
 
-        private void createSlider(float circleSize = 2, float distance = 800, int repeats = 0, double speedMultiplier = 2)
+        private void createSlider(float circleSize = 2, float distance = 400, int repeats = 0, double speedMultiplier = 2)
         {
             repeats++; // The first run through the slider is considered a repeat
 

From dccc134efa5980ec1488117f19f0c52ee94c099d Mon Sep 17 00:00:00 2001
From: Felix Ang <felix.ang@protonmail.com>
Date: Tue, 2 Jan 2018 17:53:29 +0100
Subject: [PATCH 5/7] Don't allow auto to fail

---
 osu.Game/Rulesets/Mods/ModAutoplay.cs | 1 +
 1 file changed, 1 insertion(+)

diff --git a/osu.Game/Rulesets/Mods/ModAutoplay.cs b/osu.Game/Rulesets/Mods/ModAutoplay.cs
index d94d4ba0db..4b8512fb60 100644
--- a/osu.Game/Rulesets/Mods/ModAutoplay.cs
+++ b/osu.Game/Rulesets/Mods/ModAutoplay.cs
@@ -29,5 +29,6 @@ namespace osu.Game.Rulesets.Mods
         public override string Description => "Watch a perfect automated play through the song";
         public override double ScoreMultiplier => 0;
         public override Type[] IncompatibleMods => new[] { typeof(ModRelax), typeof(ModSuddenDeath), typeof(ModNoFail) };
+        public override bool AllowFail => false;	
     }
 }

From 3e6f0c198cc1ec75ac5529cbc3f5b9913000c33c Mon Sep 17 00:00:00 2001
From: Felix Ang <felix.ang@protonmail.com>
Date: Tue, 2 Jan 2018 18:02:04 +0100
Subject: [PATCH 6/7] Remove tab

---
 osu.Game/Rulesets/Mods/ModAutoplay.cs | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/osu.Game/Rulesets/Mods/ModAutoplay.cs b/osu.Game/Rulesets/Mods/ModAutoplay.cs
index 4b8512fb60..5963c5cf82 100644
--- a/osu.Game/Rulesets/Mods/ModAutoplay.cs
+++ b/osu.Game/Rulesets/Mods/ModAutoplay.cs
@@ -29,6 +29,6 @@ namespace osu.Game.Rulesets.Mods
         public override string Description => "Watch a perfect automated play through the song";
         public override double ScoreMultiplier => 0;
         public override Type[] IncompatibleMods => new[] { typeof(ModRelax), typeof(ModSuddenDeath), typeof(ModNoFail) };
-        public override bool AllowFail => false;	
+        public override bool AllowFail => false;
     }
 }

From 7b018d4d43cd20d19a350f4ccde5de0d7f576ad4 Mon Sep 17 00:00:00 2001
From: Dean Herbert <pe@ppy.sh>
Date: Wed, 3 Jan 2018 12:56:08 +0900
Subject: [PATCH 7/7] Update framework

---
 osu-framework | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/osu-framework b/osu-framework
index 6134dafccb..66421b8944 160000
--- a/osu-framework
+++ b/osu-framework
@@ -1 +1 @@
-Subproject commit 6134dafccb3368dac96d837537325c04b89fb8ee
+Subproject commit 66421b894444cb9c4b792f9b93a786dcff5589dd