1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-06 02:42:54 +08:00
osu-lazer/osu.Game/Screens/Multi/Lounge/Components/RoomsContainer.cs

137 lines
4.2 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;
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
using osu.Framework.Extensions.IEnumerableExtensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics.UserInterface;
using osu.Game.Online.Multiplayer;
using osuTK;
namespace osu.Game.Screens.Multi.Lounge.Components
{
public class RoomsContainer : CompositeDrawable
{
2018-12-20 19:58:34 +08:00
public Action<Room> JoinRequested;
2019-01-07 17:50:27 +08:00
private readonly IBindableList<Room> rooms = new BindableList<Room>();
private readonly FillFlowContainer<DrawableRoom> roomFlow;
2018-12-25 10:59:08 +08:00
public IReadOnlyList<DrawableRoom> Rooms => roomFlow;
2019-11-15 10:07:16 +08:00
[Resolved(CanBeNull = true)]
2019-11-15 07:23:56 +08:00
private Bindable<FilterCriteria> filter { get; set; }
2019-02-08 13:57:51 +08:00
[Resolved]
private Bindable<Room> selectedRoom { get; set; }
2019-02-08 13:57:51 +08:00
[Resolved]
private IRoomManager roomManager { get; set; }
public RoomsContainer()
{
RelativeSizeAxes = Axes.X;
AutoSizeAxes = Axes.Y;
InternalChild = roomFlow = new FillFlowContainer<DrawableRoom>
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(2),
};
}
protected override void LoadComplete()
{
rooms.ItemsAdded += addRooms;
rooms.ItemsRemoved += removeRooms;
2018-12-28 00:45:19 +08:00
roomManager.RoomsUpdated += updateSorting;
rooms.BindTo(roomManager.Rooms);
2020-02-06 13:28:09 +08:00
filter?.BindValueChanged(criteria => Filter(criteria.NewValue));
2019-11-15 07:23:56 +08:00
}
public void Filter(FilterCriteria criteria)
{
2018-12-19 15:56:51 +08:00
roomFlow.Children.ForEach(r =>
{
if (criteria == null)
r.MatchingFilter = true;
else
{
bool matchingFilter = true;
matchingFilter &= r.Room.Playlist.Count == 0 || r.Room.Playlist.Any(i => i.Ruleset.Value.Equals(criteria.Ruleset));
if (!string.IsNullOrEmpty(criteria.SearchString))
matchingFilter &= r.FilterTerms.Any(term => term.IndexOf(criteria.SearchString, StringComparison.InvariantCultureIgnoreCase) >= 0);
2018-12-19 15:56:51 +08:00
switch (criteria.SecondaryFilter)
{
default:
case SecondaryFilter.Public:
matchingFilter &= r.Room.Availability.Value == RoomAvailability.Public;
2018-12-19 15:56:51 +08:00
break;
}
r.MatchingFilter = matchingFilter;
}
});
}
private void addRooms(IEnumerable<Room> rooms)
{
foreach (var r in rooms)
roomFlow.Add(new DrawableRoom(r) { Action = () => selectRoom(r) });
Filter(filter?.Value);
}
private void removeRooms(IEnumerable<Room> rooms)
{
foreach (var r in rooms)
{
var toRemove = roomFlow.Single(d => d.Room == r);
toRemove.Action = null;
roomFlow.Remove(toRemove);
2018-12-20 14:17:08 +08:00
2018-12-20 19:58:34 +08:00
selectRoom(null);
}
}
2018-12-28 00:45:19 +08:00
private void updateSorting()
{
foreach (var room in roomFlow)
2019-02-21 17:56:34 +08:00
roomFlow.SetLayoutPosition(room, room.Room.Position.Value);
2018-12-28 00:45:19 +08:00
}
private void selectRoom(Room room)
{
var drawable = roomFlow.FirstOrDefault(r => r.Room == room);
if (drawable != null && drawable.State == SelectionState.Selected)
2018-12-20 19:58:34 +08:00
JoinRequested?.Invoke(room);
else
roomFlow.Children.ForEach(r => r.State = r.Room == room ? SelectionState.Selected : SelectionState.NotSelected);
2018-12-20 19:58:34 +08:00
selectedRoom.Value = room;
}
2018-12-28 00:45:19 +08:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
if (roomManager != null)
roomManager.RoomsUpdated -= updateSorting;
}
}
}