// Copyright (c) ppy Pty Ltd . 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; using osu.Framework.Bindables; using osu.Framework.Extensions.IEnumerableExtensions; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Input.Bindings; using osu.Framework.Threading; using osu.Game.Extensions; using osu.Game.Graphics.UserInterface; using osu.Game.Input.Bindings; using osu.Game.Online.Multiplayer; using osuTK; namespace osu.Game.Screens.Multi.Lounge.Components { public class RoomsContainer : CompositeDrawable, IKeyBindingHandler { public Action JoinRequested; private readonly IBindableList rooms = new BindableList(); private readonly FillFlowContainer roomFlow; public IReadOnlyList Rooms => roomFlow; [Resolved(CanBeNull = true)] private Bindable filter { get; set; } [Resolved] private Bindable selectedRoom { get; set; } [Resolved] private IRoomManager roomManager { get; set; } public RoomsContainer() { RelativeSizeAxes = Axes.X; AutoSizeAxes = Axes.Y; InternalChild = roomFlow = new FillFlowContainer { RelativeSizeAxes = Axes.X, AutoSizeAxes = Axes.Y, Direction = FillDirection.Vertical, Spacing = new Vector2(2), }; } protected override void LoadComplete() { rooms.ItemsAdded += addRooms; rooms.ItemsRemoved += removeRooms; roomManager.RoomsUpdated += updateSorting; rooms.BindTo(roomManager.Rooms); filter?.BindValueChanged(criteria => Filter(criteria.NewValue)); } public void Filter(FilterCriteria criteria) { 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); switch (criteria.SecondaryFilter) { default: case SecondaryFilter.Public: matchingFilter &= r.Room.Availability.Value == RoomAvailability.Public; break; } r.MatchingFilter = matchingFilter; } }); } private void addRooms(IEnumerable rooms) { foreach (var room in rooms) { roomFlow.Add(new DrawableRoom(room) { Action = () => { if (room == selectedRoom.Value) { JoinRequested?.Invoke(room); return; } selectRoom(room); } }); } Filter(filter?.Value); } private void removeRooms(IEnumerable rooms) { foreach (var r in rooms) { var toRemove = roomFlow.Single(d => d.Room == r); toRemove.Action = null; roomFlow.Remove(toRemove); selectRoom(null); } } private void updateSorting() { foreach (var room in roomFlow) roomFlow.SetLayoutPosition(room, room.Room.Position.Value); } private void selectRoom(Room room) { roomFlow.Children.ForEach(r => r.State = r.Room == room ? SelectionState.Selected : SelectionState.NotSelected); selectedRoom.Value = room; } #region Key selection logic public bool OnPressed(GlobalAction action) { switch (action) { case GlobalAction.SelectNext: beginRepeatSelection(() => selectNext(1), action); return true; case GlobalAction.SelectPrevious: beginRepeatSelection(() => selectNext(-1), action); return true; } return false; } public void OnReleased(GlobalAction action) { switch (action) { case GlobalAction.SelectNext: case GlobalAction.SelectPrevious: endRepeatSelection(action); break; } } private ScheduledDelegate repeatDelegate; private object lastRepeatSource; /// /// Begin repeating the specified selection action. /// /// The action to perform. /// The source of the action. Used in conjunction with to only cancel the correct action (most recently pressed key). private void beginRepeatSelection(Action action, object source) { endRepeatSelection(); lastRepeatSource = source; repeatDelegate = this.BeginKeyRepeat(Scheduler, action); } private void endRepeatSelection(object source = null) { // only the most recent source should be able to cancel the current action. if (source != null && !EqualityComparer.Default.Equals(lastRepeatSource, source)) return; repeatDelegate?.Cancel(); repeatDelegate = null; lastRepeatSource = null; } private void selectNext(int direction) { var visibleRooms = Rooms.AsEnumerable().Where(r => r.IsPresent); Room room; if (selectedRoom.Value == null) room = visibleRooms.FirstOrDefault()?.Room; else { if (direction < 0) visibleRooms = visibleRooms.Reverse(); room = visibleRooms.SkipWhile(r => r.Room != selectedRoom.Value).Skip(1).FirstOrDefault()?.Room; } // we already have a valid selection only change selection if we still have a room to switch to. if (room != null) selectRoom(room); } #endregion protected override void Dispose(bool isDisposing) { base.Dispose(isDisposing); if (roomManager != null) roomManager.RoomsUpdated -= updateSorting; } } }