1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-21 05:22:55 +08:00
osu-lazer/osu.Game/Screens/OnlinePlay/Lounge/Components/DrawableRoomParticipantsList.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

382 lines
14 KiB
C#
Raw Normal View History

2021-07-13 14:10:44 +08:00
// 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;
2024-11-13 16:32:32 +08:00
using System.ComponentModel;
2021-07-13 14:10:44 +08:00
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
2021-07-13 14:10:44 +08:00
using osu.Game.Graphics.Sprites;
using osu.Game.Online.API.Requests.Responses;
2024-11-13 16:32:32 +08:00
using osu.Game.Online.Rooms;
using osu.Game.Overlays;
2021-07-13 14:10:44 +08:00
using osu.Game.Users.Drawables;
using osuTK;
2024-11-13 16:32:32 +08:00
using Container = osu.Framework.Graphics.Containers.Container;
2021-07-13 14:10:44 +08:00
2021-07-13 15:02:18 +08:00
namespace osu.Game.Screens.OnlinePlay.Lounge.Components
2021-07-13 14:10:44 +08:00
{
2024-11-15 12:20:09 +08:00
public partial class DrawableRoomParticipantsList : CompositeDrawable
2021-07-13 14:10:44 +08:00
{
public const float SHEAR_WIDTH = 12f;
2021-07-13 14:10:44 +08:00
private const float avatar_size = 36;
private const float height = 60f;
private static readonly Vector2 shear = new Vector2(SHEAR_WIDTH / height, 0);
2024-11-13 16:32:32 +08:00
private readonly Room room;
2021-08-12 17:51:50 +08:00
2024-11-13 16:32:32 +08:00
private FillFlowContainer<CircularAvatar> avatarFlow = null!;
private CircularAvatar hostAvatar = null!;
private LinkFlowContainer hostText = null!;
private HiddenUserCount hiddenUsers = null!;
private OsuSpriteText totalCount = null!;
2021-07-13 14:10:44 +08:00
2024-11-13 16:32:32 +08:00
public DrawableRoomParticipantsList(Room room)
2021-07-13 14:10:44 +08:00
{
2024-11-13 16:32:32 +08:00
this.room = room;
2021-07-13 14:10:44 +08:00
AutoSizeAxes = Axes.X;
Height = height;
2021-07-13 14:10:44 +08:00
}
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colours)
2021-07-13 14:10:44 +08:00
{
InternalChildren = new Drawable[]
{
new Container
{
RelativeSizeAxes = Axes.Both,
Masking = true,
CornerRadius = 10,
Shear = shear,
2021-07-13 14:10:44 +08:00
Child = new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colours.Background4,
2021-07-13 14:10:44 +08:00
}
},
new FillFlowContainer
{
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
2021-07-13 14:10:44 +08:00
Children = new Drawable[]
{
new FillFlowContainer
2021-07-13 14:10:44 +08:00
{
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Spacing = new Vector2(8),
Padding = new MarginPadding
{
Left = 8,
Right = 16
},
Children = new Drawable[]
{
hostAvatar = new CircularAvatar
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
},
hostText = new LinkFlowContainer
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
2021-09-30 10:52:14 +08:00
AutoSizeAxes = Axes.Both
}
}
2021-07-13 14:10:44 +08:00
},
new Container
2021-08-12 17:51:50 +08:00
{
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Children = new Drawable[]
{
new Container
{
RelativeSizeAxes = Axes.Both,
Masking = true,
CornerRadius = 10,
Shear = shear,
Child = new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colours.Background3,
}
},
new FillFlowContainer
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
AutoSizeAxes = Axes.Both,
Direction = FillDirection.Horizontal,
Spacing = new Vector2(4),
Padding = new MarginPadding
{
Left = 8,
Right = 16
},
Children = new Drawable[]
{
new SpriteIcon
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Size = new Vector2(16),
Icon = FontAwesome.Solid.User,
},
totalCount = new OsuSpriteText
{
Font = OsuFont.Default.With(weight: FontWeight.Bold),
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
},
avatarFlow = new FillFlowContainer<CircularAvatar>
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
AutoSizeAxes = Axes.Both,
Direction = FillDirection.Horizontal,
Spacing = new Vector2(4),
Margin = new MarginPadding { Left = 4 },
},
hiddenUsers = new HiddenUserCount
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
}
}
}
}
2021-08-12 17:51:50 +08:00
},
2021-07-13 14:10:44 +08:00
}
}
};
}
protected override void LoadComplete()
{
base.LoadComplete();
2024-11-13 16:32:32 +08:00
room.PropertyChanged += onRoomPropertyChanged;
2024-11-13 16:32:32 +08:00
updateRoomHost();
updateRoomParticipantCount();
updateRoomParticipants();
2021-07-13 14:10:44 +08:00
}
2021-08-12 17:47:22 +08:00
private int numberOfCircles = 4;
2021-07-13 14:10:44 +08:00
/// <summary>
/// The maximum number of circles visible (including the "hidden count" circle in the overflow case).
/// </summary>
public int NumberOfCircles
2021-07-13 14:10:44 +08:00
{
get => numberOfCircles;
2021-07-13 14:10:44 +08:00
set
{
numberOfCircles = value;
2021-07-13 14:10:44 +08:00
if (LoadState < LoadState.Loaded)
return;
// Reinitialising the list looks janky, but this is unlikely to be used in a setting where it's visible.
clearUsers();
foreach (var u in room.RecentParticipants)
2021-07-13 14:10:44 +08:00
addUser(u);
2021-08-12 17:47:22 +08:00
updateHiddenUsers();
2021-07-13 14:10:44 +08:00
}
}
private void updateRoomParticipants()
2021-07-13 14:10:44 +08:00
{
HashSet<APIUser> newUsers = room.RecentParticipants.ToHashSet();
2021-07-13 14:10:44 +08:00
avatarFlow.RemoveAll(a =>
{
// Avatar with no user. Really shouldn't ever be the case but asserting it correctly is difficult.
if (a.User == null)
return false;
// User was previously and still is a participant. Keep them around but remove them from the new set.
// This will be useful when we add all remaining users (now just the new participants) to the flow.
if (newUsers.Contains(a.User))
{
newUsers.Remove(a.User);
return false;
}
2021-07-13 14:10:44 +08:00
// User is no longer a participant. Remove them from the flow.
return true;
}, true);
2021-07-13 14:10:44 +08:00
// Add all remaining users to the flow.
foreach (var u in newUsers)
addUser(u);
2021-08-12 17:47:22 +08:00
updateHiddenUsers();
2021-07-13 14:10:44 +08:00
}
2021-08-12 17:47:22 +08:00
private int displayedCircles => avatarFlow.Count + (hiddenUsers.Count > 0 ? 1 : 0);
private void addUser(APIUser user)
2021-07-13 14:10:44 +08:00
{
2021-08-12 17:47:22 +08:00
if (displayedCircles < NumberOfCircles)
avatarFlow.Add(new CircularAvatar { User = user });
2021-07-13 14:10:44 +08:00
}
private void clearUsers()
{
avatarFlow.Clear();
2021-08-12 17:47:22 +08:00
updateHiddenUsers();
2021-07-13 14:10:44 +08:00
}
2021-08-12 17:47:22 +08:00
private void updateHiddenUsers()
{
2021-08-12 17:47:22 +08:00
int hiddenCount = 0;
if (room.RecentParticipants.Count > NumberOfCircles)
hiddenCount = room.ParticipantCount - NumberOfCircles + 1;
2021-08-12 17:47:22 +08:00
hiddenUsers.Count = hiddenCount;
2021-08-12 17:47:22 +08:00
if (displayedCircles > NumberOfCircles)
avatarFlow.Remove(avatarFlow.Last(), true);
2021-08-12 17:47:22 +08:00
else if (displayedCircles < NumberOfCircles)
{
var nextUser = room.RecentParticipants.FirstOrDefault(u => avatarFlow.All(a => a.User != u));
2021-08-12 17:47:22 +08:00
if (nextUser != null) addUser(nextUser);
}
}
2024-11-13 16:32:32 +08:00
private void onRoomPropertyChanged(object? sender, PropertyChangedEventArgs e)
{
switch (e.PropertyName)
{
case nameof(Room.Host):
updateRoomHost();
break;
case nameof(Room.ParticipantCount):
updateRoomParticipantCount();
break;
case nameof(Room.RecentParticipants):
updateRoomParticipants();
break;
}
2024-11-13 16:32:32 +08:00
}
private void updateRoomHost()
{
hostAvatar.User = room.Host;
hostText.Clear();
2024-11-13 16:32:32 +08:00
if (room.Host != null)
2021-09-30 11:04:30 +08:00
{
hostText.AddText("hosted by ");
2024-11-13 16:32:32 +08:00
hostText.AddUserLink(room.Host);
2021-09-30 11:04:30 +08:00
}
}
private void updateRoomParticipantCount()
{
updateHiddenUsers();
totalCount.Text = room.ParticipantCount.ToString();
}
2024-11-13 16:32:32 +08:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
room.PropertyChanged -= onRoomPropertyChanged;
}
2021-07-13 14:10:44 +08:00
private partial class CircularAvatar : CompositeDrawable
{
2024-11-13 16:32:32 +08:00
public APIUser? User
2021-07-13 14:10:44 +08:00
{
get => avatar.User;
set => avatar.User = value;
}
private readonly UpdateableAvatar avatar = new UpdateableAvatar(showUserPanelOnHover: true) { RelativeSizeAxes = Axes.Both };
2021-07-13 14:10:44 +08:00
2021-08-12 17:47:22 +08:00
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colours)
2021-07-13 14:10:44 +08:00
{
Size = new Vector2(avatar_size);
InternalChild = new CircularContainer
{
RelativeSizeAxes = Axes.Both,
Masking = true,
2021-08-12 17:47:22 +08:00
Children = new Drawable[]
{
new Box
{
Colour = colours.Background5,
RelativeSizeAxes = Axes.Both,
},
avatar
}
2021-07-13 14:10:44 +08:00
};
}
}
public partial class HiddenUserCount : CompositeDrawable
{
public int Count
{
get => count;
set
{
count = value;
countText.Text = $"+{count}";
if (count > 0)
Show();
else
Hide();
}
}
private int count;
private readonly SpriteText countText = new OsuSpriteText
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Font = OsuFont.Default.With(weight: FontWeight.Bold),
};
2021-07-13 14:10:44 +08:00
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colours)
2021-07-13 14:10:44 +08:00
{
Size = new Vector2(avatar_size);
Alpha = 0;
InternalChild = new CircularContainer
{
RelativeSizeAxes = Axes.Both,
Masking = true,
Children = new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colours.Background5,
2021-07-13 14:10:44 +08:00
},
countText
2021-07-13 14:10:44 +08:00
}
};
}
}
}
}