mirror of
https://github.com/ppy/osu.git
synced 2025-03-15 14:47:18 +08:00
Merge branch 'master' into fix-song-select-desync
This commit is contained in:
commit
dd7c6a83bd
@ -65,7 +65,7 @@ namespace osu.Game.Rulesets.Osu.UI
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
if (!IsHittable(hitObject, hitObject.HitObject.StartTime + hitObject.Result.TimeOffset))
|
if (!IsHittable(hitObject, hitObject.HitObject.StartTime + hitObject.Result.TimeOffset))
|
||||||
throw new InvalidOperationException($"A {hitObject} was hit before it become hittable!");
|
throw new InvalidOperationException($"A {hitObject} was hit before it became hittable!");
|
||||||
|
|
||||||
foreach (var obj in enumerateHitObjectsUpTo(hitObject.HitObject.StartTime))
|
foreach (var obj in enumerateHitObjectsUpTo(hitObject.HitObject.StartTime))
|
||||||
{
|
{
|
||||||
|
98
osu.Game.Tests/Visual/Gameplay/TestSceneKeyBindings.cs
Normal file
98
osu.Game.Tests/Visual/Gameplay/TestSceneKeyBindings.cs
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
// 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 NUnit.Framework;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Input.Bindings;
|
||||||
|
using osu.Framework.Testing;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Input.Bindings;
|
||||||
|
using osu.Game.Rulesets;
|
||||||
|
using osu.Game.Rulesets.Difficulty;
|
||||||
|
using osu.Game.Rulesets.Mods;
|
||||||
|
using osu.Game.Rulesets.UI;
|
||||||
|
using osuTK.Input;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
|
{
|
||||||
|
[HeadlessTest]
|
||||||
|
public class TestSceneKeyBindings : OsuManualInputManagerTestScene
|
||||||
|
{
|
||||||
|
private readonly ActionReceiver receiver;
|
||||||
|
|
||||||
|
public TestSceneKeyBindings()
|
||||||
|
{
|
||||||
|
Add(new TestKeyBindingContainer
|
||||||
|
{
|
||||||
|
Child = receiver = new ActionReceiver()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestDefaultsWhenNotDatabased()
|
||||||
|
{
|
||||||
|
AddStep("fire key", () =>
|
||||||
|
{
|
||||||
|
InputManager.PressKey(Key.A);
|
||||||
|
InputManager.ReleaseKey(Key.A);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddAssert("received key", () => receiver.ReceivedAction);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TestRuleset : Ruleset
|
||||||
|
{
|
||||||
|
public override IEnumerable<Mod> GetModsFor(ModType type) =>
|
||||||
|
throw new System.NotImplementedException();
|
||||||
|
|
||||||
|
public override DrawableRuleset CreateDrawableRulesetWith(IBeatmap beatmap, IReadOnlyList<Mod> mods = null) =>
|
||||||
|
throw new System.NotImplementedException();
|
||||||
|
|
||||||
|
public override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap) =>
|
||||||
|
throw new System.NotImplementedException();
|
||||||
|
|
||||||
|
public override DifficultyCalculator CreateDifficultyCalculator(WorkingBeatmap beatmap) =>
|
||||||
|
throw new System.NotImplementedException();
|
||||||
|
|
||||||
|
public override IEnumerable<KeyBinding> GetDefaultKeyBindings(int variant = 0)
|
||||||
|
{
|
||||||
|
return new[]
|
||||||
|
{
|
||||||
|
new KeyBinding(InputKey.A, TestAction.Down),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string Description => "test";
|
||||||
|
public override string ShortName => "test";
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum TestAction
|
||||||
|
{
|
||||||
|
Down,
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TestKeyBindingContainer : DatabasedKeyBindingContainer<TestAction>
|
||||||
|
{
|
||||||
|
public TestKeyBindingContainer()
|
||||||
|
: base(new TestRuleset().RulesetInfo, 0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ActionReceiver : CompositeDrawable, IKeyBindingHandler<TestAction>
|
||||||
|
{
|
||||||
|
public bool ReceivedAction;
|
||||||
|
|
||||||
|
public bool OnPressed(TestAction action)
|
||||||
|
{
|
||||||
|
ReceivedAction = action == TestAction.Down;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnReleased(TestAction action)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
85
osu.Game.Tests/Visual/Online/TestSceneNowPlayingCommand.cs
Normal file
85
osu.Game.Tests/Visual/Online/TestSceneNowPlayingCommand.cs
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
// 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 NUnit.Framework;
|
||||||
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Testing;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Online.Chat;
|
||||||
|
using osu.Game.Rulesets;
|
||||||
|
using osu.Game.Users;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Visual.Online
|
||||||
|
{
|
||||||
|
[HeadlessTest]
|
||||||
|
public class TestSceneNowPlayingCommand : OsuTestScene
|
||||||
|
{
|
||||||
|
[Cached(typeof(IChannelPostTarget))]
|
||||||
|
private PostTarget postTarget { get; set; }
|
||||||
|
|
||||||
|
public TestSceneNowPlayingCommand()
|
||||||
|
{
|
||||||
|
Add(postTarget = new PostTarget());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestGenericActivity()
|
||||||
|
{
|
||||||
|
AddStep("Set activity", () => API.Activity.Value = new UserActivity.InLobby());
|
||||||
|
|
||||||
|
AddStep("Run command", () => Add(new NowPlayingCommand()));
|
||||||
|
|
||||||
|
AddAssert("Check correct response", () => postTarget.LastMessage.Contains("is listening"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestEditActivity()
|
||||||
|
{
|
||||||
|
AddStep("Set activity", () => API.Activity.Value = new UserActivity.Editing(new BeatmapInfo()));
|
||||||
|
|
||||||
|
AddStep("Run command", () => Add(new NowPlayingCommand()));
|
||||||
|
|
||||||
|
AddAssert("Check correct response", () => postTarget.LastMessage.Contains("is editing"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestPlayActivity()
|
||||||
|
{
|
||||||
|
AddStep("Set activity", () => API.Activity.Value = new UserActivity.SoloGame(new BeatmapInfo(), new RulesetInfo()));
|
||||||
|
|
||||||
|
AddStep("Run command", () => Add(new NowPlayingCommand()));
|
||||||
|
|
||||||
|
AddAssert("Check correct response", () => postTarget.LastMessage.Contains("is playing"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[TestCase(true)]
|
||||||
|
[TestCase(false)]
|
||||||
|
public void TestLinkPresence(bool hasOnlineId)
|
||||||
|
{
|
||||||
|
AddStep("Set activity", () => API.Activity.Value = new UserActivity.InLobby());
|
||||||
|
|
||||||
|
AddStep("Set beatmap", () => Beatmap.Value = new DummyWorkingBeatmap(null, null)
|
||||||
|
{
|
||||||
|
BeatmapInfo = { OnlineBeatmapID = hasOnlineId ? 1234 : (int?)null }
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("Run command", () => Add(new NowPlayingCommand()));
|
||||||
|
|
||||||
|
if (hasOnlineId)
|
||||||
|
AddAssert("Check link presence", () => postTarget.LastMessage.Contains("https://osu.ppy.sh/b/1234"));
|
||||||
|
else
|
||||||
|
AddAssert("Check link not present", () => !postTarget.LastMessage.Contains("https://"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public class PostTarget : Component, IChannelPostTarget
|
||||||
|
{
|
||||||
|
public void PostMessage(string text, bool isAction = false, Channel target = null)
|
||||||
|
{
|
||||||
|
LastMessage = text;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string LastMessage { get; private set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -62,6 +62,14 @@ namespace osu.Game.Input.Bindings
|
|||||||
store.KeyBindingChanged -= ReloadMappings;
|
store.KeyBindingChanged -= ReloadMappings;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void ReloadMappings() => KeyBindings = store.Query(ruleset?.ID, variant).ToList();
|
protected override void ReloadMappings()
|
||||||
|
{
|
||||||
|
if (ruleset != null && !ruleset.ID.HasValue)
|
||||||
|
// if the provided ruleset is not stored to the database, we have no way to retrieve custom bindings.
|
||||||
|
// fallback to defaults instead.
|
||||||
|
KeyBindings = DefaultKeyBindings;
|
||||||
|
else
|
||||||
|
KeyBindings = store.Query(ruleset?.ID, variant).ToList();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,7 @@ namespace osu.Game.Online.Chat
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Manages everything channel related
|
/// Manages everything channel related
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class ChannelManager : PollingComponent
|
public class ChannelManager : PollingComponent, IChannelPostTarget
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The channels the player joins on startup
|
/// The channels the player joins on startup
|
||||||
@ -204,6 +204,10 @@ namespace osu.Game.Online.Chat
|
|||||||
|
|
||||||
switch (command)
|
switch (command)
|
||||||
{
|
{
|
||||||
|
case "np":
|
||||||
|
AddInternal(new NowPlayingCommand());
|
||||||
|
break;
|
||||||
|
|
||||||
case "me":
|
case "me":
|
||||||
if (string.IsNullOrWhiteSpace(content))
|
if (string.IsNullOrWhiteSpace(content))
|
||||||
{
|
{
|
||||||
@ -234,7 +238,7 @@ namespace osu.Game.Online.Chat
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case "help":
|
case "help":
|
||||||
target.AddNewMessages(new InfoMessage("Supported commands: /help, /me [action], /join [channel]"));
|
target.AddNewMessages(new InfoMessage("Supported commands: /help, /me [action], /join [channel], /np"));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
19
osu.Game/Online/Chat/IChannelPostTarget.cs
Normal file
19
osu.Game/Online/Chat/IChannelPostTarget.cs
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// 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.Allocation;
|
||||||
|
|
||||||
|
namespace osu.Game.Online.Chat
|
||||||
|
{
|
||||||
|
[Cached(typeof(IChannelPostTarget))]
|
||||||
|
public interface IChannelPostTarget
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Posts a message to the currently opened channel.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="text">The message text that is going to be posted</param>
|
||||||
|
/// <param name="isAction">Is true if the message is an action, e.g.: user is currently eating </param>
|
||||||
|
/// <param name="target">An optional target channel. If null, <see cref="ChannelManager.CurrentChannel"/> will be used.</param>
|
||||||
|
void PostMessage(string text, bool isAction = false, Channel target = null);
|
||||||
|
}
|
||||||
|
}
|
55
osu.Game/Online/Chat/NowPlayingCommand.cs
Normal file
55
osu.Game/Online/Chat/NowPlayingCommand.cs
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
// 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.Allocation;
|
||||||
|
using osu.Framework.Bindables;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Game.Beatmaps;
|
||||||
|
using osu.Game.Online.API;
|
||||||
|
using osu.Game.Users;
|
||||||
|
|
||||||
|
namespace osu.Game.Online.Chat
|
||||||
|
{
|
||||||
|
public class NowPlayingCommand : Component
|
||||||
|
{
|
||||||
|
[Resolved]
|
||||||
|
private IChannelPostTarget channelManager { get; set; }
|
||||||
|
|
||||||
|
[Resolved]
|
||||||
|
private IAPIProvider api { get; set; }
|
||||||
|
|
||||||
|
[Resolved]
|
||||||
|
private Bindable<WorkingBeatmap> currentBeatmap { get; set; }
|
||||||
|
|
||||||
|
protected override void LoadComplete()
|
||||||
|
{
|
||||||
|
base.LoadComplete();
|
||||||
|
|
||||||
|
string verb;
|
||||||
|
BeatmapInfo beatmap;
|
||||||
|
|
||||||
|
switch (api.Activity.Value)
|
||||||
|
{
|
||||||
|
case UserActivity.SoloGame solo:
|
||||||
|
verb = "playing";
|
||||||
|
beatmap = solo.Beatmap;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case UserActivity.Editing edit:
|
||||||
|
verb = "editing";
|
||||||
|
beatmap = edit.Beatmap;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
verb = "listening to";
|
||||||
|
beatmap = currentBeatmap.Value.BeatmapInfo;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
var beatmapString = beatmap.OnlineBeatmapID.HasValue ? $"[https://osu.ppy.sh/b/{beatmap.OnlineBeatmapID} {beatmap}]" : beatmap.ToString();
|
||||||
|
|
||||||
|
channelManager.PostMessage($"is {verb} {beatmapString}", true);
|
||||||
|
Expire();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Threading;
|
using osu.Framework.Threading;
|
||||||
|
|
||||||
namespace osu.Game.Online
|
namespace osu.Game.Online
|
||||||
@ -11,7 +11,7 @@ namespace osu.Game.Online
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// A component which requires a constant polling process.
|
/// A component which requires a constant polling process.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public abstract class PollingComponent : Component
|
public abstract class PollingComponent : CompositeDrawable // switch away from Component because InternalChildren are used in usages.
|
||||||
{
|
{
|
||||||
private double? lastTimePolled;
|
private double? lastTimePolled;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user