2016-03-20 12:31:23 +00:00
|
|
|
#ifndef CNODECOPYMIMEDATA
|
|
|
|
#define CNODECOPYMIMEDATA
|
|
|
|
|
|
|
|
#include <Common/TString.h>
|
2018-12-12 05:50:46 +00:00
|
|
|
#include <Common/Math/CVector3f.h>
|
2016-03-20 12:31:23 +00:00
|
|
|
#include <Core/Resource/Cooker/CScriptCooker.h>
|
|
|
|
#include <Core/Resource/Factory/CScriptLoader.h>
|
|
|
|
#include <Core/Scene/CSceneNode.h>
|
|
|
|
#include "Editor/CSelectionIterator.h"
|
|
|
|
#include "Editor/WorldEditor/CWorldEditor.h"
|
|
|
|
|
|
|
|
#include <QMimeData>
|
|
|
|
|
|
|
|
class CNodeCopyMimeData : public QMimeData
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
struct SCopiedNode
|
|
|
|
{
|
|
|
|
ENodeType Type;
|
|
|
|
TString Name;
|
|
|
|
CVector3f Position;
|
|
|
|
CQuaternion Rotation;
|
|
|
|
CVector3f Scale;
|
|
|
|
|
2018-12-12 05:50:46 +00:00
|
|
|
uint32 OriginalInstanceID;
|
2016-03-20 12:31:23 +00:00
|
|
|
std::vector<char> InstanceData;
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
|
|
|
CWorldEditor *mpEditor;
|
2016-07-25 09:25:22 +00:00
|
|
|
CAssetID mAreaID;
|
2016-03-20 12:31:23 +00:00
|
|
|
QVector<SCopiedNode> mCopiedNodes;
|
|
|
|
EGame mGame;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CNodeCopyMimeData(const CNodeCopyMimeData& rkSrc)
|
|
|
|
: mpEditor(rkSrc.mpEditor)
|
|
|
|
, mAreaID(rkSrc.mAreaID)
|
|
|
|
, mCopiedNodes(rkSrc.mCopiedNodes)
|
|
|
|
, mGame(rkSrc.mGame)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
CNodeCopyMimeData(CWorldEditor *pEditor)
|
|
|
|
: mpEditor(pEditor)
|
2016-07-25 09:25:22 +00:00
|
|
|
, mAreaID(pEditor->ActiveArea()->ID())
|
2016-03-20 12:31:23 +00:00
|
|
|
, mGame(pEditor->CurrentGame())
|
|
|
|
{
|
|
|
|
CNodeSelection *pSelection = pEditor->Selection();
|
|
|
|
mCopiedNodes.resize(pSelection->Size());
|
|
|
|
|
2018-12-12 05:50:46 +00:00
|
|
|
uint32 NodeIndex = 0;
|
2016-03-20 12:31:23 +00:00
|
|
|
CVector3f FirstNodePos;
|
|
|
|
bool SetFirstNodePos = false;
|
|
|
|
|
|
|
|
for (CSelectionIterator It(pEditor->Selection()); It; ++It)
|
|
|
|
{
|
|
|
|
SCopiedNode& rNode = mCopiedNodes[NodeIndex];
|
|
|
|
rNode.Type = It->NodeType();
|
|
|
|
rNode.Name = It->Name();
|
|
|
|
rNode.Position = It->LocalPosition();
|
|
|
|
rNode.Rotation = It->LocalRotation();
|
|
|
|
rNode.Scale = It->LocalScale();
|
|
|
|
|
2018-12-16 21:00:40 +00:00
|
|
|
if (rNode.Type == ENodeType::Script)
|
2016-03-20 12:31:23 +00:00
|
|
|
{
|
2016-03-27 19:09:38 +00:00
|
|
|
CScriptObject *pInst = static_cast<CScriptNode*>(*It)->Instance();
|
2016-03-20 12:31:23 +00:00
|
|
|
rNode.OriginalInstanceID = pInst->InstanceID();
|
|
|
|
|
2018-12-16 21:00:40 +00:00
|
|
|
CVectorOutStream Out(&rNode.InstanceData, EEndian::BigEndian);
|
2017-06-27 12:13:42 +00:00
|
|
|
|
|
|
|
CScriptCooker Cooker(mGame);
|
|
|
|
Cooker.WriteInstance(Out, static_cast<CScriptNode*>(*It)->Instance());
|
2016-03-20 12:31:23 +00:00
|
|
|
|
|
|
|
// Replace instance ID with 0xFFFFFFFF to force it to generate a new one.
|
2018-12-12 05:50:46 +00:00
|
|
|
Out.Seek(mGame <= EGame::Prime ? 0x5 : 0x6, SEEK_SET);
|
2016-03-20 12:31:23 +00:00
|
|
|
Out.WriteLong(0xFFFFFFFF);
|
|
|
|
|
|
|
|
if (!SetFirstNodePos)
|
|
|
|
{
|
|
|
|
FirstNodePos = rNode.Position;
|
|
|
|
SetFirstNodePos = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
rNode.Position -= FirstNodePos;
|
|
|
|
}
|
|
|
|
|
|
|
|
NodeIndex++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-12 05:50:46 +00:00
|
|
|
int IndexOfInstanceID(uint32 InstanceID) const
|
2016-03-20 12:31:23 +00:00
|
|
|
{
|
|
|
|
for (int iNode = 0; iNode < mCopiedNodes.size(); iNode++)
|
|
|
|
{
|
|
|
|
if (mCopiedNodes[iNode].OriginalInstanceID == InstanceID)
|
|
|
|
return iNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-07-25 09:25:22 +00:00
|
|
|
CAssetID AreaID() const { return mAreaID; }
|
2016-03-27 19:09:38 +00:00
|
|
|
EGame Game() const { return mGame; }
|
2016-03-20 12:31:23 +00:00
|
|
|
const QVector<SCopiedNode>& CopiedNodes() const { return mCopiedNodes; }
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // CNODECOPYMIMEDATA
|
|
|
|
|