Merge pull request #2789 from bidifx/lobby
update player list in lobby gracefully for better user experience
This commit is contained in:
@@ -41,6 +41,11 @@ namespace OpenRA.Widgets
|
|||||||
|
|
||||||
widget.ContentHeight = Math.Max(widget.ContentHeight, pos.Y + widget.ItemSpacing + w.Bounds.Height);
|
widget.ContentHeight = Math.Max(widget.ContentHeight, pos.Y + widget.ItemSpacing + w.Bounds.Height);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void AdjustChildren()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,18 @@ namespace OpenRA.Widgets
|
|||||||
if (widget.Children.Count == 0)
|
if (widget.Children.Count == 0)
|
||||||
widget.ContentHeight = widget.ItemSpacing;
|
widget.ContentHeight = widget.ItemSpacing;
|
||||||
|
|
||||||
w.Bounds.Y += widget.ContentHeight;
|
w.Bounds.Y = widget.ContentHeight;
|
||||||
widget.ContentHeight += w.Bounds.Height + widget.ItemSpacing;
|
widget.ContentHeight += w.Bounds.Height + widget.ItemSpacing;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void AdjustChildren()
|
||||||
|
{
|
||||||
|
widget.ContentHeight = widget.ItemSpacing;
|
||||||
|
foreach (var w in widget.Children)
|
||||||
|
{
|
||||||
|
w.Bounds.Y = widget.ContentHeight;
|
||||||
|
widget.ContentHeight += w.Bounds.Height + widget.ItemSpacing;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ using OpenRA.Graphics;
|
|||||||
|
|
||||||
namespace OpenRA.Widgets
|
namespace OpenRA.Widgets
|
||||||
{
|
{
|
||||||
public interface ILayout { void AdjustChild(Widget w); }
|
public interface ILayout { void AdjustChild(Widget w); void AdjustChildren(); }
|
||||||
|
|
||||||
public class ScrollPanelWidget : Widget
|
public class ScrollPanelWidget : Widget
|
||||||
{
|
{
|
||||||
@@ -50,6 +50,25 @@ namespace OpenRA.Widgets
|
|||||||
base.AddChild(child);
|
base.AddChild(child);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public override void RemoveChild(Widget child)
|
||||||
|
{
|
||||||
|
base.RemoveChild(child);
|
||||||
|
Layout.AdjustChildren();
|
||||||
|
Scroll(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ReplaceChild(Widget oldChild, Widget newChild)
|
||||||
|
{
|
||||||
|
|
||||||
|
oldChild.Removed();
|
||||||
|
newChild.Parent = this;
|
||||||
|
Children[Children.IndexOf(oldChild)] = newChild;
|
||||||
|
Layout.AdjustChildren();
|
||||||
|
Scroll(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public override void DrawOuter()
|
public override void DrawOuter()
|
||||||
{
|
{
|
||||||
if (!IsVisible())
|
if (!IsVisible())
|
||||||
|
|||||||
@@ -325,21 +325,24 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
|
|
||||||
void UpdatePlayerList()
|
void UpdatePlayerList()
|
||||||
{
|
{
|
||||||
// This causes problems for people who are in the process of editing their names (the widgets vanish from beneath them)
|
var idx = 0;
|
||||||
// Todo: handle this nicer
|
|
||||||
Players.RemoveChildren();
|
|
||||||
|
|
||||||
foreach (var kv in orderManager.LobbyInfo.Slots)
|
foreach (var kv in orderManager.LobbyInfo.Slots)
|
||||||
{
|
{
|
||||||
var key = kv.Key;
|
var key = kv.Key;
|
||||||
var slot = kv.Value;
|
var slot = kv.Value;
|
||||||
var client = orderManager.LobbyInfo.ClientInSlot(key);
|
var client = orderManager.LobbyInfo.ClientInSlot(key);
|
||||||
Widget template;
|
Widget template = null;
|
||||||
|
|
||||||
|
// get template for possible reuse
|
||||||
|
if (idx < Players.Children.Count)
|
||||||
|
template = Players.Children [idx];
|
||||||
|
|
||||||
// Empty slot
|
// Empty slot
|
||||||
if (client == null)
|
if (client == null)
|
||||||
{
|
{
|
||||||
template = EmptySlotTemplate.Clone();
|
if (template == null || template.Id != EmptySlotTemplate.Id)
|
||||||
|
template = EmptySlotTemplate.Clone();
|
||||||
|
|
||||||
Func<string> getText = () => slot.Closed ? "Closed" : "Open";
|
Func<string> getText = () => slot.Closed ? "Closed" : "Open";
|
||||||
var ready = orderManager.LocalClient.IsReady;
|
var ready = orderManager.LocalClient.IsReady;
|
||||||
|
|
||||||
@@ -367,7 +370,9 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
else if ((client.Index == orderManager.LocalClient.Index) ||
|
else if ((client.Index == orderManager.LocalClient.Index) ||
|
||||||
(client.Bot != null && Game.IsHost))
|
(client.Bot != null && Game.IsHost))
|
||||||
{
|
{
|
||||||
template = EditablePlayerTemplate.Clone();
|
if (template == null || template.Id != EditablePlayerTemplate.Id)
|
||||||
|
template = EditablePlayerTemplate.Clone();
|
||||||
|
|
||||||
var botReady = client.Bot != null && Game.IsHost && orderManager.LocalClient.IsReady;
|
var botReady = client.Bot != null && Game.IsHost && orderManager.LocalClient.IsReady;
|
||||||
var ready = botReady || client.IsReady;
|
var ready = botReady || client.IsReady;
|
||||||
|
|
||||||
@@ -422,7 +427,9 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{ // Non-editable player in slot
|
{ // Non-editable player in slot
|
||||||
template = NonEditablePlayerTemplate.Clone();
|
if (template == null || template.Id != NonEditablePlayerTemplate.Id)
|
||||||
|
template = NonEditablePlayerTemplate.Clone();
|
||||||
|
|
||||||
template.Get<LabelWidget>("NAME").GetText = () => client.Name;
|
template.Get<LabelWidget>("NAME").GetText = () => client.Name;
|
||||||
if (client.IsAdmin)
|
if (client.IsAdmin)
|
||||||
template.Get<LabelWidget>("NAME").Font = "Bold";
|
template.Get<LabelWidget>("NAME").Font = "Bold";
|
||||||
@@ -449,20 +456,32 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
}
|
}
|
||||||
|
|
||||||
template.IsVisible = () => true;
|
template.IsVisible = () => true;
|
||||||
Players.AddChild(template);
|
|
||||||
|
if (idx >= Players.Children.Count)
|
||||||
|
Players.AddChild(template);
|
||||||
|
else if (Players.Children[idx].Id != template.Id)
|
||||||
|
Players.ReplaceChild(Players.Children[idx], template);
|
||||||
|
|
||||||
|
idx++;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add spectators
|
// Add spectators
|
||||||
foreach (var client in orderManager.LobbyInfo.Clients.Where(client => client.Slot == null))
|
foreach (var client in orderManager.LobbyInfo.Clients.Where(client => client.Slot == null))
|
||||||
{
|
{
|
||||||
Widget template;
|
Widget template = null;
|
||||||
var c = client;
|
var c = client;
|
||||||
var ready = c.IsReady;
|
var ready = c.IsReady;
|
||||||
|
|
||||||
|
// get template for possible reuse
|
||||||
|
if (idx < Players.Children.Count)
|
||||||
|
template = Players.Children[idx];
|
||||||
|
|
||||||
// Editable spectator
|
// Editable spectator
|
||||||
if (c.Index == orderManager.LocalClient.Index)
|
if (c.Index == orderManager.LocalClient.Index)
|
||||||
{
|
{
|
||||||
template = EditableSpectatorTemplate.Clone();
|
if (template == null || template.Id != EditableSpectatorTemplate.Id)
|
||||||
|
template = EditableSpectatorTemplate.Clone();
|
||||||
|
|
||||||
var name = template.Get<TextFieldWidget>("NAME");
|
var name = template.Get<TextFieldWidget>("NAME");
|
||||||
name.IsDisabled = () => ready;
|
name.IsDisabled = () => ready;
|
||||||
LobbyUtils.SetupNameWidget(orderManager, c, name);
|
LobbyUtils.SetupNameWidget(orderManager, c, name);
|
||||||
@@ -481,7 +500,9 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
// Non-editable spectator
|
// Non-editable spectator
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
template = NonEditableSpectatorTemplate.Clone();
|
if (template == null || template.Id != NonEditableSpectatorTemplate.Id)
|
||||||
|
template = NonEditableSpectatorTemplate.Clone();
|
||||||
|
|
||||||
template.Get<LabelWidget>("NAME").GetText = () => c.Name;
|
template.Get<LabelWidget>("NAME").GetText = () => c.Name;
|
||||||
if (client.IsAdmin)
|
if (client.IsAdmin)
|
||||||
template.Get<LabelWidget>("NAME").Font = "Bold";
|
template.Get<LabelWidget>("NAME").Font = "Bold";
|
||||||
@@ -497,19 +518,40 @@ namespace OpenRA.Mods.RA.Widgets.Logic
|
|||||||
}
|
}
|
||||||
|
|
||||||
template.IsVisible = () => true;
|
template.IsVisible = () => true;
|
||||||
Players.AddChild(template);
|
|
||||||
|
if (idx >= Players.Children.Count)
|
||||||
|
Players.AddChild(template);
|
||||||
|
else if (Players.Children[idx].Id != template.Id)
|
||||||
|
Players.ReplaceChild(Players.Children[idx], template);
|
||||||
|
|
||||||
|
idx++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Spectate button
|
// Spectate button
|
||||||
if (orderManager.LocalClient.Slot != null)
|
if (orderManager.LocalClient.Slot != null)
|
||||||
{
|
{
|
||||||
var spec = NewSpectatorTemplate.Clone();
|
Widget spec = null;
|
||||||
|
if (idx < Players.Children.Count)
|
||||||
|
spec = Players.Children[idx];
|
||||||
|
if (spec == null || spec.Id != NewSpectatorTemplate.Id)
|
||||||
|
spec = NewSpectatorTemplate.Clone();
|
||||||
|
|
||||||
var btn = spec.Get<ButtonWidget>("SPECTATE");
|
var btn = spec.Get<ButtonWidget>("SPECTATE");
|
||||||
btn.OnClick = () => orderManager.IssueOrder(Order.Command("spectate"));
|
btn.OnClick = () => orderManager.IssueOrder(Order.Command("spectate"));
|
||||||
btn.IsDisabled = () => orderManager.LocalClient.IsReady;
|
btn.IsDisabled = () => orderManager.LocalClient.IsReady;
|
||||||
spec.IsVisible = () => true;
|
spec.IsVisible = () => true;
|
||||||
Players.AddChild(spec);
|
|
||||||
|
if (idx >= Players.Children.Count)
|
||||||
|
Players.AddChild(spec);
|
||||||
|
else if (Players.Children[idx].Id != spec.Id)
|
||||||
|
Players.ReplaceChild(Players.Children[idx], spec);
|
||||||
|
|
||||||
|
idx++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
while (Players.Children.Count > idx)
|
||||||
|
Players.RemoveChild(Players.Children[idx]);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CycleReady()
|
void CycleReady()
|
||||||
|
|||||||
Reference in New Issue
Block a user