mirror of
https://github.com/ppy/osu.git
synced 2024-11-11 09:17:51 +08:00
Merge branch 'master' into score-statistics-updates-working-2
This commit is contained in:
commit
59efd22ba2
@ -1,111 +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.
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
using System.Linq;
|
|
||||||
using NUnit.Framework;
|
|
||||||
using osu.Framework.Bindables;
|
|
||||||
using osu.Framework.Graphics;
|
|
||||||
using osu.Framework.Graphics.Containers;
|
|
||||||
using osu.Framework.Testing;
|
|
||||||
using osu.Framework.Utils;
|
|
||||||
using osu.Game.Beatmaps;
|
|
||||||
using osu.Game.Beatmaps.ControlPoints;
|
|
||||||
using osu.Game.Rulesets.Catch.Judgements;
|
|
||||||
using osu.Game.Rulesets.Catch.Objects;
|
|
||||||
using osu.Game.Rulesets.Catch.Objects.Drawables;
|
|
||||||
using osu.Game.Rulesets.Catch.Skinning;
|
|
||||||
using osu.Game.Rulesets.Catch.UI;
|
|
||||||
using osu.Game.Skinning;
|
|
||||||
using osu.Game.Tests.Visual;
|
|
||||||
using Direction = osu.Game.Rulesets.Catch.UI.Direction;
|
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.Catch.Tests
|
|
||||||
{
|
|
||||||
public partial class TestSceneCatchSkinConfiguration : OsuTestScene
|
|
||||||
{
|
|
||||||
private Catcher catcher;
|
|
||||||
|
|
||||||
private readonly Container container;
|
|
||||||
|
|
||||||
public TestSceneCatchSkinConfiguration()
|
|
||||||
{
|
|
||||||
Add(container = new Container { RelativeSizeAxes = Axes.Both });
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestCase(false)]
|
|
||||||
[TestCase(true)]
|
|
||||||
public void TestCatcherPlateFlipping(bool flip)
|
|
||||||
{
|
|
||||||
AddStep("setup catcher", () =>
|
|
||||||
{
|
|
||||||
var skin = new TestSkin { FlipCatcherPlate = flip };
|
|
||||||
container.Child = new SkinProvidingContainer(skin)
|
|
||||||
{
|
|
||||||
Child = catcher = new Catcher(new DroppedObjectContainer())
|
|
||||||
{
|
|
||||||
Anchor = Anchor.Centre
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
Fruit fruit = new Fruit();
|
|
||||||
|
|
||||||
AddStep("catch fruit", () => catchFruit(fruit, 20));
|
|
||||||
|
|
||||||
float position = 0;
|
|
||||||
|
|
||||||
AddStep("record fruit position", () => position = getCaughtObjectPosition(fruit));
|
|
||||||
|
|
||||||
AddStep("face left", () => catcher.VisualDirection = Direction.Left);
|
|
||||||
|
|
||||||
if (flip)
|
|
||||||
AddAssert("fruit position changed", () => !Precision.AlmostEquals(getCaughtObjectPosition(fruit), position));
|
|
||||||
else
|
|
||||||
AddAssert("fruit position unchanged", () => Precision.AlmostEquals(getCaughtObjectPosition(fruit), position));
|
|
||||||
|
|
||||||
AddStep("face right", () => catcher.VisualDirection = Direction.Right);
|
|
||||||
|
|
||||||
AddAssert("fruit position restored", () => Precision.AlmostEquals(getCaughtObjectPosition(fruit), position));
|
|
||||||
}
|
|
||||||
|
|
||||||
private float getCaughtObjectPosition(Fruit fruit)
|
|
||||||
{
|
|
||||||
var caughtObject = catcher.ChildrenOfType<CaughtObject>().Single(c => c.HitObject == fruit);
|
|
||||||
return caughtObject.Parent!.ToSpaceOfOtherDrawable(caughtObject.Position, catcher).X;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void catchFruit(Fruit fruit, float x)
|
|
||||||
{
|
|
||||||
fruit.ApplyDefaults(new ControlPointInfo(), new BeatmapDifficulty());
|
|
||||||
var drawableFruit = new DrawableFruit(fruit) { X = x };
|
|
||||||
var judgement = fruit.CreateJudgement();
|
|
||||||
catcher.OnNewResult(drawableFruit, new CatchJudgementResult(fruit, judgement)
|
|
||||||
{
|
|
||||||
Type = judgement.MaxResult
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private class TestSkin : TrianglesSkin
|
|
||||||
{
|
|
||||||
public bool FlipCatcherPlate { get; set; }
|
|
||||||
|
|
||||||
public TestSkin()
|
|
||||||
: base(null!)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public override IBindable<TValue> GetConfig<TLookup, TValue>(TLookup lookup)
|
|
||||||
{
|
|
||||||
if (lookup is CatchSkinConfiguration config)
|
|
||||||
{
|
|
||||||
if (config == CatchSkinConfiguration.FlipCatcherPlate)
|
|
||||||
return SkinUtils.As<TValue>(new Bindable<bool>(FlipCatcherPlate));
|
|
||||||
}
|
|
||||||
|
|
||||||
return base.GetConfig<TLookup, TValue>(lookup);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,13 +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.Catch.Skinning
|
|
||||||
{
|
|
||||||
public enum CatchSkinConfiguration
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Whether the contents of the catcher plate should be visually flipped when the catcher direction is changed.
|
|
||||||
/// </summary>
|
|
||||||
FlipCatcherPlate
|
|
||||||
}
|
|
||||||
}
|
|
@ -122,19 +122,6 @@ namespace osu.Game.Rulesets.Catch.Skinning.Legacy
|
|||||||
|
|
||||||
result.Value = LegacyColourCompatibility.DisallowZeroAlpha(result.Value);
|
result.Value = LegacyColourCompatibility.DisallowZeroAlpha(result.Value);
|
||||||
return (IBindable<TValue>)result;
|
return (IBindable<TValue>)result;
|
||||||
|
|
||||||
case CatchSkinConfiguration config:
|
|
||||||
switch (config)
|
|
||||||
{
|
|
||||||
case CatchSkinConfiguration.FlipCatcherPlate:
|
|
||||||
// Don't flip catcher plate contents if the catcher is provided by this legacy skin.
|
|
||||||
if (GetDrawableComponent(new CatchSkinComponentLookup(CatchSkinComponents.Catcher)) != null)
|
|
||||||
return (IBindable<TValue>)new Bindable<bool>();
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return base.GetConfig<TLookup, TValue>(lookup);
|
return base.GetConfig<TLookup, TValue>(lookup);
|
||||||
|
@ -112,11 +112,6 @@ namespace osu.Game.Rulesets.Catch.UI
|
|||||||
|
|
||||||
public Vector2 BodyScale => Scale * body.Scale;
|
public Vector2 BodyScale => Scale * body.Scale;
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Whether the contents of the catcher plate should be visually flipped when the catcher direction is changed.
|
|
||||||
/// </summary>
|
|
||||||
private bool flipCatcherPlate;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Width of the area that can be used to attempt catches during gameplay.
|
/// Width of the area that can be used to attempt catches during gameplay.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -339,8 +334,6 @@ namespace osu.Game.Rulesets.Catch.UI
|
|||||||
skin.GetConfig<CatchSkinColour, Color4>(CatchSkinColour.HyperDash)?.Value ??
|
skin.GetConfig<CatchSkinColour, Color4>(CatchSkinColour.HyperDash)?.Value ??
|
||||||
DEFAULT_HYPER_DASH_COLOUR;
|
DEFAULT_HYPER_DASH_COLOUR;
|
||||||
|
|
||||||
flipCatcherPlate = skin.GetConfig<CatchSkinConfiguration, bool>(CatchSkinConfiguration.FlipCatcherPlate)?.Value ?? true;
|
|
||||||
|
|
||||||
runHyperDashStateTransition(HyperDashing);
|
runHyperDashStateTransition(HyperDashing);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -352,8 +345,7 @@ namespace osu.Game.Rulesets.Catch.UI
|
|||||||
|
|
||||||
body.Scale = scaleFromDirection;
|
body.Scale = scaleFromDirection;
|
||||||
// Inverse of catcher scale is applied here, as catcher gets scaled by circle size and so do the incoming fruit.
|
// Inverse of catcher scale is applied here, as catcher gets scaled by circle size and so do the incoming fruit.
|
||||||
caughtObjectContainer.Scale = (1 / Scale.X) * (flipCatcherPlate ? scaleFromDirection : Vector2.One);
|
caughtObjectContainer.Scale = new Vector2(1 / Scale.X);
|
||||||
hitExplosionContainer.Scale = flipCatcherPlate ? scaleFromDirection : Vector2.One;
|
|
||||||
|
|
||||||
// Correct overshooting.
|
// Correct overshooting.
|
||||||
if ((hyperDashDirection > 0 && hyperDashTargetPosition < X) ||
|
if ((hyperDashDirection > 0 && hyperDashTargetPosition < X) ||
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
// 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;
|
||||||
@ -38,12 +38,18 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
private ReplayState<OsuAction> state = null!;
|
private ReplayState<OsuAction> state = null!;
|
||||||
private double lastStateChangeTime;
|
private double lastStateChangeTime;
|
||||||
|
|
||||||
|
private DrawableOsuRuleset ruleset = null!;
|
||||||
|
private IPressHandler pressHandler = null!;
|
||||||
|
|
||||||
private bool hasReplay;
|
private bool hasReplay;
|
||||||
|
private bool legacyReplay;
|
||||||
|
|
||||||
public void ApplyToDrawableRuleset(DrawableRuleset<OsuHitObject> drawableRuleset)
|
public void ApplyToDrawableRuleset(DrawableRuleset<OsuHitObject> drawableRuleset)
|
||||||
{
|
{
|
||||||
|
ruleset = (DrawableOsuRuleset)drawableRuleset;
|
||||||
|
|
||||||
// grab the input manager for future use.
|
// grab the input manager for future use.
|
||||||
osuInputManager = ((DrawableOsuRuleset)drawableRuleset).KeyBindingInputManager;
|
osuInputManager = ruleset.KeyBindingInputManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void ApplyToPlayer(Player player)
|
public void ApplyToPlayer(Player player)
|
||||||
@ -51,15 +57,22 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
if (osuInputManager.ReplayInputHandler != null)
|
if (osuInputManager.ReplayInputHandler != null)
|
||||||
{
|
{
|
||||||
hasReplay = true;
|
hasReplay = true;
|
||||||
|
|
||||||
|
Debug.Assert(ruleset.ReplayScore != null);
|
||||||
|
legacyReplay = ruleset.ReplayScore.ScoreInfo.IsLegacyScore;
|
||||||
|
|
||||||
|
pressHandler = legacyReplay ? new LegacyReplayPressHandler(this) : new PressHandler(this);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pressHandler = new PressHandler(this);
|
||||||
osuInputManager.AllowGameplayInputs = false;
|
osuInputManager.AllowGameplayInputs = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Update(Playfield playfield)
|
public void Update(Playfield playfield)
|
||||||
{
|
{
|
||||||
if (hasReplay)
|
if (hasReplay && !legacyReplay)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bool requiresHold = false;
|
bool requiresHold = false;
|
||||||
@ -132,11 +145,62 @@ namespace osu.Game.Rulesets.Osu.Mods
|
|||||||
|
|
||||||
if (down)
|
if (down)
|
||||||
{
|
{
|
||||||
state.PressedActions.Add(wasLeft ? OsuAction.LeftButton : OsuAction.RightButton);
|
pressHandler.HandlePress(wasLeft);
|
||||||
wasLeft = !wasLeft;
|
wasLeft = !wasLeft;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pressHandler.HandleRelease(wasLeft);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
state.Apply(osuInputManager.CurrentState, osuInputManager);
|
private interface IPressHandler
|
||||||
|
{
|
||||||
|
void HandlePress(bool wasLeft);
|
||||||
|
void HandleRelease(bool wasLeft);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class PressHandler : IPressHandler
|
||||||
|
{
|
||||||
|
private readonly OsuModRelax mod;
|
||||||
|
|
||||||
|
public PressHandler(OsuModRelax mod)
|
||||||
|
{
|
||||||
|
this.mod = mod;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void HandlePress(bool wasLeft)
|
||||||
|
{
|
||||||
|
mod.state.PressedActions.Add(wasLeft ? OsuAction.LeftButton : OsuAction.RightButton);
|
||||||
|
mod.state.Apply(mod.osuInputManager.CurrentState, mod.osuInputManager);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void HandleRelease(bool wasLeft)
|
||||||
|
{
|
||||||
|
mod.state.Apply(mod.osuInputManager.CurrentState, mod.osuInputManager);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// legacy replays do not contain key-presses with Relax mod, so they need to be triggered by themselves.
|
||||||
|
private class LegacyReplayPressHandler : IPressHandler
|
||||||
|
{
|
||||||
|
private readonly OsuModRelax mod;
|
||||||
|
|
||||||
|
public LegacyReplayPressHandler(OsuModRelax mod)
|
||||||
|
{
|
||||||
|
this.mod = mod;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void HandlePress(bool wasLeft)
|
||||||
|
{
|
||||||
|
mod.osuInputManager.KeyBindingContainer.TriggerPressed(wasLeft ? OsuAction.LeftButton : OsuAction.RightButton);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void HandleRelease(bool wasLeft)
|
||||||
|
{
|
||||||
|
// this intentionally releases right when `wasLeft` is true because `wasLeft` is set at point of press and not at point of release
|
||||||
|
mod.osuInputManager.KeyBindingContainer.TriggerReleased(wasLeft ? OsuAction.RightButton : OsuAction.LeftButton);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -454,5 +454,111 @@ namespace osu.Game.Tests.NonVisual.Filtering
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static readonly object[] correct_date_query_examples =
|
||||||
|
{
|
||||||
|
new object[] { "600" },
|
||||||
|
new object[] { "0.5s" },
|
||||||
|
new object[] { "120m" },
|
||||||
|
new object[] { "48h120s" },
|
||||||
|
new object[] { "10y24M" },
|
||||||
|
new object[] { "10y60d120s" },
|
||||||
|
new object[] { "0y0M2d" },
|
||||||
|
new object[] { "1y1M2d" }
|
||||||
|
};
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
[TestCaseSource(nameof(correct_date_query_examples))]
|
||||||
|
public void TestValidDateQueries(string dateQuery)
|
||||||
|
{
|
||||||
|
string query = $"played<{dateQuery} time";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.AreEqual(true, filterCriteria.LastPlayed.HasFilter);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static readonly object[] incorrect_date_query_examples =
|
||||||
|
{
|
||||||
|
new object[] { ".5s" },
|
||||||
|
new object[] { "7m27" },
|
||||||
|
new object[] { "7m7m7m" },
|
||||||
|
new object[] { "5s6m" },
|
||||||
|
new object[] { "7d7y" },
|
||||||
|
new object[] { "0:3:6" },
|
||||||
|
new object[] { "0:3:" },
|
||||||
|
new object[] { "\"three days\"" },
|
||||||
|
new object[] { "0.1y0.1M2d" },
|
||||||
|
new object[] { "0.99y0.99M2d" },
|
||||||
|
new object[] { string.Empty }
|
||||||
|
};
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
[TestCaseSource(nameof(incorrect_date_query_examples))]
|
||||||
|
public void TestInvalidDateQueries(string dateQuery)
|
||||||
|
{
|
||||||
|
string query = $"played<{dateQuery} time";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.AreEqual(false, filterCriteria.LastPlayed.HasFilter);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestGreaterDateQuery()
|
||||||
|
{
|
||||||
|
const string query = "played>50";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Max, Is.Not.Null);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Min, Is.Null);
|
||||||
|
// the parser internally references `DateTimeOffset.Now`, so to not make things too annoying for tests, just assume some tolerance
|
||||||
|
// (irrelevant in proportion to the actual filter proscribed).
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Max, Is.EqualTo(DateTimeOffset.Now.AddDays(-50)).Within(TimeSpan.FromSeconds(5)));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestLowerDateQuery()
|
||||||
|
{
|
||||||
|
const string query = "played<50";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Max, Is.Null);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Min, Is.Not.Null);
|
||||||
|
// the parser internally references `DateTimeOffset.Now`, so to not make things too annoying for tests, just assume some tolerance
|
||||||
|
// (irrelevant in proportion to the actual filter proscribed).
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Min, Is.EqualTo(DateTimeOffset.Now.AddDays(-50)).Within(TimeSpan.FromSeconds(5)));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestBothSidesDateQuery()
|
||||||
|
{
|
||||||
|
const string query = "played>3M played<1y6M";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Min, Is.Not.Null);
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Max, Is.Not.Null);
|
||||||
|
// the parser internally references `DateTimeOffset.Now`, so to not make things too annoying for tests, just assume some tolerance
|
||||||
|
// (irrelevant in proportion to the actual filter proscribed).
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Min, Is.EqualTo(DateTimeOffset.Now.AddYears(-1).AddMonths(-6)).Within(TimeSpan.FromSeconds(5)));
|
||||||
|
Assert.That(filterCriteria.LastPlayed.Max, Is.EqualTo(DateTimeOffset.Now.AddMonths(-3)).Within(TimeSpan.FromSeconds(5)));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestEqualDateQuery()
|
||||||
|
{
|
||||||
|
const string query = "played=50";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.AreEqual(false, filterCriteria.LastPlayed.HasFilter);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOutOfRangeDateQuery()
|
||||||
|
{
|
||||||
|
const string query = "played<10000y";
|
||||||
|
var filterCriteria = new FilterCriteria();
|
||||||
|
FilterQueryParser.ApplyQueries(filterCriteria, query);
|
||||||
|
Assert.AreEqual(true, filterCriteria.LastPlayed.HasFilter);
|
||||||
|
Assert.AreEqual(DateTimeOffset.MinValue.AddMilliseconds(1), filterCriteria.LastPlayed.Min);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,13 +43,13 @@ namespace osu.Game.Tests.Rulesets
|
|||||||
|
|
||||||
AddStep("setup provider", () =>
|
AddStep("setup provider", () =>
|
||||||
{
|
{
|
||||||
var rulesetSkinProvider = new RulesetSkinProvidingContainer(Ruleset.Value.CreateInstance(), Beatmap.Value.Beatmap, Beatmap.Value.Skin);
|
requester = new SkinRequester();
|
||||||
|
|
||||||
rulesetSkinProvider.Add(requester = new SkinRequester());
|
|
||||||
|
|
||||||
requester.OnLoadAsync += () => textureOnLoad = requester.GetTexture("test-image");
|
requester.OnLoadAsync += () => textureOnLoad = requester.GetTexture("test-image");
|
||||||
|
|
||||||
Child = rulesetSkinProvider;
|
Child = new RulesetSkinProvidingContainer(Ruleset.Value.CreateInstance(), Beatmap.Value.Beatmap, Beatmap.Value.Skin)
|
||||||
|
{
|
||||||
|
requester
|
||||||
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
AddAssert("requester got correct initial texture", () => textureOnLoad != null);
|
AddAssert("requester got correct initial texture", () => textureOnLoad != null);
|
||||||
|
@ -986,6 +986,29 @@ namespace osu.Game.Tests.Visual.Navigation
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestPresentBeatmapAfterDeletion()
|
||||||
|
{
|
||||||
|
BeatmapSetInfo beatmap = null;
|
||||||
|
|
||||||
|
Screens.Select.SongSelect songSelect = null;
|
||||||
|
PushAndConfirm(() => songSelect = new TestPlaySongSelect());
|
||||||
|
AddUntilStep("wait for song select", () => songSelect.BeatmapSetsLoaded);
|
||||||
|
|
||||||
|
AddStep("import beatmap", () => BeatmapImportHelper.LoadQuickOszIntoOsu(Game).WaitSafely());
|
||||||
|
AddUntilStep("wait for selected", () => !Game.Beatmap.IsDefault);
|
||||||
|
|
||||||
|
AddStep("delete selected beatmap", () =>
|
||||||
|
{
|
||||||
|
beatmap = Game.Beatmap.Value.BeatmapSetInfo;
|
||||||
|
Game.BeatmapManager.Delete(Game.Beatmap.Value.BeatmapSetInfo);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddUntilStep("nothing selected", () => Game.Beatmap.IsDefault);
|
||||||
|
AddStep("present deleted beatmap", () => Game.PresentBeatmap(beatmap));
|
||||||
|
AddAssert("still nothing selected", () => Game.Beatmap.IsDefault);
|
||||||
|
}
|
||||||
|
|
||||||
private Func<Player> playToResults()
|
private Func<Player> playToResults()
|
||||||
{
|
{
|
||||||
var player = playToCompletion();
|
var player = playToCompletion();
|
||||||
|
@ -170,6 +170,24 @@ namespace osu.Game.Tests.Visual.Online
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestPostAsOwner()
|
||||||
|
{
|
||||||
|
setUpCommentsResponse(getExampleComments());
|
||||||
|
AddStep("show comments", () => commentsContainer.ShowComments(CommentableType.Beatmapset, 123));
|
||||||
|
|
||||||
|
setUpPostResponse(true);
|
||||||
|
AddStep("enter text", () => editorTextBox.Current.Value = "comm");
|
||||||
|
AddStep("submit", () => commentsContainer.ChildrenOfType<CommentEditor>().Single().ChildrenOfType<RoundedButton>().First().TriggerClick());
|
||||||
|
|
||||||
|
AddUntilStep("comment sent", () =>
|
||||||
|
{
|
||||||
|
string writtenText = editorTextBox.Current.Value;
|
||||||
|
var comment = commentsContainer.ChildrenOfType<DrawableComment>().LastOrDefault();
|
||||||
|
return comment != null && comment.ChildrenOfType<SpriteText>().Any(y => y.Text == writtenText) && comment.ChildrenOfType<SpriteText>().Any(y => y.Text == "MAPPER");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
private void setUpCommentsResponse(CommentBundle commentBundle)
|
private void setUpCommentsResponse(CommentBundle commentBundle)
|
||||||
=> AddStep("set up response", () =>
|
=> AddStep("set up response", () =>
|
||||||
{
|
{
|
||||||
@ -183,7 +201,7 @@ namespace osu.Game.Tests.Visual.Online
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
private void setUpPostResponse()
|
private void setUpPostResponse(bool asOwner = false)
|
||||||
=> AddStep("set up response", () =>
|
=> AddStep("set up response", () =>
|
||||||
{
|
{
|
||||||
dummyAPI.HandleRequest = request =>
|
dummyAPI.HandleRequest = request =>
|
||||||
@ -191,7 +209,7 @@ namespace osu.Game.Tests.Visual.Online
|
|||||||
if (!(request is CommentPostRequest req))
|
if (!(request is CommentPostRequest req))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
req.TriggerSuccess(new CommentBundle
|
var bundle = new CommentBundle
|
||||||
{
|
{
|
||||||
Comments = new List<Comment>
|
Comments = new List<Comment>
|
||||||
{
|
{
|
||||||
@ -202,9 +220,26 @@ namespace osu.Game.Tests.Visual.Online
|
|||||||
LegacyName = "FirstUser",
|
LegacyName = "FirstUser",
|
||||||
CreatedAt = DateTimeOffset.Now,
|
CreatedAt = DateTimeOffset.Now,
|
||||||
VotesCount = 98,
|
VotesCount = 98,
|
||||||
|
CommentableId = 2001,
|
||||||
|
CommentableType = "test",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
};
|
||||||
|
|
||||||
|
if (asOwner)
|
||||||
|
{
|
||||||
|
bundle.Comments[0].UserId = 1001;
|
||||||
|
bundle.Comments[0].User = new APIUser { Id = 1001, Username = "FirstUser" };
|
||||||
|
bundle.CommentableMeta.Add(new CommentableMeta
|
||||||
|
{
|
||||||
|
Id = 2001,
|
||||||
|
OwnerId = 1001,
|
||||||
|
OwnerTitle = "MAPPER",
|
||||||
|
Type = "test",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
req.TriggerSuccess(bundle);
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
@ -4,62 +4,66 @@
|
|||||||
#nullable disable
|
#nullable disable
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using NUnit.Framework;
|
using System.Linq;
|
||||||
using osu.Framework.Allocation;
|
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Graphics.Shapes;
|
using osu.Game.Graphics.Containers;
|
||||||
using osu.Game.Online.API.Requests.Responses;
|
using osu.Game.Online.API.Requests.Responses;
|
||||||
using osu.Game.Overlays;
|
|
||||||
using osu.Game.Overlays.Comments;
|
using osu.Game.Overlays.Comments;
|
||||||
|
using osu.Game.Tests.Visual.UserInterface;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Online
|
namespace osu.Game.Tests.Visual.Online
|
||||||
{
|
{
|
||||||
public partial class TestSceneDrawableComment : OsuTestScene
|
public partial class TestSceneDrawableComment : ThemeComparisonTestScene
|
||||||
{
|
{
|
||||||
[Cached]
|
public TestSceneDrawableComment()
|
||||||
private readonly OverlayColourProvider colourProvider = new OverlayColourProvider(OverlayColourScheme.Purple);
|
: base(false)
|
||||||
|
|
||||||
private Container container;
|
|
||||||
|
|
||||||
[SetUp]
|
|
||||||
public void SetUp() => Schedule(() =>
|
|
||||||
{
|
{
|
||||||
Children = new Drawable[]
|
|
||||||
{
|
|
||||||
new Box
|
|
||||||
{
|
|
||||||
RelativeSizeAxes = Axes.Both,
|
|
||||||
Colour = colourProvider.Background4,
|
|
||||||
},
|
|
||||||
container = new Container
|
|
||||||
{
|
|
||||||
RelativeSizeAxes = Axes.X,
|
|
||||||
AutoSizeAxes = Axes.Y,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
[TestCaseSource(nameof(comments))]
|
|
||||||
public void TestComment(string description, string text)
|
|
||||||
{
|
|
||||||
AddStep(description, () =>
|
|
||||||
{
|
|
||||||
comment.Pinned = description == "Pinned";
|
|
||||||
comment.Message = text;
|
|
||||||
container.Add(new DrawableComment(comment));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static readonly Comment comment = new Comment
|
protected override Drawable CreateContent() => new OsuScrollContainer(Direction.Vertical)
|
||||||
{
|
{
|
||||||
Id = 1,
|
RelativeSizeAxes = Axes.Both,
|
||||||
LegacyName = "Test User",
|
Child = new FillFlowContainer
|
||||||
CreatedAt = DateTimeOffset.Now,
|
{
|
||||||
VotesCount = 0,
|
RelativeSizeAxes = Axes.X,
|
||||||
|
AutoSizeAxes = Axes.Y,
|
||||||
|
Direction = FillDirection.Vertical,
|
||||||
|
ChildrenEnumerable = comments.Select(info =>
|
||||||
|
{
|
||||||
|
var comment = new Comment
|
||||||
|
{
|
||||||
|
Id = 1,
|
||||||
|
UserId = 1000,
|
||||||
|
User = new APIUser { Id = 1000, Username = "Someone" },
|
||||||
|
CreatedAt = DateTimeOffset.Now,
|
||||||
|
VotesCount = 0,
|
||||||
|
Pinned = info[0] == "Pinned",
|
||||||
|
Message = info[1],
|
||||||
|
CommentableId = 2001,
|
||||||
|
CommentableType = "test"
|
||||||
|
};
|
||||||
|
|
||||||
|
return new[]
|
||||||
|
{
|
||||||
|
new DrawableComment(comment, Array.Empty<CommentableMeta>()),
|
||||||
|
new DrawableComment(comment, new[]
|
||||||
|
{
|
||||||
|
new CommentableMeta
|
||||||
|
{
|
||||||
|
Id = 2001,
|
||||||
|
OwnerId = comment.UserId,
|
||||||
|
OwnerTitle = "MAPPER",
|
||||||
|
Type = "test",
|
||||||
|
},
|
||||||
|
new CommentableMeta { Title = "Other Meta" },
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
}).SelectMany(c => c)
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private static object[] comments =
|
private static readonly string[][] comments =
|
||||||
{
|
{
|
||||||
new[] { "Plain", "This is plain comment" },
|
new[] { "Plain", "This is plain comment" },
|
||||||
new[] { "Pinned", "This is pinned comment" },
|
new[] { "Pinned", "This is pinned comment" },
|
||||||
|
@ -14,31 +14,39 @@ namespace osu.Game.Tests.Visual.UserInterface
|
|||||||
{
|
{
|
||||||
public abstract partial class ThemeComparisonTestScene : OsuGridTestScene
|
public abstract partial class ThemeComparisonTestScene : OsuGridTestScene
|
||||||
{
|
{
|
||||||
protected ThemeComparisonTestScene()
|
private readonly bool showWithoutColourProvider;
|
||||||
: base(1, 2)
|
|
||||||
|
protected ThemeComparisonTestScene(bool showWithoutColourProvider = true)
|
||||||
|
: base(1, showWithoutColourProvider ? 2 : 1)
|
||||||
{
|
{
|
||||||
|
this.showWithoutColourProvider = showWithoutColourProvider;
|
||||||
}
|
}
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load(OsuColour colours)
|
private void load(OsuColour colours)
|
||||||
{
|
{
|
||||||
Cell(0, 0).AddRange(new[]
|
if (showWithoutColourProvider)
|
||||||
{
|
{
|
||||||
new Box
|
Cell(0, 0).AddRange(new[]
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.Both,
|
new Box
|
||||||
Colour = colours.GreySeaFoam
|
{
|
||||||
},
|
RelativeSizeAxes = Axes.Both,
|
||||||
CreateContent()
|
Colour = colours.GreySeaFoam
|
||||||
});
|
},
|
||||||
|
CreateContent()
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void CreateThemedContent(OverlayColourScheme colourScheme)
|
protected void CreateThemedContent(OverlayColourScheme colourScheme)
|
||||||
{
|
{
|
||||||
var colourProvider = new OverlayColourProvider(colourScheme);
|
var colourProvider = new OverlayColourProvider(colourScheme);
|
||||||
|
|
||||||
Cell(0, 1).Clear();
|
int col = showWithoutColourProvider ? 1 : 0;
|
||||||
Cell(0, 1).Add(new DependencyProvidingContainer
|
|
||||||
|
Cell(0, col).Clear();
|
||||||
|
Cell(0, col).Add(new DependencyProvidingContainer
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.Both,
|
RelativeSizeAxes = Axes.Both,
|
||||||
CachedDependencies = new (Type, object)[]
|
CachedDependencies = new (Type, object)[]
|
||||||
|
@ -33,7 +33,7 @@ namespace osu.Game.Online.API.Requests.Responses
|
|||||||
[JsonProperty(@"votes_count")]
|
[JsonProperty(@"votes_count")]
|
||||||
public int VotesCount { get; set; }
|
public int VotesCount { get; set; }
|
||||||
|
|
||||||
[JsonProperty(@"commenatble_type")]
|
[JsonProperty(@"commentable_type")]
|
||||||
public string CommentableType { get; set; } = null!;
|
public string CommentableType { get; set; } = null!;
|
||||||
|
|
||||||
[JsonProperty(@"commentable_id")]
|
[JsonProperty(@"commentable_id")]
|
||||||
|
@ -11,6 +11,9 @@ namespace osu.Game.Online.API.Requests.Responses
|
|||||||
{
|
{
|
||||||
public class CommentBundle
|
public class CommentBundle
|
||||||
{
|
{
|
||||||
|
[JsonProperty(@"commentable_meta")]
|
||||||
|
public List<CommentableMeta> CommentableMeta { get; set; } = new List<CommentableMeta>();
|
||||||
|
|
||||||
[JsonProperty(@"comments")]
|
[JsonProperty(@"comments")]
|
||||||
public List<Comment> Comments { get; set; }
|
public List<Comment> Comments { get; set; }
|
||||||
|
|
||||||
|
28
osu.Game/Online/API/Requests/Responses/CommentableMeta.cs
Normal file
28
osu.Game/Online/API/Requests/Responses/CommentableMeta.cs
Normal file
@ -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 Newtonsoft.Json;
|
||||||
|
|
||||||
|
namespace osu.Game.Online.API.Requests.Responses
|
||||||
|
{
|
||||||
|
public class CommentableMeta
|
||||||
|
{
|
||||||
|
[JsonProperty("id")]
|
||||||
|
public long Id { get; set; }
|
||||||
|
|
||||||
|
[JsonProperty("owner_id")]
|
||||||
|
public long? OwnerId { get; set; }
|
||||||
|
|
||||||
|
[JsonProperty("owner_title")]
|
||||||
|
public string? OwnerTitle { get; set; }
|
||||||
|
|
||||||
|
[JsonProperty("title")]
|
||||||
|
public string Title { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[JsonProperty("type")]
|
||||||
|
public string Type { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
[JsonProperty("url")]
|
||||||
|
public string Url { get; set; } = string.Empty;
|
||||||
|
}
|
||||||
|
}
|
@ -631,6 +631,12 @@ namespace osu.Game
|
|||||||
|
|
||||||
var detachedSet = databasedSet.PerformRead(s => s.Detach());
|
var detachedSet = databasedSet.PerformRead(s => s.Detach());
|
||||||
|
|
||||||
|
if (detachedSet.DeletePending)
|
||||||
|
{
|
||||||
|
Logger.Log("The requested beatmap has since been deleted.", LoggingTarget.Information);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
PerformFromScreen(screen =>
|
PerformFromScreen(screen =>
|
||||||
{
|
{
|
||||||
// Find beatmaps that match our predicate.
|
// Find beatmaps that match our predicate.
|
||||||
|
180
osu.Game/Overlays/Comments/CommentAuthorLine.cs
Normal file
180
osu.Game/Overlays/Comments/CommentAuthorLine.cs
Normal file
@ -0,0 +1,180 @@
|
|||||||
|
// 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.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Graphics.Cursor;
|
||||||
|
using osu.Framework.Graphics.Shapes;
|
||||||
|
using osu.Framework.Graphics.Sprites;
|
||||||
|
using osu.Framework.Localisation;
|
||||||
|
using osu.Game.Graphics;
|
||||||
|
using osu.Game.Graphics.Containers;
|
||||||
|
using osu.Game.Graphics.Sprites;
|
||||||
|
using osu.Game.Online.API.Requests.Responses;
|
||||||
|
using osu.Game.Resources.Localisation.Web;
|
||||||
|
using osuTK;
|
||||||
|
|
||||||
|
namespace osu.Game.Overlays.Comments
|
||||||
|
{
|
||||||
|
public partial class CommentAuthorLine : FillFlowContainer
|
||||||
|
{
|
||||||
|
private readonly Comment comment;
|
||||||
|
private readonly IReadOnlyList<CommentableMeta> meta;
|
||||||
|
|
||||||
|
private OsuSpriteText deletedLabel = null!;
|
||||||
|
|
||||||
|
public CommentAuthorLine(Comment comment, IReadOnlyList<CommentableMeta> meta)
|
||||||
|
{
|
||||||
|
this.comment = comment;
|
||||||
|
this.meta = meta;
|
||||||
|
}
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load()
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
Direction = FillDirection.Horizontal;
|
||||||
|
Spacing = new Vector2(4, 0);
|
||||||
|
|
||||||
|
Add(new LinkFlowContainer(s => s.Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold))
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both
|
||||||
|
}.With(username =>
|
||||||
|
{
|
||||||
|
if (comment.UserId.HasValue)
|
||||||
|
username.AddUserLink(comment.User);
|
||||||
|
else
|
||||||
|
username.AddText(comment.LegacyName!);
|
||||||
|
}));
|
||||||
|
|
||||||
|
var ownerMeta = meta.FirstOrDefault(m => m.Id == comment.CommentableId && m.Type == comment.CommentableType);
|
||||||
|
|
||||||
|
if (ownerMeta?.OwnerId != null && ownerMeta.OwnerId == comment.UserId)
|
||||||
|
{
|
||||||
|
Add(new OwnerTitleBadge(ownerMeta.OwnerTitle ?? string.Empty)
|
||||||
|
{
|
||||||
|
// add top space to align with username
|
||||||
|
Margin = new MarginPadding { Top = 1f },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (comment.Pinned)
|
||||||
|
Add(new PinnedCommentNotice());
|
||||||
|
|
||||||
|
Add(new ParentUsername(comment));
|
||||||
|
|
||||||
|
Add(deletedLabel = new OsuSpriteText
|
||||||
|
{
|
||||||
|
Alpha = 0f,
|
||||||
|
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold),
|
||||||
|
Text = CommentsStrings.Deleted
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void MarkDeleted()
|
||||||
|
{
|
||||||
|
deletedLabel.Show();
|
||||||
|
}
|
||||||
|
|
||||||
|
private partial class OwnerTitleBadge : CircularContainer
|
||||||
|
{
|
||||||
|
private readonly string title;
|
||||||
|
|
||||||
|
public OwnerTitleBadge(string title)
|
||||||
|
{
|
||||||
|
this.title = title;
|
||||||
|
}
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load(OverlayColourProvider colourProvider)
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
Masking = true;
|
||||||
|
|
||||||
|
InternalChildren = new Drawable[]
|
||||||
|
{
|
||||||
|
new Box
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
Colour = colourProvider.Light1,
|
||||||
|
},
|
||||||
|
new OsuSpriteText
|
||||||
|
{
|
||||||
|
Text = title,
|
||||||
|
Font = OsuFont.Default.With(size: 10, weight: FontWeight.Bold),
|
||||||
|
Margin = new MarginPadding { Vertical = 2, Horizontal = 5 },
|
||||||
|
Colour = colourProvider.Background6,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private partial class PinnedCommentNotice : FillFlowContainer
|
||||||
|
{
|
||||||
|
public PinnedCommentNotice()
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
Direction = FillDirection.Horizontal;
|
||||||
|
Spacing = new Vector2(2, 0);
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
new SpriteIcon
|
||||||
|
{
|
||||||
|
Icon = FontAwesome.Solid.Thumbtack,
|
||||||
|
Size = new Vector2(14),
|
||||||
|
Anchor = Anchor.CentreLeft,
|
||||||
|
Origin = Anchor.CentreLeft,
|
||||||
|
},
|
||||||
|
new OsuSpriteText
|
||||||
|
{
|
||||||
|
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold),
|
||||||
|
Text = CommentsStrings.Pinned,
|
||||||
|
Anchor = Anchor.CentreLeft,
|
||||||
|
Origin = Anchor.CentreLeft,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private partial class ParentUsername : FillFlowContainer, IHasTooltip
|
||||||
|
{
|
||||||
|
public LocalisableString TooltipText => getParentMessage();
|
||||||
|
|
||||||
|
private readonly Comment? parentComment;
|
||||||
|
|
||||||
|
public ParentUsername(Comment comment)
|
||||||
|
{
|
||||||
|
parentComment = comment.ParentComment;
|
||||||
|
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
Direction = FillDirection.Horizontal;
|
||||||
|
Spacing = new Vector2(3, 0);
|
||||||
|
Alpha = comment.ParentId == null ? 0 : 1;
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
new SpriteIcon
|
||||||
|
{
|
||||||
|
Icon = FontAwesome.Solid.Reply,
|
||||||
|
Size = new Vector2(14),
|
||||||
|
},
|
||||||
|
new OsuSpriteText
|
||||||
|
{
|
||||||
|
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold, italics: true),
|
||||||
|
Text = parentComment?.User?.Username ?? parentComment?.LegacyName!
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private LocalisableString getParentMessage()
|
||||||
|
{
|
||||||
|
if (parentComment == null)
|
||||||
|
return string.Empty;
|
||||||
|
|
||||||
|
return parentComment.HasMessage ? parentComment.Message : parentComment.IsDeleted ? CommentsStrings.Deleted : string.Empty;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -301,7 +301,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
|
|
||||||
void addNewComment(Comment comment)
|
void addNewComment(Comment comment)
|
||||||
{
|
{
|
||||||
var drawableComment = GetDrawableComment(comment);
|
var drawableComment = GetDrawableComment(comment, bundle.CommentableMeta);
|
||||||
|
|
||||||
if (comment.ParentId == null)
|
if (comment.ParentId == null)
|
||||||
{
|
{
|
||||||
@ -333,7 +333,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
if (CommentDictionary.ContainsKey(comment.Id))
|
if (CommentDictionary.ContainsKey(comment.Id))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
topLevelComments.Add(GetDrawableComment(comment));
|
topLevelComments.Add(GetDrawableComment(comment, bundle.CommentableMeta));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (topLevelComments.Any())
|
if (topLevelComments.Any())
|
||||||
@ -351,12 +351,12 @@ namespace osu.Game.Overlays.Comments
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public DrawableComment GetDrawableComment(Comment comment)
|
public DrawableComment GetDrawableComment(Comment comment, IReadOnlyList<CommentableMeta> meta)
|
||||||
{
|
{
|
||||||
if (CommentDictionary.TryGetValue(comment.Id, out var existing))
|
if (CommentDictionary.TryGetValue(comment.Id, out var existing))
|
||||||
return existing;
|
return existing;
|
||||||
|
|
||||||
return CommentDictionary[comment.Id] = new DrawableComment(comment)
|
return CommentDictionary[comment.Id] = new DrawableComment(comment, meta)
|
||||||
{
|
{
|
||||||
ShowDeleted = { BindTarget = ShowDeleted },
|
ShowDeleted = { BindTarget = ShowDeleted },
|
||||||
Sort = { BindTarget = Sort },
|
Sort = { BindTarget = Sort },
|
||||||
|
@ -4,12 +4,10 @@
|
|||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Game.Graphics;
|
using osu.Game.Graphics;
|
||||||
using osu.Framework.Graphics.Sprites;
|
|
||||||
using osuTK;
|
using osuTK;
|
||||||
using osu.Game.Online.API.Requests.Responses;
|
using osu.Game.Online.API.Requests.Responses;
|
||||||
using osu.Game.Users.Drawables;
|
using osu.Game.Users.Drawables;
|
||||||
using osu.Game.Graphics.Containers;
|
using osu.Game.Graphics.Containers;
|
||||||
using osu.Framework.Graphics.Cursor;
|
|
||||||
using osu.Framework.Bindables;
|
using osu.Framework.Bindables;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using osu.Game.Graphics.Sprites;
|
using osu.Game.Graphics.Sprites;
|
||||||
@ -21,7 +19,6 @@ using osu.Framework.Extensions.IEnumerableExtensions;
|
|||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using osu.Framework.Extensions.LocalisationExtensions;
|
using osu.Framework.Extensions.LocalisationExtensions;
|
||||||
using osu.Framework.Localisation;
|
|
||||||
using osu.Framework.Logging;
|
using osu.Framework.Logging;
|
||||||
using osu.Framework.Platform;
|
using osu.Framework.Platform;
|
||||||
using osu.Game.Graphics.UserInterface;
|
using osu.Game.Graphics.UserInterface;
|
||||||
@ -42,6 +39,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
public Action<DrawableComment, int> RepliesRequested = null!;
|
public Action<DrawableComment, int> RepliesRequested = null!;
|
||||||
|
|
||||||
public readonly Comment Comment;
|
public readonly Comment Comment;
|
||||||
|
public readonly IReadOnlyList<CommentableMeta> Meta;
|
||||||
|
|
||||||
public readonly BindableBool ShowDeleted = new BindableBool();
|
public readonly BindableBool ShowDeleted = new BindableBool();
|
||||||
public readonly Bindable<CommentsSortCriteria> Sort = new Bindable<CommentsSortCriteria>();
|
public readonly Bindable<CommentsSortCriteria> Sort = new Bindable<CommentsSortCriteria>();
|
||||||
@ -72,7 +70,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
private LinkFlowContainer actionsContainer = null!;
|
private LinkFlowContainer actionsContainer = null!;
|
||||||
private LoadingSpinner actionsLoading = null!;
|
private LoadingSpinner actionsLoading = null!;
|
||||||
private DeletedCommentsCounter deletedCommentsCounter = null!;
|
private DeletedCommentsCounter deletedCommentsCounter = null!;
|
||||||
private OsuSpriteText deletedLabel = null!;
|
private CommentAuthorLine author = null!;
|
||||||
private GridContainer content = null!;
|
private GridContainer content = null!;
|
||||||
private VotePill votePill = null!;
|
private VotePill votePill = null!;
|
||||||
private Container<CommentEditor> replyEditorContainer = null!;
|
private Container<CommentEditor> replyEditorContainer = null!;
|
||||||
@ -90,15 +88,15 @@ namespace osu.Game.Overlays.Comments
|
|||||||
[Resolved]
|
[Resolved]
|
||||||
private OnScreenDisplay? onScreenDisplay { get; set; }
|
private OnScreenDisplay? onScreenDisplay { get; set; }
|
||||||
|
|
||||||
public DrawableComment(Comment comment)
|
public DrawableComment(Comment comment, IReadOnlyList<CommentableMeta> meta)
|
||||||
{
|
{
|
||||||
Comment = comment;
|
Comment = comment;
|
||||||
|
Meta = meta;
|
||||||
}
|
}
|
||||||
|
|
||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load(OverlayColourProvider colourProvider, DrawableComment? parentComment)
|
private void load(OverlayColourProvider colourProvider, DrawableComment? parentComment)
|
||||||
{
|
{
|
||||||
LinkFlowContainer username;
|
|
||||||
FillFlowContainer info;
|
FillFlowContainer info;
|
||||||
CommentMarkdownContainer message;
|
CommentMarkdownContainer message;
|
||||||
|
|
||||||
@ -174,27 +172,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
},
|
},
|
||||||
Children = new Drawable[]
|
Children = new Drawable[]
|
||||||
{
|
{
|
||||||
new FillFlowContainer
|
author = new CommentAuthorLine(Comment, Meta),
|
||||||
{
|
|
||||||
AutoSizeAxes = Axes.Both,
|
|
||||||
Direction = FillDirection.Horizontal,
|
|
||||||
Spacing = new Vector2(10, 0),
|
|
||||||
Children = new[]
|
|
||||||
{
|
|
||||||
username = new LinkFlowContainer(s => s.Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold))
|
|
||||||
{
|
|
||||||
AutoSizeAxes = Axes.Both
|
|
||||||
},
|
|
||||||
Comment.Pinned ? new PinnedCommentNotice() : Empty(),
|
|
||||||
new ParentUsername(Comment),
|
|
||||||
deletedLabel = new OsuSpriteText
|
|
||||||
{
|
|
||||||
Alpha = 0f,
|
|
||||||
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold),
|
|
||||||
Text = CommentsStrings.Deleted
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
message = new CommentMarkdownContainer
|
message = new CommentMarkdownContainer
|
||||||
{
|
{
|
||||||
RelativeSizeAxes = Axes.X,
|
RelativeSizeAxes = Axes.X,
|
||||||
@ -218,7 +196,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
{
|
{
|
||||||
new DrawableDate(Comment.CreatedAt, 12, false)
|
new DrawableDate(Comment.CreatedAt, 12, false)
|
||||||
{
|
{
|
||||||
Colour = colourProvider.Foreground1
|
Colour = colourProvider.Foreground1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -311,11 +289,6 @@ namespace osu.Game.Overlays.Comments
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (Comment.UserId.HasValue)
|
|
||||||
username.AddUserLink(Comment.User);
|
|
||||||
else
|
|
||||||
username.AddText(Comment.LegacyName!);
|
|
||||||
|
|
||||||
if (Comment.EditedAt.HasValue && Comment.EditedUser != null)
|
if (Comment.EditedAt.HasValue && Comment.EditedUser != null)
|
||||||
{
|
{
|
||||||
var font = OsuFont.GetFont(size: 12, weight: FontWeight.Regular);
|
var font = OsuFont.GetFont(size: 12, weight: FontWeight.Regular);
|
||||||
@ -400,7 +373,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
private void makeDeleted()
|
private void makeDeleted()
|
||||||
{
|
{
|
||||||
deletedLabel.Show();
|
author.MarkDeleted();
|
||||||
content.FadeColour(OsuColour.Gray(0.5f));
|
content.FadeColour(OsuColour.Gray(0.5f));
|
||||||
votePill.Hide();
|
votePill.Hide();
|
||||||
actionsContainer.Expire();
|
actionsContainer.Expire();
|
||||||
@ -547,70 +520,5 @@ namespace osu.Game.Overlays.Comments
|
|||||||
Top = 10
|
Top = 10
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private partial class PinnedCommentNotice : FillFlowContainer
|
|
||||||
{
|
|
||||||
public PinnedCommentNotice()
|
|
||||||
{
|
|
||||||
AutoSizeAxes = Axes.Both;
|
|
||||||
Direction = FillDirection.Horizontal;
|
|
||||||
Spacing = new Vector2(2, 0);
|
|
||||||
Children = new Drawable[]
|
|
||||||
{
|
|
||||||
new SpriteIcon
|
|
||||||
{
|
|
||||||
Icon = FontAwesome.Solid.Thumbtack,
|
|
||||||
Size = new Vector2(14),
|
|
||||||
Anchor = Anchor.CentreLeft,
|
|
||||||
Origin = Anchor.CentreLeft,
|
|
||||||
},
|
|
||||||
new OsuSpriteText
|
|
||||||
{
|
|
||||||
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold),
|
|
||||||
Text = CommentsStrings.Pinned,
|
|
||||||
Anchor = Anchor.CentreLeft,
|
|
||||||
Origin = Anchor.CentreLeft,
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private partial class ParentUsername : FillFlowContainer, IHasTooltip
|
|
||||||
{
|
|
||||||
public LocalisableString TooltipText => getParentMessage();
|
|
||||||
|
|
||||||
private readonly Comment? parentComment;
|
|
||||||
|
|
||||||
public ParentUsername(Comment comment)
|
|
||||||
{
|
|
||||||
parentComment = comment.ParentComment;
|
|
||||||
|
|
||||||
AutoSizeAxes = Axes.Both;
|
|
||||||
Direction = FillDirection.Horizontal;
|
|
||||||
Spacing = new Vector2(3, 0);
|
|
||||||
Alpha = comment.ParentId == null ? 0 : 1;
|
|
||||||
Children = new Drawable[]
|
|
||||||
{
|
|
||||||
new SpriteIcon
|
|
||||||
{
|
|
||||||
Icon = FontAwesome.Solid.Reply,
|
|
||||||
Size = new Vector2(14),
|
|
||||||
},
|
|
||||||
new OsuSpriteText
|
|
||||||
{
|
|
||||||
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold, italics: true),
|
|
||||||
Text = parentComment?.User?.Username ?? parentComment?.LegacyName!
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
private LocalisableString getParentMessage()
|
|
||||||
{
|
|
||||||
if (parentComment == null)
|
|
||||||
return string.Empty;
|
|
||||||
|
|
||||||
return parentComment.HasMessage ? parentComment.Message : parentComment.IsDeleted ? CommentsStrings.Deleted : string.Empty;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -60,7 +60,7 @@ namespace osu.Game.Overlays.Comments
|
|||||||
foreach (var comment in cb.Comments)
|
foreach (var comment in cb.Comments)
|
||||||
comment.ParentComment = parentComment;
|
comment.ParentComment = parentComment;
|
||||||
|
|
||||||
var drawables = cb.Comments.Select(commentsContainer.GetDrawableComment).ToArray();
|
var drawables = cb.Comments.Select(c => commentsContainer.GetDrawableComment(c, cb.CommentableMeta)).ToArray();
|
||||||
OnPost?.Invoke(drawables);
|
OnPost?.Invoke(drawables);
|
||||||
|
|
||||||
OnCancel!.Invoke();
|
OnCancel!.Invoke();
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
// 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.Linq;
|
||||||
using osu.Game.Beatmaps;
|
using osu.Game.Beatmaps;
|
||||||
using osu.Game.Screens.Select.Filter;
|
using osu.Game.Screens.Select.Filter;
|
||||||
@ -64,6 +65,7 @@ namespace osu.Game.Screens.Select.Carousel
|
|||||||
match &= !criteria.CircleSize.HasFilter || criteria.CircleSize.IsInRange(BeatmapInfo.Difficulty.CircleSize);
|
match &= !criteria.CircleSize.HasFilter || criteria.CircleSize.IsInRange(BeatmapInfo.Difficulty.CircleSize);
|
||||||
match &= !criteria.OverallDifficulty.HasFilter || criteria.OverallDifficulty.IsInRange(BeatmapInfo.Difficulty.OverallDifficulty);
|
match &= !criteria.OverallDifficulty.HasFilter || criteria.OverallDifficulty.IsInRange(BeatmapInfo.Difficulty.OverallDifficulty);
|
||||||
match &= !criteria.Length.HasFilter || criteria.Length.IsInRange(BeatmapInfo.Length);
|
match &= !criteria.Length.HasFilter || criteria.Length.IsInRange(BeatmapInfo.Length);
|
||||||
|
match &= !criteria.LastPlayed.HasFilter || criteria.LastPlayed.IsInRange(BeatmapInfo.LastPlayed ?? DateTimeOffset.MinValue);
|
||||||
match &= !criteria.BPM.HasFilter || criteria.BPM.IsInRange(BeatmapInfo.BPM);
|
match &= !criteria.BPM.HasFilter || criteria.BPM.IsInRange(BeatmapInfo.BPM);
|
||||||
|
|
||||||
match &= !criteria.BeatDivisor.HasFilter || criteria.BeatDivisor.IsInRange(BeatmapInfo.BeatDivisor);
|
match &= !criteria.BeatDivisor.HasFilter || criteria.BeatDivisor.IsInRange(BeatmapInfo.BeatDivisor);
|
||||||
|
@ -35,6 +35,7 @@ namespace osu.Game.Screens.Select
|
|||||||
public OptionalRange<double> BPM;
|
public OptionalRange<double> BPM;
|
||||||
public OptionalRange<int> BeatDivisor;
|
public OptionalRange<int> BeatDivisor;
|
||||||
public OptionalRange<BeatmapOnlineStatus> OnlineStatus;
|
public OptionalRange<BeatmapOnlineStatus> OnlineStatus;
|
||||||
|
public OptionalRange<DateTimeOffset> LastPlayed;
|
||||||
public OptionalTextFilter Creator;
|
public OptionalTextFilter Creator;
|
||||||
public OptionalTextFilter Artist;
|
public OptionalTextFilter Artist;
|
||||||
public OptionalTextFilter Title;
|
public OptionalTextFilter Title;
|
||||||
|
@ -61,6 +61,10 @@ namespace osu.Game.Screens.Select
|
|||||||
case "length":
|
case "length":
|
||||||
return tryUpdateLengthRange(criteria, op, value);
|
return tryUpdateLengthRange(criteria, op, value);
|
||||||
|
|
||||||
|
case "played":
|
||||||
|
case "lastplayed":
|
||||||
|
return tryUpdateDateAgoRange(ref criteria.LastPlayed, op, value);
|
||||||
|
|
||||||
case "divisor":
|
case "divisor":
|
||||||
return TryUpdateCriteriaRange(ref criteria.BeatDivisor, op, value, tryParseInt);
|
return TryUpdateCriteriaRange(ref criteria.BeatDivisor, op, value, tryParseInt);
|
||||||
|
|
||||||
@ -376,5 +380,107 @@ namespace osu.Game.Screens.Select
|
|||||||
|
|
||||||
return tryUpdateCriteriaRange(ref criteria.Length, op, totalLength, minScale / 2.0);
|
return tryUpdateCriteriaRange(ref criteria.Length, op, totalLength, minScale / 2.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This function is intended for parsing "days / months / years ago" type filters.
|
||||||
|
/// </summary>
|
||||||
|
private static bool tryUpdateDateAgoRange(ref FilterCriteria.OptionalRange<DateTimeOffset> dateRange, Operator op, string val)
|
||||||
|
{
|
||||||
|
switch (op)
|
||||||
|
{
|
||||||
|
case Operator.Equal:
|
||||||
|
// an equality filter is difficult to define for support here.
|
||||||
|
// if "3 months 2 days ago" means a single concrete time instant, such a filter is basically useless.
|
||||||
|
// if it means a range of 24 hours, then that is annoying to write and also comes with its own implications
|
||||||
|
// (does it mean "time instant 3 months 2 days ago, within 12 hours of tolerance either direction"?
|
||||||
|
// does it mean "the full calendar day, from midnight to midnight, 3 months 2 days ago"?)
|
||||||
|
// as such, for simplicity, just refuse to support this.
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// for the remaining operators, since the value provided to this function is an "ago" type value
|
||||||
|
// (as in, referring to some amount of time back),
|
||||||
|
// we'll want to flip the operator, such that `>5d` means "more than five days ago", as in "*before* five days ago",
|
||||||
|
// as intended by the user.
|
||||||
|
case Operator.Less:
|
||||||
|
op = Operator.Greater;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Operator.LessOrEqual:
|
||||||
|
op = Operator.GreaterOrEqual;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Operator.Greater:
|
||||||
|
op = Operator.Less;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Operator.GreaterOrEqual:
|
||||||
|
op = Operator.LessOrEqual;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupCollection? match = null;
|
||||||
|
|
||||||
|
match ??= tryMatchRegex(val, @"^((?<years>\d+)y)?((?<months>\d+)M)?((?<days>\d+(\.\d+)?)d)?((?<hours>\d+(\.\d+)?)h)?((?<minutes>\d+(\.\d+)?)m)?((?<seconds>\d+(\.\d+)?)s)?$");
|
||||||
|
match ??= tryMatchRegex(val, @"^(?<days>\d+(\.\d+)?)$");
|
||||||
|
|
||||||
|
if (match == null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
DateTimeOffset? dateTimeOffset = null;
|
||||||
|
DateTimeOffset now = DateTimeOffset.Now;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
List<string> keys = new List<string> { @"seconds", @"minutes", @"hours", @"days", @"months", @"years" };
|
||||||
|
|
||||||
|
foreach (string key in keys)
|
||||||
|
{
|
||||||
|
if (!match.TryGetValue(key, out var group) || !group.Success)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (group.Success)
|
||||||
|
{
|
||||||
|
if (!tryParseDoubleWithPoint(group.Value, out double length))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
switch (key)
|
||||||
|
{
|
||||||
|
case @"seconds":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddSeconds(-length);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case @"minutes":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddMinutes(-length);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case @"hours":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddHours(-length);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case @"days":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddDays(-length);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case @"months":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddMonths(-(int)length);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case @"years":
|
||||||
|
dateTimeOffset = (dateTimeOffset ?? now).AddYears(-(int)length);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (ArgumentOutOfRangeException)
|
||||||
|
{
|
||||||
|
dateTimeOffset = DateTimeOffset.MinValue.AddMilliseconds(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!dateTimeOffset.HasValue)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return tryUpdateCriteriaRange(ref dateRange, op, dateTimeOffset.Value);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user