mirror of
https://github.com/ppy/osu.git
synced 2025-01-07 16:52:54 +08:00
288 lines
10 KiB
C#
288 lines
10 KiB
C#
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Caching;
|
|
using osu.Framework.Extensions.Color4Extensions;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Containers;
|
|
using osu.Framework.Graphics.Cursor;
|
|
using osu.Framework.Graphics.Lines;
|
|
using osu.Framework.Graphics.UserInterface;
|
|
using osu.Framework.Input.States;
|
|
using osu.Game.Graphics;
|
|
using osu.Game.Graphics.UserInterface;
|
|
using osu.Game.Tournament.Components;
|
|
using osu.Game.Tournament.Screens.Ladder.Components;
|
|
using OpenTK;
|
|
using OpenTK.Graphics;
|
|
using SixLabors.Primitives;
|
|
|
|
namespace osu.Game.Tournament.Screens.Ladder
|
|
{
|
|
public class LadderManager : CompositeDrawable, IHasContextMenu
|
|
{
|
|
public readonly List<TournamentTeam> Teams;
|
|
private readonly Container<DrawableMatchPairing> pairingsContainer;
|
|
private readonly Container<Path> paths;
|
|
private readonly Container headings;
|
|
|
|
private readonly ScrollableContainer scrollContent;
|
|
|
|
[Cached]
|
|
private readonly LadderEditorInfo editorInfo = new LadderEditorInfo();
|
|
|
|
public LadderManager(LadderInfo info)
|
|
{
|
|
editorInfo.Teams = Teams = info.Teams;
|
|
editorInfo.Groupings = info.Groupings;
|
|
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
InternalChild = new Container
|
|
{
|
|
RelativeSizeAxes = Axes.Both,
|
|
Children = new Drawable[]
|
|
{
|
|
scrollContent = new ScrollableContainer
|
|
{
|
|
RelativeSizeAxes = Axes.Both,
|
|
Children = new Drawable[]
|
|
{
|
|
paths = new Container<Path> { RelativeSizeAxes = Axes.Both },
|
|
headings = new Container { RelativeSizeAxes = Axes.Both },
|
|
pairingsContainer = new Container<DrawableMatchPairing> { RelativeSizeAxes = Axes.Both },
|
|
}
|
|
},
|
|
new LadderEditorSettings
|
|
{
|
|
Anchor = Anchor.TopRight,
|
|
Origin = Anchor.TopRight,
|
|
Margin = new MarginPadding(5)
|
|
}
|
|
}
|
|
};
|
|
|
|
// assign teams
|
|
foreach (var pairing in info.Pairings)
|
|
{
|
|
pairing.Team1.Value = info.Teams.FirstOrDefault(t => t.Acronym == pairing.Team1Acronym);
|
|
pairing.Team2.Value = info.Teams.FirstOrDefault(t => t.Acronym == pairing.Team2Acronym);
|
|
}
|
|
|
|
// assign progressions
|
|
foreach (var pair in info.Progressions)
|
|
{
|
|
var src = info.Pairings.FirstOrDefault(p => p.ID == pair.Item1);
|
|
var dest = info.Pairings.FirstOrDefault(p => p.ID == pair.Item2);
|
|
|
|
if (src == null) throw new InvalidOperationException();
|
|
|
|
if (dest != null)
|
|
{
|
|
if (pair.Losers)
|
|
src.LosersProgression.Value = dest;
|
|
else
|
|
src.Progression.Value = dest;
|
|
}
|
|
}
|
|
|
|
foreach (var pairing in info.Pairings)
|
|
addPairing(pairing);
|
|
|
|
foreach (var group in info.Groupings)
|
|
foreach (var id in group.Pairings)
|
|
info.Pairings.Single(p => p.ID == id).Grouping.Value = group;
|
|
|
|
// todo: fix this
|
|
Scheduler.AddDelayed(() => layout.Invalidate(), 100, true);
|
|
}
|
|
|
|
public LadderInfo CreateInfo()
|
|
{
|
|
var pairings = pairingsContainer.Select(p => p.Pairing).ToList();
|
|
|
|
foreach (var g in editorInfo.Groupings)
|
|
g.Pairings = pairings.Where(p => p.Grouping.Value == g).Select(p => p.ID).ToList();
|
|
|
|
return new LadderInfo
|
|
{
|
|
Pairings = pairings,
|
|
Progressions = pairings.Where(p => p.Progression.Value != null).Select(p => new TournamentProgression(p.ID, p.Progression.Value.ID)).Concat(
|
|
pairings.Where(p => p.LosersProgression.Value != null).Select(p => new TournamentProgression(p.ID, p.LosersProgression.Value.ID, true)))
|
|
.ToList(),
|
|
Teams = editorInfo.Teams,
|
|
Groupings = editorInfo.Groupings
|
|
};
|
|
}
|
|
|
|
private void addPairing(MatchPairing pairing) => pairingsContainer.Add(new DrawableMatchPairing(pairing));
|
|
|
|
public MenuItem[] ContextMenuItems
|
|
{
|
|
get
|
|
{
|
|
if (!editorInfo.EditingEnabled)
|
|
return new MenuItem[0];
|
|
|
|
return new MenuItem[]
|
|
{
|
|
new OsuMenuItem("Create new match", MenuItemType.Highlighted, () =>
|
|
{
|
|
var pos = pairingsContainer.ToLocalSpace(GetContainingInputManager().CurrentState.Mouse.Position);
|
|
addPairing(new MatchPairing { Position = new Point((int)pos.X, (int)pos.Y) });
|
|
}),
|
|
};
|
|
}
|
|
}
|
|
|
|
private Cached layout = new Cached();
|
|
|
|
protected override void Update()
|
|
{
|
|
base.Update();
|
|
|
|
if (!layout.IsValid)
|
|
updateLayout();
|
|
}
|
|
|
|
private Color4 normalPathColour;
|
|
private Color4 losersPathColour;
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(OsuColour colours)
|
|
{
|
|
normalPathColour = colours.BlueDarker.Darken(2);
|
|
losersPathColour = colours.YellowDarker.Darken(2);
|
|
}
|
|
|
|
private void updateLayout()
|
|
{
|
|
paths.Clear();
|
|
headings.Clear();
|
|
|
|
int id = 1;
|
|
foreach (var pairing in pairingsContainer.OrderBy(d => d.Y).ThenBy(d => d.X))
|
|
{
|
|
pairing.Pairing.ID = id++;
|
|
|
|
if (pairing.Pairing.Progression.Value != null)
|
|
{
|
|
var dest = pairingsContainer.FirstOrDefault(p => p.Pairing == pairing.Pairing.Progression.Value);
|
|
|
|
if (dest == null)
|
|
// clean up outdated progressions.
|
|
pairing.Pairing.Progression.Value = null;
|
|
else
|
|
paths.Add(new ProgressionPath(pairing, dest) { Colour = pairing.Pairing.Losers ? losersPathColour : normalPathColour });
|
|
}
|
|
}
|
|
|
|
foreach (var group in editorInfo.Groupings)
|
|
{
|
|
var topPairing = pairingsContainer.Where(p => !p.Pairing.Losers && p.Pairing.Grouping.Value == group).OrderBy(p => p.Y).FirstOrDefault();
|
|
|
|
if (topPairing == null) continue;
|
|
|
|
headings.Add(new DrawableTournamentGrouping(group)
|
|
{
|
|
Position = headings.ToLocalSpace((topPairing.ScreenSpaceDrawQuad.TopLeft + topPairing.ScreenSpaceDrawQuad.TopRight) / 2),
|
|
Margin = new MarginPadding { Bottom = 10 },
|
|
Origin = Anchor.BottomCentre,
|
|
});
|
|
}
|
|
|
|
foreach (var group in editorInfo.Groupings)
|
|
{
|
|
var topPairing = pairingsContainer.Where(p => p.Pairing.Losers && p.Pairing.Grouping.Value == group).OrderBy(p => p.Y).FirstOrDefault();
|
|
|
|
if (topPairing == null) continue;
|
|
|
|
headings.Add(new DrawableTournamentGrouping(group, true)
|
|
{
|
|
Position = headings.ToLocalSpace((topPairing.ScreenSpaceDrawQuad.TopLeft + topPairing.ScreenSpaceDrawQuad.TopRight) / 2),
|
|
Margin = new MarginPadding { Bottom = 10 },
|
|
Origin = Anchor.BottomCentre,
|
|
});
|
|
}
|
|
|
|
layout.Validate();
|
|
}
|
|
|
|
public void RequestJoin(MatchPairing pairing, bool losers) => scrollContent.Add(new JoinRequestHandler(pairingsContainer, pairing, losers));
|
|
|
|
private class JoinRequestHandler : CompositeDrawable
|
|
{
|
|
private readonly Container<DrawableMatchPairing> pairingsContainer;
|
|
public readonly MatchPairing Source;
|
|
private readonly bool losers;
|
|
|
|
private ProgressionPath path;
|
|
|
|
public JoinRequestHandler(Container<DrawableMatchPairing> pairingsContainer, MatchPairing source, bool losers)
|
|
{
|
|
this.pairingsContainer = pairingsContainer;
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
Source = source;
|
|
this.losers = losers;
|
|
if (losers)
|
|
Source.LosersProgression.Value = null;
|
|
else
|
|
Source.Progression.Value = null;
|
|
}
|
|
|
|
private DrawableMatchPairing findTarget(InputState state) => pairingsContainer.FirstOrDefault(d => d.ReceiveMouseInputAt(state.Mouse.NativeState.Position));
|
|
|
|
public override bool ReceiveMouseInputAt(Vector2 screenSpacePos) => true;
|
|
|
|
protected override bool OnMouseMove(InputState state)
|
|
{
|
|
var found = findTarget(state);
|
|
|
|
if (found == path?.Destination)
|
|
return false;
|
|
|
|
path?.Expire();
|
|
path = null;
|
|
|
|
if (found == null)
|
|
return false;
|
|
|
|
AddInternal(path = new ProgressionPath(pairingsContainer.First(c => c.Pairing == Source), found)
|
|
{
|
|
Colour = Color4.Yellow,
|
|
});
|
|
|
|
return base.OnMouseMove(state);
|
|
}
|
|
|
|
protected override bool OnClick(InputState state)
|
|
{
|
|
var found = findTarget(state);
|
|
|
|
if (found != null)
|
|
{
|
|
if (found.Pairing != Source)
|
|
{
|
|
if (losers)
|
|
Source.LosersProgression.Value = found.Pairing;
|
|
else
|
|
Source.Progression.Value = found.Pairing;
|
|
}
|
|
|
|
Expire();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void Remove(MatchPairing pairing) => pairingsContainer.FirstOrDefault(p => p.Pairing == pairing)?.Remove();
|
|
}
|
|
}
|