2015-08-16 04:29:37 +00:00
|
|
|
#include "CGizmo.h"
|
2015-08-17 18:10:42 +00:00
|
|
|
#include <Common/Math.h>
|
2015-08-16 04:29:37 +00:00
|
|
|
#include <Core/CRenderer.h>
|
|
|
|
#include <Core/CResCache.h>
|
2015-08-20 01:01:58 +00:00
|
|
|
#include <Core/CDrawUtil.h>
|
2015-08-24 01:02:14 +00:00
|
|
|
#include <Core/Log.h>
|
|
|
|
|
|
|
|
#include <QApplication>
|
|
|
|
#include <QDesktopWidget>
|
2015-08-20 01:01:58 +00:00
|
|
|
#include <iostream>
|
2015-08-16 04:29:37 +00:00
|
|
|
|
|
|
|
CGizmo::CGizmo()
|
|
|
|
{
|
|
|
|
LoadModels();
|
|
|
|
|
2015-08-17 18:10:42 +00:00
|
|
|
SetMode(eTranslate);
|
2015-08-16 04:29:37 +00:00
|
|
|
mSelectedAxes = eNone;
|
2015-08-24 01:02:14 +00:00
|
|
|
mTransformSpace = eWorldTransform;
|
2015-08-16 04:29:37 +00:00
|
|
|
mGizmoSize = 1.f;
|
|
|
|
mCameraDist = 0.f;
|
2015-08-24 01:02:14 +00:00
|
|
|
mIsTransforming = false;
|
|
|
|
mHasTransformed = false;
|
|
|
|
mWrapOffset = 0.f;
|
|
|
|
mEnableCursorWrap = true;
|
2015-08-16 04:29:37 +00:00
|
|
|
|
|
|
|
mPosition = CVector3f::skZero;
|
|
|
|
mRotation = CQuaternion::skIdentity;
|
2015-08-28 22:57:24 +00:00
|
|
|
mLocalRotation = CQuaternion::skIdentity;
|
2015-08-16 04:29:37 +00:00
|
|
|
mScale = CVector3f::skOne;
|
2015-08-28 22:57:24 +00:00
|
|
|
mFlipScaleX = false;
|
|
|
|
mFlipScaleY = false;
|
|
|
|
mFlipScaleZ = false;
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
mDeltaTranslation = CVector3f::skZero;
|
2015-08-16 04:29:37 +00:00
|
|
|
mDeltaRotation = CQuaternion::skIdentity;
|
|
|
|
mDeltaScale = CVector3f::skOne;
|
2015-08-24 14:18:23 +00:00
|
|
|
mTotalScale = CVector3f::skOne;
|
2015-08-20 01:01:58 +00:00
|
|
|
mSetOffset = false;
|
2015-08-16 04:29:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CGizmo::~CGizmo()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-11-25 21:37:34 +00:00
|
|
|
void CGizmo::AddToRenderer(CRenderer *pRenderer, const SViewInfo&)
|
2015-08-16 04:29:37 +00:00
|
|
|
{
|
|
|
|
// Transform is updated every frame even if the user doesn't modify the gizmo
|
|
|
|
// in order to account for scale changes based on camera distance
|
|
|
|
UpdateTransform();
|
2015-08-17 18:10:42 +00:00
|
|
|
SModelPart *pPart = mpCurrentParts;
|
2015-08-16 04:29:37 +00:00
|
|
|
|
|
|
|
// Add all parts to renderer
|
2015-08-17 18:10:42 +00:00
|
|
|
for (u32 iPart = 0; iPart < mNumCurrentParts; iPart++)
|
2015-08-16 04:29:37 +00:00
|
|
|
{
|
2015-08-17 18:10:42 +00:00
|
|
|
CModel *pModel = pPart->pModel;
|
2015-08-16 04:29:37 +00:00
|
|
|
|
|
|
|
// Determine whether to use the mat set for regular (0) or highlight (1)
|
2015-08-24 01:02:14 +00:00
|
|
|
EGizmoAxes partAxes = pPart->modelAxes;
|
|
|
|
bool isHighlighted = (partAxes != eNone) && ((mSelectedAxes & partAxes) == pPart->modelAxes);
|
2015-08-16 04:29:37 +00:00
|
|
|
u32 setID = (isHighlighted ? 1 : 0);
|
|
|
|
|
|
|
|
// Add to renderer...
|
|
|
|
if (pModel->HasTransparency(setID))
|
2015-11-27 23:28:35 +00:00
|
|
|
pRenderer->AddTransparentMesh(this, iPart, pModel->AABox().Transformed(mTransform), eDrawMesh);
|
2015-08-16 04:29:37 +00:00
|
|
|
else
|
2015-11-27 23:28:35 +00:00
|
|
|
pRenderer->AddOpaqueMesh(this, iPart, pModel->AABox().Transformed(mTransform), eDrawMesh);
|
2015-08-16 04:29:37 +00:00
|
|
|
|
2015-08-17 18:10:42 +00:00
|
|
|
pPart++;
|
2015-08-16 04:29:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-27 23:28:35 +00:00
|
|
|
void CGizmo::Draw(ERenderOptions /*Options*/, int ComponentIndex, const SViewInfo& /*ViewInfo*/)
|
2015-08-16 04:29:37 +00:00
|
|
|
{
|
|
|
|
// Determine which SModelPart array to use
|
2015-11-27 23:28:35 +00:00
|
|
|
if (ComponentIndex >= (int) mNumCurrentParts) return;
|
2015-08-17 18:10:42 +00:00
|
|
|
SModelPart *pPart = mpCurrentParts;
|
2015-08-16 04:29:37 +00:00
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
// Set model matrix
|
2015-11-27 23:28:35 +00:00
|
|
|
if (pPart[ComponentIndex].isBillboard)
|
2015-08-28 22:57:24 +00:00
|
|
|
CGraphics::sMVPBlock.ModelMatrix = mBillboardTransform.ToMatrix4f();
|
2015-11-27 23:28:35 +00:00
|
|
|
else if ((mMode == eScale) && ((mSelectedAxes & pPart[ComponentIndex].modelAxes) != 0))
|
2015-08-28 22:57:24 +00:00
|
|
|
CGraphics::sMVPBlock.ModelMatrix = mScaledTransform.ToMatrix4f();
|
|
|
|
else
|
|
|
|
CGraphics::sMVPBlock.ModelMatrix = mTransform.ToMatrix4f();
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
CGraphics::UpdateMVPBlock();
|
|
|
|
|
2015-11-26 12:35:02 +00:00
|
|
|
// Clear tint color
|
|
|
|
CGraphics::sPixelBlock.TintColor = CColor::skWhite.ToVector4f();
|
|
|
|
CGraphics::UpdatePixelBlock();
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
// Choose material set
|
2015-11-27 23:28:35 +00:00
|
|
|
EGizmoAxes partAxes = pPart[ComponentIndex].modelAxes;
|
|
|
|
bool isHighlighted = (partAxes != eNone) && ((mSelectedAxes & partAxes) == pPart[ComponentIndex].modelAxes);
|
2015-08-16 04:29:37 +00:00
|
|
|
u32 setID = (isHighlighted ? 1 : 0);
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
// Draw model
|
2015-11-27 23:28:35 +00:00
|
|
|
pPart[ComponentIndex].pModel->Draw((ERenderOptions) 0, setID);
|
2015-08-16 04:29:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::IncrementSize()
|
|
|
|
{
|
|
|
|
static const float skIncAmount = 1.3f;
|
|
|
|
static const float skMaxSize = powf(skIncAmount, 4);
|
|
|
|
|
|
|
|
mGizmoSize *= skIncAmount;
|
|
|
|
if (mGizmoSize > skMaxSize) mGizmoSize = skMaxSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::DecrementSize()
|
|
|
|
{
|
|
|
|
static const float skDecAmount = (1.f / 1.3f);
|
|
|
|
static const float skMinSize = powf(skDecAmount, 4);
|
|
|
|
|
|
|
|
mGizmoSize *= skDecAmount;
|
|
|
|
if (mGizmoSize < skMinSize) mGizmoSize = skMinSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::UpdateForCamera(const CCamera &camera)
|
|
|
|
{
|
|
|
|
CVector3f camPos = camera.Position();
|
2015-08-28 22:57:24 +00:00
|
|
|
CVector3f cameraToGizmo = (mPosition - camPos).Normalized();
|
|
|
|
mFlipScaleX = (mRotation.XAxis().Dot(cameraToGizmo) >= 0.f);
|
|
|
|
mFlipScaleY = (mRotation.YAxis().Dot(cameraToGizmo) >= 0.f);
|
|
|
|
mFlipScaleZ = (mRotation.ZAxis().Dot(cameraToGizmo) >= 0.f);
|
2015-08-16 04:29:37 +00:00
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
if ((!mIsTransforming) || (mMode != eTranslate))
|
|
|
|
mCameraDist = mPosition.Distance(camPos);
|
|
|
|
|
2015-08-16 04:29:37 +00:00
|
|
|
// todo: make this cleaner...
|
2015-08-28 22:57:24 +00:00
|
|
|
CVector3f billDir = (camPos - mPosition).Normalized();
|
2015-08-16 04:29:37 +00:00
|
|
|
CVector3f axis = CVector3f::skForward.Cross(billDir);
|
2015-08-24 01:02:14 +00:00
|
|
|
float angle = acosf(CVector3f::skForward.Dot(billDir));
|
2015-08-16 04:29:37 +00:00
|
|
|
mBillboardRotation = CQuaternion::FromAxisAngle(angle, axis);
|
|
|
|
}
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
bool CGizmo::CheckSelectedAxes(const CRay &ray)
|
2015-08-17 18:10:42 +00:00
|
|
|
{
|
|
|
|
CRay localRay = ray.Transformed(mTransform.Inverse());
|
2015-08-24 01:02:14 +00:00
|
|
|
CRay billRay = ray.Transformed(mBillboardTransform.Inverse());
|
2015-08-17 18:10:42 +00:00
|
|
|
|
|
|
|
// Do raycast on each model
|
|
|
|
SModelPart *pPart = mpCurrentParts;
|
|
|
|
|
|
|
|
struct SResult {
|
|
|
|
SModelPart *pPart;
|
|
|
|
float dist;
|
|
|
|
};
|
|
|
|
std::list<SResult> results;
|
|
|
|
|
|
|
|
for (u32 iPart = 0; iPart < mNumCurrentParts; iPart++)
|
|
|
|
{
|
2015-08-20 01:01:58 +00:00
|
|
|
if (!pPart->enableRayCast)
|
|
|
|
{
|
|
|
|
pPart++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-08-17 18:10:42 +00:00
|
|
|
CModel *pModel = pPart->pModel;
|
2015-08-24 01:02:14 +00:00
|
|
|
CRay& partRay = (pPart->isBillboard ? billRay : localRay);
|
2015-08-17 18:10:42 +00:00
|
|
|
|
|
|
|
// Ray/Model AABox test - allow buffer room because lines are small
|
|
|
|
CAABox AABox = pModel->AABox();
|
|
|
|
AABox.ExpandBy(CVector3f::skOne);
|
2015-08-24 01:02:14 +00:00
|
|
|
bool modelBoxCheck = Math::RayBoxIntersection(partRay, AABox).first;
|
2015-08-17 18:10:42 +00:00
|
|
|
|
|
|
|
if (modelBoxCheck)
|
|
|
|
{
|
|
|
|
bool hit = false;
|
|
|
|
float dist;
|
|
|
|
|
|
|
|
for (u32 iSurf = 0; iSurf < pModel->GetSurfaceCount(); iSurf++)
|
|
|
|
{
|
2015-08-20 01:01:58 +00:00
|
|
|
// Skip surface/box check - since we use lines the boxes might be too small
|
2015-08-17 18:10:42 +00:00
|
|
|
SSurface *pSurf = pModel->GetSurface(iSurf);
|
2015-11-24 10:22:37 +00:00
|
|
|
std::pair<bool,float> surfCheck = pSurf->IntersectsRay(partRay, false, 0.05f);
|
2015-08-17 18:10:42 +00:00
|
|
|
|
|
|
|
if (surfCheck.first)
|
|
|
|
{
|
|
|
|
if ((!hit) || (surfCheck.second < dist))
|
|
|
|
dist = surfCheck.second;
|
|
|
|
|
|
|
|
hit = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hit)
|
|
|
|
{
|
|
|
|
SResult result;
|
|
|
|
result.pPart = pPart;
|
|
|
|
result.dist = dist;
|
|
|
|
results.push_back(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pPart++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Results list empty = no hits
|
|
|
|
if (results.empty())
|
|
|
|
{
|
|
|
|
mSelectedAxes = eNone;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we have at least one hit - sort results and set selected axes
|
|
|
|
results.sort([](const SResult& a, SResult& b) -> bool
|
|
|
|
{
|
|
|
|
return (a.dist < b.dist);
|
|
|
|
});
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
CRay& partRay = (pPart->isBillboard ? billRay : localRay);
|
2015-08-17 18:10:42 +00:00
|
|
|
mSelectedAxes = results.front().pPart->modelAxes;
|
2015-08-24 01:02:14 +00:00
|
|
|
mHitPoint = mTransform * partRay.PointOnRay(results.front().dist);
|
|
|
|
|
|
|
|
return (mSelectedAxes != eNone);
|
2015-08-17 18:10:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
u32 CGizmo::NumSelectedAxes()
|
|
|
|
{
|
|
|
|
u32 out = 0;
|
|
|
|
|
|
|
|
for (u32 iAxis = 1; iAxis < 8; iAxis <<= 1)
|
|
|
|
if (mSelectedAxes & (EGizmoAxes) iAxis) out++;
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::ResetSelectedAxes()
|
|
|
|
{
|
|
|
|
mSelectedAxes = eNone;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::StartTransform()
|
|
|
|
{
|
2015-08-24 01:02:14 +00:00
|
|
|
mIsTransforming = true;
|
|
|
|
mHasTransformed = false;
|
|
|
|
mWrapOffset = CVector2f::skZero;
|
2015-08-20 01:01:58 +00:00
|
|
|
mSetOffset = false;
|
|
|
|
mTotalTranslation = CVector3f::skZero;
|
2015-08-24 01:02:14 +00:00
|
|
|
mTotalRotation = CVector3f::skZero;
|
|
|
|
mCurrentRotation = CQuaternion::skIdentity;
|
2015-08-20 01:01:58 +00:00
|
|
|
mTotalScale = CVector3f::skOne;
|
2015-08-24 01:02:14 +00:00
|
|
|
|
2015-08-28 22:57:24 +00:00
|
|
|
// Set rotation direction
|
2015-08-24 01:02:14 +00:00
|
|
|
if (mMode == eRotate)
|
|
|
|
{
|
|
|
|
CVector3f axis;
|
|
|
|
if (mSelectedAxes & eX) axis = mRotation.XAxis();
|
|
|
|
else if (mSelectedAxes & eY) axis = mRotation.YAxis();
|
|
|
|
else axis = mRotation.ZAxis();
|
|
|
|
|
|
|
|
CVector3f gizmoToHit = (mHitPoint - mPosition).Normalized();
|
2015-08-28 22:57:24 +00:00
|
|
|
mMoveDir = axis.Cross(gizmoToHit);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set scale direction
|
|
|
|
else if (mMode == eScale)
|
|
|
|
{
|
|
|
|
// Only need to set scale direction if < 3 axes selected
|
|
|
|
if (NumSelectedAxes() != 3)
|
|
|
|
{
|
|
|
|
// One axis; direction = selected axis
|
|
|
|
if (NumSelectedAxes() == 1)
|
|
|
|
{
|
|
|
|
if (mSelectedAxes & eX) mMoveDir = mRotation.XAxis();
|
|
|
|
else if (mSelectedAxes & eY) mMoveDir = mRotation.YAxis();
|
|
|
|
else mMoveDir = mRotation.ZAxis();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Two axes; interpolate between the two selected axes
|
|
|
|
else if (NumSelectedAxes() == 2)
|
|
|
|
{
|
|
|
|
CVector3f axisA = (mSelectedAxes & eX ? mRotation.XAxis() : mRotation.YAxis());
|
|
|
|
CVector3f axisB = (mSelectedAxes & eZ ? mRotation.ZAxis() : mRotation.YAxis());
|
|
|
|
mMoveDir = (axisA + axisB) / 2.f;
|
|
|
|
}
|
|
|
|
}
|
2015-08-24 01:02:14 +00:00
|
|
|
}
|
2015-08-20 01:01:58 +00:00
|
|
|
}
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
bool CGizmo::TransformFromInput(const CRay& ray, CCamera& camera)
|
2015-08-20 01:01:58 +00:00
|
|
|
{
|
2015-08-24 01:02:14 +00:00
|
|
|
// Wrap cursor (this has no effect until the next time this function is called)
|
|
|
|
if (mEnableCursorWrap && (mMode != eTranslate))
|
|
|
|
WrapCursor();
|
|
|
|
|
|
|
|
// Calculate normalized cursor position
|
|
|
|
QPoint cursorPos = QCursor::pos();
|
|
|
|
QRect geom = QApplication::desktop()->screenGeometry();
|
|
|
|
CVector2f mouseCoords(
|
|
|
|
(((2.f * cursorPos.x()) / geom.width()) - 1.f),
|
|
|
|
(1.f - ((2.f * cursorPos.y()) / geom.height()))
|
|
|
|
);
|
|
|
|
|
|
|
|
// Translate
|
2015-08-20 01:01:58 +00:00
|
|
|
if (mMode == eTranslate)
|
|
|
|
{
|
|
|
|
// Create translate plane
|
|
|
|
CVector3f axisA, axisB;
|
|
|
|
u32 numAxes = NumSelectedAxes();
|
|
|
|
|
|
|
|
if (numAxes == 1)
|
|
|
|
{
|
|
|
|
if (mSelectedAxes & eX) axisB = mRotation.XAxis();
|
|
|
|
else if (mSelectedAxes & eY) axisB = mRotation.YAxis();
|
|
|
|
else axisB = mRotation.ZAxis();
|
|
|
|
|
|
|
|
CVector3f gizmoToCamera = (mPosition - camera.Position()).Normalized();
|
|
|
|
axisA = axisB.Cross(gizmoToCamera);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (numAxes == 2)
|
|
|
|
{
|
|
|
|
axisA = (mSelectedAxes & eX ? mRotation.XAxis() : mRotation.YAxis());
|
|
|
|
axisB = (mSelectedAxes & eZ ? mRotation.ZAxis() : mRotation.YAxis());
|
|
|
|
}
|
|
|
|
|
|
|
|
CVector3f planeNormal = axisA.Cross(axisB);
|
|
|
|
mTranslatePlane.Redefine(planeNormal, mPosition);
|
|
|
|
|
|
|
|
// Do translate
|
|
|
|
std::pair<bool,float> result = Math::RayPlaneIntersecton(ray, mTranslatePlane);
|
|
|
|
|
|
|
|
if (result.first)
|
|
|
|
{
|
|
|
|
CVector3f hit = ray.PointOnRay(result.second);
|
|
|
|
CVector3f localDelta = mRotation.Inverse() * (hit - mPosition);
|
|
|
|
|
|
|
|
// Calculate new position
|
|
|
|
CVector3f newPos = mPosition;
|
|
|
|
if (mSelectedAxes & eX) newPos += mRotation.XAxis() * localDelta.x;
|
|
|
|
if (mSelectedAxes & eY) newPos += mRotation.YAxis() * localDelta.y;
|
|
|
|
if (mSelectedAxes & eZ) newPos += mRotation.ZAxis() * localDelta.z;
|
|
|
|
|
|
|
|
// Check relativity of new pos to camera to reduce issue where the gizmo might
|
|
|
|
// go flying off into the distance if newPosToCamera is parallel to the plane
|
|
|
|
CVector3f newPosToCamera = (newPos - camera.Position()).Normalized();
|
|
|
|
float dot = Math::Abs(planeNormal.Dot(newPosToCamera));
|
|
|
|
if (dot < 0.02f) return false;
|
|
|
|
|
|
|
|
// Set offset
|
|
|
|
if (!mSetOffset)
|
|
|
|
{
|
|
|
|
mTranslateOffset = mPosition - newPos;
|
|
|
|
mDeltaTranslation = CVector3f::skZero;
|
|
|
|
mSetOffset = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply translation
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mDeltaTranslation = mRotation.Inverse() * (newPos - mPosition + mTranslateOffset);
|
2015-08-24 01:02:14 +00:00
|
|
|
if (!(mSelectedAxes & eX)) mDeltaTranslation.x = 0.f;
|
|
|
|
if (!(mSelectedAxes & eY)) mDeltaTranslation.y = 0.f;
|
|
|
|
if (!(mSelectedAxes & eZ)) mDeltaTranslation.z = 0.f;
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
mTotalTranslation += mDeltaTranslation;
|
2015-08-24 01:02:14 +00:00
|
|
|
mPosition += mRotation * mDeltaTranslation;
|
|
|
|
|
|
|
|
if (!mHasTransformed && (mDeltaTranslation != CVector3f::skZero))
|
|
|
|
mHasTransformed = true;
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mDeltaTranslation = CVector3f::skZero;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
// Rotate
|
|
|
|
else if (mMode == eRotate)
|
|
|
|
{
|
|
|
|
// Choose rotation axis
|
|
|
|
CVector3f axis;
|
|
|
|
if (mSelectedAxes & eX) axis = CVector3f::skUnitX;
|
|
|
|
else if (mSelectedAxes & eY) axis = CVector3f::skUnitY;
|
|
|
|
else axis = CVector3f::skUnitZ;
|
|
|
|
|
2015-08-28 22:57:24 +00:00
|
|
|
// Convert hit point + move direction into a line in screen space
|
2015-08-24 01:02:14 +00:00
|
|
|
// Clockwise direction is set in StartTransform(). Is there a cleaner way to calculate the direction?
|
|
|
|
CMatrix4f VP = camera.ViewMatrix().Transpose() * camera.ProjectionMatrix().Transpose();
|
|
|
|
CVector2f lineOrigin = (mHitPoint * VP).xy();
|
2015-08-28 22:57:24 +00:00
|
|
|
CVector2f lineDir = (((mHitPoint + mMoveDir) * VP).xy() - lineOrigin).Normalized();
|
2015-08-24 01:02:14 +00:00
|
|
|
float rotAmount = lineDir.Dot(mouseCoords + mWrapOffset - lineOrigin) * 180.f;
|
|
|
|
|
|
|
|
// Set offset
|
|
|
|
if (!mSetOffset)
|
|
|
|
{
|
|
|
|
mRotateOffset = -rotAmount;
|
|
|
|
mDeltaRotation = CQuaternion::skIdentity;
|
|
|
|
mSetOffset = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply rotation
|
|
|
|
rotAmount += mRotateOffset;
|
|
|
|
CQuaternion oldRot = mCurrentRotation;
|
2015-08-28 22:57:24 +00:00
|
|
|
mCurrentRotation = CQuaternion::FromAxisAngle(Math::DegreesToRadians(rotAmount), axis);
|
2015-08-24 01:02:14 +00:00
|
|
|
mDeltaRotation = mCurrentRotation * oldRot.Inverse();
|
|
|
|
|
|
|
|
if (mTransformSpace == eLocalTransform)
|
|
|
|
mRotation *= mDeltaRotation;
|
|
|
|
|
|
|
|
// Add to total
|
|
|
|
if (mSelectedAxes & eX) mTotalRotation.x = rotAmount;
|
|
|
|
else if (mSelectedAxes & eY) mTotalRotation.y = rotAmount;
|
|
|
|
else mTotalRotation.z = rotAmount;
|
|
|
|
|
|
|
|
if (!mHasTransformed && (rotAmount != 0.f))
|
|
|
|
mHasTransformed = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-08-28 22:57:24 +00:00
|
|
|
// Scale
|
|
|
|
else if (mMode == eScale)
|
|
|
|
{
|
|
|
|
// Create a line in screen space. First step: line origin
|
|
|
|
CMatrix4f VP = camera.ViewMatrix().Transpose() * camera.ProjectionMatrix().Transpose();
|
|
|
|
CVector2f lineOrigin = (mPosition * VP).xy();
|
|
|
|
|
|
|
|
// Next step: determine the appropriate world space direction using the selected axes and then convert to screen space
|
|
|
|
// Since the axes can be flipped while the gizmo is transforming, this has to be done every frame rather than
|
|
|
|
// pre-saving the world space direction like the rotate gizmo does.
|
|
|
|
CVector3f dirX = (mFlipScaleX ? -mRotation.XAxis() : mRotation.XAxis());
|
|
|
|
CVector3f dirY = (mFlipScaleY ? -mRotation.YAxis() : mRotation.YAxis());
|
|
|
|
CVector3f dirZ = (mFlipScaleZ ? -mRotation.ZAxis() : mRotation.ZAxis());
|
|
|
|
CVector2f lineDir;
|
|
|
|
|
|
|
|
// One axis - world space direction is just the selected axis
|
|
|
|
if (NumSelectedAxes() == 1)
|
|
|
|
{
|
|
|
|
CVector3f worldDir;
|
|
|
|
if (mSelectedAxes & eX) worldDir = dirX;
|
|
|
|
else if (mSelectedAxes & eY) worldDir = dirY;
|
|
|
|
else worldDir = dirZ;
|
|
|
|
lineDir = (((mPosition + worldDir) * VP).xy() - lineOrigin).Normalized();
|
|
|
|
}
|
|
|
|
// Two axes - take the two selected axes and convert them to world space, then average them for the line direction
|
|
|
|
else if (NumSelectedAxes() == 2)
|
|
|
|
{
|
|
|
|
CVector3f axisA = (mSelectedAxes & eX ? dirX : dirY);
|
|
|
|
CVector3f axisB = (mSelectedAxes & eZ ? dirZ : dirY);
|
|
|
|
CVector2f screenA = (((mPosition + axisA) * VP).xy() - lineOrigin).Normalized();
|
|
|
|
CVector2f screenB = (((mPosition + axisB) * VP).xy() - lineOrigin).Normalized();
|
|
|
|
lineDir = ((screenA + screenB) / 2.f).Normalized();
|
|
|
|
}
|
|
|
|
// Three axes - use straight up
|
|
|
|
else lineDir = CVector2f::skUp;
|
|
|
|
|
|
|
|
float scaleAmount = lineDir.Dot(mouseCoords + mWrapOffset - lineOrigin) * 5.f;
|
|
|
|
|
|
|
|
// Set offset
|
|
|
|
if (!mSetOffset)
|
|
|
|
{
|
|
|
|
mScaleOffset = -scaleAmount;
|
|
|
|
mDeltaScale = CVector3f::skOne;
|
|
|
|
mSetOffset = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply scale
|
|
|
|
scaleAmount = scaleAmount + mScaleOffset + 1.f;
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
// A multiplier is applied to the scale amount of it's less than 1 to prevent it from going negative
|
2015-08-28 22:57:24 +00:00
|
|
|
if (scaleAmount < 1.f)
|
|
|
|
scaleAmount = 1.f / (-(scaleAmount - 1.f) + 1.f);
|
|
|
|
|
|
|
|
CVector3f oldScale = mTotalScale;
|
|
|
|
|
|
|
|
mTotalScale = CVector3f::skOne;
|
|
|
|
if (mSelectedAxes & eX) mTotalScale.x = scaleAmount;
|
|
|
|
if (mSelectedAxes & eY) mTotalScale.y = scaleAmount;
|
|
|
|
if (mSelectedAxes & eZ) mTotalScale.z = scaleAmount;
|
|
|
|
|
|
|
|
mDeltaScale = mTotalScale / oldScale;
|
|
|
|
|
2015-09-01 17:05:48 +00:00
|
|
|
if (!mHasTransformed && (scaleAmount != 1.f))
|
2015-08-28 22:57:24 +00:00
|
|
|
mHasTransformed = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-08-20 01:01:58 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CGizmo::EndTransform()
|
|
|
|
{
|
2015-08-28 22:57:24 +00:00
|
|
|
mTotalScale = CVector3f::skOne;
|
2015-08-24 01:02:14 +00:00
|
|
|
mIsTransforming = false;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
bool CGizmo::IsTransforming() const
|
2015-09-01 17:05:48 +00:00
|
|
|
{
|
|
|
|
return mIsTransforming;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
bool CGizmo::HasTransformed() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mHasTransformed;
|
2015-08-20 01:01:58 +00:00
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CGizmo::EGizmoMode CGizmo::Mode() const
|
2015-08-16 04:29:37 +00:00
|
|
|
{
|
|
|
|
return mMode;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
ETransformSpace CGizmo::TransformSpace() const
|
2015-09-01 17:05:48 +00:00
|
|
|
{
|
|
|
|
return mTransformSpace;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::Position() const
|
2015-08-20 01:01:58 +00:00
|
|
|
{
|
|
|
|
return mPosition;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::DeltaTranslation() const
|
2015-08-20 01:01:58 +00:00
|
|
|
{
|
|
|
|
return mDeltaTranslation;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::TotalTranslation() const
|
2015-08-20 01:01:58 +00:00
|
|
|
{
|
|
|
|
return mTotalTranslation;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CQuaternion CGizmo::Rotation() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mRotation;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CQuaternion CGizmo::DeltaRotation() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mDeltaRotation;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::TotalRotation() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mTotalRotation;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::Scale() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mScale;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::DeltaScale() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mDeltaScale;
|
|
|
|
}
|
|
|
|
|
2015-09-29 10:48:28 +00:00
|
|
|
CVector3f CGizmo::TotalScale() const
|
2015-08-24 01:02:14 +00:00
|
|
|
{
|
|
|
|
return mTotalScale;
|
|
|
|
}
|
|
|
|
|
2015-08-16 04:29:37 +00:00
|
|
|
void CGizmo::SetMode(EGizmoMode mode)
|
|
|
|
{
|
|
|
|
mMode = mode;
|
2015-08-17 18:10:42 +00:00
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case eTranslate:
|
|
|
|
mpCurrentParts = smTranslateModels;
|
|
|
|
mNumCurrentParts = 9;
|
2015-08-20 01:01:58 +00:00
|
|
|
mDeltaRotation = CQuaternion::skIdentity;
|
|
|
|
mDeltaScale = CVector3f::skOne;
|
2015-08-17 18:10:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case eRotate:
|
|
|
|
mpCurrentParts = smRotateModels;
|
2015-08-24 01:02:14 +00:00
|
|
|
mNumCurrentParts = 5;
|
2015-08-20 01:01:58 +00:00
|
|
|
mDeltaTranslation = CVector3f::skZero;
|
|
|
|
mDeltaScale = CVector3f::skOne;
|
2015-08-17 18:10:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case eScale:
|
|
|
|
mpCurrentParts = smScaleModels;
|
|
|
|
mNumCurrentParts = 10;
|
2015-08-20 01:01:58 +00:00
|
|
|
mDeltaTranslation = CVector3f::skZero;
|
|
|
|
mDeltaRotation = CQuaternion::skIdentity;
|
2015-08-17 18:10:42 +00:00
|
|
|
break;
|
|
|
|
}
|
2015-08-16 04:29:37 +00:00
|
|
|
}
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
void CGizmo::SetTransformSpace(ETransformSpace space)
|
|
|
|
{
|
|
|
|
mTransformSpace = space;
|
2015-08-28 22:57:24 +00:00
|
|
|
|
|
|
|
if (space == eWorldTransform)
|
|
|
|
mRotation = CQuaternion::skIdentity;
|
|
|
|
else
|
|
|
|
mRotation = mLocalRotation;
|
2015-08-24 01:02:14 +00:00
|
|
|
}
|
|
|
|
|
2015-08-16 04:29:37 +00:00
|
|
|
void CGizmo::SetPosition(const CVector3f& position)
|
|
|
|
{
|
|
|
|
mPosition = position;
|
|
|
|
}
|
|
|
|
|
2015-08-28 22:57:24 +00:00
|
|
|
void CGizmo::SetLocalRotation(const CQuaternion& orientation)
|
2015-08-17 18:10:42 +00:00
|
|
|
{
|
2015-08-28 22:57:24 +00:00
|
|
|
mLocalRotation = orientation;
|
|
|
|
|
|
|
|
if (mTransformSpace == eLocalTransform)
|
|
|
|
mRotation = orientation;
|
2015-08-17 18:10:42 +00:00
|
|
|
}
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
void CGizmo::EnableCursorWrap(bool wrap)
|
|
|
|
{
|
|
|
|
mEnableCursorWrap = wrap;
|
|
|
|
}
|
|
|
|
|
2015-08-16 04:29:37 +00:00
|
|
|
// ************ PRIVATE STATIC ************
|
|
|
|
void CGizmo::LoadModels()
|
|
|
|
{
|
|
|
|
if (!smModelsLoaded)
|
|
|
|
{
|
2015-08-24 01:02:14 +00:00
|
|
|
Log::Write("Loading transform gizmo models");
|
|
|
|
|
2015-12-13 20:52:17 +00:00
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_X] = SModelPart(eX, true, false, gResCache.GetResource("../resources/editor/TranslateX.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_Y] = SModelPart(eY, true, false, gResCache.GetResource("../resources/editor/TranslateY.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_Z] = SModelPart(eZ, true, false, gResCache.GetResource("../resources/editor/TranslateZ.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_LINES_XY] = SModelPart(eXY, true, false, gResCache.GetResource("../resources/editor/TranslateLinesXY.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_LINES_XZ] = SModelPart(eXZ, true, false, gResCache.GetResource("../resources/editor/TranslateLinesXZ.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_LINES_YZ] = SModelPart(eYZ, true, false, gResCache.GetResource("../resources/editor/TranslateLinesYZ.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_POLY_XY] = SModelPart(eXY, false, false, gResCache.GetResource("../resources/editor/TranslatePolyXY.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_POLY_XZ] = SModelPart(eXZ, false, false, gResCache.GetResource("../resources/editor/TranslatePolyXZ.CMDL"));
|
|
|
|
smTranslateModels[CGIZMO_TRANSLATE_POLY_YZ] = SModelPart(eYZ, false, false, gResCache.GetResource("../resources/editor/TranslatePolyYZ.CMDL"));
|
|
|
|
|
|
|
|
smRotateModels[CGIZMO_ROTATE_OUTLINE] = SModelPart(eNone, true, true, gResCache.GetResource("../resources/editor/RotateClipOutline.CMDL"));
|
|
|
|
smRotateModels[CGIZMO_ROTATE_X] = SModelPart(eX, true, false, gResCache.GetResource("../resources/editor/RotateX.CMDL"));
|
|
|
|
smRotateModels[CGIZMO_ROTATE_Y] = SModelPart(eY, true, false, gResCache.GetResource("../resources/editor/RotateY.CMDL"));
|
|
|
|
smRotateModels[CGIZMO_ROTATE_Z] = SModelPart(eZ, true, false, gResCache.GetResource("../resources/editor/RotateZ.CMDL"));
|
|
|
|
smRotateModels[CGIZMO_ROTATE_XYZ] = SModelPart(eXYZ, false, false, gResCache.GetResource("../resources/editor/RotateXYZ.CMDL"));
|
|
|
|
|
|
|
|
smScaleModels[CGIZMO_SCALE_X] = SModelPart(eX, true, false, gResCache.GetResource("../resources/editor/ScaleX.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_Y] = SModelPart(eY, true, false, gResCache.GetResource("../resources/editor/ScaleY.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_Z] = SModelPart(eZ, true, false, gResCache.GetResource("../resources/editor/ScaleZ.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_LINES_XY] = SModelPart(eXY, true, false, gResCache.GetResource("../resources/editor/ScaleLinesXY.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_LINES_XZ] = SModelPart(eXZ, true, false, gResCache.GetResource("../resources/editor/ScaleLinesXZ.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_LINES_YZ] = SModelPart(eYZ, true, false, gResCache.GetResource("../resources/editor/ScaleLinesYZ.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_POLY_XY] = SModelPart(eXY, true, false, gResCache.GetResource("../resources/editor/ScalePolyXY.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_POLY_XZ] = SModelPart(eXZ, true, false, gResCache.GetResource("../resources/editor/ScalePolyXZ.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_POLY_YZ] = SModelPart(eYZ, true, false, gResCache.GetResource("../resources/editor/ScalePolyYZ.CMDL"));
|
|
|
|
smScaleModels[CGIZMO_SCALE_XYZ] = SModelPart(eXYZ, true, false, gResCache.GetResource("../resources/editor/ScaleXYZ.CMDL"));
|
2015-08-16 04:29:37 +00:00
|
|
|
|
|
|
|
smModelsLoaded = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ************ PROTECTED ************
|
|
|
|
void CGizmo::UpdateTransform()
|
|
|
|
{
|
|
|
|
// Scale is recalculated every frame because it changes frequently, based on camera distance
|
|
|
|
// Rotation and position values are just saved directly
|
|
|
|
mScale = mGizmoSize * (mCameraDist / 10.f);
|
|
|
|
|
2015-08-28 22:57:24 +00:00
|
|
|
// Scale also factors in axis flip if mode is Scale.
|
2015-08-16 04:29:37 +00:00
|
|
|
if (mMode == eScale)
|
|
|
|
{
|
|
|
|
if (mFlipScaleX) mScale.x = -mScale.x;
|
|
|
|
if (mFlipScaleY) mScale.y = -mScale.y;
|
|
|
|
if (mFlipScaleZ) mScale.z = -mScale.z;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create transform
|
|
|
|
mTransform = CTransform4f::skIdentity;
|
|
|
|
mTransform.Scale(mScale);
|
|
|
|
mTransform.Rotate(mRotation);
|
|
|
|
mTransform.Translate(mPosition);
|
|
|
|
|
|
|
|
// Create billboard transform for rotation gizmo
|
|
|
|
if (mMode == eRotate)
|
|
|
|
{
|
|
|
|
mBillboardTransform = CTransform4f::skIdentity;
|
|
|
|
mBillboardTransform.Scale(mScale);
|
|
|
|
mBillboardTransform.Rotate(mBillboardRotation);
|
|
|
|
mBillboardTransform.Translate(mPosition);
|
|
|
|
}
|
2015-08-28 22:57:24 +00:00
|
|
|
|
|
|
|
// Create scaled transform for scale gizmo
|
|
|
|
else if (mMode == eScale)
|
|
|
|
{
|
|
|
|
mScaledTransform = CTransform4f::skIdentity;
|
|
|
|
mScaledTransform.Scale(mScale * mTotalScale);
|
|
|
|
mScaledTransform.Rotate(mRotation);
|
|
|
|
mScaledTransform.Translate(mPosition);
|
|
|
|
}
|
2015-08-16 04:29:37 +00:00
|
|
|
}
|
|
|
|
|
2015-08-24 01:02:14 +00:00
|
|
|
void CGizmo::WrapCursor()
|
|
|
|
{
|
|
|
|
QRect geom = QApplication::desktop()->screenGeometry();
|
|
|
|
QPoint cursorPos = QCursor::pos();
|
|
|
|
|
|
|
|
// Horizontal
|
|
|
|
if (cursorPos.x() == geom.width() - 1)
|
|
|
|
{
|
|
|
|
QCursor::setPos(1, cursorPos.y());
|
|
|
|
mWrapOffset.x += 2.f;
|
|
|
|
}
|
|
|
|
else if (cursorPos.x() == 0)
|
|
|
|
{
|
|
|
|
QCursor::setPos(geom.width() - 2, cursorPos.y());
|
|
|
|
mWrapOffset.x -= 2.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Vertical
|
|
|
|
cursorPos = QCursor::pos(); // Grab again to account for changes on horizontal wrap
|
|
|
|
|
|
|
|
if (cursorPos.y() == geom.height() - 1)
|
|
|
|
{
|
|
|
|
QCursor::setPos(cursorPos.x(), 1);
|
|
|
|
mWrapOffset.y -= 2.f;
|
|
|
|
}
|
|
|
|
else if (cursorPos.y() == 0)
|
|
|
|
{
|
|
|
|
QCursor::setPos(cursorPos.x(), geom.height() - 2);
|
|
|
|
mWrapOffset.y += 2.f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-16 04:29:37 +00:00
|
|
|
// ************ STATIC MEMBER INITIALIZATION ************
|
|
|
|
bool CGizmo::smModelsLoaded = false;
|
2015-08-17 18:10:42 +00:00
|
|
|
CGizmo::SModelPart CGizmo::smTranslateModels[9];
|
2015-08-24 01:02:14 +00:00
|
|
|
CGizmo::SModelPart CGizmo::smRotateModels[5];
|
2015-08-17 18:10:42 +00:00
|
|
|
CGizmo::SModelPart CGizmo::smScaleModels[10];
|