mirror of
https://github.com/ppy/osu.git
synced 2025-01-18 08:32:54 +08:00
Implement spectator list display
- First step for https://github.com/ppy/osu/issues/22087 - Supersedes / closes https://github.com/ppy/osu/pull/22795 Roughly uses design shown in https://github.com/ppy/osu/pull/22795#issuecomment-1579936284 with some modifications to better fit everything else, and some customisation options so it can fit better on other skins.
This commit is contained in:
parent
0e20c0e307
commit
582c5180b9
49
osu.Game.Tests/Visual/Gameplay/TestSceneSpectatorList.cs
Normal file
49
osu.Game.Tests/Visual/Gameplay/TestSceneSpectatorList.cs
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
// 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.Threading;
|
||||||
|
using NUnit.Framework;
|
||||||
|
using osu.Framework.Bindables;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Utils;
|
||||||
|
using osu.Game.Graphics;
|
||||||
|
using osu.Game.Screens.Play;
|
||||||
|
using osu.Game.Screens.Play.HUD;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
|
{
|
||||||
|
[TestFixture]
|
||||||
|
public partial class TestSceneSpectatorList : OsuTestScene
|
||||||
|
{
|
||||||
|
private readonly BindableList<SpectatorList.Spectator> spectators = new BindableList<SpectatorList.Spectator>();
|
||||||
|
private readonly Bindable<LocalUserPlayingState> localUserPlayingState = new Bindable<LocalUserPlayingState>();
|
||||||
|
|
||||||
|
private int counter;
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestBasics()
|
||||||
|
{
|
||||||
|
SpectatorList list = null!;
|
||||||
|
AddStep("create spectator list", () => Child = list = new SpectatorList
|
||||||
|
{
|
||||||
|
Anchor = Anchor.Centre,
|
||||||
|
Origin = Anchor.Centre,
|
||||||
|
Spectators = { BindTarget = spectators },
|
||||||
|
UserPlayingState = { BindTarget = localUserPlayingState }
|
||||||
|
});
|
||||||
|
|
||||||
|
AddStep("start playing", () => localUserPlayingState.Value = LocalUserPlayingState.Playing);
|
||||||
|
AddStep("add a user", () =>
|
||||||
|
{
|
||||||
|
int id = Interlocked.Increment(ref counter);
|
||||||
|
spectators.Add(new SpectatorList.Spectator(id, $"User {id}"));
|
||||||
|
});
|
||||||
|
AddStep("remove random user", () => spectators.RemoveAt(RNG.Next(0, spectators.Count)));
|
||||||
|
AddStep("enter break", () => localUserPlayingState.Value = LocalUserPlayingState.Break);
|
||||||
|
AddStep("stop playing", () => localUserPlayingState.Value = LocalUserPlayingState.NotPlaying);
|
||||||
|
AddStep("change font to venera", () => list.Font.Value = Typeface.Venera);
|
||||||
|
AddStep("change font to torus", () => list.Font.Value = Typeface.Torus);
|
||||||
|
AddStep("change header colour", () => list.HeaderColour.Value = new Colour4(RNG.NextSingle(), RNG.NextSingle(), RNG.NextSingle(), 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
19
osu.Game/Localisation/HUD/SpectatorListStrings.cs
Normal file
19
osu.Game/Localisation/HUD/SpectatorListStrings.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.Localisation;
|
||||||
|
|
||||||
|
namespace osu.Game.Localisation.HUD
|
||||||
|
{
|
||||||
|
public static class SpectatorListStrings
|
||||||
|
{
|
||||||
|
private const string prefix = @"osu.Game.Resources.Localisation.SpectatorList";
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// "Spectators ({0})"
|
||||||
|
/// </summary>
|
||||||
|
public static LocalisableString SpectatorCount(int arg0) => new TranslatableString(getKey(@"spectator_count"), @"Spectators ({0})", arg0);
|
||||||
|
|
||||||
|
private static string getKey(string key) => $@"{prefix}:{key}";
|
||||||
|
}
|
||||||
|
}
|
@ -14,6 +14,7 @@ using osu.Game.Graphics.Containers;
|
|||||||
using osu.Game.Graphics.UserInterface;
|
using osu.Game.Graphics.UserInterface;
|
||||||
using osu.Game.Overlays;
|
using osu.Game.Overlays;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
using osuTK.Graphics;
|
||||||
|
|
||||||
namespace osu.Game.Online.Chat
|
namespace osu.Game.Online.Chat
|
||||||
{
|
{
|
||||||
@ -27,6 +28,18 @@ namespace osu.Game.Online.Chat
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public readonly SlimReadOnlyListWrapper<Drawable> Parts;
|
public readonly SlimReadOnlyListWrapper<Drawable> Parts;
|
||||||
|
|
||||||
|
public new Color4 IdleColour
|
||||||
|
{
|
||||||
|
get => base.IdleColour;
|
||||||
|
set => base.IdleColour = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public new Color4 HoverColour
|
||||||
|
{
|
||||||
|
get => base.HoverColour;
|
||||||
|
set => base.HoverColour = value;
|
||||||
|
}
|
||||||
|
|
||||||
[Resolved]
|
[Resolved]
|
||||||
private OverlayColourProvider? overlayColourProvider { get; set; }
|
private OverlayColourProvider? overlayColourProvider { get; set; }
|
||||||
|
|
||||||
@ -56,7 +69,8 @@ namespace osu.Game.Online.Chat
|
|||||||
[BackgroundDependencyLoader]
|
[BackgroundDependencyLoader]
|
||||||
private void load(OsuColour colours)
|
private void load(OsuColour colours)
|
||||||
{
|
{
|
||||||
IdleColour = overlayColourProvider?.Light2 ?? colours.Blue;
|
if (IdleColour == default)
|
||||||
|
IdleColour = overlayColourProvider?.Light2 ?? colours.Blue;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override IEnumerable<Drawable> EffectTargets => Parts;
|
protected override IEnumerable<Drawable> EffectTargets => Parts;
|
||||||
|
219
osu.Game/Screens/Play/HUD/SpectatorList.cs
Normal file
219
osu.Game/Screens/Play/HUD/SpectatorList.cs
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
// 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.Specialized;
|
||||||
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Bindables;
|
||||||
|
using osu.Framework.Extensions.LocalisationExtensions;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Graphics.Pooling;
|
||||||
|
using osu.Game.Configuration;
|
||||||
|
using osu.Game.Graphics;
|
||||||
|
using osu.Game.Graphics.Sprites;
|
||||||
|
using osu.Game.Online.Chat;
|
||||||
|
using osu.Game.Users;
|
||||||
|
using osu.Game.Localisation.HUD;
|
||||||
|
using osu.Game.Localisation.SkinComponents;
|
||||||
|
|
||||||
|
namespace osu.Game.Screens.Play.HUD
|
||||||
|
{
|
||||||
|
public partial class SpectatorList : CompositeDrawable
|
||||||
|
{
|
||||||
|
private const int max_spectators_displayed = 10;
|
||||||
|
|
||||||
|
public BindableList<Spectator> Spectators { get; } = new BindableList<Spectator>();
|
||||||
|
public Bindable<LocalUserPlayingState> UserPlayingState { get; } = new Bindable<LocalUserPlayingState>();
|
||||||
|
|
||||||
|
[SettingSource(typeof(SkinnableComponentStrings), nameof(SkinnableComponentStrings.Font), nameof(SkinnableComponentStrings.FontDescription))]
|
||||||
|
public Bindable<Typeface> Font { get; } = new Bindable<Typeface>(Typeface.Torus);
|
||||||
|
|
||||||
|
[SettingSource(typeof(SkinnableComponentStrings), nameof(SkinnableComponentStrings.TextColour), nameof(SkinnableComponentStrings.TextColourDescription))]
|
||||||
|
public BindableColour4 HeaderColour { get; } = new BindableColour4(Colour4.White);
|
||||||
|
|
||||||
|
protected OsuSpriteText Header { get; private set; } = null!;
|
||||||
|
|
||||||
|
private FillFlowContainer mainFlow = null!;
|
||||||
|
private FillFlowContainer<SpectatorListEntry> spectatorsFlow = null!;
|
||||||
|
private DrawablePool<SpectatorListEntry> pool = null!;
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load(OsuColour colours)
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
|
||||||
|
InternalChildren = new Drawable[]
|
||||||
|
{
|
||||||
|
mainFlow = new FillFlowContainer
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both,
|
||||||
|
AutoSizeDuration = 250,
|
||||||
|
AutoSizeEasing = Easing.OutQuint,
|
||||||
|
Direction = FillDirection.Vertical,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
Header = new OsuSpriteText
|
||||||
|
{
|
||||||
|
Colour = colours.Blue0,
|
||||||
|
Font = OsuFont.GetFont(size: 12, weight: FontWeight.Bold),
|
||||||
|
},
|
||||||
|
spectatorsFlow = new FillFlowContainer<SpectatorListEntry>
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both,
|
||||||
|
Direction = FillDirection.Vertical,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
pool = new DrawablePool<SpectatorListEntry>(max_spectators_displayed),
|
||||||
|
};
|
||||||
|
|
||||||
|
HeaderColour.Value = Header.Colour;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void LoadComplete()
|
||||||
|
{
|
||||||
|
base.LoadComplete();
|
||||||
|
|
||||||
|
Spectators.BindCollectionChanged(onSpectatorsChanged, true);
|
||||||
|
UserPlayingState.BindValueChanged(_ => updateVisibility());
|
||||||
|
|
||||||
|
Font.BindValueChanged(_ => updateAppearance());
|
||||||
|
HeaderColour.BindValueChanged(_ => updateAppearance(), true);
|
||||||
|
FinishTransforms(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void onSpectatorsChanged(object? sender, NotifyCollectionChangedEventArgs e)
|
||||||
|
{
|
||||||
|
switch (e.Action)
|
||||||
|
{
|
||||||
|
case NotifyCollectionChangedAction.Add:
|
||||||
|
{
|
||||||
|
for (int i = 0; i < e.NewItems!.Count; i++)
|
||||||
|
{
|
||||||
|
var spectator = (Spectator)e.NewItems![i]!;
|
||||||
|
int index = e.NewStartingIndex + i;
|
||||||
|
|
||||||
|
if (index >= max_spectators_displayed)
|
||||||
|
break;
|
||||||
|
|
||||||
|
spectatorsFlow.Insert(e.NewStartingIndex + i, pool.Get(entry =>
|
||||||
|
{
|
||||||
|
entry.Current.Value = spectator;
|
||||||
|
entry.UserPlayingState = UserPlayingState;
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case NotifyCollectionChangedAction.Remove:
|
||||||
|
{
|
||||||
|
spectatorsFlow.RemoveAll(entry => e.OldItems!.Contains(entry.Current.Value), false);
|
||||||
|
|
||||||
|
for (int i = 0; i < spectatorsFlow.Count; i++)
|
||||||
|
spectatorsFlow.SetLayoutPosition(spectatorsFlow[i], i);
|
||||||
|
|
||||||
|
if (Spectators.Count >= max_spectators_displayed && spectatorsFlow.Count < max_spectators_displayed)
|
||||||
|
{
|
||||||
|
for (int i = spectatorsFlow.Count; i < max_spectators_displayed; i++)
|
||||||
|
{
|
||||||
|
var spectator = Spectators[i];
|
||||||
|
spectatorsFlow.Insert(i, pool.Get(entry =>
|
||||||
|
{
|
||||||
|
entry.Current.Value = spectator;
|
||||||
|
entry.UserPlayingState = UserPlayingState;
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case NotifyCollectionChangedAction.Reset:
|
||||||
|
{
|
||||||
|
spectatorsFlow.Clear(false);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new NotSupportedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
Header.Text = SpectatorListStrings.SpectatorCount(Spectators.Count).ToUpper();
|
||||||
|
updateVisibility();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateVisibility()
|
||||||
|
{
|
||||||
|
mainFlow.FadeTo(Spectators.Count > 0 && UserPlayingState.Value != LocalUserPlayingState.NotPlaying ? 1 : 0, 250, Easing.OutQuint);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateAppearance()
|
||||||
|
{
|
||||||
|
Header.Font = OsuFont.GetFont(Font.Value, 12, FontWeight.Bold);
|
||||||
|
Header.Colour = HeaderColour.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private partial class SpectatorListEntry : PoolableDrawable
|
||||||
|
{
|
||||||
|
public Bindable<Spectator> Current { get; } = new Bindable<Spectator>();
|
||||||
|
|
||||||
|
private readonly BindableWithCurrent<LocalUserPlayingState> current = new BindableWithCurrent<LocalUserPlayingState>();
|
||||||
|
|
||||||
|
public Bindable<LocalUserPlayingState> UserPlayingState
|
||||||
|
{
|
||||||
|
get => current.Current;
|
||||||
|
set => current.Current = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private OsuSpriteText username = null!;
|
||||||
|
private DrawableLinkCompiler? linkCompiler;
|
||||||
|
|
||||||
|
[Resolved]
|
||||||
|
private OsuGame? game { get; set; }
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load()
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
|
||||||
|
InternalChildren = new Drawable[]
|
||||||
|
{
|
||||||
|
username = new OsuSpriteText(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void LoadComplete()
|
||||||
|
{
|
||||||
|
base.LoadComplete();
|
||||||
|
UserPlayingState.BindValueChanged(_ => updateEnabledState());
|
||||||
|
Current.BindValueChanged(_ => updateState(), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateState()
|
||||||
|
{
|
||||||
|
username.Text = Current.Value.Username;
|
||||||
|
linkCompiler?.Expire();
|
||||||
|
AddInternal(linkCompiler = new DrawableLinkCompiler([username])
|
||||||
|
{
|
||||||
|
IdleColour = Colour4.White,
|
||||||
|
Action = () => game?.HandleLink(new LinkDetails(LinkAction.OpenUserProfile, Current.Value)),
|
||||||
|
});
|
||||||
|
updateEnabledState();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateEnabledState()
|
||||||
|
{
|
||||||
|
if (linkCompiler != null)
|
||||||
|
linkCompiler.Enabled.Value = UserPlayingState.Value != LocalUserPlayingState.Playing;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public record Spectator(int OnlineID, string Username) : IUser
|
||||||
|
{
|
||||||
|
public CountryCode CountryCode => CountryCode.Unknown;
|
||||||
|
public bool IsBot => false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user