Map Editor - Tiles' filters multiple selection
This commit is contained in:
@@ -38,26 +38,17 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
}
|
||||
|
||||
readonly DropDownButtonWidget ownersDropDown;
|
||||
readonly ScrollPanelWidget panel;
|
||||
readonly ScrollItemWidget itemTemplate;
|
||||
readonly Ruleset mapRules;
|
||||
|
||||
readonly ActorSelectorActor[] allActors;
|
||||
|
||||
PlayerReference selectedOwner;
|
||||
|
||||
[ObjectCreator.UseCtor]
|
||||
public ActorSelectorLogic(Widget widget, World world, WorldRenderer worldRenderer)
|
||||
: base(widget, world, worldRenderer)
|
||||
: base(widget, world, worldRenderer, "ACTORTEMPLATE_LIST", "ACTORPREVIEW_TEMPLATE")
|
||||
{
|
||||
mapRules = world.Map.Rules;
|
||||
|
||||
ownersDropDown = widget.Get<DropDownButtonWidget>("OWNERS_DROPDOWN");
|
||||
|
||||
panel = widget.Get<ScrollPanelWidget>("ACTORTEMPLATE_LIST");
|
||||
itemTemplate = panel.Get<ScrollItemWidget>("ACTORPREVIEW_TEMPLATE");
|
||||
panel.Layout = new GridLayout(panel);
|
||||
|
||||
var editorLayer = world.WorldActor.Trait<EditorActorLayer>();
|
||||
|
||||
selectedOwner = editorLayer.Players.Players.Values.First();
|
||||
@@ -136,10 +127,9 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
FilteredCategories.Add(c);
|
||||
}
|
||||
|
||||
var searchTextField = widget.Get<TextFieldWidget>("SEARCH_TEXTFIELD");
|
||||
searchTextField.OnTextEdited = () =>
|
||||
SearchTextField.OnTextEdited = () =>
|
||||
{
|
||||
searchFilter = searchTextField.Text.Trim();
|
||||
searchFilter = SearchTextField.Text.Trim();
|
||||
FilteredCategories.Clear();
|
||||
|
||||
if (!string.IsNullOrEmpty(searchFilter))
|
||||
@@ -155,40 +145,6 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
InitializePreviews();
|
||||
};
|
||||
|
||||
searchTextField.OnEscKey = () =>
|
||||
{
|
||||
searchTextField.Text = "";
|
||||
searchTextField.YieldKeyboardFocus();
|
||||
return true;
|
||||
};
|
||||
|
||||
var actorCategorySelector = widget.Get<DropDownButtonWidget>("CATEGORIES_DROPDOWN");
|
||||
actorCategorySelector.GetText = () =>
|
||||
{
|
||||
if (SelectedCategories.Count == 0)
|
||||
return "None";
|
||||
|
||||
if (!string.IsNullOrEmpty(searchFilter))
|
||||
return "Search Results";
|
||||
|
||||
if (SelectedCategories.Count == 1)
|
||||
return SelectedCategories.First();
|
||||
|
||||
if (SelectedCategories.Count == allCategories.Length)
|
||||
return "All";
|
||||
|
||||
return "Multiple";
|
||||
};
|
||||
|
||||
actorCategorySelector.OnMouseDown = _ =>
|
||||
{
|
||||
if (searchTextField != null)
|
||||
searchTextField.YieldKeyboardFocus();
|
||||
|
||||
actorCategorySelector.RemovePanel();
|
||||
actorCategorySelector.AttachPanel(CreateCategoriesPanel(panel));
|
||||
};
|
||||
|
||||
InitializePreviews();
|
||||
}
|
||||
|
||||
@@ -202,7 +158,7 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
protected override void InitializePreviews()
|
||||
{
|
||||
panel.RemoveChildren();
|
||||
Panel.RemoveChildren();
|
||||
if (!SelectedCategories.Any())
|
||||
return;
|
||||
|
||||
@@ -224,7 +180,7 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
try
|
||||
{
|
||||
var item = ScrollItemWidget.Setup(itemTemplate,
|
||||
var item = ScrollItemWidget.Setup(ItemTemplate,
|
||||
() => { var brush = Editor.CurrentBrush as EditorActorBrush; return brush != null && brush.Actor == actor; },
|
||||
() => Editor.SetBrush(new EditorActorBrush(Editor, actor, selectedOwner, WorldRenderer)));
|
||||
|
||||
@@ -233,8 +189,8 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
// Scale templates to fit within the panel
|
||||
var scale = 1f;
|
||||
if (scale * preview.IdealPreviewSize.X > itemTemplate.Bounds.Width)
|
||||
scale = (itemTemplate.Bounds.Width - panel.ItemSpacing) / (float)preview.IdealPreviewSize.X;
|
||||
if (scale * preview.IdealPreviewSize.X > ItemTemplate.Bounds.Width)
|
||||
scale = (ItemTemplate.Bounds.Width - Panel.ItemSpacing) / (float)preview.IdealPreviewSize.X;
|
||||
|
||||
preview.GetScale = () => scale;
|
||||
preview.Bounds.Width = (int)(scale * preview.IdealPreviewSize.X);
|
||||
@@ -246,7 +202,7 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
item.GetTooltipText = () => a.Tooltip;
|
||||
|
||||
panel.AddChild(item);
|
||||
Panel.AddChild(item);
|
||||
}
|
||||
catch
|
||||
{
|
||||
|
||||
@@ -11,6 +11,7 @@
|
||||
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using OpenRA.Graphics;
|
||||
using OpenRA.Widgets;
|
||||
|
||||
@@ -19,9 +20,12 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
public abstract class CommonSelectorLogic : ChromeLogic
|
||||
{
|
||||
protected readonly Widget Widget;
|
||||
protected readonly TextFieldWidget SearchTextField;
|
||||
protected readonly World World;
|
||||
protected readonly WorldRenderer WorldRenderer;
|
||||
protected readonly EditorViewportControllerWidget Editor;
|
||||
protected readonly ScrollPanelWidget Panel;
|
||||
protected readonly ScrollItemWidget ItemTemplate;
|
||||
|
||||
protected readonly HashSet<string> SelectedCategories = new HashSet<string>();
|
||||
protected readonly List<string> FilteredCategories = new List<string>();
|
||||
@@ -29,17 +33,55 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
protected string[] allCategories;
|
||||
protected string searchFilter;
|
||||
|
||||
public CommonSelectorLogic(Widget widget, World world, WorldRenderer worldRenderer)
|
||||
public CommonSelectorLogic(Widget widget, World world, WorldRenderer worldRenderer, string templateListId, string previewTemplateId)
|
||||
{
|
||||
this.Widget = widget;
|
||||
this.World = world;
|
||||
this.WorldRenderer = worldRenderer;
|
||||
Editor = widget.Parent.Get<EditorViewportControllerWidget>("MAP_EDITOR");
|
||||
Panel = widget.Get<ScrollPanelWidget>(templateListId);
|
||||
ItemTemplate = Panel.Get<ScrollItemWidget>(previewTemplateId);
|
||||
Panel.Layout = new GridLayout(Panel);
|
||||
|
||||
SearchTextField = widget.Get<TextFieldWidget>("SEARCH_TEXTFIELD");
|
||||
SearchTextField.OnEscKey = () =>
|
||||
{
|
||||
SearchTextField.Text = "";
|
||||
SearchTextField.YieldKeyboardFocus();
|
||||
return true;
|
||||
};
|
||||
|
||||
var categorySelector = widget.Get<DropDownButtonWidget>("CATEGORIES_DROPDOWN");
|
||||
categorySelector.GetText = () =>
|
||||
{
|
||||
if (SelectedCategories.Count == 0)
|
||||
return "None";
|
||||
|
||||
if (!string.IsNullOrEmpty(searchFilter))
|
||||
return "Search Results";
|
||||
|
||||
if (SelectedCategories.Count == 1)
|
||||
return SelectedCategories.First();
|
||||
|
||||
if (SelectedCategories.Count == allCategories.Length)
|
||||
return "All";
|
||||
|
||||
return "Multiple";
|
||||
};
|
||||
|
||||
categorySelector.OnMouseDown = _ =>
|
||||
{
|
||||
if (SearchTextField != null)
|
||||
SearchTextField.YieldKeyboardFocus();
|
||||
|
||||
categorySelector.RemovePanel();
|
||||
categorySelector.AttachPanel(CreateCategoriesPanel(Panel));
|
||||
};
|
||||
}
|
||||
|
||||
protected Widget CreateCategoriesPanel(ScrollPanelWidget panel)
|
||||
{
|
||||
var categoriesPanel = Ui.LoadWidget("ACTOR_CATEGORY_FILTER_PANEL", null, new WidgetArgs());
|
||||
var categoriesPanel = Ui.LoadWidget("CATEGORY_FILTER_PANEL", null, new WidgetArgs());
|
||||
var categoryTemplate = categoriesPanel.Get<CheckboxWidget>("CATEGORY_TEMPLATE");
|
||||
|
||||
var selectButtons = categoriesPanel.Get<ContainerWidget>("SELECT_CATEGORIES_BUTTONS");
|
||||
|
||||
@@ -35,87 +35,44 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
}
|
||||
|
||||
readonly TileSet tileset;
|
||||
readonly ScrollPanelWidget panel;
|
||||
readonly ScrollItemWidget itemTemplate;
|
||||
readonly TileSelectorTemplate[] allTemplates;
|
||||
|
||||
string selectedCategory;
|
||||
string userSelectedCategory;
|
||||
|
||||
[ObjectCreator.UseCtor]
|
||||
public TileSelectorLogic(Widget widget, World world, WorldRenderer worldRenderer) :
|
||||
base(widget, world, worldRenderer)
|
||||
public TileSelectorLogic(Widget widget, World world, WorldRenderer worldRenderer)
|
||||
: base(widget, world, worldRenderer, "TILETEMPLATE_LIST", "TILEPREVIEW_TEMPLATE")
|
||||
{
|
||||
tileset = world.Map.Rules.TileSet;
|
||||
|
||||
panel = widget.Get<ScrollPanelWidget>("TILETEMPLATE_LIST");
|
||||
itemTemplate = panel.Get<ScrollItemWidget>("TILEPREVIEW_TEMPLATE");
|
||||
panel.Layout = new GridLayout(panel);
|
||||
|
||||
allTemplates = tileset.Templates.Values.Select(t => new TileSelectorTemplate(t)).ToArray();
|
||||
|
||||
var orderedCategories = allTemplates.SelectMany(t => t.Categories)
|
||||
allCategories = allTemplates.SelectMany(t => t.Categories)
|
||||
.Distinct()
|
||||
.OrderBy(CategoryOrder)
|
||||
.ToArray();
|
||||
|
||||
var searchTextField = widget.Get<TextFieldWidget>("SEARCH_TEXTFIELD");
|
||||
searchTextField.OnTextEdited = () =>
|
||||
foreach (var c in allCategories)
|
||||
{
|
||||
searchFilter = searchTextField.Text.Trim();
|
||||
selectedCategory = string.IsNullOrEmpty(searchFilter) ? userSelectedCategory : null;
|
||||
SelectedCategories.Add(c);
|
||||
FilteredCategories.Add(c);
|
||||
}
|
||||
|
||||
InitializePreviews();
|
||||
};
|
||||
|
||||
searchTextField.OnEscKey = () =>
|
||||
SearchTextField.OnTextEdited = () =>
|
||||
{
|
||||
searchTextField.Text = "";
|
||||
searchTextField.YieldKeyboardFocus();
|
||||
return true;
|
||||
};
|
||||
searchFilter = SearchTextField.Text.Trim();
|
||||
FilteredCategories.Clear();
|
||||
|
||||
Func<string, string> categoryTitle = s => s != null ? s : "Search Results";
|
||||
Func<string, ScrollItemWidget, ScrollItemWidget> setupItem = (option, template) =>
|
||||
{
|
||||
var item = ScrollItemWidget.Setup(template, () => selectedCategory == option, () =>
|
||||
{
|
||||
selectedCategory = option;
|
||||
if (option != null)
|
||||
userSelectedCategory = option;
|
||||
|
||||
InitializePreviews();
|
||||
});
|
||||
|
||||
var title = categoryTitle(option);
|
||||
item.Get<LabelWidget>("LABEL").GetText = () => title;
|
||||
return item;
|
||||
};
|
||||
|
||||
var tileCategorySelector = widget.Get<DropDownButtonWidget>("CATEGORIES_DROPDOWN");
|
||||
tileCategorySelector.OnClick = () =>
|
||||
{
|
||||
if (searchTextField != null)
|
||||
searchTextField.YieldKeyboardFocus();
|
||||
|
||||
var categories = orderedCategories.AsEnumerable();
|
||||
if (!string.IsNullOrEmpty(searchFilter))
|
||||
{
|
||||
var filteredCategories = allTemplates.Where(t => t.SearchTerms.Any(
|
||||
FilteredCategories.AddRange(
|
||||
allTemplates.Where(t => t.SearchTerms.Any(
|
||||
s => s.IndexOf(searchFilter, StringComparison.OrdinalIgnoreCase) >= 0))
|
||||
.SelectMany(t => t.Categories)
|
||||
.Distinct()
|
||||
.OrderBy(CategoryOrder);
|
||||
categories = new string[] { null }.Concat(filteredCategories);
|
||||
}
|
||||
.OrderBy(CategoryOrder));
|
||||
else
|
||||
FilteredCategories.AddRange(allCategories);
|
||||
|
||||
tileCategorySelector.ShowDropDown("LABEL_DROPDOWN_TEMPLATE", 270, categories, setupItem);
|
||||
InitializePreviews();
|
||||
};
|
||||
|
||||
var actorCategorySelector = widget.Get<DropDownButtonWidget>("CATEGORIES_DROPDOWN");
|
||||
actorCategorySelector.GetText = () => categoryTitle(selectedCategory);
|
||||
|
||||
selectedCategory = userSelectedCategory = orderedCategories.First();
|
||||
InitializePreviews();
|
||||
}
|
||||
|
||||
@@ -127,18 +84,20 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
protected override void InitializePreviews()
|
||||
{
|
||||
panel.RemoveChildren();
|
||||
Panel.RemoveChildren();
|
||||
if (!SelectedCategories.Any())
|
||||
return;
|
||||
|
||||
foreach (var t in allTemplates)
|
||||
{
|
||||
if (selectedCategory != null && !t.Categories.Contains(selectedCategory))
|
||||
if (!SelectedCategories.Overlaps(t.Categories))
|
||||
continue;
|
||||
|
||||
if (!string.IsNullOrEmpty(searchFilter) && !t.SearchTerms.Any(s => s.IndexOf(searchFilter, StringComparison.OrdinalIgnoreCase) >= 0))
|
||||
continue;
|
||||
|
||||
var tileId = t.Template.Id;
|
||||
var item = ScrollItemWidget.Setup(itemTemplate,
|
||||
var item = ScrollItemWidget.Setup(ItemTemplate,
|
||||
() => { var brush = Editor.CurrentBrush as EditorTileBrush; return brush != null && brush.Template == tileId; },
|
||||
() => Editor.SetBrush(new EditorTileBrush(Editor, tileId, WorldRenderer)));
|
||||
|
||||
@@ -149,7 +108,7 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
|
||||
// Scale templates to fit within the panel
|
||||
var scale = 1f;
|
||||
while (scale * bounds.Width > itemTemplate.Bounds.Width)
|
||||
while (scale * bounds.Width > ItemTemplate.Bounds.Width)
|
||||
scale /= 2;
|
||||
|
||||
preview.Template = template;
|
||||
@@ -162,7 +121,7 @@ namespace OpenRA.Mods.Common.Widgets.Logic
|
||||
item.IsVisible = () => true;
|
||||
item.GetTooltipText = () => t.Tooltip;
|
||||
|
||||
panel.AddChild(item);
|
||||
Panel.AddChild(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -569,7 +569,7 @@ Container@EDITOR_WORLD_ROOT:
|
||||
Font: Bold
|
||||
Contrast: true
|
||||
|
||||
ScrollPanel@ACTOR_CATEGORY_FILTER_PANEL:
|
||||
ScrollPanel@CATEGORY_FILTER_PANEL:
|
||||
Width: 190
|
||||
Children:
|
||||
Container@SELECT_CATEGORIES_BUTTONS:
|
||||
|
||||
@@ -548,7 +548,7 @@ Container@EDITOR_WORLD_ROOT:
|
||||
Font: Bold
|
||||
Contrast: true
|
||||
|
||||
ScrollPanel@ACTOR_CATEGORY_FILTER_PANEL:
|
||||
ScrollPanel@CATEGORY_FILTER_PANEL:
|
||||
Width: 170
|
||||
Children:
|
||||
Container@SELECT_CATEGORIES_BUTTONS:
|
||||
|
||||
Reference in New Issue
Block a user