You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
172 lines
5.5 KiB
C++
172 lines
5.5 KiB
C++
/*
|
|
* Copyright (c) Contributors to the Open 3D Engine Project.
|
|
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0 OR MIT
|
|
*
|
|
*/
|
|
#include "EditorCommon.h"
|
|
#include <AzCore/std/sort.h>
|
|
|
|
CommandHierarchyItemReparent::CommandHierarchyItemReparent(UndoStack* stack,
|
|
HierarchyWidget* hierarchy,
|
|
const HierarchyItemRawPtrList& items)
|
|
: QUndoCommand()
|
|
, m_stack(stack)
|
|
, m_hierarchy(hierarchy)
|
|
, m_listToValidate()
|
|
, m_isFirstExecution(true)
|
|
, m_sourceChildrenSorted(false)
|
|
, m_destinationChildrenSorted(false)
|
|
{
|
|
for (auto item : items)
|
|
{
|
|
// Add source child info
|
|
ChildItem sourceChildItem;
|
|
sourceChildItem.m_id = item->GetEntityId();
|
|
sourceChildItem.m_parentId = item->GetPreMoveParentId();
|
|
sourceChildItem.m_row = item->GetPreMoveChildRow();
|
|
m_sourceChildren.push_back(sourceChildItem);
|
|
|
|
// Add destination child info
|
|
ChildItem destChildItem;
|
|
destChildItem.m_id = sourceChildItem.m_id;
|
|
|
|
QTreeWidgetItem* itemParent = item->QTreeWidgetItem::parent();
|
|
if (!itemParent)
|
|
{
|
|
itemParent = m_hierarchy->invisibleRootItem();
|
|
}
|
|
HierarchyItem* postMoveParent = HierarchyItem::RttiCast(itemParent);
|
|
destChildItem.m_parentId = (postMoveParent ? postMoveParent->GetEntityId() : AZ::EntityId());
|
|
|
|
destChildItem.m_row = itemParent->indexOfChild(item);
|
|
|
|
m_destinationChildren.push_back(destChildItem);
|
|
|
|
// Add items to validate
|
|
if (sourceChildItem.m_id.IsValid())
|
|
{
|
|
m_listToValidate.push_back(sourceChildItem.m_id);
|
|
}
|
|
|
|
if (sourceChildItem.m_parentId.IsValid())
|
|
{
|
|
m_listToValidate.push_back(sourceChildItem.m_parentId);
|
|
}
|
|
|
|
if (destChildItem.m_parentId.IsValid())
|
|
{
|
|
m_listToValidate.push_back(destChildItem.m_parentId);
|
|
}
|
|
}
|
|
|
|
if (items.size() == 1)
|
|
{
|
|
setText(QString("move \"%1\"").arg(items.front()->GetElement()->GetName().c_str()));
|
|
|
|
}
|
|
else
|
|
{
|
|
setText(QString("move elements"));
|
|
}
|
|
}
|
|
|
|
void CommandHierarchyItemReparent::undo()
|
|
{
|
|
UndoStackExecutionScope s(m_stack);
|
|
|
|
Reparent(m_destinationChildren, m_sourceChildren, m_sourceChildrenSorted);
|
|
}
|
|
|
|
void CommandHierarchyItemReparent::redo()
|
|
{
|
|
UndoStackExecutionScope s(m_stack);
|
|
|
|
Reparent(m_sourceChildren, m_destinationChildren, m_destinationChildrenSorted);
|
|
}
|
|
|
|
void CommandHierarchyItemReparent::Reparent(ChildItemList& sourceChildren, ChildItemList& destChildren, bool& childrenSorted)
|
|
{
|
|
if (!HierarchyHelpers::AllItemExists(m_hierarchy, m_listToValidate))
|
|
{
|
|
// At least one item is missing.
|
|
// Nothing to do.
|
|
return;
|
|
}
|
|
|
|
if (!childrenSorted)
|
|
{
|
|
// Sort children by destination index to be inserted in that order
|
|
destChildren.sort([](const ChildItem& i1, const ChildItem& i2) { return i1.m_row < i2.m_row; });
|
|
|
|
childrenSorted = true;
|
|
}
|
|
|
|
if (m_isFirstExecution)
|
|
{
|
|
m_isFirstExecution = false;
|
|
}
|
|
else
|
|
{
|
|
// Editor-side.
|
|
//
|
|
// IMPORTANT: The editor-side MUST be done BEFORE the runtime-side.
|
|
|
|
// First remove all items
|
|
for (auto child : sourceChildren)
|
|
{
|
|
HierarchyItem* item = HierarchyItem::RttiCast(HierarchyHelpers::ElementToItem(m_hierarchy, child.m_id, false));
|
|
QTreeWidgetItem* sourceParentItem = HierarchyHelpers::ElementToItem(m_hierarchy, child.m_parentId, true);
|
|
sourceParentItem->removeChild(item);
|
|
}
|
|
|
|
// Add items to new parents
|
|
for (auto child : destChildren)
|
|
{
|
|
HierarchyItem* item = HierarchyItem::RttiCast(HierarchyHelpers::ElementToItem(m_hierarchy, child.m_id, false));
|
|
QTreeWidgetItem* destinationParentItem = HierarchyHelpers::ElementToItem(m_hierarchy, child.m_parentId, true);
|
|
destinationParentItem->insertChild(child.m_row, item);
|
|
}
|
|
}
|
|
|
|
// Runtime-side.
|
|
//
|
|
// IMPORTANT: The runtime-side depends on the editor-side being done FIRST.
|
|
|
|
// First remove all elements
|
|
for (auto child : sourceChildren)
|
|
{
|
|
EBUS_EVENT_ID(child.m_id, UiElementBus, RemoveFromParent);
|
|
}
|
|
|
|
// Add elements to new parents
|
|
LyShine::EntityArray selectedElements;
|
|
for (auto child : destChildren)
|
|
{
|
|
QTreeWidgetItem* destinationParentItem = HierarchyHelpers::ElementToItem(m_hierarchy, child.m_parentId, true);
|
|
HierarchyItem* destinationParent = HierarchyItem::RttiCast(destinationParentItem);
|
|
AZ::Entity* parentElement = (destinationParent ? destinationParent->GetElement() : nullptr);
|
|
|
|
// Add the element to its new parent
|
|
EBUS_EVENT_ID(child.m_id, UiElementBus, AddToParentAtIndex, parentElement, child.m_row);
|
|
|
|
// Add item to selected item list
|
|
HierarchyItem* item = HierarchyItem::RttiCast(HierarchyHelpers::ElementToItem(m_hierarchy, child.m_id, false));
|
|
selectedElements.push_back(item->GetElement());
|
|
}
|
|
|
|
// Set the focus to the currently manipulated items
|
|
HierarchyHelpers::SetSelectedItems(m_hierarchy, &selectedElements);
|
|
}
|
|
|
|
void CommandHierarchyItemReparent::Push(UndoStack* stack,
|
|
HierarchyWidget* hierarchy,
|
|
const HierarchyItemRawPtrList& items,
|
|
[[maybe_unused]] const QTreeWidgetItemRawPtrList& itemParents)
|
|
{
|
|
stack->push(new CommandHierarchyItemReparent(stack,
|
|
hierarchy,
|
|
items));
|
|
}
|