mirror of
https://github.com/ppy/osu.git
synced 2025-01-08 03:52:56 +08:00
Merge branch 'master' into video-offset
This commit is contained in:
commit
6d81da5419
@ -3,24 +3,17 @@
|
|||||||
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using osu.Framework.Extensions.IEnumerableExtensions;
|
using osu.Framework.Extensions.IEnumerableExtensions;
|
||||||
using osu.Framework.Graphics.Sprites;
|
|
||||||
using osu.Framework.Utils;
|
using osu.Framework.Utils;
|
||||||
using osu.Game.Beatmaps;
|
using osu.Game.Beatmaps;
|
||||||
using osu.Game.Graphics;
|
|
||||||
using osu.Game.Rulesets.Mania.Beatmaps;
|
using osu.Game.Rulesets.Mania.Beatmaps;
|
||||||
using osu.Game.Rulesets.Mania.Objects;
|
using osu.Game.Rulesets.Mania.Objects;
|
||||||
using osu.Game.Rulesets.Mods;
|
using osu.Game.Rulesets.Mods;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Mania.Mods
|
namespace osu.Game.Rulesets.Mania.Mods
|
||||||
{
|
{
|
||||||
public class ManiaModRandom : Mod, IApplicableToBeatmap
|
public class ManiaModRandom : ModRandom, IApplicableToBeatmap
|
||||||
{
|
{
|
||||||
public override string Name => "Random";
|
|
||||||
public override string Acronym => "RD";
|
|
||||||
public override ModType Type => ModType.Conversion;
|
|
||||||
public override IconUsage? Icon => OsuIcon.Dice;
|
|
||||||
public override string Description => @"Shuffle around the keys!";
|
public override string Description => @"Shuffle around the keys!";
|
||||||
public override double ScoreMultiplier => 1;
|
|
||||||
|
|
||||||
public void ApplyToBeatmap(IBeatmap beatmap)
|
public void ApplyToBeatmap(IBeatmap beatmap)
|
||||||
{
|
{
|
||||||
|
@ -4,48 +4,43 @@
|
|||||||
using System;
|
using System;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Testing;
|
|
||||||
using osu.Game.Beatmaps;
|
using osu.Game.Beatmaps;
|
||||||
using osu.Game.Beatmaps.ControlPoints;
|
using osu.Game.Beatmaps.ControlPoints;
|
||||||
using osu.Game.Rulesets.Osu.Objects;
|
using osu.Game.Rulesets.Osu.Objects;
|
||||||
using osu.Game.Rulesets.Osu.Objects.Drawables;
|
using osu.Game.Rulesets.Osu.Objects.Drawables;
|
||||||
using osu.Game.Rulesets.Scoring;
|
using osu.Game.Rulesets.Scoring;
|
||||||
using osu.Game.Skinning;
|
using osu.Game.Skinning;
|
||||||
|
using osu.Game.Tests.Visual;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Osu.Tests
|
namespace osu.Game.Rulesets.Osu.Tests
|
||||||
{
|
{
|
||||||
public class TestSceneHitCircleArea : ManualInputManagerTestScene
|
public class TestSceneHitCircleArea : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private HitCircle hitCircle;
|
private HitCircle hitCircle;
|
||||||
private DrawableHitCircle drawableHitCircle;
|
private DrawableHitCircle drawableHitCircle;
|
||||||
private DrawableHitCircle.HitReceptor hitAreaReceptor => drawableHitCircle.HitArea;
|
private DrawableHitCircle.HitReceptor hitAreaReceptor => drawableHitCircle.HitArea;
|
||||||
|
|
||||||
[SetUp]
|
[SetUp]
|
||||||
public new void SetUp()
|
public void SetUp() => Schedule(() =>
|
||||||
{
|
{
|
||||||
base.SetUp();
|
hitCircle = new HitCircle
|
||||||
|
|
||||||
Schedule(() =>
|
|
||||||
{
|
{
|
||||||
hitCircle = new HitCircle
|
Position = new Vector2(100, 100),
|
||||||
{
|
StartTime = Time.Current + 500
|
||||||
Position = new Vector2(100, 100),
|
};
|
||||||
StartTime = Time.Current + 500
|
|
||||||
};
|
|
||||||
|
|
||||||
hitCircle.ApplyDefaults(new ControlPointInfo(), new BeatmapDifficulty());
|
hitCircle.ApplyDefaults(new ControlPointInfo(), new BeatmapDifficulty());
|
||||||
|
|
||||||
Child = new SkinProvidingContainer(new DefaultSkin())
|
Child = new SkinProvidingContainer(new DefaultSkin())
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
Child = drawableHitCircle = new DrawableHitCircle(hitCircle)
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.Both,
|
Size = new Vector2(100)
|
||||||
Child = drawableHitCircle = new DrawableHitCircle(hitCircle)
|
}
|
||||||
{
|
};
|
||||||
Size = new Vector2(100)
|
});
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void TestCircleHitCentre()
|
public void TestCircleHitCentre()
|
||||||
|
@ -23,7 +23,7 @@ using osuTK.Graphics;
|
|||||||
|
|
||||||
namespace osu.Game.Rulesets.Osu.Tests
|
namespace osu.Game.Rulesets.Osu.Tests
|
||||||
{
|
{
|
||||||
public class TestSceneOsuDistanceSnapGrid : ManualInputManagerTestScene
|
public class TestSceneOsuDistanceSnapGrid : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private const double beat_length = 100;
|
private const double beat_length = 100;
|
||||||
private static readonly Vector2 grid_position = new Vector2(512, 384);
|
private static readonly Vector2 grid_position = new Vector2(512, 384);
|
||||||
|
@ -12,7 +12,7 @@ using osu.Game.Tests.Visual;
|
|||||||
|
|
||||||
namespace osu.Game.Rulesets.Osu.Tests
|
namespace osu.Game.Rulesets.Osu.Tests
|
||||||
{
|
{
|
||||||
public class TestSceneResumeOverlay : ManualInputManagerTestScene
|
public class TestSceneResumeOverlay : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -126,7 +126,7 @@ namespace osu.Game.Rulesets.Osu.Objects.Drawables.Pieces
|
|||||||
.FadeTo(tracking_alpha, 250, Easing.OutQuint);
|
.FadeTo(tracking_alpha, 250, Easing.OutQuint);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.RotateTo(currentRotation / 2, validAndTracking ? 500 : 1500, Easing.OutExpo);
|
Rotation = (float)Interpolation.Lerp(Rotation, currentRotation / 2, Math.Clamp(Math.Abs(Time.Elapsed) / 40, 0, 1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,7 @@ namespace osu.Game.Rulesets.Taiko.Tests.Mods
|
|||||||
|
|
||||||
[TestCase(false)]
|
[TestCase(false)]
|
||||||
[TestCase(true)]
|
[TestCase(true)]
|
||||||
public void TestHit(bool shouldMiss) => CreateHitObjectTest(new HitObjectTestData(new CentreHit { StartTime = 1000 }), shouldMiss);
|
public void TestHit(bool shouldMiss) => CreateHitObjectTest(new HitObjectTestData(new Hit { StartTime = 1000, Type = HitType.Centre }), shouldMiss);
|
||||||
|
|
||||||
[TestCase(false)]
|
[TestCase(false)]
|
||||||
[TestCase(true)]
|
[TestCase(true)]
|
||||||
|
@ -27,8 +27,8 @@ namespace osu.Game.Rulesets.Taiko.Tests
|
|||||||
{
|
{
|
||||||
StartTime = hitObject.StartTime,
|
StartTime = hitObject.StartTime,
|
||||||
EndTime = hitObject.GetEndTime(),
|
EndTime = hitObject.GetEndTime(),
|
||||||
IsRim = hitObject is RimHit,
|
IsRim = (hitObject as Hit)?.Type == HitType.Rim,
|
||||||
IsCentre = hitObject is CentreHit,
|
IsCentre = (hitObject as Hit)?.Type == HitType.Centre,
|
||||||
IsDrumRoll = hitObject is DrumRoll,
|
IsDrumRoll = hitObject is DrumRoll,
|
||||||
IsSwell = hitObject is Swell,
|
IsSwell = hitObject is Swell,
|
||||||
IsStrong = ((TaikoHitObject)hitObject).IsStrong
|
IsStrong = ((TaikoHitObject)hitObject).IsStrong
|
||||||
|
@ -69,7 +69,7 @@ namespace osu.Game.Rulesets.Taiko.Tests
|
|||||||
|
|
||||||
WorkingBeatmap beatmap = CreateWorkingBeatmap(new Beatmap
|
WorkingBeatmap beatmap = CreateWorkingBeatmap(new Beatmap
|
||||||
{
|
{
|
||||||
HitObjects = new List<HitObject> { new CentreHit() },
|
HitObjects = new List<HitObject> { new Hit { Type = HitType.Centre } },
|
||||||
BeatmapInfo = new BeatmapInfo
|
BeatmapInfo = new BeatmapInfo
|
||||||
{
|
{
|
||||||
BaseDifficulty = new BeatmapDifficulty(),
|
BaseDifficulty = new BeatmapDifficulty(),
|
||||||
|
@ -124,24 +124,13 @@ namespace osu.Game.Rulesets.Taiko.Beatmaps
|
|||||||
bool isRim = currentSamples.Any(s => s.Name == HitSampleInfo.HIT_CLAP || s.Name == HitSampleInfo.HIT_WHISTLE);
|
bool isRim = currentSamples.Any(s => s.Name == HitSampleInfo.HIT_CLAP || s.Name == HitSampleInfo.HIT_WHISTLE);
|
||||||
strong = currentSamples.Any(s => s.Name == HitSampleInfo.HIT_FINISH);
|
strong = currentSamples.Any(s => s.Name == HitSampleInfo.HIT_FINISH);
|
||||||
|
|
||||||
if (isRim)
|
yield return new Hit
|
||||||
{
|
{
|
||||||
yield return new RimHit
|
StartTime = j,
|
||||||
{
|
Type = isRim ? HitType.Rim : HitType.Centre,
|
||||||
StartTime = j,
|
Samples = currentSamples,
|
||||||
Samples = currentSamples,
|
IsStrong = strong
|
||||||
IsStrong = strong
|
};
|
||||||
};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
yield return new CentreHit
|
|
||||||
{
|
|
||||||
StartTime = j,
|
|
||||||
Samples = currentSamples,
|
|
||||||
IsStrong = strong
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
i = (i + 1) % allSamples.Count;
|
i = (i + 1) % allSamples.Count;
|
||||||
}
|
}
|
||||||
@ -180,24 +169,13 @@ namespace osu.Game.Rulesets.Taiko.Beatmaps
|
|||||||
{
|
{
|
||||||
bool isRim = samples.Any(s => s.Name == HitSampleInfo.HIT_CLAP || s.Name == HitSampleInfo.HIT_WHISTLE);
|
bool isRim = samples.Any(s => s.Name == HitSampleInfo.HIT_CLAP || s.Name == HitSampleInfo.HIT_WHISTLE);
|
||||||
|
|
||||||
if (isRim)
|
yield return new Hit
|
||||||
{
|
{
|
||||||
yield return new RimHit
|
StartTime = obj.StartTime,
|
||||||
{
|
Type = isRim ? HitType.Rim : HitType.Centre,
|
||||||
StartTime = obj.StartTime,
|
Samples = obj.Samples,
|
||||||
Samples = obj.Samples,
|
IsStrong = strong
|
||||||
IsStrong = strong
|
};
|
||||||
};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
yield return new CentreHit
|
|
||||||
{
|
|
||||||
StartTime = obj.StartTime,
|
|
||||||
Samples = obj.Samples,
|
|
||||||
IsStrong = strong
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ namespace osu.Game.Rulesets.Taiko.Difficulty.Preprocessing
|
|||||||
public TaikoDifficultyHitObject(HitObject hitObject, HitObject lastObject, double clockRate)
|
public TaikoDifficultyHitObject(HitObject hitObject, HitObject lastObject, double clockRate)
|
||||||
: base(hitObject, lastObject, clockRate)
|
: base(hitObject, lastObject, clockRate)
|
||||||
{
|
{
|
||||||
HasTypeChange = lastObject is RimHit != hitObject is RimHit;
|
HasTypeChange = (lastObject as Hit)?.Type != (hitObject as Hit)?.Type;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
27
osu.Game.Rulesets.Taiko/Mods/TaikoModRandom.cs
Normal file
27
osu.Game.Rulesets.Taiko/Mods/TaikoModRandom.cs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using osu.Framework.Utils;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Rulesets.Mods;
|
||||||
|
using osu.Game.Rulesets.Taiko.Beatmaps;
|
||||||
|
using osu.Game.Rulesets.Taiko.Objects;
|
||||||
|
|
||||||
|
namespace osu.Game.Rulesets.Taiko.Mods
|
||||||
|
{
|
||||||
|
public class TaikoModRandom : ModRandom, IApplicableToBeatmap
|
||||||
|
{
|
||||||
|
public override string Description => @"Shuffle around the colours!";
|
||||||
|
|
||||||
|
public void ApplyToBeatmap(IBeatmap beatmap)
|
||||||
|
{
|
||||||
|
var taikoBeatmap = (TaikoBeatmap)beatmap;
|
||||||
|
|
||||||
|
foreach (var obj in taikoBeatmap.HitObjects)
|
||||||
|
{
|
||||||
|
if (obj is Hit hit)
|
||||||
|
hit.Type = RNG.Next(2) == 0 ? HitType.Centre : HitType.Rim;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,9 +0,0 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Taiko.Objects
|
|
||||||
{
|
|
||||||
public class CentreHit : Hit
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@ -5,5 +5,9 @@ namespace osu.Game.Rulesets.Taiko.Objects
|
|||||||
{
|
{
|
||||||
public class Hit : TaikoHitObject
|
public class Hit : TaikoHitObject
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The <see cref="HitType"/> that actuates this <see cref="Hit"/>.
|
||||||
|
/// </summary>
|
||||||
|
public HitType Type { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
21
osu.Game.Rulesets.Taiko/Objects/HitType.cs
Normal file
21
osu.Game.Rulesets.Taiko/Objects/HitType.cs
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
namespace osu.Game.Rulesets.Taiko.Objects
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The type of a <see cref="Hit"/>.
|
||||||
|
/// </summary>
|
||||||
|
public enum HitType
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// A <see cref="Hit"/> that can be hit by the centre portion of the drum.
|
||||||
|
/// </summary>
|
||||||
|
Centre,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// A <see cref="Hit"/> that can be hit by the rim portion of the drum.
|
||||||
|
/// </summary>
|
||||||
|
Rim
|
||||||
|
}
|
||||||
|
}
|
@ -1,9 +0,0 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Taiko.Objects
|
|
||||||
{
|
|
||||||
public class RimHit : Hit
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@ -97,7 +97,7 @@ namespace osu.Game.Rulesets.Taiko.Replays
|
|||||||
{
|
{
|
||||||
TaikoAction[] actions;
|
TaikoAction[] actions;
|
||||||
|
|
||||||
if (hit is CentreHit)
|
if (hit.Type == HitType.Centre)
|
||||||
{
|
{
|
||||||
actions = h.IsStrong
|
actions = h.IsStrong
|
||||||
? new[] { TaikoAction.LeftCentre, TaikoAction.RightCentre }
|
? new[] { TaikoAction.LeftCentre, TaikoAction.RightCentre }
|
||||||
|
@ -114,6 +114,7 @@ namespace osu.Game.Rulesets.Taiko
|
|||||||
case ModType.Conversion:
|
case ModType.Conversion:
|
||||||
return new Mod[]
|
return new Mod[]
|
||||||
{
|
{
|
||||||
|
new TaikoModRandom(),
|
||||||
new TaikoModDifficultyAdjust(),
|
new TaikoModDifficultyAdjust(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -48,11 +48,11 @@ namespace osu.Game.Rulesets.Taiko.UI
|
|||||||
{
|
{
|
||||||
switch (h)
|
switch (h)
|
||||||
{
|
{
|
||||||
case CentreHit centreHit:
|
case Hit hit:
|
||||||
return new DrawableCentreHit(centreHit);
|
if (hit.Type == HitType.Centre)
|
||||||
|
return new DrawableCentreHit(hit);
|
||||||
case RimHit rimHit:
|
else
|
||||||
return new DrawableRimHit(rimHit);
|
return new DrawableRimHit(hit);
|
||||||
|
|
||||||
case DrumRoll drumRoll:
|
case DrumRoll drumRoll:
|
||||||
return new DrawableDrumRoll(drumRoll);
|
return new DrawableDrumRoll(drumRoll);
|
||||||
|
@ -14,9 +14,9 @@ using osu.Game.Rulesets.Objects.Drawables;
|
|||||||
using osu.Game.Rulesets.Judgements;
|
using osu.Game.Rulesets.Judgements;
|
||||||
using osu.Game.Rulesets.UI;
|
using osu.Game.Rulesets.UI;
|
||||||
using osu.Game.Rulesets.UI.Scrolling;
|
using osu.Game.Rulesets.UI.Scrolling;
|
||||||
using osu.Game.Rulesets.Taiko.Objects;
|
|
||||||
using osu.Game.Rulesets.Taiko.Objects.Drawables;
|
using osu.Game.Rulesets.Taiko.Objects.Drawables;
|
||||||
using osu.Game.Rulesets.Taiko.Judgements;
|
using osu.Game.Rulesets.Taiko.Judgements;
|
||||||
|
using osu.Game.Rulesets.Taiko.Objects;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
using osuTK.Graphics;
|
using osuTK.Graphics;
|
||||||
|
|
||||||
@ -245,7 +245,7 @@ namespace osu.Game.Rulesets.Taiko.UI
|
|||||||
if (!result.IsHit)
|
if (!result.IsHit)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
bool isRim = judgedObject.HitObject is RimHit;
|
bool isRim = (judgedObject.HitObject as Hit)?.Type == HitType.Rim;
|
||||||
|
|
||||||
hitExplosionContainer.Add(new HitExplosion(judgedObject, isRim));
|
hitExplosionContainer.Add(new HitExplosion(judgedObject, isRim));
|
||||||
|
|
||||||
|
@ -31,29 +31,29 @@ namespace osu.Game.Tests.Beatmaps.Formats
|
|||||||
StoryboardLayer background = storyboard.Layers.FirstOrDefault(l => l.Depth == 3);
|
StoryboardLayer background = storyboard.Layers.FirstOrDefault(l => l.Depth == 3);
|
||||||
Assert.IsNotNull(background);
|
Assert.IsNotNull(background);
|
||||||
Assert.AreEqual(16, background.Elements.Count);
|
Assert.AreEqual(16, background.Elements.Count);
|
||||||
Assert.IsTrue(background.EnabledWhenFailing);
|
Assert.IsTrue(background.VisibleWhenFailing);
|
||||||
Assert.IsTrue(background.EnabledWhenPassing);
|
Assert.IsTrue(background.VisibleWhenPassing);
|
||||||
Assert.AreEqual("Background", background.Name);
|
Assert.AreEqual("Background", background.Name);
|
||||||
|
|
||||||
StoryboardLayer fail = storyboard.Layers.FirstOrDefault(l => l.Depth == 2);
|
StoryboardLayer fail = storyboard.Layers.FirstOrDefault(l => l.Depth == 2);
|
||||||
Assert.IsNotNull(fail);
|
Assert.IsNotNull(fail);
|
||||||
Assert.AreEqual(0, fail.Elements.Count);
|
Assert.AreEqual(0, fail.Elements.Count);
|
||||||
Assert.IsTrue(fail.EnabledWhenFailing);
|
Assert.IsTrue(fail.VisibleWhenFailing);
|
||||||
Assert.IsFalse(fail.EnabledWhenPassing);
|
Assert.IsFalse(fail.VisibleWhenPassing);
|
||||||
Assert.AreEqual("Fail", fail.Name);
|
Assert.AreEqual("Fail", fail.Name);
|
||||||
|
|
||||||
StoryboardLayer pass = storyboard.Layers.FirstOrDefault(l => l.Depth == 1);
|
StoryboardLayer pass = storyboard.Layers.FirstOrDefault(l => l.Depth == 1);
|
||||||
Assert.IsNotNull(pass);
|
Assert.IsNotNull(pass);
|
||||||
Assert.AreEqual(0, pass.Elements.Count);
|
Assert.AreEqual(0, pass.Elements.Count);
|
||||||
Assert.IsFalse(pass.EnabledWhenFailing);
|
Assert.IsFalse(pass.VisibleWhenFailing);
|
||||||
Assert.IsTrue(pass.EnabledWhenPassing);
|
Assert.IsTrue(pass.VisibleWhenPassing);
|
||||||
Assert.AreEqual("Pass", pass.Name);
|
Assert.AreEqual("Pass", pass.Name);
|
||||||
|
|
||||||
StoryboardLayer foreground = storyboard.Layers.FirstOrDefault(l => l.Depth == 0);
|
StoryboardLayer foreground = storyboard.Layers.FirstOrDefault(l => l.Depth == 0);
|
||||||
Assert.IsNotNull(foreground);
|
Assert.IsNotNull(foreground);
|
||||||
Assert.AreEqual(151, foreground.Elements.Count);
|
Assert.AreEqual(151, foreground.Elements.Count);
|
||||||
Assert.IsTrue(foreground.EnabledWhenFailing);
|
Assert.IsTrue(foreground.VisibleWhenFailing);
|
||||||
Assert.IsTrue(foreground.EnabledWhenPassing);
|
Assert.IsTrue(foreground.VisibleWhenPassing);
|
||||||
Assert.AreEqual("Foreground", foreground.Name);
|
Assert.AreEqual("Foreground", foreground.Name);
|
||||||
|
|
||||||
int spriteCount = background.Elements.Count(x => x.GetType() == typeof(StoryboardSprite));
|
int spriteCount = background.Elements.Count(x => x.GetType() == typeof(StoryboardSprite));
|
||||||
|
@ -37,7 +37,7 @@ using osuTK.Graphics;
|
|||||||
namespace osu.Game.Tests.Visual.Background
|
namespace osu.Game.Tests.Visual.Background
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneUserDimBackgrounds : ManualInputManagerTestScene
|
public class TestSceneUserDimBackgrounds : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
@ -278,6 +278,7 @@ namespace osu.Game.Tests.Visual.Background
|
|||||||
|
|
||||||
private void setupUserSettings()
|
private void setupUserSettings()
|
||||||
{
|
{
|
||||||
|
AddUntilStep("Song select is current", () => songSelect.IsCurrentScreen());
|
||||||
AddUntilStep("Song select has selection", () => songSelect.Carousel?.SelectedBeatmap != null);
|
AddUntilStep("Song select has selection", () => songSelect.Carousel?.SelectedBeatmap != null);
|
||||||
AddStep("Set default user settings", () =>
|
AddStep("Set default user settings", () =>
|
||||||
{
|
{
|
||||||
|
@ -12,7 +12,7 @@ using osuTK.Graphics;
|
|||||||
namespace osu.Game.Tests.Visual.Components
|
namespace osu.Game.Tests.Visual.Components
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneIdleTracker : ManualInputManagerTestScene
|
public class TestSceneIdleTracker : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private IdleTrackingBox box1;
|
private IdleTrackingBox box1;
|
||||||
private IdleTrackingBox box2;
|
private IdleTrackingBox box2;
|
||||||
|
@ -3,27 +3,83 @@
|
|||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using osu.Framework.Allocation;
|
using System.Linq;
|
||||||
|
using NUnit.Framework;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Graphics.Shapes;
|
||||||
|
using osu.Framework.Graphics.UserInterface;
|
||||||
|
using osu.Framework.Testing;
|
||||||
using osu.Game.Screens.Edit;
|
using osu.Game.Screens.Edit;
|
||||||
using osu.Game.Screens.Edit.Compose.Components;
|
using osu.Game.Screens.Edit.Compose.Components;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
using osuTK.Input;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Editor
|
namespace osu.Game.Tests.Visual.Editor
|
||||||
{
|
{
|
||||||
public class TestSceneBeatDivisorControl : OsuTestScene
|
public class TestSceneBeatDivisorControl : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[] { typeof(BindableBeatDivisor) };
|
public override IReadOnlyList<Type> RequiredTypes => new[] { typeof(BindableBeatDivisor) };
|
||||||
|
private BeatDivisorControl beatDivisorControl;
|
||||||
|
private BindableBeatDivisor bindableBeatDivisor;
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
private SliderBar<int> tickSliderBar;
|
||||||
private void load()
|
private EquilateralTriangle tickMarkerHead;
|
||||||
|
|
||||||
|
[SetUp]
|
||||||
|
public void SetUp() => Schedule(() =>
|
||||||
{
|
{
|
||||||
Child = new BeatDivisorControl(new BindableBeatDivisor())
|
Child = beatDivisorControl = new BeatDivisorControl(bindableBeatDivisor = new BindableBeatDivisor(16))
|
||||||
{
|
{
|
||||||
Anchor = Anchor.Centre,
|
Anchor = Anchor.Centre,
|
||||||
Origin = Anchor.Centre,
|
Origin = Anchor.Centre,
|
||||||
Size = new Vector2(90, 90)
|
Size = new Vector2(90, 90)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
tickSliderBar = beatDivisorControl.ChildrenOfType<SliderBar<int>>().Single();
|
||||||
|
tickMarkerHead = tickSliderBar.ChildrenOfType<EquilateralTriangle>().Single();
|
||||||
|
});
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestBindableBeatDivisor()
|
||||||
|
{
|
||||||
|
AddRepeatStep("move previous", () => bindableBeatDivisor.Previous(), 4);
|
||||||
|
AddAssert("divisor is 4", () => bindableBeatDivisor.Value == 4);
|
||||||
|
AddRepeatStep("move next", () => bindableBeatDivisor.Next(), 3);
|
||||||
|
AddAssert("divisor is 12", () => bindableBeatDivisor.Value == 12);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestMouseInput()
|
||||||
|
{
|
||||||
|
AddStep("hold marker", () =>
|
||||||
|
{
|
||||||
|
InputManager.MoveMouseTo(tickMarkerHead.ScreenSpaceDrawQuad.Centre);
|
||||||
|
InputManager.PressButton(MouseButton.Left);
|
||||||
|
});
|
||||||
|
AddStep("move to 8 and release", () =>
|
||||||
|
{
|
||||||
|
InputManager.MoveMouseTo(tickSliderBar.ScreenSpaceDrawQuad.Centre);
|
||||||
|
InputManager.ReleaseButton(MouseButton.Left);
|
||||||
|
});
|
||||||
|
AddAssert("divisor is 8", () => bindableBeatDivisor.Value == 8);
|
||||||
|
AddStep("hold marker", () => InputManager.PressButton(MouseButton.Left));
|
||||||
|
AddStep("move to 16", () => InputManager.MoveMouseTo(getPositionForDivisor(16)));
|
||||||
|
AddStep("move to ~10 and release", () =>
|
||||||
|
{
|
||||||
|
InputManager.MoveMouseTo(getPositionForDivisor(10));
|
||||||
|
InputManager.ReleaseButton(MouseButton.Left);
|
||||||
|
});
|
||||||
|
AddAssert("divisor clamped to 8", () => bindableBeatDivisor.Value == 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Vector2 getPositionForDivisor(int divisor)
|
||||||
|
{
|
||||||
|
var relativePosition = (float)Math.Clamp(divisor, 0, 16) / 16;
|
||||||
|
var sliderDrawQuad = tickSliderBar.ScreenSpaceDrawQuad;
|
||||||
|
return new Vector2(
|
||||||
|
sliderDrawQuad.TopLeft.X + sliderDrawQuad.Width * relativePosition,
|
||||||
|
sliderDrawQuad.Centre.Y
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,7 +17,7 @@ using osuTK.Graphics;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Editor
|
namespace osu.Game.Tests.Visual.Editor
|
||||||
{
|
{
|
||||||
public class TestSceneZoomableScrollContainer : ManualInputManagerTestScene
|
public class TestSceneZoomableScrollContainer : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private ZoomableScrollContainer scrollContainer;
|
private ZoomableScrollContainer scrollContainer;
|
||||||
private Drawable innerBox;
|
private Drawable innerBox;
|
||||||
|
@ -18,7 +18,7 @@ using osuTK.Input;
|
|||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
[Description("player pause/fail screens")]
|
[Description("player pause/fail screens")]
|
||||||
public class TestSceneGameplayMenuOverlay : ManualInputManagerTestScene
|
public class TestSceneGameplayMenuOverlay : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[] { typeof(FailOverlay), typeof(PauseOverlay) };
|
public override IReadOnlyList<Type> RequiredTypes => new[] { typeof(FailOverlay), typeof(PauseOverlay) };
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
public class TestSceneHUDOverlay : ManualInputManagerTestScene
|
public class TestSceneHUDOverlay : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private HUDOverlay hudOverlay;
|
private HUDOverlay hudOverlay;
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ using osuTK.Input;
|
|||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
[Description("'Hold to Quit' UI element")]
|
[Description("'Hold to Quit' UI element")]
|
||||||
public class TestSceneHoldForMenuButton : ManualInputManagerTestScene
|
public class TestSceneHoldForMenuButton : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private bool exitAction;
|
private bool exitAction;
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ using osuTK.Input;
|
|||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneKeyCounter : ManualInputManagerTestScene
|
public class TestSceneKeyCounter : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -29,7 +29,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
public class TestScenePlayerLoader : ManualInputManagerTestScene
|
public class TestScenePlayerLoader : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private TestPlayerLoader loader;
|
private TestPlayerLoader loader;
|
||||||
private TestPlayerLoaderContainer container;
|
private TestPlayerLoaderContainer container;
|
||||||
|
@ -14,7 +14,7 @@ using osuTK.Input;
|
|||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneSkipOverlay : ManualInputManagerTestScene
|
public class TestSceneSkipOverlay : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private SkipOverlay skip;
|
private SkipOverlay skip;
|
||||||
private int requestCount;
|
private int requestCount;
|
||||||
|
@ -20,7 +20,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Multiplayer
|
namespace osu.Game.Tests.Visual.Multiplayer
|
||||||
{
|
{
|
||||||
public class TestSceneDrawableRoomPlaylist : ManualInputManagerTestScene
|
public class TestSceneDrawableRoomPlaylist : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -27,7 +27,7 @@ namespace osu.Game.Tests.Visual.Navigation
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// A scene which tests full game flow.
|
/// A scene which tests full game flow.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public abstract class OsuGameTestScene : ManualInputManagerTestScene
|
public abstract class OsuGameTestScene : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private GameHost host;
|
private GameHost host;
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Online
|
namespace osu.Game.Tests.Visual.Online
|
||||||
{
|
{
|
||||||
public class TestSceneChatOverlay : ManualInputManagerTestScene
|
public class TestSceneChatOverlay : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
87
osu.Game.Tests/Visual/Online/TestSceneFriendDisplay.cs
Normal file
87
osu.Game.Tests/Visual/Online/TestSceneFriendDisplay.cs
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Game.Overlays.Dashboard.Friends;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Game.Users;
|
||||||
|
using osu.Game.Overlays;
|
||||||
|
using osu.Framework.Allocation;
|
||||||
|
using NUnit.Framework;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Visual.Online
|
||||||
|
{
|
||||||
|
public class TestSceneFriendDisplay : OsuTestScene
|
||||||
|
{
|
||||||
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
|
{
|
||||||
|
typeof(FriendDisplay),
|
||||||
|
typeof(FriendOnlineStreamControl),
|
||||||
|
typeof(UserListToolbar)
|
||||||
|
};
|
||||||
|
|
||||||
|
protected override bool UseOnlineAPI => true;
|
||||||
|
|
||||||
|
[Cached]
|
||||||
|
private readonly OverlayColourProvider colourProvider = new OverlayColourProvider(OverlayColourScheme.Purple);
|
||||||
|
|
||||||
|
private FriendDisplay display;
|
||||||
|
|
||||||
|
[SetUp]
|
||||||
|
public void Setup() => Schedule(() =>
|
||||||
|
{
|
||||||
|
Child = new BasicScrollContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
Child = display = new FriendDisplay()
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOffline()
|
||||||
|
{
|
||||||
|
AddStep("Populate", () => display.Users = getUsers());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOnline()
|
||||||
|
{
|
||||||
|
AddStep("Fetch online", () => display?.Fetch());
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<User> getUsers() => new List<User>
|
||||||
|
{
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
Username = "flyte",
|
||||||
|
Id = 3103765,
|
||||||
|
IsOnline = true,
|
||||||
|
CurrentModeRank = 1111,
|
||||||
|
Country = new Country { FlagName = "JP" },
|
||||||
|
CoverUrl = "https://osu.ppy.sh/images/headers/profile-covers/c6.jpg"
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
Username = "peppy",
|
||||||
|
Id = 2,
|
||||||
|
IsOnline = false,
|
||||||
|
CurrentModeRank = 2222,
|
||||||
|
Country = new Country { FlagName = "AU" },
|
||||||
|
CoverUrl = "https://osu.ppy.sh/images/headers/profile-covers/c3.jpg",
|
||||||
|
IsSupporter = true,
|
||||||
|
SupportLevel = 3,
|
||||||
|
},
|
||||||
|
new User
|
||||||
|
{
|
||||||
|
Username = "Evast",
|
||||||
|
Id = 8195163,
|
||||||
|
Country = new Country { FlagName = "BY" },
|
||||||
|
CoverUrl = "https://assets.ppy.sh/user-profile-covers/8195163/4a8e2ad5a02a2642b631438cfa6c6bd7e2f9db289be881cb27df18331f64144c.jpeg",
|
||||||
|
IsOnline = false,
|
||||||
|
LastVisit = DateTimeOffset.Now
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -227,6 +227,34 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
waitForSelection(set_count);
|
waitForSelection(set_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestSelectionEnteringFromEmptyRuleset()
|
||||||
|
{
|
||||||
|
var sets = new List<BeatmapSetInfo>();
|
||||||
|
|
||||||
|
AddStep("Create beatmaps for taiko only", () =>
|
||||||
|
{
|
||||||
|
sets.Clear();
|
||||||
|
|
||||||
|
var rulesetBeatmapSet = createTestBeatmapSet(1);
|
||||||
|
var taikoRuleset = rulesets.AvailableRulesets.ElementAt(1);
|
||||||
|
rulesetBeatmapSet.Beatmaps.ForEach(b =>
|
||||||
|
{
|
||||||
|
b.Ruleset = taikoRuleset;
|
||||||
|
b.RulesetID = 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
sets.Add(rulesetBeatmapSet);
|
||||||
|
});
|
||||||
|
|
||||||
|
loadBeatmaps(sets, () => new FilterCriteria { Ruleset = rulesets.AvailableRulesets.ElementAt(0) });
|
||||||
|
|
||||||
|
AddStep("Set non-empty mode filter", () =>
|
||||||
|
carousel.Filter(new FilterCriteria { Ruleset = rulesets.AvailableRulesets.ElementAt(1) }, false));
|
||||||
|
|
||||||
|
AddAssert("Something is selected", () => carousel.SelectedBeatmap != null);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Test sorting
|
/// Test sorting
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -399,27 +427,32 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
AddStep("filter to ruleset 0", () =>
|
AddStep("filter to ruleset 0", () =>
|
||||||
carousel.Filter(new FilterCriteria { Ruleset = rulesets.AvailableRulesets.ElementAt(0) }, false));
|
carousel.Filter(new FilterCriteria { Ruleset = rulesets.AvailableRulesets.ElementAt(0) }, false));
|
||||||
AddStep("select filtered map skipping filtered", () => carousel.SelectBeatmap(testMixed.Beatmaps[1], false));
|
AddStep("select filtered map skipping filtered", () => carousel.SelectBeatmap(testMixed.Beatmaps[1], false));
|
||||||
AddAssert("unfiltered beatmap not selected", () => carousel.SelectedBeatmap == null);
|
AddAssert("unfiltered beatmap not selected", () => carousel.SelectedBeatmap.RulesetID == 0);
|
||||||
|
|
||||||
AddStep("remove mixed set", () =>
|
AddStep("remove mixed set", () =>
|
||||||
{
|
{
|
||||||
carousel.RemoveBeatmapSet(testMixed);
|
carousel.RemoveBeatmapSet(testMixed);
|
||||||
testMixed = null;
|
testMixed = null;
|
||||||
});
|
});
|
||||||
var testSingle = createTestBeatmapSet(set_count + 2);
|
BeatmapSetInfo testSingle = null;
|
||||||
testSingle.Beatmaps.ForEach(b =>
|
AddStep("add single ruleset beatmapset", () =>
|
||||||
{
|
{
|
||||||
b.Ruleset = rulesets.AvailableRulesets.ElementAt(1);
|
testSingle = createTestBeatmapSet(set_count + 2);
|
||||||
b.RulesetID = b.Ruleset.ID ?? 1;
|
testSingle.Beatmaps.ForEach(b =>
|
||||||
|
{
|
||||||
|
b.Ruleset = rulesets.AvailableRulesets.ElementAt(1);
|
||||||
|
b.RulesetID = b.Ruleset.ID ?? 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
carousel.UpdateBeatmapSet(testSingle);
|
||||||
});
|
});
|
||||||
AddStep("add single ruleset beatmapset", () => carousel.UpdateBeatmapSet(testSingle));
|
|
||||||
AddStep("select filtered map skipping filtered", () => carousel.SelectBeatmap(testSingle.Beatmaps[0], false));
|
AddStep("select filtered map skipping filtered", () => carousel.SelectBeatmap(testSingle.Beatmaps[0], false));
|
||||||
checkNoSelection();
|
checkNoSelection();
|
||||||
AddStep("remove single ruleset set", () => carousel.RemoveBeatmapSet(testSingle));
|
AddStep("remove single ruleset set", () => carousel.RemoveBeatmapSet(testSingle));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void TestCarouselRootIsRandom()
|
public void TestCarouselRemembersSelection()
|
||||||
{
|
{
|
||||||
List<BeatmapSetInfo> manySets = new List<BeatmapSetInfo>();
|
List<BeatmapSetInfo> manySets = new List<BeatmapSetInfo>();
|
||||||
|
|
||||||
@ -429,12 +462,74 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
loadBeatmaps(manySets);
|
loadBeatmaps(manySets);
|
||||||
|
|
||||||
advanceSelection(direction: 1, diff: false);
|
advanceSelection(direction: 1, diff: false);
|
||||||
checkNonmatchingFilter();
|
|
||||||
checkNonmatchingFilter();
|
for (int i = 0; i < 5; i++)
|
||||||
checkNonmatchingFilter();
|
{
|
||||||
checkNonmatchingFilter();
|
AddStep("Toggle non-matching filter", () =>
|
||||||
checkNonmatchingFilter();
|
{
|
||||||
AddAssert("Selection was random", () => eagerSelectedIDs.Count > 1);
|
carousel.Filter(new FilterCriteria { SearchText = Guid.NewGuid().ToString() }, false);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("Restore no filter", () =>
|
||||||
|
{
|
||||||
|
carousel.Filter(new FilterCriteria(), false);
|
||||||
|
eagerSelectedIDs.Add(carousel.SelectedBeatmapSet.ID);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// always returns to same selection as long as it's available.
|
||||||
|
AddAssert("Selection was remembered", () => eagerSelectedIDs.Count == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestRandomFallbackOnNonMatchingPrevious()
|
||||||
|
{
|
||||||
|
List<BeatmapSetInfo> manySets = new List<BeatmapSetInfo>();
|
||||||
|
|
||||||
|
AddStep("populate maps", () =>
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
var set = createTestBeatmapSet(i);
|
||||||
|
|
||||||
|
foreach (var b in set.Beatmaps)
|
||||||
|
{
|
||||||
|
// all taiko except for first
|
||||||
|
int ruleset = i > 0 ? 1 : 0;
|
||||||
|
|
||||||
|
b.Ruleset = rulesets.GetRuleset(ruleset);
|
||||||
|
b.RulesetID = ruleset;
|
||||||
|
}
|
||||||
|
|
||||||
|
manySets.Add(set);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
loadBeatmaps(manySets);
|
||||||
|
|
||||||
|
for (int i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
AddStep("Reset filter", () => carousel.Filter(new FilterCriteria(), false));
|
||||||
|
|
||||||
|
AddStep("select first beatmap", () => carousel.SelectBeatmap(manySets.First().Beatmaps.First()));
|
||||||
|
|
||||||
|
AddStep("Toggle non-matching filter", () =>
|
||||||
|
{
|
||||||
|
carousel.Filter(new FilterCriteria { SearchText = Guid.NewGuid().ToString() }, false);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddAssert("selection lost", () => carousel.SelectedBeatmap == null);
|
||||||
|
|
||||||
|
AddStep("Restore different ruleset filter", () =>
|
||||||
|
{
|
||||||
|
carousel.Filter(new FilterCriteria { Ruleset = rulesets.GetRuleset(1) }, false);
|
||||||
|
eagerSelectedIDs.Add(carousel.SelectedBeatmapSet.ID);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddAssert("selection changed", () => carousel.SelectedBeatmap != manySets.First().Beatmaps.First());
|
||||||
|
}
|
||||||
|
|
||||||
|
AddAssert("Selection was random", () => eagerSelectedIDs.Count > 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
@ -484,7 +579,7 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
checkVisibleItemCount(true, 15);
|
checkVisibleItemCount(true, 15);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void loadBeatmaps(List<BeatmapSetInfo> beatmapSets = null)
|
private void loadBeatmaps(List<BeatmapSetInfo> beatmapSets = null, Func<FilterCriteria> initialCriteria = null)
|
||||||
{
|
{
|
||||||
createCarousel();
|
createCarousel();
|
||||||
|
|
||||||
@ -499,7 +594,7 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
bool changed = false;
|
bool changed = false;
|
||||||
AddStep($"Load {(beatmapSets.Count > 0 ? beatmapSets.Count.ToString() : "some")} beatmaps", () =>
|
AddStep($"Load {(beatmapSets.Count > 0 ? beatmapSets.Count.ToString() : "some")} beatmaps", () =>
|
||||||
{
|
{
|
||||||
carousel.Filter(new FilterCriteria());
|
carousel.Filter(initialCriteria?.Invoke() ?? new FilterCriteria());
|
||||||
carousel.BeatmapSetsChanged = () => changed = true;
|
carousel.BeatmapSetsChanged = () => changed = true;
|
||||||
carousel.BeatmapSets = beatmapSets;
|
carousel.BeatmapSets = beatmapSets;
|
||||||
});
|
});
|
||||||
@ -593,16 +688,6 @@ namespace osu.Game.Tests.Visual.SongSelect
|
|||||||
AddAssert("Selection is visible", selectedBeatmapVisible);
|
AddAssert("Selection is visible", selectedBeatmapVisible);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void checkNonmatchingFilter()
|
|
||||||
{
|
|
||||||
AddStep("Toggle non-matching filter", () =>
|
|
||||||
{
|
|
||||||
carousel.Filter(new FilterCriteria { SearchText = "Dingo" }, false);
|
|
||||||
carousel.Filter(new FilterCriteria(), false);
|
|
||||||
eagerSelectedIDs.Add(carousel.SelectedBeatmapSet.ID);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private BeatmapSetInfo createTestBeatmapSet(int id)
|
private BeatmapSetInfo createTestBeatmapSet(int id)
|
||||||
{
|
{
|
||||||
return new BeatmapSetInfo
|
return new BeatmapSetInfo
|
||||||
|
@ -15,7 +15,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
public class TestSceneCommentEditor : ManualInputManagerTestScene
|
public class TestSceneCommentEditor : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -17,7 +17,7 @@ using osuTK.Graphics;
|
|||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneCursors : ManualInputManagerTestScene
|
public class TestSceneCursors : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private readonly MenuCursorContainer menuCursorContainer;
|
private readonly MenuCursorContainer menuCursorContainer;
|
||||||
private readonly CustomCursorBox[] cursorBoxes = new CustomCursorBox[6];
|
private readonly CustomCursorBox[] cursorBoxes = new CustomCursorBox[6];
|
||||||
|
@ -27,7 +27,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
public class TestSceneDeleteLocalScore : ManualInputManagerTestScene
|
public class TestSceneDeleteLocalScore : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -8,7 +8,7 @@ using NUnit.Framework;
|
|||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Game.Overlays;
|
using osu.Game.Overlays;
|
||||||
using osu.Game.Overlays.Home.Friends;
|
using osu.Game.Overlays.Dashboard.Friends;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
@ -17,20 +17,20 @@ namespace osu.Game.Tests.Visual.UserInterface
|
|||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
typeof(FriendsOnlineStatusControl),
|
typeof(FriendOnlineStreamControl),
|
||||||
typeof(FriendsOnlineStatusItem),
|
typeof(FriendsOnlineStatusItem),
|
||||||
typeof(OverlayStreamControl<>),
|
typeof(OverlayStreamControl<>),
|
||||||
typeof(OverlayStreamItem<>),
|
typeof(OverlayStreamItem<>),
|
||||||
typeof(FriendsBundle)
|
typeof(FriendStream)
|
||||||
};
|
};
|
||||||
|
|
||||||
[Cached]
|
[Cached]
|
||||||
private readonly OverlayColourProvider colourProvider = new OverlayColourProvider(OverlayColourScheme.Blue);
|
private readonly OverlayColourProvider colourProvider = new OverlayColourProvider(OverlayColourScheme.Blue);
|
||||||
|
|
||||||
private FriendsOnlineStatusControl control;
|
private FriendOnlineStreamControl control;
|
||||||
|
|
||||||
[SetUp]
|
[SetUp]
|
||||||
public void SetUp() => Schedule(() => Child = control = new FriendsOnlineStatusControl
|
public void SetUp() => Schedule(() => Child = control = new FriendOnlineStreamControl
|
||||||
{
|
{
|
||||||
Anchor = Anchor.Centre,
|
Anchor = Anchor.Centre,
|
||||||
Origin = Anchor.Centre,
|
Origin = Anchor.Centre,
|
||||||
@ -55,9 +55,9 @@ namespace osu.Game.Tests.Visual.UserInterface
|
|||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
AddAssert("3 users", () => control.Items.FirstOrDefault(item => item.Status == FriendsOnlineStatus.All)?.Count == 3);
|
AddAssert("3 users", () => control.Items.FirstOrDefault(item => item.Status == OnlineStatus.All)?.Count == 3);
|
||||||
AddAssert("1 online user", () => control.Items.FirstOrDefault(item => item.Status == FriendsOnlineStatus.Online)?.Count == 1);
|
AddAssert("1 online user", () => control.Items.FirstOrDefault(item => item.Status == OnlineStatus.Online)?.Count == 1);
|
||||||
AddAssert("2 offline users", () => control.Items.FirstOrDefault(item => item.Status == FriendsOnlineStatus.Offline)?.Count == 2);
|
AddAssert("2 offline users", () => control.Items.FirstOrDefault(item => item.Status == OnlineStatus.Offline)?.Count == 2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,7 @@ using osuTK.Graphics;
|
|||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public class TestSceneOsuHoverContainer : ManualInputManagerTestScene
|
public class TestSceneOsuHoverContainer : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private OsuHoverTestContainer hoverContainer;
|
private OsuHoverTestContainer hoverContainer;
|
||||||
private Box colourContainer;
|
private Box colourContainer;
|
||||||
|
@ -12,7 +12,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
public class TestSceneStatefulMenuItem : ManualInputManagerTestScene
|
public class TestSceneStatefulMenuItem : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
public override IReadOnlyList<Type> RequiredTypes => new[]
|
||||||
{
|
{
|
||||||
|
@ -9,7 +9,7 @@ using osuTK.Input;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
{
|
{
|
||||||
public class TestSceneSwitchButton : ManualInputManagerTestScene
|
public class TestSceneSwitchButton : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
private SwitchButton switchButton;
|
private SwitchButton switchButton;
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ using osu.Framework.Graphics;
|
|||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Game.Graphics.Sprites;
|
using osu.Game.Graphics.Sprites;
|
||||||
using osu.Game.Overlays;
|
using osu.Game.Overlays;
|
||||||
using osu.Game.Overlays.Home.Friends;
|
using osu.Game.Overlays.Dashboard.Friends;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.UserInterface
|
namespace osu.Game.Tests.Visual.UserInterface
|
||||||
|
@ -1,16 +1,147 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using System.Linq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Utils;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Rulesets;
|
||||||
using osu.Game.Tournament.Models;
|
using osu.Game.Tournament.Models;
|
||||||
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tournament.Tests
|
namespace osu.Game.Tournament.Tests
|
||||||
{
|
{
|
||||||
[TestFixture]
|
[TestFixture]
|
||||||
public abstract class LadderTestScene : TournamentTestScene
|
public abstract class LadderTestScene : TournamentTestScene
|
||||||
{
|
{
|
||||||
|
[Cached]
|
||||||
|
protected LadderInfo Ladder { get; private set; } = new LadderInfo();
|
||||||
|
|
||||||
[Resolved]
|
[Resolved]
|
||||||
protected LadderInfo Ladder { get; private set; }
|
private RulesetStore rulesetStore { get; set; }
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load()
|
||||||
|
{
|
||||||
|
if (Ladder.Ruleset.Value == null)
|
||||||
|
Ladder.Ruleset.Value = rulesetStore.AvailableRulesets.First();
|
||||||
|
|
||||||
|
Ruleset.BindTo(Ladder.Ruleset);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void LoadComplete()
|
||||||
|
{
|
||||||
|
base.LoadComplete();
|
||||||
|
|
||||||
|
TournamentMatch match = CreateSampleMatch();
|
||||||
|
|
||||||
|
Ladder.Rounds.Add(match.Round.Value);
|
||||||
|
Ladder.Matches.Add(match);
|
||||||
|
Ladder.Teams.Add(match.Team1.Value);
|
||||||
|
Ladder.Teams.Add(match.Team2.Value);
|
||||||
|
|
||||||
|
Ladder.CurrentMatch.Value = match;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static TournamentMatch CreateSampleMatch() => new TournamentMatch
|
||||||
|
{
|
||||||
|
Team1 =
|
||||||
|
{
|
||||||
|
Value = new TournamentTeam
|
||||||
|
{
|
||||||
|
FlagName = { Value = "JP" },
|
||||||
|
FullName = { Value = "Japan" },
|
||||||
|
LastYearPlacing = { Value = 10 },
|
||||||
|
Seed = { Value = "Low" },
|
||||||
|
SeedingResults =
|
||||||
|
{
|
||||||
|
new SeedingResult
|
||||||
|
{
|
||||||
|
Mod = { Value = "NM" },
|
||||||
|
Seed = { Value = 10 },
|
||||||
|
Beatmaps =
|
||||||
|
{
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 12345672,
|
||||||
|
Seed = { Value = 24 },
|
||||||
|
},
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 1234567,
|
||||||
|
Seed = { Value = 12 },
|
||||||
|
},
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 1234567,
|
||||||
|
Seed = { Value = 16 },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
new SeedingResult
|
||||||
|
{
|
||||||
|
Mod = { Value = "DT" },
|
||||||
|
Seed = { Value = 5 },
|
||||||
|
Beatmaps =
|
||||||
|
{
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 234567,
|
||||||
|
Seed = { Value = 3 },
|
||||||
|
},
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 234567,
|
||||||
|
Seed = { Value = 6 },
|
||||||
|
},
|
||||||
|
new SeedingBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Score = 234567,
|
||||||
|
Seed = { Value = 12 },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Players =
|
||||||
|
{
|
||||||
|
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 12 } } },
|
||||||
|
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 16 } } },
|
||||||
|
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 20 } } },
|
||||||
|
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 24 } } },
|
||||||
|
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 30 } } },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Team2 =
|
||||||
|
{
|
||||||
|
Value = new TournamentTeam
|
||||||
|
{
|
||||||
|
FlagName = { Value = "US" },
|
||||||
|
FullName = { Value = "United States" },
|
||||||
|
Players =
|
||||||
|
{
|
||||||
|
new User { Username = "Hello" },
|
||||||
|
new User { Username = "Hello" },
|
||||||
|
new User { Username = "Hello" },
|
||||||
|
new User { Username = "Hello" },
|
||||||
|
new User { Username = "Hello" },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Round =
|
||||||
|
{
|
||||||
|
Value = new TournamentRound { Name = { Value = "Quarterfinals" } }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public static BeatmapInfo CreateSampleBeatmapInfo() =>
|
||||||
|
new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist", ID = RNG.Next(0, 1000000) } };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,24 +1,140 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
using System;
|
using System.Linq;
|
||||||
using System.Collections.Generic;
|
using NUnit.Framework;
|
||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Testing;
|
||||||
|
using osu.Game.Tournament.Components;
|
||||||
|
using osu.Game.Tournament.Models;
|
||||||
using osu.Game.Tournament.Screens.MapPool;
|
using osu.Game.Tournament.Screens.MapPool;
|
||||||
|
|
||||||
namespace osu.Game.Tournament.Tests.Screens
|
namespace osu.Game.Tournament.Tests.Screens
|
||||||
{
|
{
|
||||||
public class TestSceneMapPoolScreen : LadderTestScene
|
public class TestSceneMapPoolScreen : LadderTestScene
|
||||||
{
|
{
|
||||||
public override IReadOnlyList<Type> RequiredTypes => new[]
|
private MapPoolScreen screen;
|
||||||
{
|
|
||||||
typeof(MapPoolScreen)
|
|
||||||
};
|
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load()
|
private void load()
|
||||||
{
|
{
|
||||||
Add(new MapPoolScreen { Width = 0.7f });
|
Add(screen = new MapPoolScreen { Width = 0.7f });
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestFewMaps()
|
||||||
|
{
|
||||||
|
AddStep("load few maps", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
addBeatmap();
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("reset match", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value = new TournamentMatch();
|
||||||
|
Ladder.CurrentMatch.Value = Ladder.Matches.First();
|
||||||
|
});
|
||||||
|
|
||||||
|
assertTwoWide();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestJustEnoughMaps()
|
||||||
|
{
|
||||||
|
AddStep("load just enough maps", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < 18; i++)
|
||||||
|
addBeatmap();
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("reset match", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value = new TournamentMatch();
|
||||||
|
Ladder.CurrentMatch.Value = Ladder.Matches.First();
|
||||||
|
});
|
||||||
|
|
||||||
|
assertTwoWide();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestManyMaps()
|
||||||
|
{
|
||||||
|
AddStep("load many maps", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < 19; i++)
|
||||||
|
addBeatmap();
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("reset match", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value = new TournamentMatch();
|
||||||
|
Ladder.CurrentMatch.Value = Ladder.Matches.First();
|
||||||
|
});
|
||||||
|
|
||||||
|
assertThreeWide();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestJustEnoughMods()
|
||||||
|
{
|
||||||
|
AddStep("load many maps", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < 11; i++)
|
||||||
|
addBeatmap(i > 4 ? $"M{i}" : "NM");
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("reset match", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value = new TournamentMatch();
|
||||||
|
Ladder.CurrentMatch.Value = Ladder.Matches.First();
|
||||||
|
});
|
||||||
|
|
||||||
|
assertTwoWide();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertTwoWide() =>
|
||||||
|
AddAssert("ensure layout width is 2", () => screen.ChildrenOfType<FillFlowContainer<FillFlowContainer<TournamentBeatmapPanel>>>().First().Padding.Left > 0);
|
||||||
|
|
||||||
|
private void assertThreeWide() =>
|
||||||
|
AddAssert("ensure layout width is 3", () => screen.ChildrenOfType<FillFlowContainer<FillFlowContainer<TournamentBeatmapPanel>>>().First().Padding.Left == 0);
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestManyMods()
|
||||||
|
{
|
||||||
|
AddStep("load many maps", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Clear();
|
||||||
|
|
||||||
|
for (int i = 0; i < 12; i++)
|
||||||
|
addBeatmap(i > 4 ? $"M{i}" : "NM");
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("reset match", () =>
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value = new TournamentMatch();
|
||||||
|
Ladder.CurrentMatch.Value = Ladder.Matches.First();
|
||||||
|
});
|
||||||
|
|
||||||
|
assertThreeWide();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addBeatmap(string mods = "nm")
|
||||||
|
{
|
||||||
|
Ladder.CurrentMatch.Value.Round.Value.Beatmaps.Add(new RoundBeatmap
|
||||||
|
{
|
||||||
|
BeatmapInfo = CreateSampleBeatmapInfo(),
|
||||||
|
Mods = mods
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ namespace osu.Game.Tournament.Tests.Screens
|
|||||||
|
|
||||||
public TestSceneSeedingEditorScreen()
|
public TestSceneSeedingEditorScreen()
|
||||||
{
|
{
|
||||||
var match = TestSceneSeedingScreen.CreateSampleSeededMatch();
|
var match = CreateSampleMatch();
|
||||||
|
|
||||||
Add(new SeedingEditorScreen(match.Team1.Value)
|
Add(new SeedingEditorScreen(match.Team1.Value)
|
||||||
{
|
{
|
||||||
|
@ -3,10 +3,8 @@
|
|||||||
|
|
||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Game.Beatmaps;
|
|
||||||
using osu.Game.Tournament.Models;
|
using osu.Game.Tournament.Models;
|
||||||
using osu.Game.Tournament.Screens.TeamIntro;
|
using osu.Game.Tournament.Screens.TeamIntro;
|
||||||
using osu.Game.Users;
|
|
||||||
|
|
||||||
namespace osu.Game.Tournament.Tests.Screens
|
namespace osu.Game.Tournament.Tests.Screens
|
||||||
{
|
{
|
||||||
@ -18,110 +16,11 @@ namespace osu.Game.Tournament.Tests.Screens
|
|||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load()
|
private void load()
|
||||||
{
|
{
|
||||||
ladder.CurrentMatch.Value = CreateSampleSeededMatch();
|
|
||||||
|
|
||||||
Add(new SeedingScreen
|
Add(new SeedingScreen
|
||||||
{
|
{
|
||||||
FillMode = FillMode.Fit,
|
FillMode = FillMode.Fit,
|
||||||
FillAspectRatio = 16 / 9f
|
FillAspectRatio = 16 / 9f
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public static TournamentMatch CreateSampleSeededMatch() => new TournamentMatch
|
|
||||||
{
|
|
||||||
Team1 =
|
|
||||||
{
|
|
||||||
Value = new TournamentTeam
|
|
||||||
{
|
|
||||||
FlagName = { Value = "JP" },
|
|
||||||
FullName = { Value = "Japan" },
|
|
||||||
LastYearPlacing = { Value = 10 },
|
|
||||||
Seed = { Value = "Low" },
|
|
||||||
SeedingResults =
|
|
||||||
{
|
|
||||||
new SeedingResult
|
|
||||||
{
|
|
||||||
Mod = { Value = "NM" },
|
|
||||||
Seed = { Value = 10 },
|
|
||||||
Beatmaps =
|
|
||||||
{
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 12345672,
|
|
||||||
Seed = { Value = 24 },
|
|
||||||
},
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 1234567,
|
|
||||||
Seed = { Value = 12 },
|
|
||||||
},
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 1234567,
|
|
||||||
Seed = { Value = 16 },
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
new SeedingResult
|
|
||||||
{
|
|
||||||
Mod = { Value = "DT" },
|
|
||||||
Seed = { Value = 5 },
|
|
||||||
Beatmaps =
|
|
||||||
{
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 234567,
|
|
||||||
Seed = { Value = 3 },
|
|
||||||
},
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 234567,
|
|
||||||
Seed = { Value = 6 },
|
|
||||||
},
|
|
||||||
new SeedingBeatmap
|
|
||||||
{
|
|
||||||
BeatmapInfo = new BeatmapInfo { Metadata = new BeatmapMetadata { Title = "Test Title", Artist = "Test Artist" } },
|
|
||||||
Score = 234567,
|
|
||||||
Seed = { Value = 12 },
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Players =
|
|
||||||
{
|
|
||||||
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 12 } } },
|
|
||||||
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 16 } } },
|
|
||||||
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 20 } } },
|
|
||||||
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 24 } } },
|
|
||||||
new User { Username = "Hello", Statistics = new UserStatistics { Ranks = new UserStatistics.UserRanks { Global = 30 } } },
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Team2 =
|
|
||||||
{
|
|
||||||
Value = new TournamentTeam
|
|
||||||
{
|
|
||||||
FlagName = { Value = "US" },
|
|
||||||
FullName = { Value = "United States" },
|
|
||||||
Players =
|
|
||||||
{
|
|
||||||
new User { Username = "Hello" },
|
|
||||||
new User { Username = "Hello" },
|
|
||||||
new User { Username = "Hello" },
|
|
||||||
new User { Username = "Hello" },
|
|
||||||
new User { Username = "Hello" },
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Round =
|
|
||||||
{
|
|
||||||
Value = new TournamentRound { Name = { Value = "Quarterfinals" } }
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -70,6 +70,17 @@ namespace osu.Game.Tournament.Components
|
|||||||
|
|
||||||
protected override ChatLine CreateMessage(Message message) => new MatchMessage(message);
|
protected override ChatLine CreateMessage(Message message) => new MatchMessage(message);
|
||||||
|
|
||||||
|
protected override StandAloneDrawableChannel CreateDrawableChannel(Channel channel) => new MatchChannel(channel);
|
||||||
|
|
||||||
|
public class MatchChannel : StandAloneDrawableChannel
|
||||||
|
{
|
||||||
|
public MatchChannel(Channel channel)
|
||||||
|
: base(channel)
|
||||||
|
{
|
||||||
|
ScrollbarVisible = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected class MatchMessage : StandAloneMessage
|
protected class MatchMessage : StandAloneMessage
|
||||||
{
|
{
|
||||||
public MatchMessage(Message message)
|
public MatchMessage(Message message)
|
||||||
|
@ -24,7 +24,13 @@ namespace osu.Game.Tournament.Models
|
|||||||
// only used for serialisation
|
// only used for serialisation
|
||||||
public List<TournamentProgression> Progressions = new List<TournamentProgression>();
|
public List<TournamentProgression> Progressions = new List<TournamentProgression>();
|
||||||
|
|
||||||
[JsonIgnore]
|
[JsonIgnore] // updated manually in TournamentGameBase
|
||||||
public Bindable<TournamentMatch> CurrentMatch = new Bindable<TournamentMatch>();
|
public Bindable<TournamentMatch> CurrentMatch = new Bindable<TournamentMatch>();
|
||||||
|
|
||||||
|
public Bindable<int> ChromaKeyWidth = new BindableInt(1024)
|
||||||
|
{
|
||||||
|
MinValue = 640,
|
||||||
|
MaxValue = 1366,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ using osu.Framework.Graphics.Shapes;
|
|||||||
using osu.Framework.Platform;
|
using osu.Framework.Platform;
|
||||||
using osu.Framework.Threading;
|
using osu.Framework.Threading;
|
||||||
using osu.Game.Graphics.UserInterface;
|
using osu.Game.Graphics.UserInterface;
|
||||||
|
using osu.Game.Overlays.Settings;
|
||||||
using osu.Game.Tournament.Components;
|
using osu.Game.Tournament.Components;
|
||||||
using osu.Game.Tournament.IPC;
|
using osu.Game.Tournament.IPC;
|
||||||
using osu.Game.Tournament.Models;
|
using osu.Game.Tournament.Models;
|
||||||
@ -35,6 +36,8 @@ namespace osu.Game.Tournament.Screens.Gameplay
|
|||||||
[Resolved]
|
[Resolved]
|
||||||
private TournamentMatchChatDisplay chat { get; set; }
|
private TournamentMatchChatDisplay chat { get; set; }
|
||||||
|
|
||||||
|
private Box chroma;
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load(LadderInfo ladder, MatchIPCInfo ipc, Storage storage)
|
private void load(LadderInfo ladder, MatchIPCInfo ipc, Storage storage)
|
||||||
{
|
{
|
||||||
@ -60,11 +63,10 @@ namespace osu.Game.Tournament.Screens.Gameplay
|
|||||||
Origin = Anchor.TopCentre,
|
Origin = Anchor.TopCentre,
|
||||||
Children = new Drawable[]
|
Children = new Drawable[]
|
||||||
{
|
{
|
||||||
new Box
|
chroma = new Box
|
||||||
{
|
{
|
||||||
// chroma key area for stable gameplay
|
// chroma key area for stable gameplay
|
||||||
Name = "chroma",
|
Name = "chroma",
|
||||||
RelativeSizeAxes = Axes.X,
|
|
||||||
Anchor = Anchor.TopCentre,
|
Anchor = Anchor.TopCentre,
|
||||||
Origin = Anchor.TopCentre,
|
Origin = Anchor.TopCentre,
|
||||||
Height = 512,
|
Height = 512,
|
||||||
@ -93,6 +95,12 @@ namespace osu.Game.Tournament.Screens.Gameplay
|
|||||||
RelativeSizeAxes = Axes.X,
|
RelativeSizeAxes = Axes.X,
|
||||||
Text = "Toggle chat",
|
Text = "Toggle chat",
|
||||||
Action = () => { State.Value = State.Value == TourneyState.Idle ? TourneyState.Playing : TourneyState.Idle; }
|
Action = () => { State.Value = State.Value == TourneyState.Idle ? TourneyState.Playing : TourneyState.Idle; }
|
||||||
|
},
|
||||||
|
new SettingsSlider<int>
|
||||||
|
{
|
||||||
|
LabelText = "Chroma Width",
|
||||||
|
Bindable = LadderInfo.ChromaKeyWidth,
|
||||||
|
KeyboardStep = 1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -101,6 +109,8 @@ namespace osu.Game.Tournament.Screens.Gameplay
|
|||||||
State.BindTo(ipc.State);
|
State.BindTo(ipc.State);
|
||||||
State.BindValueChanged(stateChanged, true);
|
State.BindValueChanged(stateChanged, true);
|
||||||
|
|
||||||
|
ladder.ChromaKeyWidth.BindValueChanged(width => chroma.Width = width.NewValue, true);
|
||||||
|
|
||||||
currentMatch.BindValueChanged(m =>
|
currentMatch.BindValueChanged(m =>
|
||||||
{
|
{
|
||||||
warmup.Value = m.NewValue.Team1Score.Value + m.NewValue.Team2Score.Value == 0;
|
warmup.Value = m.NewValue.Team1Score.Value + m.NewValue.Team2Score.Value == 0;
|
||||||
|
@ -50,11 +50,11 @@ namespace osu.Game.Tournament.Screens.MapPool
|
|||||||
new MatchHeader(),
|
new MatchHeader(),
|
||||||
mapFlows = new FillFlowContainer<FillFlowContainer<TournamentBeatmapPanel>>
|
mapFlows = new FillFlowContainer<FillFlowContainer<TournamentBeatmapPanel>>
|
||||||
{
|
{
|
||||||
Y = 140,
|
Y = 160,
|
||||||
Spacing = new Vector2(10, 10),
|
Spacing = new Vector2(10, 10),
|
||||||
Padding = new MarginPadding(25),
|
|
||||||
Direction = FillDirection.Vertical,
|
Direction = FillDirection.Vertical,
|
||||||
RelativeSizeAxes = Axes.Both,
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
},
|
},
|
||||||
new ControlPanel
|
new ControlPanel
|
||||||
{
|
{
|
||||||
@ -95,6 +95,7 @@ namespace osu.Game.Tournament.Screens.MapPool
|
|||||||
Text = "Reset",
|
Text = "Reset",
|
||||||
Action = reset
|
Action = reset
|
||||||
},
|
},
|
||||||
|
new ControlPanel.Spacer(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -211,11 +212,15 @@ namespace osu.Game.Tournament.Screens.MapPool
|
|||||||
{
|
{
|
||||||
mapFlows.Clear();
|
mapFlows.Clear();
|
||||||
|
|
||||||
|
int totalRows = 0;
|
||||||
|
|
||||||
if (match.NewValue.Round.Value != null)
|
if (match.NewValue.Round.Value != null)
|
||||||
{
|
{
|
||||||
FillFlowContainer<TournamentBeatmapPanel> currentFlow = null;
|
FillFlowContainer<TournamentBeatmapPanel> currentFlow = null;
|
||||||
string currentMod = null;
|
string currentMod = null;
|
||||||
|
|
||||||
|
int flowCount = 0;
|
||||||
|
|
||||||
foreach (var b in match.NewValue.Round.Value.Beatmaps)
|
foreach (var b in match.NewValue.Round.Value.Beatmaps)
|
||||||
{
|
{
|
||||||
if (currentFlow == null || currentMod != b.Mods)
|
if (currentFlow == null || currentMod != b.Mods)
|
||||||
@ -229,6 +234,15 @@ namespace osu.Game.Tournament.Screens.MapPool
|
|||||||
});
|
});
|
||||||
|
|
||||||
currentMod = b.Mods;
|
currentMod = b.Mods;
|
||||||
|
|
||||||
|
totalRows++;
|
||||||
|
flowCount = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (++flowCount > 2)
|
||||||
|
{
|
||||||
|
totalRows++;
|
||||||
|
flowCount = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
currentFlow.Add(new TournamentBeatmapPanel(b.BeatmapInfo, b.Mods)
|
currentFlow.Add(new TournamentBeatmapPanel(b.BeatmapInfo, b.Mods)
|
||||||
@ -239,6 +253,12 @@ namespace osu.Game.Tournament.Screens.MapPool
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mapFlows.Padding = new MarginPadding(5)
|
||||||
|
{
|
||||||
|
// remove horizontal padding to increase flow width to 3 panels
|
||||||
|
Horizontal = totalRows > 9 ? 0 : 100
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -116,7 +116,7 @@ namespace osu.Game.Tournament.Screens
|
|||||||
{
|
{
|
||||||
windowSize.Value = new Size((int)(1920 / TournamentSceneManager.STREAM_AREA_WIDTH * TournamentSceneManager.REQUIRED_WIDTH), 1080);
|
windowSize.Value = new Size((int)(1920 / TournamentSceneManager.STREAM_AREA_WIDTH * TournamentSceneManager.REQUIRED_WIDTH), 1080);
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,7 +26,7 @@ namespace osu.Game.Online.Chat
|
|||||||
|
|
||||||
protected ChannelManager ChannelManager;
|
protected ChannelManager ChannelManager;
|
||||||
|
|
||||||
private DrawableChannel drawableChannel;
|
private StandAloneDrawableChannel drawableChannel;
|
||||||
|
|
||||||
private readonly bool postingTextbox;
|
private readonly bool postingTextbox;
|
||||||
|
|
||||||
@ -77,6 +77,9 @@ namespace osu.Game.Online.Chat
|
|||||||
ChannelManager = manager;
|
ChannelManager = manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected virtual StandAloneDrawableChannel CreateDrawableChannel(Channel channel) =>
|
||||||
|
new StandAloneDrawableChannel(channel);
|
||||||
|
|
||||||
private void postMessage(TextBox sender, bool newtext)
|
private void postMessage(TextBox sender, bool newtext)
|
||||||
{
|
{
|
||||||
var text = textbox.Text.Trim();
|
var text = textbox.Text.Trim();
|
||||||
@ -100,14 +103,14 @@ namespace osu.Game.Online.Chat
|
|||||||
|
|
||||||
if (e.NewValue == null) return;
|
if (e.NewValue == null) return;
|
||||||
|
|
||||||
AddInternal(drawableChannel = new StandAloneDrawableChannel(e.NewValue)
|
drawableChannel = CreateDrawableChannel(e.NewValue);
|
||||||
{
|
drawableChannel.CreateChatLineAction = CreateMessage;
|
||||||
CreateChatLineAction = CreateMessage,
|
drawableChannel.Padding = new MarginPadding { Bottom = postingTextbox ? textbox_height : 0 };
|
||||||
Padding = new MarginPadding { Bottom = postingTextbox ? textbox_height : 0 }
|
|
||||||
});
|
AddInternal(drawableChannel);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected class StandAloneDrawableChannel : DrawableChannel
|
public class StandAloneDrawableChannel : DrawableChannel
|
||||||
{
|
{
|
||||||
public Func<Message, ChatLine> CreateChatLineAction;
|
public Func<Message, ChatLine> CreateChatLineAction;
|
||||||
|
|
||||||
|
@ -47,6 +47,8 @@ namespace osu.Game
|
|||||||
{
|
{
|
||||||
public const string CLIENT_STREAM_NAME = "lazer";
|
public const string CLIENT_STREAM_NAME = "lazer";
|
||||||
|
|
||||||
|
public const int SAMPLE_CONCURRENCY = 6;
|
||||||
|
|
||||||
protected OsuConfigManager LocalConfig;
|
protected OsuConfigManager LocalConfig;
|
||||||
|
|
||||||
protected BeatmapManager BeatmapManager;
|
protected BeatmapManager BeatmapManager;
|
||||||
@ -153,6 +155,8 @@ namespace osu.Game
|
|||||||
AddFont(Resources, @"Fonts/Venera-Bold");
|
AddFont(Resources, @"Fonts/Venera-Bold");
|
||||||
AddFont(Resources, @"Fonts/Venera-Black");
|
AddFont(Resources, @"Fonts/Venera-Black");
|
||||||
|
|
||||||
|
Audio.Samples.PlaybackConcurrency = SAMPLE_CONCURRENCY;
|
||||||
|
|
||||||
runMigrations();
|
runMigrations();
|
||||||
|
|
||||||
dependencies.Cache(SkinManager = new SkinManager(Storage, contextFactory, Host, Audio, new NamespacedResourceStore<byte[]>(Resources, "Skins/Legacy")));
|
dependencies.Cache(SkinManager = new SkinManager(Storage, contextFactory, Host, Audio, new NamespacedResourceStore<byte[]>(Resources, "Skins/Legacy")));
|
||||||
|
@ -26,6 +26,20 @@ namespace osu.Game.Overlays.Chat
|
|||||||
protected FillFlowContainer ChatLineFlow;
|
protected FillFlowContainer ChatLineFlow;
|
||||||
private OsuScrollContainer scroll;
|
private OsuScrollContainer scroll;
|
||||||
|
|
||||||
|
private bool scrollbarVisible = true;
|
||||||
|
|
||||||
|
public bool ScrollbarVisible
|
||||||
|
{
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if (scrollbarVisible == value) return;
|
||||||
|
|
||||||
|
scrollbarVisible = value;
|
||||||
|
if (scroll != null)
|
||||||
|
scroll.ScrollbarVisible = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
[Resolved]
|
[Resolved]
|
||||||
private OsuColour colours { get; set; }
|
private OsuColour colours { get; set; }
|
||||||
|
|
||||||
@ -44,6 +58,7 @@ namespace osu.Game.Overlays.Chat
|
|||||||
Masking = true,
|
Masking = true,
|
||||||
Child = scroll = new OsuScrollContainer
|
Child = scroll = new OsuScrollContainer
|
||||||
{
|
{
|
||||||
|
ScrollbarVisible = scrollbarVisible,
|
||||||
RelativeSizeAxes = Axes.Both,
|
RelativeSizeAxes = Axes.Both,
|
||||||
// Some chat lines have effects that slightly protrude to the bottom,
|
// Some chat lines have effects that slightly protrude to the bottom,
|
||||||
// which we do not want to mask away, hence the padding.
|
// which we do not want to mask away, hence the padding.
|
||||||
|
267
osu.Game/Overlays/Dashboard/Friends/FriendDisplay.cs
Normal file
267
osu.Game/Overlays/Dashboard/Friends/FriendDisplay.cs
Normal file
@ -0,0 +1,267 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading;
|
||||||
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Graphics.Shapes;
|
||||||
|
using osu.Game.Graphics.UserInterface;
|
||||||
|
using osu.Game.Online.API;
|
||||||
|
using osu.Game.Online.API.Requests;
|
||||||
|
using osu.Game.Users;
|
||||||
|
using osuTK;
|
||||||
|
|
||||||
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
|
{
|
||||||
|
public class FriendDisplay : CompositeDrawable
|
||||||
|
{
|
||||||
|
private List<User> users = new List<User>();
|
||||||
|
|
||||||
|
public List<User> Users
|
||||||
|
{
|
||||||
|
get => users;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
users = value;
|
||||||
|
|
||||||
|
onlineStreamControl.Populate(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Resolved]
|
||||||
|
private IAPIProvider api { get; set; }
|
||||||
|
|
||||||
|
private GetFriendsRequest request;
|
||||||
|
private CancellationTokenSource cancellationToken;
|
||||||
|
|
||||||
|
private Drawable currentContent;
|
||||||
|
|
||||||
|
private readonly FriendOnlineStreamControl onlineStreamControl;
|
||||||
|
private readonly Box background;
|
||||||
|
private readonly Box controlBackground;
|
||||||
|
private readonly UserListToolbar userListToolbar;
|
||||||
|
private readonly Container itemsPlaceholder;
|
||||||
|
private readonly LoadingLayer loading;
|
||||||
|
|
||||||
|
public FriendDisplay()
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X;
|
||||||
|
AutoSizeAxes = Axes.Y;
|
||||||
|
InternalChild = new FillFlowContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
new Container
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
controlBackground = new Box
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both
|
||||||
|
},
|
||||||
|
new Container
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Padding = new MarginPadding
|
||||||
|
{
|
||||||
|
Top = 20,
|
||||||
|
Horizontal = 45
|
||||||
|
},
|
||||||
|
Child = onlineStreamControl = new FriendOnlineStreamControl(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
new Container
|
||||||
|
{
|
||||||
|
Name = "User List",
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
background = new Box
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both
|
||||||
|
},
|
||||||
|
new FillFlowContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Direction = FillDirection.Vertical,
|
||||||
|
Margin = new MarginPadding { Bottom = 20 },
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
new Container
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Padding = new MarginPadding
|
||||||
|
{
|
||||||
|
Horizontal = 40,
|
||||||
|
Vertical = 20
|
||||||
|
},
|
||||||
|
Child = userListToolbar = new UserListToolbar
|
||||||
|
{
|
||||||
|
Anchor = Anchor.CentreRight,
|
||||||
|
Origin = Anchor.CentreRight,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
new Container
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
itemsPlaceholder = new Container
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Padding = new MarginPadding { Horizontal = 50 }
|
||||||
|
},
|
||||||
|
loading = new LoadingLayer(itemsPlaceholder)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load(OverlayColourProvider colourProvider)
|
||||||
|
{
|
||||||
|
background.Colour = colourProvider.Background4;
|
||||||
|
controlBackground.Colour = colourProvider.Background5;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void LoadComplete()
|
||||||
|
{
|
||||||
|
base.LoadComplete();
|
||||||
|
|
||||||
|
onlineStreamControl.Current.BindValueChanged(_ => recreatePanels());
|
||||||
|
userListToolbar.DisplayStyle.BindValueChanged(_ => recreatePanels());
|
||||||
|
userListToolbar.SortCriteria.BindValueChanged(_ => recreatePanels());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Fetch()
|
||||||
|
{
|
||||||
|
if (!api.IsLoggedIn)
|
||||||
|
return;
|
||||||
|
|
||||||
|
request = new GetFriendsRequest();
|
||||||
|
request.Success += response => Schedule(() => Users = response);
|
||||||
|
api.Queue(request);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void recreatePanels()
|
||||||
|
{
|
||||||
|
if (!users.Any())
|
||||||
|
return;
|
||||||
|
|
||||||
|
cancellationToken?.Cancel();
|
||||||
|
|
||||||
|
if (itemsPlaceholder.Any())
|
||||||
|
loading.Show();
|
||||||
|
|
||||||
|
var sortedUsers = sortUsers(getUsersInCurrentGroup());
|
||||||
|
|
||||||
|
LoadComponentAsync(createTable(sortedUsers), addContentToPlaceholder, (cancellationToken = new CancellationTokenSource()).Token);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<User> getUsersInCurrentGroup()
|
||||||
|
{
|
||||||
|
switch (onlineStreamControl.Current.Value?.Status)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case OnlineStatus.All:
|
||||||
|
return users;
|
||||||
|
|
||||||
|
case OnlineStatus.Offline:
|
||||||
|
return users.Where(u => !u.IsOnline).ToList();
|
||||||
|
|
||||||
|
case OnlineStatus.Online:
|
||||||
|
return users.Where(u => u.IsOnline).ToList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addContentToPlaceholder(Drawable content)
|
||||||
|
{
|
||||||
|
loading.Hide();
|
||||||
|
|
||||||
|
var lastContent = currentContent;
|
||||||
|
|
||||||
|
if (lastContent != null)
|
||||||
|
{
|
||||||
|
lastContent.FadeOut(100, Easing.OutQuint).Expire();
|
||||||
|
lastContent.Delay(25).Schedule(() => lastContent.BypassAutoSizeAxes = Axes.Y);
|
||||||
|
}
|
||||||
|
|
||||||
|
itemsPlaceholder.Add(currentContent = content);
|
||||||
|
currentContent.FadeIn(200, Easing.OutQuint);
|
||||||
|
}
|
||||||
|
|
||||||
|
private FillFlowContainer createTable(List<User> users)
|
||||||
|
{
|
||||||
|
var style = userListToolbar.DisplayStyle.Value;
|
||||||
|
|
||||||
|
return new FillFlowContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Spacing = new Vector2(style == OverlayPanelDisplayStyle.Card ? 10 : 2),
|
||||||
|
Children = users.Select(u => createUserPanel(u, style)).ToList()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private UserPanel createUserPanel(User user, OverlayPanelDisplayStyle style)
|
||||||
|
{
|
||||||
|
switch (style)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case OverlayPanelDisplayStyle.Card:
|
||||||
|
return new UserGridPanel(user).With(panel =>
|
||||||
|
{
|
||||||
|
panel.Anchor = Anchor.TopCentre;
|
||||||
|
panel.Origin = Anchor.TopCentre;
|
||||||
|
panel.Width = 290;
|
||||||
|
});
|
||||||
|
|
||||||
|
case OverlayPanelDisplayStyle.List:
|
||||||
|
return new UserListPanel(user);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<User> sortUsers(List<User> unsorted)
|
||||||
|
{
|
||||||
|
switch (userListToolbar.SortCriteria.Value)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case UserSortCriteria.LastVisit:
|
||||||
|
return unsorted.OrderByDescending(u => u.LastVisit).ToList();
|
||||||
|
|
||||||
|
case UserSortCriteria.Rank:
|
||||||
|
return unsorted.OrderByDescending(u => u.CurrentModeRank.HasValue).ThenBy(u => u.CurrentModeRank ?? 0).ToList();
|
||||||
|
|
||||||
|
case UserSortCriteria.Username:
|
||||||
|
return unsorted.OrderBy(u => u.Username).ToList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Dispose(bool isDisposing)
|
||||||
|
{
|
||||||
|
request?.Cancel();
|
||||||
|
cancellationToken?.Cancel();
|
||||||
|
|
||||||
|
base.Dispose(isDisposing);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,28 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using osu.Game.Users;
|
||||||
|
|
||||||
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
|
{
|
||||||
|
public class FriendOnlineStreamControl : OverlayStreamControl<FriendStream>
|
||||||
|
{
|
||||||
|
protected override OverlayStreamItem<FriendStream> CreateStreamItem(FriendStream value) => new FriendsOnlineStatusItem(value);
|
||||||
|
|
||||||
|
public void Populate(List<User> users)
|
||||||
|
{
|
||||||
|
Clear();
|
||||||
|
|
||||||
|
var userCount = users.Count;
|
||||||
|
var onlineUsersCount = users.Count(user => user.IsOnline);
|
||||||
|
|
||||||
|
AddItem(new FriendStream(OnlineStatus.All, userCount));
|
||||||
|
AddItem(new FriendStream(OnlineStatus.Online, onlineUsersCount));
|
||||||
|
AddItem(new FriendStream(OnlineStatus.Offline, userCount - onlineUsersCount));
|
||||||
|
|
||||||
|
Current.Value = Items.FirstOrDefault();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
18
osu.Game/Overlays/Dashboard/Friends/FriendStream.cs
Normal file
18
osu.Game/Overlays/Dashboard/Friends/FriendStream.cs
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
|
{
|
||||||
|
public class FriendStream
|
||||||
|
{
|
||||||
|
public OnlineStatus Status { get; }
|
||||||
|
|
||||||
|
public int Count { get; }
|
||||||
|
|
||||||
|
public FriendStream(OnlineStatus status, int count)
|
||||||
|
{
|
||||||
|
Status = status;
|
||||||
|
Count = count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -5,11 +5,11 @@ using System;
|
|||||||
using osu.Game.Graphics;
|
using osu.Game.Graphics;
|
||||||
using osuTK.Graphics;
|
using osuTK.Graphics;
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Home.Friends
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
{
|
{
|
||||||
public class FriendsOnlineStatusItem : OverlayStreamItem<FriendsBundle>
|
public class FriendsOnlineStatusItem : OverlayStreamItem<FriendStream>
|
||||||
{
|
{
|
||||||
public FriendsOnlineStatusItem(FriendsBundle value)
|
public FriendsOnlineStatusItem(FriendStream value)
|
||||||
: base(value)
|
: base(value)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
@ -22,13 +22,13 @@ namespace osu.Game.Overlays.Home.Friends
|
|||||||
{
|
{
|
||||||
switch (Value.Status)
|
switch (Value.Status)
|
||||||
{
|
{
|
||||||
case FriendsOnlineStatus.All:
|
case OnlineStatus.All:
|
||||||
return Color4.White;
|
return Color4.White;
|
||||||
|
|
||||||
case FriendsOnlineStatus.Online:
|
case OnlineStatus.Online:
|
||||||
return colours.GreenLight;
|
return colours.GreenLight;
|
||||||
|
|
||||||
case FriendsOnlineStatus.Offline:
|
case OnlineStatus.Offline:
|
||||||
return Color4.Black;
|
return Color4.Black;
|
||||||
|
|
||||||
default:
|
default:
|
12
osu.Game/Overlays/Dashboard/Friends/OnlineStatus.cs
Normal file
12
osu.Game/Overlays/Dashboard/Friends/OnlineStatus.cs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
|
{
|
||||||
|
public enum OnlineStatus
|
||||||
|
{
|
||||||
|
All,
|
||||||
|
Online,
|
||||||
|
Offline
|
||||||
|
}
|
||||||
|
}
|
@ -6,7 +6,7 @@ using osu.Framework.Graphics.Containers;
|
|||||||
using osuTK;
|
using osuTK;
|
||||||
using osu.Framework.Bindables;
|
using osu.Framework.Bindables;
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Home.Friends
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
{
|
{
|
||||||
public class UserListToolbar : CompositeDrawable
|
public class UserListToolbar : CompositeDrawable
|
||||||
{
|
{
|
@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
using System.ComponentModel;
|
using System.ComponentModel;
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Home.Friends
|
namespace osu.Game.Overlays.Dashboard.Friends
|
||||||
{
|
{
|
||||||
public class UserSortTabControl : OverlaySortTabControl<UserSortCriteria>
|
public class UserSortTabControl : OverlaySortTabControl<UserSortCriteria>
|
||||||
{
|
{
|
@ -1,25 +0,0 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Home.Friends
|
|
||||||
{
|
|
||||||
public class FriendsBundle
|
|
||||||
{
|
|
||||||
public FriendsOnlineStatus Status { get; }
|
|
||||||
|
|
||||||
public int Count { get; }
|
|
||||||
|
|
||||||
public FriendsBundle(FriendsOnlineStatus status, int count)
|
|
||||||
{
|
|
||||||
Status = status;
|
|
||||||
Count = count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public enum FriendsOnlineStatus
|
|
||||||
{
|
|
||||||
All,
|
|
||||||
Online,
|
|
||||||
Offline
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,26 +0,0 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
||||||
// See the LICENCE file in the repository root for full licence text.
|
|
||||||
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using osu.Game.Users;
|
|
||||||
|
|
||||||
namespace osu.Game.Overlays.Home.Friends
|
|
||||||
{
|
|
||||||
public class FriendsOnlineStatusControl : OverlayStreamControl<FriendsBundle>
|
|
||||||
{
|
|
||||||
protected override OverlayStreamItem<FriendsBundle> CreateStreamItem(FriendsBundle value) => new FriendsOnlineStatusItem(value);
|
|
||||||
|
|
||||||
public void Populate(List<User> users)
|
|
||||||
{
|
|
||||||
var userCount = users.Count;
|
|
||||||
var onlineUsersCount = users.Count(user => user.IsOnline);
|
|
||||||
|
|
||||||
AddItem(new FriendsBundle(FriendsOnlineStatus.All, userCount));
|
|
||||||
AddItem(new FriendsBundle(FriendsOnlineStatus.Online, onlineUsersCount));
|
|
||||||
AddItem(new FriendsBundle(FriendsOnlineStatus.Offline, userCount - onlineUsersCount));
|
|
||||||
|
|
||||||
Current.Value = Items.FirstOrDefault();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -8,16 +8,14 @@ namespace osu.Game.Overlays.Settings
|
|||||||
{
|
{
|
||||||
public class SettingsCheckbox : SettingsItem<bool>
|
public class SettingsCheckbox : SettingsItem<bool>
|
||||||
{
|
{
|
||||||
private OsuCheckbox checkbox;
|
|
||||||
|
|
||||||
private string labelText;
|
private string labelText;
|
||||||
|
|
||||||
protected override Drawable CreateControl() => checkbox = new OsuCheckbox();
|
protected override Drawable CreateControl() => new OsuCheckbox();
|
||||||
|
|
||||||
public override string LabelText
|
public override string LabelText
|
||||||
{
|
{
|
||||||
get => labelText;
|
get => labelText;
|
||||||
set => checkbox.LabelText = labelText = value;
|
set => ((OsuCheckbox)Control).LabelText = labelText = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -33,22 +33,24 @@ namespace osu.Game.Overlays.Settings
|
|||||||
|
|
||||||
protected readonly FillFlowContainer FlowContent;
|
protected readonly FillFlowContainer FlowContent;
|
||||||
|
|
||||||
private SpriteText text;
|
private SpriteText labelText;
|
||||||
|
|
||||||
public bool ShowsDefaultIndicator = true;
|
public bool ShowsDefaultIndicator = true;
|
||||||
|
|
||||||
public virtual string LabelText
|
public virtual string LabelText
|
||||||
{
|
{
|
||||||
get => text?.Text ?? string.Empty;
|
get => labelText?.Text ?? string.Empty;
|
||||||
set
|
set
|
||||||
{
|
{
|
||||||
if (text == null)
|
if (labelText == null)
|
||||||
{
|
{
|
||||||
// construct lazily for cases where the label is not needed (may be provided by the Control).
|
// construct lazily for cases where the label is not needed (may be provided by the Control).
|
||||||
FlowContent.Insert(-1, text = new OsuSpriteText());
|
FlowContent.Insert(-1, labelText = new OsuSpriteText());
|
||||||
|
|
||||||
|
updateDisabled();
|
||||||
}
|
}
|
||||||
|
|
||||||
text.Text = value;
|
labelText.Text = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -96,13 +98,19 @@ namespace osu.Game.Overlays.Settings
|
|||||||
if (controlWithCurrent != null)
|
if (controlWithCurrent != null)
|
||||||
{
|
{
|
||||||
controlWithCurrent.Current.ValueChanged += _ => SettingChanged?.Invoke();
|
controlWithCurrent.Current.ValueChanged += _ => SettingChanged?.Invoke();
|
||||||
controlWithCurrent.Current.DisabledChanged += disabled => { Colour = disabled ? Color4.Gray : Color4.White; };
|
controlWithCurrent.Current.DisabledChanged += _ => updateDisabled();
|
||||||
|
|
||||||
if (ShowsDefaultIndicator)
|
if (ShowsDefaultIndicator)
|
||||||
restoreDefaultButton.Bindable = controlWithCurrent.Current;
|
restoreDefaultButton.Bindable = controlWithCurrent.Current;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void updateDisabled()
|
||||||
|
{
|
||||||
|
if (labelText != null)
|
||||||
|
labelText.Alpha = controlWithCurrent.Current.Disabled ? 0.3f : 1;
|
||||||
|
}
|
||||||
|
|
||||||
private class RestoreDefaultValueButton : Container, IHasTooltip
|
private class RestoreDefaultValueButton : Container, IHasTooltip
|
||||||
{
|
{
|
||||||
private Bindable<T> bindable;
|
private Bindable<T> bindable;
|
||||||
|
17
osu.Game/Rulesets/Mods/ModRandom.cs
Normal file
17
osu.Game/Rulesets/Mods/ModRandom.cs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||||
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
|
using osu.Framework.Graphics.Sprites;
|
||||||
|
using osu.Game.Graphics;
|
||||||
|
|
||||||
|
namespace osu.Game.Rulesets.Mods
|
||||||
|
{
|
||||||
|
public abstract class ModRandom : Mod
|
||||||
|
{
|
||||||
|
public override string Name => "Random";
|
||||||
|
public override string Acronym => "RD";
|
||||||
|
public override ModType Type => ModType.Conversion;
|
||||||
|
public override IconUsage? Icon => OsuIcon.Dice;
|
||||||
|
public override double ScoreMultiplier => 1;
|
||||||
|
}
|
||||||
|
}
|
@ -158,6 +158,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
dependencies.Cache(textureStore);
|
dependencies.Cache(textureStore);
|
||||||
|
|
||||||
localSampleStore = dependencies.Get<AudioManager>().GetSampleStore(new NamespacedResourceStore<byte[]>(resources, "Samples"));
|
localSampleStore = dependencies.Get<AudioManager>().GetSampleStore(new NamespacedResourceStore<byte[]>(resources, "Samples"));
|
||||||
|
localSampleStore.PlaybackConcurrency = OsuGameBase.SAMPLE_CONCURRENCY;
|
||||||
dependencies.CacheAs<ISampleStore>(new FallbackSampleStore(localSampleStore, dependencies.Get<ISampleStore>()));
|
dependencies.CacheAs<ISampleStore>(new FallbackSampleStore(localSampleStore, dependencies.Get<ISampleStore>()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -279,6 +279,11 @@ namespace osu.Game.Screens.Edit.Compose.Components
|
|||||||
handleMouseInput(e.ScreenSpaceMousePosition);
|
handleMouseInput(e.ScreenSpaceMousePosition);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override void OnDragEnd(DragEndEvent e)
|
||||||
|
{
|
||||||
|
handleMouseInput(e.ScreenSpaceMousePosition);
|
||||||
|
}
|
||||||
|
|
||||||
private void handleMouseInput(Vector2 screenSpaceMousePosition)
|
private void handleMouseInput(Vector2 screenSpaceMousePosition)
|
||||||
{
|
{
|
||||||
// copied from SliderBar so we can do custom spacing logic.
|
// copied from SliderBar so we can do custom spacing logic.
|
||||||
|
@ -44,7 +44,6 @@ namespace osu.Game.Screens.Play
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
drawableStoryboard = storyboard.CreateDrawable();
|
drawableStoryboard = storyboard.CreateDrawable();
|
||||||
drawableStoryboard.Masking = true;
|
|
||||||
|
|
||||||
if (async)
|
if (async)
|
||||||
LoadComponentAsync(drawableStoryboard, Add);
|
LoadComponentAsync(drawableStoryboard, Add);
|
||||||
|
@ -333,8 +333,7 @@ namespace osu.Game.Screens.Select
|
|||||||
else
|
else
|
||||||
set = visibleSets.ElementAt(RNG.Next(visibleSets.Count));
|
set = visibleSets.ElementAt(RNG.Next(visibleSets.Count));
|
||||||
|
|
||||||
var visibleBeatmaps = set.Beatmaps.Where(s => !s.Filtered.Value).ToList();
|
select(set);
|
||||||
select(visibleBeatmaps[RNG.Next(visibleBeatmaps.Count)]);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -751,13 +750,17 @@ namespace osu.Game.Screens.Select
|
|||||||
|
|
||||||
public CarouselRoot(BeatmapCarousel carousel)
|
public CarouselRoot(BeatmapCarousel carousel)
|
||||||
{
|
{
|
||||||
|
// root should always remain selected. if not, PerformSelection will not be called.
|
||||||
|
State.Value = CarouselItemState.Selected;
|
||||||
|
State.ValueChanged += state => State.Value = CarouselItemState.Selected;
|
||||||
|
|
||||||
this.carousel = carousel;
|
this.carousel = carousel;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void PerformSelection()
|
protected override void PerformSelection()
|
||||||
{
|
{
|
||||||
if (LastSelected == null)
|
if (LastSelected == null || LastSelected.Filtered.Value)
|
||||||
carousel.SelectNextRandom();
|
carousel?.SelectNextRandom();
|
||||||
else
|
else
|
||||||
base.PerformSelection();
|
base.PerformSelection();
|
||||||
}
|
}
|
||||||
|
@ -104,7 +104,8 @@ namespace osu.Game.Screens.Select.Carousel
|
|||||||
|
|
||||||
private void updateSelected(CarouselItem newSelection)
|
private void updateSelected(CarouselItem newSelection)
|
||||||
{
|
{
|
||||||
LastSelected = newSelection;
|
if (newSelection != null)
|
||||||
|
LastSelected = newSelection;
|
||||||
updateSelectedIndex();
|
updateSelectedIndex();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,6 +16,7 @@ using Container = osu.Framework.Graphics.Containers.Container;
|
|||||||
using osu.Framework.Graphics.Shapes;
|
using osu.Framework.Graphics.Shapes;
|
||||||
using osu.Game.Configuration;
|
using osu.Game.Configuration;
|
||||||
using osu.Game.Rulesets;
|
using osu.Game.Rulesets;
|
||||||
|
using osu.Framework.Input.Events;
|
||||||
|
|
||||||
namespace osu.Game.Screens.Select
|
namespace osu.Game.Screens.Select
|
||||||
{
|
{
|
||||||
@ -136,6 +137,8 @@ namespace osu.Game.Screens.Select
|
|||||||
|
|
||||||
public void Deactivate()
|
public void Deactivate()
|
||||||
{
|
{
|
||||||
|
searchTextBox.ReadOnly = true;
|
||||||
|
|
||||||
searchTextBox.HoldFocus = false;
|
searchTextBox.HoldFocus = false;
|
||||||
if (searchTextBox.HasFocus)
|
if (searchTextBox.HasFocus)
|
||||||
GetContainingInputManager().ChangeFocus(searchTextBox);
|
GetContainingInputManager().ChangeFocus(searchTextBox);
|
||||||
@ -143,6 +146,7 @@ namespace osu.Game.Screens.Select
|
|||||||
|
|
||||||
public void Activate()
|
public void Activate()
|
||||||
{
|
{
|
||||||
|
searchTextBox.ReadOnly = false;
|
||||||
searchTextBox.HoldFocus = true;
|
searchTextBox.HoldFocus = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -184,5 +188,9 @@ namespace osu.Game.Screens.Select
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void updateCriteria() => FilterChanged?.Invoke(CreateCriteria());
|
private void updateCriteria() => FilterChanged?.Invoke(CreateCriteria());
|
||||||
|
|
||||||
|
protected override bool OnClick(ClickEvent e) => true;
|
||||||
|
|
||||||
|
protected override bool OnHover(HoverEvent e) => true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -107,5 +107,7 @@ namespace osu.Game.Screens.Select
|
|||||||
protected override bool OnMouseDown(MouseDownEvent e) => true;
|
protected override bool OnMouseDown(MouseDownEvent e) => true;
|
||||||
|
|
||||||
protected override bool OnClick(ClickEvent e) => true;
|
protected override bool OnClick(ClickEvent e) => true;
|
||||||
|
|
||||||
|
protected override bool OnHover(HoverEvent e) => true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -56,6 +56,7 @@ namespace osu.Game.Screens.Select
|
|||||||
switch (e.Key)
|
switch (e.Key)
|
||||||
{
|
{
|
||||||
case Key.Enter:
|
case Key.Enter:
|
||||||
|
case Key.KeypadEnter:
|
||||||
// this is a special hard-coded case; we can't rely on OnPressed (of SongSelect) as GlobalActionContainer is
|
// this is a special hard-coded case; we can't rely on OnPressed (of SongSelect) as GlobalActionContainer is
|
||||||
// matching with exact modifier consideration (so Ctrl+Enter would be ignored).
|
// matching with exact modifier consideration (so Ctrl+Enter would be ignored).
|
||||||
FinaliseSelection();
|
FinaliseSelection();
|
||||||
|
@ -150,6 +150,7 @@ namespace osu.Game.Screens.Select
|
|||||||
},
|
},
|
||||||
Child = Carousel = new BeatmapCarousel
|
Child = Carousel = new BeatmapCarousel
|
||||||
{
|
{
|
||||||
|
AllowSelection = false, // delay any selection until our bindables are ready to make a good choice.
|
||||||
Anchor = Anchor.CentreRight,
|
Anchor = Anchor.CentreRight,
|
||||||
Origin = Anchor.CentreRight,
|
Origin = Anchor.CentreRight,
|
||||||
RelativeSizeAxes = Axes.Both,
|
RelativeSizeAxes = Axes.Both,
|
||||||
@ -655,6 +656,8 @@ namespace osu.Game.Screens.Select
|
|||||||
{
|
{
|
||||||
bindBindables();
|
bindBindables();
|
||||||
|
|
||||||
|
Carousel.AllowSelection = true;
|
||||||
|
|
||||||
// If a selection was already obtained, do not attempt to update the selected beatmap.
|
// If a selection was already obtained, do not attempt to update the selected beatmap.
|
||||||
if (Carousel.SelectedBeatmapSet != null)
|
if (Carousel.SelectedBeatmapSet != null)
|
||||||
return;
|
return;
|
||||||
|
@ -52,7 +52,11 @@ namespace osu.Game.Skinning
|
|||||||
|
|
||||||
if (storage != null)
|
if (storage != null)
|
||||||
{
|
{
|
||||||
Samples = audioManager?.GetSampleStore(storage);
|
var samples = audioManager?.GetSampleStore(storage);
|
||||||
|
if (samples != null)
|
||||||
|
samples.PlaybackConcurrency = OsuGameBase.SAMPLE_CONCURRENCY;
|
||||||
|
|
||||||
|
Samples = samples;
|
||||||
Textures = new TextureStore(new TextureLoaderStore(storage));
|
Textures = new TextureStore(new TextureLoaderStore(storage));
|
||||||
|
|
||||||
(storage as ResourceStore<byte[]>)?.AddExtension("ogg");
|
(storage as ResourceStore<byte[]>)?.AddExtension("ogg");
|
||||||
|
@ -75,7 +75,7 @@ namespace osu.Game.Storyboards.Drawables
|
|||||||
private void updateLayerVisibility()
|
private void updateLayerVisibility()
|
||||||
{
|
{
|
||||||
foreach (var layer in Children)
|
foreach (var layer in Children)
|
||||||
layer.Enabled = passing ? layer.Layer.EnabledWhenPassing : layer.Layer.EnabledWhenFailing;
|
layer.Enabled = passing ? layer.Layer.VisibleWhenPassing : layer.Layer.VisibleWhenFailing;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,8 @@ namespace osu.Game.Storyboards.Drawables
|
|||||||
RelativeSizeAxes = Axes.Both;
|
RelativeSizeAxes = Axes.Both;
|
||||||
Anchor = Anchor.Centre;
|
Anchor = Anchor.Centre;
|
||||||
Origin = Anchor.Centre;
|
Origin = Anchor.Centre;
|
||||||
Enabled = layer.EnabledWhenPassing;
|
Enabled = layer.VisibleWhenPassing;
|
||||||
|
Masking = layer.Masking;
|
||||||
}
|
}
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
|
@ -23,8 +23,8 @@ namespace osu.Game.Storyboards
|
|||||||
{
|
{
|
||||||
layers.Add("Video", new StoryboardLayer("Video", 4));
|
layers.Add("Video", new StoryboardLayer("Video", 4));
|
||||||
layers.Add("Background", new StoryboardLayer("Background", 3));
|
layers.Add("Background", new StoryboardLayer("Background", 3));
|
||||||
layers.Add("Fail", new StoryboardLayer("Fail", 2) { EnabledWhenPassing = false, });
|
layers.Add("Fail", new StoryboardLayer("Fail", 2) { VisibleWhenPassing = false, });
|
||||||
layers.Add("Pass", new StoryboardLayer("Pass", 1) { EnabledWhenFailing = false, });
|
layers.Add("Pass", new StoryboardLayer("Pass", 1) { VisibleWhenFailing = false, });
|
||||||
layers.Add("Foreground", new StoryboardLayer("Foreground", 0));
|
layers.Add("Foreground", new StoryboardLayer("Foreground", 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,17 +8,23 @@ namespace osu.Game.Storyboards
|
|||||||
{
|
{
|
||||||
public class StoryboardLayer
|
public class StoryboardLayer
|
||||||
{
|
{
|
||||||
public string Name;
|
public readonly string Name;
|
||||||
public int Depth;
|
|
||||||
public bool EnabledWhenPassing = true;
|
public readonly int Depth;
|
||||||
public bool EnabledWhenFailing = true;
|
|
||||||
|
public readonly bool Masking;
|
||||||
|
|
||||||
|
public bool VisibleWhenPassing = true;
|
||||||
|
|
||||||
|
public bool VisibleWhenFailing = true;
|
||||||
|
|
||||||
public List<IStoryboardElement> Elements = new List<IStoryboardElement>();
|
public List<IStoryboardElement> Elements = new List<IStoryboardElement>();
|
||||||
|
|
||||||
public StoryboardLayer(string name, int depth)
|
public StoryboardLayer(string name, int depth, bool masking = true)
|
||||||
{
|
{
|
||||||
Name = name;
|
Name = name;
|
||||||
Depth = depth;
|
Depth = depth;
|
||||||
|
Masking = masking;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Add(IStoryboardElement element)
|
public void Add(IStoryboardElement element)
|
||||||
|
@ -14,7 +14,7 @@ using osuTK.Graphics;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual
|
namespace osu.Game.Tests.Visual
|
||||||
{
|
{
|
||||||
public abstract class ManualInputManagerTestScene : OsuTestScene
|
public abstract class OsuManualInputManagerTestScene : OsuTestScene
|
||||||
{
|
{
|
||||||
protected override Container<Drawable> Content => content;
|
protected override Container<Drawable> Content => content;
|
||||||
private readonly Container content;
|
private readonly Container content;
|
||||||
@ -24,7 +24,7 @@ namespace osu.Game.Tests.Visual
|
|||||||
private readonly TriangleButton buttonTest;
|
private readonly TriangleButton buttonTest;
|
||||||
private readonly TriangleButton buttonLocal;
|
private readonly TriangleButton buttonLocal;
|
||||||
|
|
||||||
protected ManualInputManagerTestScene()
|
protected OsuManualInputManagerTestScene()
|
||||||
{
|
{
|
||||||
base.Content.AddRange(new Drawable[]
|
base.Content.AddRange(new Drawable[]
|
||||||
{
|
{
|
@ -11,7 +11,7 @@ namespace osu.Game.Tests.Visual
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// A test case which can be used to test a screen (that relies on OnEntering being called to execute startup instructions).
|
/// A test case which can be used to test a screen (that relies on OnEntering being called to execute startup instructions).
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public abstract class ScreenTestScene : ManualInputManagerTestScene
|
public abstract class ScreenTestScene : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
protected readonly OsuScreenStack Stack;
|
protected readonly OsuScreenStack Stack;
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ using osu.Game.Rulesets.Edit;
|
|||||||
|
|
||||||
namespace osu.Game.Tests.Visual
|
namespace osu.Game.Tests.Visual
|
||||||
{
|
{
|
||||||
public abstract class SelectionBlueprintTestScene : ManualInputManagerTestScene
|
public abstract class SelectionBlueprintTestScene : OsuManualInputManagerTestScene
|
||||||
{
|
{
|
||||||
protected override Container<Drawable> Content => content ?? base.Content;
|
protected override Container<Drawable> Content => content ?? base.Content;
|
||||||
private readonly Container content;
|
private readonly Container content;
|
||||||
|
@ -69,6 +69,9 @@ namespace osu.Game.Users
|
|||||||
[JsonProperty(@"support_level")]
|
[JsonProperty(@"support_level")]
|
||||||
public int SupportLevel;
|
public int SupportLevel;
|
||||||
|
|
||||||
|
[JsonProperty(@"current_mode_rank")]
|
||||||
|
public int? CurrentModeRank;
|
||||||
|
|
||||||
[JsonProperty(@"is_gmt")]
|
[JsonProperty(@"is_gmt")]
|
||||||
public bool IsGMT;
|
public bool IsGMT;
|
||||||
|
|
||||||
|
@ -99,6 +99,9 @@ namespace osu.Game.Users
|
|||||||
{
|
{
|
||||||
base.LoadComplete();
|
base.LoadComplete();
|
||||||
Status.TriggerChange();
|
Status.TriggerChange();
|
||||||
|
|
||||||
|
// Colour should be applied immediately on first load.
|
||||||
|
statusIcon.FinishTransforms();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override bool OnHover(HoverEvent e)
|
protected override bool OnHover(HoverEvent e)
|
||||||
|
@ -24,7 +24,7 @@
|
|||||||
<PackageReference Include="Newtonsoft.Json" Version="12.0.3" />
|
<PackageReference Include="Newtonsoft.Json" Version="12.0.3" />
|
||||||
<PackageReference Include="ppy.osu.Game.Resources" Version="2020.315.0" />
|
<PackageReference Include="ppy.osu.Game.Resources" Version="2020.315.0" />
|
||||||
<PackageReference Include="ppy.osu.Framework" Version="2020.319.0" />
|
<PackageReference Include="ppy.osu.Framework" Version="2020.319.0" />
|
||||||
<PackageReference Include="Sentry" Version="2.1.0" />
|
<PackageReference Include="Sentry" Version="2.1.1" />
|
||||||
<PackageReference Include="SharpCompress" Version="0.24.0" />
|
<PackageReference Include="SharpCompress" Version="0.24.0" />
|
||||||
<PackageReference Include="NUnit" Version="3.12.0" />
|
<PackageReference Include="NUnit" Version="3.12.0" />
|
||||||
<PackageReference Include="System.ComponentModel.Annotations" Version="4.7.0" />
|
<PackageReference Include="System.ComponentModel.Annotations" Version="4.7.0" />
|
||||||
|
Loading…
Reference in New Issue
Block a user