| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030 |
- // Urho3D editor attribute inspector window handling
- #include "Scripts/Editor/AttributeEditor.as"
- Window@ attributeInspectorWindow;
- UIElement@ parentContainer;
- UIElement@ inspectorLockButton;
- bool applyMaterialList = true;
- bool attributesDirty = false;
- bool attributesFullDirty = false;
- const String STRIKED_OUT = "——"; // Two unicode EM DASH (U+2014)
- const StringHash NODE_IDS_VAR("NodeIDs");
- const StringHash COMPONENT_IDS_VAR("ComponentIDs");
- const StringHash UI_ELEMENT_IDS_VAR("UIElementIDs");
- const int LABEL_WIDTH = 30;
- // Constants for accessing xmlResources
- Array<XMLFile@> xmlResources;
- const uint ATTRIBUTE_RES = 0;
- const uint VARIABLE_RES = 1;
- const uint STYLE_RES = 2;
- const uint TAGS_RES = 3;
- uint nodeContainerIndex = M_MAX_UNSIGNED;
- uint componentContainerStartIndex = 0;
- uint elementContainerIndex = M_MAX_UNSIGNED;
- // Node or UIElement hash-to-varname reverse mapping
- VariantMap globalVarNames;
- bool inspectorLocked = false;
- void InitXMLResources()
- {
- String[] resources = { "UI/EditorInspector_Attribute.xml", "UI/EditorInspector_Variable.xml",
- "UI/EditorInspector_Style.xml", "UI/EditorInspector_Tags.xml" };
- for (uint i = 0; i < resources.length; ++i)
- xmlResources.Push(cache.GetResource("XMLFile", resources[i]));
- }
- /// Delete all child containers in the inspector list.
- void DeleteAllContainers()
- {
- parentContainer.RemoveAllChildren();
- nodeContainerIndex = M_MAX_UNSIGNED;
- componentContainerStartIndex = 0;
- elementContainerIndex = M_MAX_UNSIGNED;
- }
- /// Get container at the specified index in the inspector list, the container must be created before.
- UIElement@ GetContainer(uint index)
- {
- return parentContainer.children[index];
- }
- /// Get node container in the inspector list, create the container if it is not yet available.
- UIElement@ GetNodeContainer()
- {
- if (nodeContainerIndex != M_MAX_UNSIGNED)
- return GetContainer(nodeContainerIndex);
- nodeContainerIndex = parentContainer.numChildren;
- parentContainer.LoadChildXML(xmlResources[ATTRIBUTE_RES], uiStyle);
- UIElement@ container = GetContainer(nodeContainerIndex);
- container.LoadChildXML(xmlResources[VARIABLE_RES], uiStyle);
- SubscribeToEvent(container.GetChild("ResetToDefault", true), "Released", "HandleResetToDefault");
- SubscribeToEvent(container.GetChild("NewVarDropDown", true), "ItemSelected", "CreateNodeVariable");
- SubscribeToEvent(container.GetChild("DeleteVarButton", true), "Released", "DeleteNodeVariable");
- ++componentContainerStartIndex;
- parentContainer.LoadChildXML(xmlResources[TAGS_RES], uiStyle);
- parentContainer.GetChild("TagsLabel", true).SetFixedWidth(LABEL_WIDTH);
- LineEdit@ tagEdit = parentContainer.GetChild("TagsEdit", true);
- SubscribeToEvent(tagEdit, "TextChanged", "HandleTagsEdit");
- UIElement@ tagSelect = parentContainer.GetChild("TagsSelect", true);
- SubscribeToEvent(tagSelect, "Released", "HandleTagsSelect");
- ++componentContainerStartIndex;
- return container;
- }
- /// Get component container at the specified index, create the container if it is not yet available at the specified index.
- UIElement@ GetComponentContainer(uint index)
- {
- if (componentContainerStartIndex + index < parentContainer.numChildren)
- return GetContainer(componentContainerStartIndex + index);
- UIElement@ container;
- for (uint i = parentContainer.numChildren; i <= componentContainerStartIndex + index; ++i)
- {
- parentContainer.LoadChildXML(xmlResources[ATTRIBUTE_RES], uiStyle);
- container = GetContainer(i);
- SubscribeToEvent(container.GetChild("ResetToDefault", true), "Released", "HandleResetToDefault");
- }
- return container;
- }
- /// Get UI-element container, create the container if it is not yet available.
- UIElement@ GetUIElementContainer()
- {
- if (elementContainerIndex != M_MAX_UNSIGNED)
- return GetContainer(elementContainerIndex);
- elementContainerIndex = parentContainer.numChildren;
- parentContainer.LoadChildXML(xmlResources[ATTRIBUTE_RES], uiStyle);
- parentContainer.LoadChildXML(xmlResources[TAGS_RES], uiStyle);
- parentContainer.GetChild("TagsLabel", true).SetFixedWidth(LABEL_WIDTH);
- UIElement@ container = GetContainer(elementContainerIndex);
- container.LoadChildXML(xmlResources[VARIABLE_RES], uiStyle);
- container.LoadChildXML(xmlResources[STYLE_RES], uiStyle);
- DropDownList@ styleList = container.GetChild("StyleDropDown", true);
- styleList.placeholderText = STRIKED_OUT;
- styleList.parent.GetChild("StyleDropDownLabel").SetFixedWidth(LABEL_WIDTH);
- PopulateStyleList(styleList);
- SubscribeToEvent(container.GetChild("ResetToDefault", true), "Released", "HandleResetToDefault");
- SubscribeToEvent(container.GetChild("NewVarDropDown", true), "ItemSelected", "CreateUIElementVariable");
- SubscribeToEvent(container.GetChild("DeleteVarButton", true), "Released", "DeleteUIElementVariable");
- SubscribeToEvent(styleList, "ItemSelected", "HandleStyleItemSelected");
- LineEdit@ tagEdit = parentContainer.GetChild("TagsEdit", true);
- SubscribeToEvent(tagEdit, "TextChanged", "HandleTagsEdit");
- UIElement@ tagSelect = parentContainer.GetChild("TagsSelect", true);
- SubscribeToEvent(tagSelect, "Released", "HandleTagsSelect");
- return container;
- }
- void CreateAttributeInspectorWindow()
- {
- if (attributeInspectorWindow !is null)
- return;
- InitResourcePicker();
- InitVectorStructs();
- InitXMLResources();
- attributeInspectorWindow = LoadEditorUI("UI/EditorInspectorWindow.xml");
- parentContainer = attributeInspectorWindow.GetChild("ParentContainer");
- ui.root.AddChild(attributeInspectorWindow);
- int height = Min(ui.root.height - 60, 500);
- attributeInspectorWindow.SetSize(344, height);
- attributeInspectorWindow.SetPosition(ui.root.width - 10 - attributeInspectorWindow.width, 100);
- attributeInspectorWindow.opacity = uiMaxOpacity;
- attributeInspectorWindow.BringToFront();
- inspectorLockButton = attributeInspectorWindow.GetChild("LockButton", true);
- UpdateAttributeInspector();
- SubscribeToEvent(inspectorLockButton, "Pressed", "ToggleInspectorLock");
- SubscribeToEvent(attributeInspectorWindow.GetChild("CloseButton", true), "Pressed", "HideAttributeInspectorWindow");
- SubscribeToEvent(attributeInspectorWindow, "LayoutUpdated", "HandleWindowLayoutUpdated");
- }
- void DisableInspectorLock()
- {
- inspectorLocked = false;
- if (inspectorLockButton !is null)
- inspectorLockButton.style = "Button";
- UpdateAttributeInspector(true);
- }
- void EnableInspectorLock()
- {
- inspectorLocked = true;
- if (inspectorLockButton !is null)
- inspectorLockButton.style = "ToggledButton";
- }
- void ToggleInspectorLock()
- {
- if (inspectorLocked)
- DisableInspectorLock();
- else
- EnableInspectorLock();
- }
- bool ToggleAttributeInspectorWindow()
- {
- if (attributeInspectorWindow.visible == false)
- ShowAttributeInspectorWindow();
- else
- HideAttributeInspectorWindow();
- return true;
- }
- void ShowAttributeInspectorWindow()
- {
- attributeInspectorWindow.visible = true;
- attributeInspectorWindow.BringToFront();
- }
- void HideAttributeInspectorWindow()
- {
- attributeInspectorWindow.visible = false;
- }
- /// Handle main window layout updated event by positioning elements that needs manually-positioning (elements that are children of UI-element container with "Free" layout-mode).
- void HandleWindowLayoutUpdated()
- {
- // When window resize and so the list's width is changed, adjust the 'Is enabled' container width and icon panel width so that their children stay at the right most position
- for (uint i = 0; i < parentContainer.numChildren; ++i)
- {
- UIElement@ container = GetContainer(i);
- ListView@ list = container.GetChild("AttributeList");
- if (list is null)
- continue;
- int width = list.width;
- // Adjust the icon panel's width
- UIElement@ panel = container.GetChild("IconsPanel", true);
- if (panel !is null)
- panel.width = width;
- // At the moment, only 'Is Enabled' container (place-holder + check box) is being created as child of the list view instead of as list item
- for (uint j = 0; j < list.numChildren; ++j)
- {
- UIElement@ element = list.children[j];
- if (!element.internal)
- {
- element.SetFixedWidth(width);
- UIElement@ title = container.GetChild("TitleText");
- element.position = IntVector2(0, (title.screenPosition - list.screenPosition).y);
- // Adjust icon panel's width one more time to cater for the space occupied by 'Is Enabled' check box
- if (panel !is null)
- panel.width = width - element.children[1].width - panel.layoutSpacing;
- break;
- }
- }
- }
- }
- Array<Serializable@> ToSerializableArray(Array<Node@> nodes)
- {
- Array<Serializable@> serializables;
- for (uint i = 0; i < nodes.length; ++i)
- serializables.Push(nodes[i]);
- return serializables;
- }
- /// Update the whole attribute inspector window, when fullUpdate flag is set to true then first delete all the containers and repopulate them again from scratch.
- /// The fullUpdate flag is usually set to true when the structure of the attributes are different than the existing attributes in the list.
- void UpdateAttributeInspector(bool fullUpdate = true)
- {
- if (inspectorLocked)
- return;
- attributesDirty = false;
- if (fullUpdate)
- attributesFullDirty = false;
- // If full update delete all containers and add them back as necessary
- if (fullUpdate)
- DeleteAllContainers();
- if (!editNodes.empty)
- {
- UIElement@ container = GetNodeContainer();
- Text@ nodeTitle = container.GetChild("TitleText");
- String nodeType;
- if (editNode !is null)
- {
- String idStr;
- if (editNode.id >= FIRST_LOCAL_ID)
- idStr = " (Local ID " + String(editNode.id) + ")";
- else
- idStr = " (ID " + String(editNode.id) + ")";
- nodeType = editNode.typeName;
- nodeTitle.text = nodeType + idStr;
- LineEdit@ tagEdit = parentContainer.GetChild("TagsEdit", true);
- tagEdit.text = Join(editNode.tags, ";");
- }
- else
- {
- nodeType = editNodes[0].typeName;
- nodeTitle.text = nodeType + " (ID " + STRIKED_OUT + " : " + editNodes.length + "x)";
- }
- IconizeUIElement(nodeTitle, nodeType);
- ListView@ list = container.GetChild("AttributeList");
- Array<Serializable@> nodes = ToSerializableArray(editNodes);
- UpdateAttributes(nodes, list, fullUpdate);
- if (fullUpdate)
- {
- //\todo Avoid hardcoding
- // Resize the node editor according to the number of variables, up to a certain maximum
- uint maxAttrs = Clamp(list.contentElement.numChildren, MIN_NODE_ATTRIBUTES, MAX_NODE_ATTRIBUTES);
- list.SetFixedHeight(maxAttrs * ATTR_HEIGHT + 2);
- container.SetFixedHeight(maxAttrs * ATTR_HEIGHT + 58);
- }
- // Set icon's target in the icon panel
- SetAttributeEditorID(container.GetChild("ResetToDefault", true), nodes);
- }
- if (!editComponents.empty)
- {
- uint numEditableComponents = editComponents.length / numEditableComponentsPerNode;
- String multiplierText;
- if (numEditableComponents > 1)
- multiplierText = " (" + numEditableComponents + "x)";
- for (uint j = 0; j < numEditableComponentsPerNode; ++j)
- {
- UIElement@ container = GetComponentContainer(j);
- Text@ componentTitle = container.GetChild("TitleText");
- componentTitle.text = GetComponentTitle(editComponents[j * numEditableComponents]) + multiplierText;
- IconizeUIElement(componentTitle, editComponents[j * numEditableComponents].typeName);
- SetIconEnabledColor(componentTitle, editComponents[j * numEditableComponents].enabledEffective);
- Array<Serializable@> components;
- for (uint i = 0; i < numEditableComponents; ++i)
- {
- Component@ component = editComponents[j * numEditableComponents + i];
- components.Push(component);
- }
- UpdateAttributes(components, container.GetChild("AttributeList"), fullUpdate);
- SetAttributeEditorID(container.GetChild("ResetToDefault", true), components);
- }
- }
- if (!editUIElements.empty)
- {
- UIElement@ container = GetUIElementContainer();
- Text@ titleText = container.GetChild("TitleText");
- DropDownList@ styleList = container.GetChild("StyleDropDown", true);
- String elementType;
- if (editUIElement !is null)
- {
- elementType = editUIElement.typeName;
- titleText.text = elementType + " [ID " + GetUIElementID(editUIElement).ToString() + "]";
- SetStyleListSelection(styleList, editUIElement.style);
- LineEdit@ tagEdit = parentContainer.GetChild("TagsEdit", true);
- tagEdit.text = Join(editUIElement.tags, ";");
- }
- else
- {
- elementType = editUIElements[0].typeName;
- String appliedStyle = cast<UIElement>(editUIElements[0]).style;
- bool sameType = true;
- bool sameStyle = true;
- for (uint i = 1; i < editUIElements.length; ++i)
- {
- if (editUIElements[i].typeName != elementType)
- {
- sameType = false;
- sameStyle = false;
- break;
- }
- if (sameStyle && cast<UIElement>(editUIElements[i]).style != appliedStyle)
- sameStyle = false;
- }
- titleText.text = (sameType ? elementType : "Mixed type") + " [ID " + STRIKED_OUT + " : " + editUIElements.length + "x]";
- SetStyleListSelection(SetEditable(styleList, sameStyle), sameStyle ? appliedStyle : STRIKED_OUT);
- if (!sameType)
- elementType.Clear(); // No icon
- }
- IconizeUIElement(titleText, elementType);
- UpdateAttributes(editUIElements, container.GetChild("AttributeList"), fullUpdate);
- SetAttributeEditorID(container.GetChild("ResetToDefault", true), editUIElements);
- }
- if (parentContainer.numChildren > 0)
- UpdateAttributeInspectorIcons();
- else
- {
- // No editables, insert a dummy component container to show the information
- Text@ titleText = GetComponentContainer(0).GetChild("TitleText");
- titleText.text = "Select editable objects";
- titleText.autoLocalizable = true;
- UIElement@ panel = titleText.GetChild("IconsPanel");
- panel.visible = false;
- }
- // Adjust size and position of manual-layout UI-elements, e.g. icons panel
- if (fullUpdate)
- HandleWindowLayoutUpdated();
- }
- /// Update the attribute list of the node container.
- void UpdateNodeAttributes()
- {
- bool fullUpdate = false;
- UpdateAttributes(ToSerializableArray(editNodes), GetNodeContainer().GetChild("AttributeList"), fullUpdate);
- if (fullUpdate)
- HandleWindowLayoutUpdated();
- }
- /// Update the icons enabled color based on the internal state of the objects.
- /// For node and component, based on "enabled" property.
- /// For ui-element, based on "visible" property.
- void UpdateAttributeInspectorIcons()
- {
- if (!editNodes.empty)
- {
- Text@ nodeTitle = GetNodeContainer().GetChild("TitleText");
- if (editNode !is null)
- SetIconEnabledColor(nodeTitle, editNode.enabled);
- else if (editNodes.length > 0)
- {
- bool hasSameEnabledState = true;
- for (uint i = 1; i < editNodes.length; ++i)
- {
- if (editNodes[i].enabled != editNodes[0].enabled)
- {
- hasSameEnabledState = false;
- break;
- }
- }
- SetIconEnabledColor(nodeTitle, editNodes[0].enabled, !hasSameEnabledState);
- }
- }
- if (!editComponents.empty)
- {
- uint numEditableComponents = editComponents.length / numEditableComponentsPerNode;
- for (uint j = 0; j < numEditableComponentsPerNode; ++j)
- {
- Text@ componentTitle = GetComponentContainer(j).GetChild("TitleText");
- bool enabledEffective = editComponents[j * numEditableComponents].enabledEffective;
- bool hasSameEnabledState = true;
- for (uint i = 1; i < numEditableComponents; ++i)
- {
- if (editComponents[j * numEditableComponents + i].enabledEffective != enabledEffective)
- {
- hasSameEnabledState = false;
- break;
- }
- }
- SetIconEnabledColor(componentTitle, enabledEffective, !hasSameEnabledState);
- }
- }
-
- if (!editUIElements.empty)
- {
- Text@ elementTitle = GetUIElementContainer().GetChild("TitleText");
- if (editUIElement !is null)
- SetIconEnabledColor(elementTitle, editUIElement.visible);
- else if (editUIElements.length > 0)
- {
- bool hasSameVisibleState = true;
- bool visible = cast<UIElement>(editUIElements[0]).visible;
- for (uint i = 1; i < editUIElements.length; ++i)
- {
- if (cast<UIElement>(editUIElements[i]).visible != visible)
- {
- hasSameVisibleState = false;
- break;
- }
- }
- SetIconEnabledColor(elementTitle, visible, !hasSameVisibleState);
- }
- }
- }
- /// Return true if the edit attribute action should continue.
- bool PreEditAttribute(Array<Serializable@>@ serializables, uint index)
- {
- return true;
- }
- /// Call after the attribute values in the target serializables have been edited.
- void PostEditAttribute(Array<Serializable@>@ serializables, uint index, const Array<Variant>& oldValues)
- {
- // Create undo actions for the edits
- EditActionGroup group;
- for (uint i = 0; i < serializables.length; ++i)
- {
- EditAttributeAction action;
- action.Define(serializables[i], index, oldValues[i]);
- group.actions.Push(action);
- }
- SaveEditActionGroup(group);
- // If a UI-element changing its 'Is Modal' attribute, clear the hierarchy list selection
- int itemType = GetType(serializables[0]);
- if (itemType == ITEM_UI_ELEMENT && serializables[0].attributeInfos[index].name == "Is Modal")
- hierarchyList.ClearSelection();
- for (uint i = 0; i < serializables.length; ++i)
- {
- PostEditAttribute(serializables[i], index);
- if (itemType == ITEM_UI_ELEMENT)
- SetUIElementModified(serializables[i]);
- }
- if (itemType != ITEM_UI_ELEMENT)
- SetSceneModified();
- }
- /// Call after the attribute values in the target serializables have been edited.
- void PostEditAttribute(Serializable@ serializable, uint index)
- {
- // If a StaticModel/AnimatedModel/Skybox model was changed, apply a possibly different material list
- if (applyMaterialList && serializable.attributeInfos[index].name == "Model")
- {
- StaticModel@ staticModel = cast<StaticModel>(serializable);
- if (staticModel !is null)
- staticModel.ApplyMaterialList();
- }
-
- // If a CollisionShape changed the shape type to trimesh or convex, and a collision model is not set,
- // try to get it from a StaticModel in the same node
- if (serializable.typeName == "CollisionShape" && serializable.attributeInfos[index].name == "Shape Type")
- {
- int shapeType = serializable.GetAttribute("Shape Type").GetInt();
- if ((shapeType == 6 || shapeType == 7) && serializable.GetAttribute("CustomGeometry ComponentID").GetInt() == 0 &&
- serializable.GetAttribute("Model").GetResourceRef().name.Trimmed().length == 0)
- {
- Node@ ownerNode = cast<Component>(serializable).node;
- if (ownerNode !is null)
- {
- StaticModel@ staticModel = ownerNode.GetComponent("StaticModel");
- if (staticModel !is null)
- {
- serializable.SetAttribute("Model", staticModel.GetAttribute("Model"));
- serializable.ApplyAttributes();
- }
- }
- }
- }
- }
- /// Store the IDs of the actual serializable objects into user-defined variable of the 'attribute editor' (e.g. line-edit, drop-down-list, etc).
- void SetAttributeEditorID(UIElement@ attrEdit, Array<Serializable@>@ serializables)
- {
- if (serializables is null || serializables.length == 0)
- return;
- // All target serializables must be either nodes, ui-elements, or components
- Array<Variant> ids;
- switch (GetType(serializables[0]))
- {
- case ITEM_NODE:
- for (uint i = 0; i < serializables.length; ++i)
- ids.Push(cast<Node>(serializables[i]).id);
- attrEdit.vars[NODE_IDS_VAR] = ids;
- break;
- case ITEM_COMPONENT:
- for (uint i = 0; i < serializables.length; ++i)
- ids.Push(cast<Component>(serializables[i]).id);
- attrEdit.vars[COMPONENT_IDS_VAR] = ids;
- break;
- case ITEM_UI_ELEMENT:
- for (uint i = 0; i < serializables.length; ++i)
- ids.Push(GetUIElementID(cast<UIElement>(serializables[i])));
- attrEdit.vars[UI_ELEMENT_IDS_VAR] = ids;
- break;
- default:
- break;
- }
- }
- /// Return the actual serializable objects based on the IDs stored in the user-defined variable of the 'attribute editor'.
- Array<Serializable@>@ GetAttributeEditorTargets(UIElement@ attrEdit)
- {
- Array<Serializable@> ret;
- Variant variant = attrEdit.GetVar(NODE_IDS_VAR);
- if (!variant.empty)
- {
- Array<Variant>@ ids = variant.GetVariantVector();
- for (uint i = 0; i < ids.length; ++i)
- {
- Node@ node = editorScene.GetNode(ids[i].GetUInt());
- if (node !is null)
- ret.Push(node);
- }
- }
- else
- {
- variant = attrEdit.GetVar(COMPONENT_IDS_VAR);
- if (!variant.empty)
- {
- Array<Variant>@ ids = variant.GetVariantVector();
- for (uint i = 0; i < ids.length; ++i)
- {
- Component@ component = editorScene.GetComponent(ids[i].GetUInt());
- if (component !is null)
- ret.Push(component);
- }
- }
- else
- {
- variant = attrEdit.GetVar(UI_ELEMENT_IDS_VAR);
- if (!variant.empty)
- {
- Array<Variant>@ ids = variant.GetVariantVector();
- for (uint i = 0; i < ids.length; ++i)
- {
- UIElement@ element = editorUIElement.GetChild(UI_ELEMENT_ID_VAR, ids[i], true);
- if (element !is null)
- ret.Push(element);
- }
- }
- }
- }
- return ret;
- }
- void HandleTagsEdit(StringHash eventType, VariantMap& eventData)
- {
- LineEdit@ lineEdit = eventData["Element"].GetPtr();
- Array<String> tags = lineEdit.text.Split(';');
-
- if (editUIElement !is null)
- {
- editUIElement.RemoveAllTags();
- for (uint i = 0; i < tags.length; i++)
- editUIElement.AddTag(tags[i].Trimmed());
- }
- else if (editNode !is null)
- {
- editNode.RemoveAllTags();
- for (uint i = 0; i < tags.length; i++)
- editNode.AddTag(tags[i].Trimmed());
- }
- }
- void HandleTagsSelect(StringHash eventType, VariantMap& eventData)
- {
- UIElement@ tagSelect = eventData["Element"].GetPtr();
- Array<UIElement@> actions;
- String Indicator = "* ";
- // In first priority changes to UIElement
- if (editUIElement !is null)
- {
- // 1. Add established tags from current editable UIElement to menu
- Array<String> elementTags = editUIElement.tags;
- for (uint i = 0; i < elementTags.length; i++)
- {
- bool isHasTag = editUIElement.HasTag(elementTags[i]);
- String taggedIndicator = (isHasTag ? Indicator : "");
- actions.Push(CreateContextMenuItem(taggedIndicator + elementTags[i], "HandleTagsMenuSelection", elementTags[i]));
- }
- // 2. Add default tags
- Array<String> stdTags = defaultTags.Split(';');
- for (uint i= 0; i < stdTags.length; i++)
- {
- bool isHasTag = editUIElement.HasTag(stdTags[i]);
- // Add this tag into menu if only Node not tadded with it yet, otherwise it showed on step 1.
- if (!isHasTag)
- {
- String taggedIndicator = (isHasTag ? Indicator : "");
- actions.Push(CreateContextMenuItem(taggedIndicator + stdTags[i], "HandleTagsMenuSelection", stdTags[i]));
- }
- }
- }
- else if (editNode !is null)
- {
- // 1. Add established tags from Node to menu
- Array<String> nodeTags = editNode.tags;
- for (uint i = 0; i < nodeTags.length; i++)
- {
- bool isHasTag = editNode.HasTag(nodeTags[i]);
- String taggedIndicator = (isHasTag ? Indicator : "");
- actions.Push(CreateContextMenuItem(taggedIndicator + nodeTags[i], "HandleTagsMenuSelection", nodeTags[i]));
- }
- Array<String> sceneTags = editorScene.tags;
- // 2. Add tags from Scene.tags (In this scenario Scene.tags used as storage for frequently used tags in current Scene only)
- for (uint i = 0; i < sceneTags.length; i++)
- {
- bool isHasTag = editNode.HasTag(sceneTags[i]);
- // Add this tag into menu if only Node not tadded with it yet, otherwise it showed on step 1.
- if (!isHasTag)
- {
- String taggedIndicator = (isHasTag ? Indicator : "");
- actions.Push(CreateContextMenuItem(taggedIndicator + sceneTags[i], "HandleTagsMenuSelection", sceneTags[i]));
- }
- }
- // 3. Add default tags
- Array<String> stdTags = defaultTags.Split(';');
- for (uint i = 0; i < stdTags.length; i++)
- {
- bool isHasTag = editNode.HasTag(stdTags[i]);
- // Add this tag into menu if only Node not tadded with it yet, otherwise it showed on step 1.
- if (!isHasTag)
- {
- String taggedIndicator = (isHasTag ? Indicator : "");
- actions.Push(CreateContextMenuItem(taggedIndicator + stdTags[i], "HandleTagsMenuSelection", stdTags[i]));
- }
- }
- }
-
- // if any action has been added, add also Reset and Cancel and show menu
- if (actions.length > 0)
- {
- actions.Push(CreateContextMenuItem("Reset", "HandleTagsMenuSelection", "Reset"));
- actions.Push(CreateContextMenuItem("Cancel", "HandleTagsMenuSelectionDivisor"));
- ActivateContextMenu(actions);
- }
-
- }
- void HandleTagsMenuSelectionDivisor()
- {
- //do nothing
- }
- void HandleTagsMenuSelection()
- {
- Menu@ menu = GetEventSender();
- if (menu is null)
- return;
- String menuSelectedTag = menu.name;
- // In first priority changes to UIElement
- if (editUIElement !is null)
- {
- if (menuSelectedTag == "Reset")
- {
- editUIElement.RemoveAllTags();
- UpdateAttributeInspector();
- return;
- }
-
- if (!editUIElement.HasTag(menuSelectedTag))
- {
- editUIElement.AddTag(menuSelectedTag.Trimmed());
- }
- else
- {
- editUIElement.RemoveTag(menuSelectedTag.Trimmed());
- }
- }
- else if (editNode !is null)
- {
- if (menuSelectedTag == "Reset")
- {
- editNode.RemoveAllTags();
- UpdateAttributeInspector();
- return;
- }
- if (!editNode.HasTag(menuSelectedTag))
- {
- editNode.AddTag(menuSelectedTag.Trimmed());
- }
- else
- {
- editNode.RemoveTag(menuSelectedTag.Trimmed());
- }
- }
- UpdateAttributeInspector();
- }
- /// Handle reset to default event, sent when reset icon in the icon-panel is clicked.
- void HandleResetToDefault(StringHash eventType, VariantMap& eventData)
- {
- ui.cursor.shape = CS_BUSY;
- UIElement@ button = eventData["Element"].GetPtr();
- Array<Serializable@>@ serializables = GetAttributeEditorTargets(button);
- if (serializables.empty)
- return;
- // Group for storing undo actions
- EditActionGroup group;
- // Reset target serializables to their default values
- for (uint i = 0; i < serializables.length; ++i)
- {
- Serializable@ target = serializables[i];
- ResetAttributesAction action;
- action.Define(target);
- group.actions.Push(action);
- target.ResetToDefault();
- if (action.targetType == ITEM_UI_ELEMENT)
- {
- action.SetInternalVars(target);
- SetUIElementModified(target);
- }
- target.ApplyAttributes();
- for (uint j = 0; j < target.numAttributes; ++j)
- PostEditAttribute(target, j);
- }
- SaveEditActionGroup(group);
- if (GetType(serializables[0]) != ITEM_UI_ELEMENT)
- SetSceneModified();
- attributesFullDirty = true;
- }
- /// Handle create new user-defined variable event for node target.
- void CreateNodeVariable(StringHash eventType, VariantMap& eventData)
- {
- if (editNodes.empty)
- return;
- String newName = ExtractVariableName(eventData);
- if (newName.empty)
- return;
- // Create scene variable
- editorScene.RegisterVar(newName);
- globalVarNames[newName] = newName;
- Variant newValue = ExtractVariantType(eventData);
- // If we overwrite an existing variable, must recreate the attribute-editor(s) for the correct type
- bool overwrite = false;
- for (uint i = 0; i < editNodes.length; ++i)
- {
- overwrite = overwrite || editNodes[i].vars.Contains(newName);
- editNodes[i].vars[newName] = newValue;
- }
- if (overwrite)
- attributesFullDirty = true;
- else
- attributesDirty = true;
- }
- /// Handle delete existing user-defined variable event for node target.
- void DeleteNodeVariable(StringHash eventType, VariantMap& eventData)
- {
- if (editNodes.empty)
- return;
- String delName = ExtractVariableName(eventData);
- if (delName.empty)
- return;
- bool erased = false;
- for (uint i = 0; i < editNodes.length; ++i)
- {
- // \todo Should first check whether var in question is editable
- erased = editNodes[i].vars.Erase(delName) || erased;
- }
- if (erased)
- {
- attributesDirty = true;
- // If the attribute is not defined in any other node, unregister from the scene
- // to prevent it from being unnecessarily saved; the global var list will still hold it
- // to keep the hash-name mapping known in case it's in use in other scenes
- Array<Node@>@ allChildren = editorScene.GetChildren(true);
- StringHash delNameHash(delName);
- bool inUse = false;
- for (uint i = 0; i < allChildren.length; ++i)
- {
- if (allChildren[i].vars.Contains(delNameHash))
- {
- inUse = true;
- break;
- }
- }
-
- if (!inUse)
- editorScene.UnregisterVar(delName);
- }
- }
- /// Handle create new user-defined variable event for ui-element target.
- void CreateUIElementVariable(StringHash eventType, VariantMap& eventData)
- {
- if (editUIElements.empty)
- return;
- String newName = ExtractVariableName(eventData);
- if (newName.empty)
- return;
- // Create UIElement variable
- globalVarNames[newName] = newName;
- Variant newValue = ExtractVariantType(eventData);
- // If we overwrite an existing variable, must recreate the attribute-editor(s) for the correct type
- bool overwrite = false;
- for (uint i = 0; i < editUIElements.length; ++i)
- {
- UIElement@ element = cast<UIElement>(editUIElements[i]);
- overwrite = overwrite || element.vars.Contains(newName);
- element.vars[newName] = newValue;
- }
- if (overwrite)
- attributesFullDirty = true;
- else
- attributesDirty = true;
- }
- /// Handle delete existing user-defined variable event for ui-element target.
- void DeleteUIElementVariable(StringHash eventType, VariantMap& eventData)
- {
- if (editUIElements.empty)
- return;
- String delName = ExtractVariableName(eventData);
- if (delName.empty)
- return;
- // Note: intentionally do not unregister the variable name here as the same variable name may still be used by other attribute list
- bool erased = false;
- for (uint i = 0; i < editUIElements.length; ++i)
- {
- // \todo Should first check whether var in question is editable
- erased = cast<UIElement>(editUIElements[i]).vars.Erase(delName) || erased;
- }
- if (erased)
- attributesDirty = true;
- }
- String ExtractVariableName(VariantMap& eventData)
- {
- UIElement@ element = eventData["Element"].GetPtr();
- LineEdit@ nameEdit = element.parent.GetChild("VarNameEdit");
- return nameEdit.text.Trimmed();
- }
- Variant ExtractVariantType(VariantMap& eventData)
- {
- DropDownList@ dropDown = eventData["Element"].GetPtr();
- switch (dropDown.selection)
- {
- case 0:
- return int(0);
- case 1:
- return false;
- case 2:
- return float(0.0);
- case 3:
- return Variant(String());
- case 4:
- return Variant(Vector3());
- case 5:
- return Variant(Color());
- }
- return Variant(); // This should not happen
- }
- /// Get back the human-readable variable name from the StringHash.
- String GetVarName(StringHash hash)
- {
- // First try to get it from scene
- String name = editorScene.GetVarName(hash);
- // Then from the global variable reverse mappings
- if (name.empty && globalVarNames.Contains(hash))
- name = globalVarNames[hash].ToString();
- return name;
- }
- bool inSetStyleListSelection = false;
- /// Select/highlight the matching style in the style drop-down-list based on specified style.
- void SetStyleListSelection(DropDownList@ styleList, const String&in style)
- {
- // Prevent infinite loop upon initial style selection
- inSetStyleListSelection = true;
- uint selection = M_MAX_UNSIGNED;
- String styleName = style.empty ? "auto" : style;
- Array<UIElement@> items = styleList.GetItems();
- for (uint i = 0; i < items.length; ++i)
- {
- Text@ element = cast<Text>(items[i]);
- if (element is null)
- continue; // It may be a divider
- if (element.text == styleName)
- {
- selection = i;
- break;
- }
- }
- styleList.selection = selection;
- inSetStyleListSelection = false;
- }
- /// Handle the style change of the target ui-elements event when a new style is picked from the style drop-down-list.
- void HandleStyleItemSelected(StringHash eventType, VariantMap& eventData)
- {
- if (inSetStyleListSelection || editUIElements.empty)
- return;
- ui.cursor.shape = CS_BUSY;
- DropDownList@ styleList = eventData["Element"].GetPtr();
- Text@ text = cast<Text>(styleList.selectedItem);
- if (text is null)
- return;
- String newStyle = text.text;
- if (newStyle == "auto")
- newStyle.Clear();
- // Group for storing undo actions
- EditActionGroup group;
- // Apply new style to selected UI-elements
- for (uint i = 0; i < editUIElements.length; ++i)
- {
- UIElement@ element = editUIElements[i];
- ApplyUIElementStyleAction action;
- action.Define(element, newStyle);
- group.actions.Push(action);
- // Use the Redo() to actually do the action
- action.Redo();
- }
- SaveEditActionGroup(group);
- }
|