osu/osu.Game/Overlays/Dashboard/CurrentlyPlayingDisplay.cs

150 lines
4.9 KiB
C#
Raw Normal View History

// 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.Specialized;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Screens;
2020-11-06 07:38:57 +00:00
using osu.Game.Database;
using osu.Game.Online.API;
using osu.Game.Online.Spectator;
using osu.Game.Screens.OnlinePlay.Match.Components;
using osu.Game.Screens.Play;
using osu.Game.Users;
using osuTK;
namespace osu.Game.Overlays.Dashboard
{
internal class CurrentlyPlayingDisplay : CompositeDrawable
{
private readonly IBindableList<int> playingUsers = new BindableList<int>();
2020-11-02 12:09:47 +00:00
private FillFlowContainer<PlayingUserPanel> userFlow;
[Resolved]
private SpectatorClient spectatorClient { get; set; }
[BackgroundDependencyLoader]
private void load()
{
RelativeSizeAxes = Axes.X;
AutoSizeAxes = Axes.Y;
2020-11-02 12:09:47 +00:00
InternalChild = userFlow = new FillFlowContainer<PlayingUserPanel>
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
2020-11-06 04:08:07 +00:00
Padding = new MarginPadding(10),
Spacing = new Vector2(10),
};
}
[Resolved]
private IAPIProvider api { get; set; }
2020-11-06 07:38:57 +00:00
[Resolved]
private UserLookupCache users { get; set; }
protected override void LoadComplete()
{
base.LoadComplete();
playingUsers.BindTo(spectatorClient.PlayingUsers);
2020-11-09 03:22:54 +00:00
playingUsers.BindCollectionChanged(onUsersChanged, true);
}
2020-11-09 03:22:54 +00:00
private void onUsersChanged(object sender, NotifyCollectionChangedEventArgs e) => Schedule(() =>
{
switch (e.Action)
{
case NotifyCollectionChangedAction.Add:
foreach (int id in e.NewItems.OfType<int>().ToArray())
2020-11-09 03:22:54 +00:00
{
users.GetUserAsync(id).ContinueWith(u =>
{
2020-11-09 03:22:54 +00:00
if (u.Result == null) return;
Schedule(() =>
{
2020-11-09 03:22:54 +00:00
// user may no longer be playing.
if (!playingUsers.Contains(u.Result.Id))
2020-11-06 08:55:29 +00:00
return;
2020-11-09 03:22:54 +00:00
userFlow.Add(createUserPanel(u.Result));
2020-11-06 07:38:57 +00:00
});
2020-11-09 03:22:54 +00:00
});
}
2020-11-09 03:22:54 +00:00
break;
2020-11-09 03:22:54 +00:00
case NotifyCollectionChangedAction.Remove:
foreach (int u in e.OldItems.OfType<int>())
2020-11-09 03:22:54 +00:00
userFlow.FirstOrDefault(card => card.User.Id == u)?.Expire();
break;
2020-11-09 03:22:54 +00:00
case NotifyCollectionChangedAction.Reset:
userFlow.Clear();
break;
}
});
2020-11-02 12:09:47 +00:00
private PlayingUserPanel createUserPanel(User user) =>
new PlayingUserPanel(user).With(panel =>
{
panel.Anchor = Anchor.TopCentre;
panel.Origin = Anchor.TopCentre;
});
2020-11-02 12:09:47 +00:00
private class PlayingUserPanel : CompositeDrawable
{
public readonly User User;
[Resolved(canBeNull: true)]
private OsuGame game { get; set; }
public PlayingUserPanel(User user)
{
User = user;
AutoSizeAxes = Axes.Both;
}
2020-11-02 12:09:47 +00:00
[BackgroundDependencyLoader]
private void load(IAPIProvider api)
{
2020-11-02 12:09:47 +00:00
InternalChildren = new Drawable[]
{
new FillFlowContainer
{
AutoSizeAxes = Axes.Y,
2020-11-02 12:09:47 +00:00
Direction = FillDirection.Vertical,
Spacing = new Vector2(2),
Width = 290,
2020-11-02 12:09:47 +00:00
Children = new Drawable[]
{
new UserGridPanel(User)
2020-11-02 12:09:47 +00:00
{
RelativeSizeAxes = Axes.X,
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
2020-11-02 12:09:47 +00:00
},
new PurpleTriangleButton
2020-11-02 12:09:47 +00:00
{
RelativeSizeAxes = Axes.X,
2020-11-02 12:09:47 +00:00
Text = "Watch",
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
2021-04-01 13:08:52 +00:00
Action = () => game?.PerformFromScreen(s => s.Push(new SoloSpectator(User))),
Enabled = { Value = User.Id != api.LocalUser.Value.Id }
2020-11-02 12:09:47 +00:00
}
}
},
};
}
}
}
}