Change free axis rotation.

pull/39/head^2
Aesylwinn 9 years ago
parent 3e4ac0c662
commit 7125775648

@ -188,6 +188,7 @@ void CSMPrefs::State::declare()
"Shift-acceleration factor during drag operations", 4.0). "Shift-acceleration factor during drag operations", 4.0).
setTooltip ("Acceleration factor during drag operations while holding down shift"). setTooltip ("Acceleration factor during drag operations while holding down shift").
setRange (0.001, 100.0); setRange (0.001, 100.0);
declareDouble ("rotate-factor", "Free rotation factor", 0.007).setPrecision(4).setRange(0.0001, 0.1);
declareCategory ("Tooltips"); declareCategory ("Tooltips");
declareBool ("scene", "Show Tooltips in 3D scenes", true); declareBool ("scene", "Show Tooltips in 3D scenes", true);

@ -323,33 +323,28 @@ void CSVRender::InstanceMode::drag (const QPoint& pos, int diffX, int diffY, dou
osg::Vec3f eye, centre, up; osg::Vec3f eye, centre, up;
getWorldspaceWidget().getCamera()->getViewMatrix().getLookAt (eye, centre, up); getWorldspaceWidget().getCamera()->getViewMatrix().getLookAt (eye, centre, up);
osg::Vec3f camBack = eye - centre; float angle;
osg::Vec3f axis;
// Convert coordinate system
osg::Vec3f screenCenter = getScreenCoords(getSelectionCenter(selection));
int widgetHeight = getWorldspaceWidget().height();
float newX = pos.x() - screenCenter.x();
float newY = (widgetHeight - pos.y()) - screenCenter.y();
float oldX = newX - diffX;
float oldY = newY - diffY; // diffY appears to already be flipped
osg::Vec3f oldVec = osg::Vec3f(oldX, oldY, 0); if (mDragAxis == -1)
oldVec.normalize(); {
// Free rotate
float rotationFactor = CSMPrefs::get()["3D Scene Input"]["rotate-factor"].toDouble() * speedFactor;
osg::Vec3f newVec = osg::Vec3f(newX, newY, 0); osg::Quat cameraRotation = getWorldspaceWidget().getCamera()->getInverseViewMatrix().getRotate();
newVec.normalize();
// Find angle and axis of rotation osg::Vec3f camForward = centre - eye;
float angle = std::acos(oldVec * newVec) * speedFactor; osg::Vec3f screenDir = cameraRotation * osg::Vec3f(diffX, diffY, 0);
if (((oldVec ^ newVec) * camBack < 0) ^ (camBack.z() < 0)) screenDir.normalize();
angle *= -1;
osg::Vec3f axis; angle = std::sqrt(diffX*diffX + diffY*diffY) * rotationFactor;
if (mDragAxis != -1) axis = screenDir ^ camForward;
}
else
{ {
// Global axis rotation
osg::Vec3f camBack = eye - centre;
for (int i = 0; i < 3; ++i) for (int i = 0; i < 3; ++i)
{ {
if (i == mDragAxis) if (i == mDragAxis)
@ -358,11 +353,32 @@ void CSVRender::InstanceMode::drag (const QPoint& pos, int diffX, int diffY, dou
axis[i] = 0; axis[i] = 0;
} }
// Flip axis if facing opposite side
if (camBack * axis < 0) if (camBack * axis < 0)
axis *= -1; axis *= -1;
// Convert coordinate system
osg::Vec3f screenCenter = getScreenCoords(getSelectionCenter(selection));
int widgetHeight = getWorldspaceWidget().height();
float newX = pos.x() - screenCenter.x();
float newY = (widgetHeight - pos.y()) - screenCenter.y();
float oldX = newX - diffX;
float oldY = newY - diffY; // diffY appears to already be flipped
osg::Vec3f oldVec = osg::Vec3f(oldX, oldY, 0);
oldVec.normalize();
osg::Vec3f newVec = osg::Vec3f(newX, newY, 0);
newVec.normalize();
// Find angle and axis of rotation
angle = std::acos(oldVec * newVec) * speedFactor;
if (((oldVec ^ newVec) * camBack < 0) ^ (camBack.z() < 0))
angle *= -1;
} }
else
axis = camBack;
rotation = osg::Quat(angle, axis); rotation = osg::Quat(angle, axis);
} }

Loading…
Cancel
Save