mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-04 18:06:22 +03:00
3476 lines
115 KiB
C++
3476 lines
115 KiB
C++
// Copyright (c) 2016-2019 OPEN CASCADE SAS
|
|
//
|
|
// This file is part of Open CASCADE Technology software library.
|
|
//
|
|
// This library is free software; you can redistribute it and/or modify it under
|
|
// the terms of the GNU Lesser General Public License version 2.1 as published
|
|
// by the Free Software Foundation, with special exception defined in the file
|
|
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
|
|
// distribution for complete text of the license and disclaimer of any warranty.
|
|
//
|
|
// Alternatively, this file may be used under the terms of Open CASCADE
|
|
// commercial license or contractual agreement.
|
|
|
|
#include "AIS_ViewController.hxx"
|
|
|
|
#include <AIS_AnimationCamera.hxx>
|
|
#include <AIS_InteractiveContext.hxx>
|
|
#include <AIS_Point.hxx>
|
|
#include <AIS_RubberBand.hxx>
|
|
#include <AIS_XRTrackedDevice.hxx>
|
|
#include <Aspect_XRSession.hxx>
|
|
#include <Aspect_Grid.hxx>
|
|
#include <Geom_CartesianPoint.hxx>
|
|
#include <Message.hxx>
|
|
#include <gp_Quaternion.hxx>
|
|
#include <V3d_View.hxx>
|
|
#include <V3d_Viewer.hxx>
|
|
#include <WNT_HIDSpaceMouse.hxx>
|
|
|
|
//=================================================================================================
|
|
|
|
AIS_ViewController::AIS_ViewController()
|
|
: myLastEventsTime(0.0),
|
|
myToAskNextFrame(false),
|
|
myIsContinuousRedraw(false),
|
|
myMinCamDistance(1.0),
|
|
myRotationMode(AIS_RotationMode_BndBoxActive),
|
|
myNavigationMode(AIS_NavigationMode_Orbit),
|
|
myMouseAccel(1.0f),
|
|
myOrbitAccel(1.0f),
|
|
myToShowPanAnchorPoint(true),
|
|
myToShowRotateCenter(true),
|
|
myToLockOrbitZUp(false),
|
|
myToInvertPitch(false),
|
|
myToAllowTouchZRotation(false),
|
|
myToAllowRotation(true),
|
|
myToAllowPanning(true),
|
|
myToAllowZooming(true),
|
|
myToAllowZFocus(true),
|
|
myToAllowHighlight(true),
|
|
myToAllowDragging(true),
|
|
myToStickToRayOnZoom(true),
|
|
myToStickToRayOnRotation(true),
|
|
//
|
|
myWalkSpeedAbsolute(1.5f),
|
|
myWalkSpeedRelative(0.1f),
|
|
myThrustSpeed(0.0f),
|
|
myHasThrust(false),
|
|
//
|
|
myViewAnimation(
|
|
new AIS_AnimationCamera("AIS_ViewController_ViewAnimation", Handle(V3d_View)())),
|
|
myObjAnimation(new AIS_Animation("AIS_ViewController_ObjectsAnimation")),
|
|
myToPauseObjAnimation(false),
|
|
myPrevMoveTo(-1, -1),
|
|
myHasHlrOnBeforeRotation(false),
|
|
//
|
|
myXRPrsDevices(0, 0),
|
|
myXRLaserTeleColor(Quantity_NOC_GREEN),
|
|
myXRLaserPickColor(Quantity_NOC_BLUE),
|
|
myXRLastTeleportHand(Aspect_XRTrackedDeviceRole_Other),
|
|
myXRLastPickingHand(Aspect_XRTrackedDeviceRole_Other),
|
|
myXRLastPickDepthLeft(Precision::Infinite()),
|
|
myXRLastPickDepthRight(Precision::Infinite()),
|
|
myXRTurnAngle(M_PI_4),
|
|
myToDisplayXRAuxDevices(false),
|
|
myToDisplayXRHands(true),
|
|
//
|
|
myMouseClickThreshold(3.0),
|
|
myMouseDoubleClickInt(0.4),
|
|
myScrollZoomRatio(15.0f),
|
|
myMouseActiveGesture(AIS_MouseGesture_NONE),
|
|
myMouseActiveIdleRotation(false),
|
|
myMouseClickCounter(0),
|
|
myMouseSingleButton(-1),
|
|
myMouseStopDragOnUnclick(false),
|
|
//
|
|
myTouchToleranceScale(1.0f),
|
|
myTouchClickThresholdPx(3.0f),
|
|
myTouchRotationThresholdPx(6.0f),
|
|
myTouchZRotationThreshold(float(2.0 * M_PI / 180.0)),
|
|
myTouchPanThresholdPx(4.0f),
|
|
myTouchZoomThresholdPx(6.0f),
|
|
myTouchZoomRatio(0.13f),
|
|
myTouchDraggingThresholdPx(6.0f),
|
|
//
|
|
myNbTouchesLast(0),
|
|
myUpdateStartPointPan(true),
|
|
myUpdateStartPointRot(true),
|
|
myUpdateStartPointZRot(true),
|
|
//
|
|
myPanPnt3d(Precision::Infinite(), 0.0, 0.0)
|
|
{
|
|
myViewAnimation->SetOwnDuration(0.5);
|
|
|
|
myAnchorPointPrs1 = new AIS_Point(new Geom_CartesianPoint(0.0, 0.0, 0.0));
|
|
myAnchorPointPrs1->SetZLayer(Graphic3d_ZLayerId_Top);
|
|
myAnchorPointPrs1->SetMutable(true);
|
|
|
|
myAnchorPointPrs2 = new AIS_Point(new Geom_CartesianPoint(0.0, 0.0, 0.0));
|
|
myAnchorPointPrs2->SetZLayer(Graphic3d_ZLayerId_Topmost);
|
|
myAnchorPointPrs2->SetMutable(true);
|
|
|
|
myRubberBand =
|
|
new AIS_RubberBand(Quantity_NOC_LIGHTBLUE, Aspect_TOL_SOLID, Quantity_NOC_LIGHTBLUE4, 0.5, 1.0);
|
|
myRubberBand->SetZLayer(Graphic3d_ZLayerId_TopOSD);
|
|
myRubberBand->SetTransformPersistence(
|
|
new Graphic3d_TransformPers(Graphic3d_TMF_2d, Aspect_TOTP_LEFT_UPPER));
|
|
myRubberBand->SetDisplayMode(0);
|
|
myRubberBand->SetMutable(true);
|
|
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton,
|
|
AIS_MouseGesture_RotateOrbit);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_CTRL,
|
|
AIS_MouseGesture_Zoom);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_SHIFT,
|
|
AIS_MouseGesture_Pan);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_ALT,
|
|
AIS_MouseGesture_SelectRectangle);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_ALT
|
|
| (Standard_UInteger)Aspect_VKeyFlags_SHIFT,
|
|
AIS_MouseGesture_SelectRectangle);
|
|
|
|
myMouseSelectionSchemes.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton,
|
|
AIS_SelectionScheme_Replace);
|
|
myMouseSelectionSchemes.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_ALT,
|
|
AIS_SelectionScheme_Replace);
|
|
myMouseSelectionSchemes.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_SHIFT,
|
|
AIS_SelectionScheme_XOR);
|
|
myMouseSelectionSchemes.Bind((Standard_UInteger)Aspect_VKeyMouse_LeftButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_ALT
|
|
| (Standard_UInteger)Aspect_VKeyFlags_SHIFT,
|
|
AIS_SelectionScheme_XOR);
|
|
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_RightButton, AIS_MouseGesture_Zoom);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_RightButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_CTRL,
|
|
AIS_MouseGesture_RotateOrbit);
|
|
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_MiddleButton, AIS_MouseGesture_Pan);
|
|
myMouseGestureMap.Bind((Standard_UInteger)Aspect_VKeyMouse_MiddleButton
|
|
| (Standard_UInteger)Aspect_VKeyFlags_CTRL,
|
|
AIS_MouseGesture_Pan);
|
|
|
|
myMouseGestureMapDrag.Bind(Aspect_VKeyMouse_LeftButton, AIS_MouseGesture_Drag);
|
|
|
|
myXRTeleportHaptic.Duration = 3600.0f;
|
|
myXRTeleportHaptic.Frequency = 0.1f;
|
|
myXRTeleportHaptic.Amplitude = 0.2f;
|
|
|
|
myXRPickingHaptic.Duration = 0.1f;
|
|
myXRPickingHaptic.Frequency = 4.0f;
|
|
myXRPickingHaptic.Amplitude = 0.1f;
|
|
|
|
myXRSelectHaptic.Duration = 0.2f;
|
|
myXRSelectHaptic.Frequency = 4.0f;
|
|
myXRSelectHaptic.Amplitude = 0.5f;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
AIS_ViewController::~AIS_ViewController()
|
|
{
|
|
//
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::ResetViewInput()
|
|
{
|
|
myKeys.Reset();
|
|
myMousePressed = Aspect_VKeyMouse_NONE;
|
|
myMouseModifiers = Aspect_VKeyFlags_NONE;
|
|
myMouseSingleButton = -1;
|
|
myUI.Dragging.ToAbort = true;
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
myMouseClickTimer.Stop();
|
|
myMouseClickCounter = 0;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::FlushViewEvents(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
Standard_Boolean theToHandle)
|
|
{
|
|
flushBuffers(theCtx, theView);
|
|
flushGestures(theCtx, theView);
|
|
|
|
if (theView->IsSubview())
|
|
{
|
|
// move input coordinates inside the view
|
|
const Graphic3d_Vec2i aDelta = theView->View()->SubviewTopLeft();
|
|
if (myGL.MoveTo.ToHilight || myGL.Dragging.ToStart)
|
|
{
|
|
myGL.MoveTo.Point -= aDelta;
|
|
}
|
|
if (myGL.Panning.ToStart)
|
|
{
|
|
myGL.Panning.PointStart -= aDelta;
|
|
}
|
|
if (myGL.Dragging.ToStart)
|
|
{
|
|
myGL.Dragging.PointStart -= aDelta;
|
|
}
|
|
if (myGL.Dragging.ToMove)
|
|
{
|
|
myGL.Dragging.PointTo -= aDelta;
|
|
}
|
|
if (myGL.OrbitRotation.ToStart)
|
|
{
|
|
myGL.OrbitRotation.PointStart -= Graphic3d_Vec2d(aDelta);
|
|
}
|
|
if (myGL.OrbitRotation.ToRotate)
|
|
{
|
|
myGL.OrbitRotation.PointTo -= Graphic3d_Vec2d(aDelta);
|
|
}
|
|
if (myGL.ViewRotation.ToStart)
|
|
{
|
|
myGL.ViewRotation.PointStart -= Graphic3d_Vec2d(aDelta);
|
|
}
|
|
if (myGL.ViewRotation.ToRotate)
|
|
{
|
|
myGL.ViewRotation.PointTo -= Graphic3d_Vec2d(aDelta);
|
|
}
|
|
for (Graphic3d_Vec2i& aPntIter : myGL.Selection.Points)
|
|
{
|
|
aPntIter -= aDelta;
|
|
}
|
|
for (Aspect_ScrollDelta& aZoomIter : myGL.ZoomActions)
|
|
{
|
|
aZoomIter.Point -= aDelta;
|
|
}
|
|
}
|
|
|
|
if (theToHandle)
|
|
{
|
|
HandleViewEvents(theCtx, theView);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::flushBuffers(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)&)
|
|
{
|
|
myToAskNextFrame = false;
|
|
|
|
myGL.IsNewGesture = myUI.IsNewGesture;
|
|
myUI.IsNewGesture = false;
|
|
|
|
myGL.ZoomActions.Clear();
|
|
myGL.ZoomActions.Append(myUI.ZoomActions);
|
|
myUI.ZoomActions.Clear();
|
|
|
|
myGL.Orientation.ToFitAll = myUI.Orientation.ToFitAll;
|
|
myUI.Orientation.ToFitAll = false;
|
|
if (myUI.Orientation.ToSetViewOrient)
|
|
{
|
|
myUI.Orientation.ToSetViewOrient = false;
|
|
myGL.Orientation.ToSetViewOrient = true;
|
|
myGL.Orientation.ViewOrient = myUI.Orientation.ViewOrient;
|
|
}
|
|
|
|
if (myUI.MoveTo.ToHilight)
|
|
{
|
|
myUI.MoveTo.ToHilight = false;
|
|
myGL.MoveTo.ToHilight = true;
|
|
myGL.MoveTo.Point = myUI.MoveTo.Point;
|
|
}
|
|
|
|
{
|
|
myGL.Selection.Tool = myUI.Selection.Tool;
|
|
myGL.Selection.Scheme = myUI.Selection.Scheme;
|
|
myGL.Selection.Points = myUI.Selection.Points;
|
|
// myGL.Selection.Scheme = AIS_SelectionScheme_UNKNOWN; // no need
|
|
if (myUI.Selection.Tool == AIS_ViewSelectionTool_Picking)
|
|
{
|
|
myUI.Selection.Points.Clear();
|
|
}
|
|
}
|
|
|
|
if (myUI.Selection.ToApplyTool)
|
|
{
|
|
myGL.Selection.ToApplyTool = true;
|
|
myUI.Selection.ToApplyTool = false;
|
|
myUI.Selection.Points.Clear();
|
|
}
|
|
|
|
if (myUI.Panning.ToStart)
|
|
{
|
|
myUI.Panning.ToStart = false;
|
|
myGL.Panning.ToStart = true;
|
|
myGL.Panning.PointStart = myUI.Panning.PointStart;
|
|
}
|
|
|
|
if (myUI.Panning.ToPan)
|
|
{
|
|
myUI.Panning.ToPan = false;
|
|
myGL.Panning.ToPan = true;
|
|
myGL.Panning.Delta = myUI.Panning.Delta;
|
|
}
|
|
|
|
if (myUI.Dragging.ToAbort)
|
|
{
|
|
myUI.Dragging.ToAbort = false;
|
|
myGL.Dragging.ToAbort = true;
|
|
}
|
|
else if (myUI.Dragging.ToStop)
|
|
{
|
|
myUI.Dragging.ToStop = false;
|
|
myGL.Dragging.ToStop = true;
|
|
}
|
|
else
|
|
{
|
|
if (myUI.Dragging.ToStart)
|
|
{
|
|
myUI.Dragging.ToStart = false;
|
|
myGL.Dragging.ToStart = true;
|
|
myGL.Dragging.PointStart = myUI.Dragging.PointStart;
|
|
}
|
|
if (myUI.Dragging.ToConfirm)
|
|
{
|
|
myUI.Dragging.ToConfirm = false;
|
|
myGL.Dragging.ToConfirm = true;
|
|
}
|
|
if (myUI.Dragging.ToMove)
|
|
{
|
|
myUI.Dragging.ToMove = false;
|
|
myGL.Dragging.ToMove = true;
|
|
}
|
|
}
|
|
|
|
myGL.Dragging.PointTo = myUI.Dragging.PointTo;
|
|
|
|
if (myUI.OrbitRotation.ToStart)
|
|
{
|
|
myUI.OrbitRotation.ToStart = false;
|
|
myGL.OrbitRotation.ToStart = true;
|
|
myGL.OrbitRotation.PointStart = myUI.OrbitRotation.PointStart;
|
|
}
|
|
|
|
if (myUI.OrbitRotation.ToRotate)
|
|
{
|
|
myUI.OrbitRotation.ToRotate = false;
|
|
myGL.OrbitRotation.ToRotate = true;
|
|
myGL.OrbitRotation.PointTo = myUI.OrbitRotation.PointTo;
|
|
}
|
|
|
|
if (myUI.ViewRotation.ToStart)
|
|
{
|
|
myUI.ViewRotation.ToStart = false;
|
|
myGL.ViewRotation.ToStart = true;
|
|
myGL.ViewRotation.PointStart = myUI.ViewRotation.PointStart;
|
|
}
|
|
|
|
if (myUI.ViewRotation.ToRotate)
|
|
{
|
|
myUI.ViewRotation.ToRotate = false;
|
|
myGL.ViewRotation.ToRotate = true;
|
|
myGL.ViewRotation.PointTo = myUI.ViewRotation.PointTo;
|
|
}
|
|
|
|
if (myUI.ZRotate.ToRotate)
|
|
{
|
|
myGL.ZRotate = myUI.ZRotate;
|
|
myUI.ZRotate.ToRotate = false;
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::flushGestures(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
const Standard_Real aTolScale = myTouchToleranceScale;
|
|
const Standard_Integer aTouchNb = myTouchPoints.Extent();
|
|
if (myNbTouchesLast != aTouchNb)
|
|
{
|
|
myNbTouchesLast = aTouchNb;
|
|
myGL.IsNewGesture = true;
|
|
}
|
|
if (aTouchNb == 1) // touch
|
|
{
|
|
Aspect_Touch& aTouch = myTouchPoints.ChangeFromIndex(1);
|
|
if (myUpdateStartPointRot)
|
|
{
|
|
// skip rotation if have active dragged object
|
|
if (myNavigationMode == AIS_NavigationMode_Orbit)
|
|
{
|
|
myGL.OrbitRotation.ToStart = true;
|
|
myGL.OrbitRotation.PointStart = myStartRotCoord;
|
|
}
|
|
else
|
|
{
|
|
myGL.ViewRotation.ToStart = true;
|
|
myGL.ViewRotation.PointStart = myStartRotCoord;
|
|
}
|
|
|
|
myUpdateStartPointRot = false;
|
|
theView->Invalidate();
|
|
}
|
|
|
|
// rotation
|
|
const Standard_Real aRotTouchTol =
|
|
!aTouch.IsPreciseDevice ? aTolScale * myTouchRotationThresholdPx : gp::Resolution();
|
|
if (Abs(aTouch.Delta().x()) + Abs(aTouch.Delta().y()) > aRotTouchTol)
|
|
{
|
|
const Standard_Real aRotAccel =
|
|
myNavigationMode == AIS_NavigationMode_FirstPersonWalk ? myMouseAccel : myOrbitAccel;
|
|
if (myNavigationMode == AIS_NavigationMode_Orbit)
|
|
{
|
|
const Graphic3d_Vec2d aRotDelta = aTouch.To - myGL.OrbitRotation.PointStart;
|
|
myGL.OrbitRotation.ToRotate = true;
|
|
myGL.OrbitRotation.PointTo = myGL.OrbitRotation.PointStart + aRotDelta * aRotAccel;
|
|
myGL.Dragging.ToMove = true;
|
|
myGL.Dragging.PointTo.SetValues((int)aTouch.To.x(), (int)aTouch.To.y());
|
|
}
|
|
else
|
|
{
|
|
const Graphic3d_Vec2d aRotDelta = aTouch.To - myGL.ViewRotation.PointStart;
|
|
myGL.ViewRotation.ToRotate = true;
|
|
myGL.ViewRotation.PointTo = myGL.ViewRotation.PointStart + aRotDelta * aRotAccel;
|
|
myGL.Dragging.ToMove = true;
|
|
myGL.Dragging.PointTo.SetValues((int)aTouch.To.x(), (int)aTouch.To.y());
|
|
}
|
|
|
|
aTouch.From = aTouch.To;
|
|
}
|
|
}
|
|
else if (aTouchNb == 2) // pinch
|
|
{
|
|
Aspect_Touch& aFirstTouch = myTouchPoints.ChangeFromIndex(1);
|
|
Aspect_Touch& aLastTouch = myTouchPoints.ChangeFromIndex(2);
|
|
const Graphic3d_Vec2d aFrom[2] = {aFirstTouch.From, aLastTouch.From};
|
|
const Graphic3d_Vec2d aTo[2] = {aFirstTouch.To, aLastTouch.To};
|
|
|
|
Graphic3d_Vec2d aPinchCenterStart((aFrom[0].x() + aFrom[1].x()) / 2.0,
|
|
(aFrom[0].y() + aFrom[1].y()) / 2.0);
|
|
|
|
Standard_Real aPinchCenterXEnd = (aTo[0].x() + aTo[1].x()) / 2.0;
|
|
Standard_Real aPinchCenterYEnd = (aTo[0].y() + aTo[1].y()) / 2.0;
|
|
|
|
Standard_Real aPinchCenterXDev = aPinchCenterXEnd - aPinchCenterStart.x();
|
|
Standard_Real aPinchCenterYDev = aPinchCenterYEnd - aPinchCenterStart.y();
|
|
|
|
Standard_Real aStartSize = (aFrom[0] - aFrom[1]).Modulus();
|
|
Standard_Real anEndSize = (aTo[0] - aTo[1]).Modulus();
|
|
|
|
Standard_Real aDeltaSize = anEndSize - aStartSize;
|
|
|
|
bool anIsClearDev = false;
|
|
|
|
if (myToAllowTouchZRotation)
|
|
{
|
|
Standard_Real A1 = aFrom[0].y() - aFrom[1].y();
|
|
Standard_Real B1 = aFrom[1].x() - aFrom[0].x();
|
|
|
|
Standard_Real A2 = aTo[0].y() - aTo[1].y();
|
|
Standard_Real B2 = aTo[1].x() - aTo[0].x();
|
|
|
|
Standard_Real aRotAngle = 0.0;
|
|
|
|
Standard_Real aDenomenator = A1 * A2 + B1 * B2;
|
|
if (aDenomenator <= Precision::Confusion())
|
|
{
|
|
aRotAngle = 0.0;
|
|
}
|
|
else
|
|
{
|
|
Standard_Real aNumerator = A1 * B2 - A2 * B1;
|
|
aRotAngle = ATan(aNumerator / aDenomenator);
|
|
}
|
|
|
|
if (Abs(aRotAngle) > Standard_Real(myTouchZRotationThreshold))
|
|
{
|
|
myGL.ZRotate.ToRotate = true;
|
|
myGL.ZRotate.Angle = aRotAngle;
|
|
anIsClearDev = true;
|
|
}
|
|
}
|
|
|
|
if (Abs(aDeltaSize) > aTolScale * myTouchZoomThresholdPx)
|
|
{
|
|
// zoom
|
|
aDeltaSize *= Standard_Real(myTouchZoomRatio);
|
|
Aspect_ScrollDelta aParams(Graphic3d_Vec2i(aPinchCenterStart), aDeltaSize);
|
|
myGL.ZoomActions.Append(aParams);
|
|
anIsClearDev = true;
|
|
}
|
|
|
|
const Standard_Real aPanTouchTol =
|
|
!aFirstTouch.IsPreciseDevice ? aTolScale * myTouchPanThresholdPx : gp::Resolution();
|
|
if (Abs(aPinchCenterXDev) + Abs(aPinchCenterYDev) > aPanTouchTol)
|
|
{
|
|
// pan
|
|
if (myUpdateStartPointPan)
|
|
{
|
|
myGL.Panning.ToStart = true;
|
|
myGL.Panning.PointStart = Graphic3d_Vec2i(myStartPanCoord);
|
|
myUpdateStartPointPan = false;
|
|
theView->Invalidate();
|
|
}
|
|
|
|
myGL.Panning.ToPan = true;
|
|
myGL.Panning.Delta.x() = int(aPinchCenterXDev);
|
|
myGL.Panning.Delta.y() = int(-aPinchCenterYDev);
|
|
anIsClearDev = true;
|
|
}
|
|
|
|
if (anIsClearDev)
|
|
{
|
|
aFirstTouch.From = aFirstTouch.To;
|
|
aLastTouch.From = aLastTouch.To;
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::UpdateViewOrientation(V3d_TypeOfOrientation theOrientation,
|
|
bool theToFitAll)
|
|
{
|
|
myUI.Orientation.ToFitAll = theToFitAll;
|
|
myUI.Orientation.ToSetViewOrient = true;
|
|
myUI.Orientation.ViewOrient = theOrientation;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::SelectInViewer(const Graphic3d_Vec2i& thePnt,
|
|
const AIS_SelectionScheme theScheme)
|
|
{
|
|
if (myUI.Selection.Tool != AIS_ViewSelectionTool_Picking)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_Picking;
|
|
myUI.Selection.Points.Clear();
|
|
}
|
|
|
|
myUI.Selection.Scheme = theScheme;
|
|
myUI.Selection.Points.Append(thePnt);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::SelectInViewer(const NCollection_Sequence<Graphic3d_Vec2i>& thePnts,
|
|
const AIS_SelectionScheme theScheme)
|
|
{
|
|
myUI.Selection.Scheme = theScheme;
|
|
myUI.Selection.Points = thePnts;
|
|
myUI.Selection.ToApplyTool = true;
|
|
if (thePnts.Length() == 1)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_Picking;
|
|
}
|
|
else if (thePnts.Length() == 2)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_RubberBand;
|
|
}
|
|
else
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_Polygon;
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::UpdateRubberBand(const Graphic3d_Vec2i& thePntFrom,
|
|
const Graphic3d_Vec2i& thePntTo)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_RubberBand;
|
|
myUI.Selection.Points.Clear();
|
|
myUI.Selection.Points.Append(thePntFrom);
|
|
myUI.Selection.Points.Append(thePntTo);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::UpdatePolySelection(const Graphic3d_Vec2i& thePnt, bool theToAppend)
|
|
{
|
|
if (myUI.Selection.Tool != AIS_ViewSelectionTool_Polygon)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_Polygon;
|
|
myUI.Selection.Points.Clear();
|
|
}
|
|
|
|
if (myUI.Selection.Points.IsEmpty())
|
|
{
|
|
myUI.Selection.Points.Append(thePnt);
|
|
}
|
|
else if (theToAppend && myUI.Selection.Points.Last() != thePnt)
|
|
{
|
|
myUI.Selection.Points.Append(thePnt);
|
|
}
|
|
else
|
|
{
|
|
myUI.Selection.Points.ChangeLast() = thePnt;
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateZoom(const Aspect_ScrollDelta& theDelta)
|
|
{
|
|
if (!myUI.ZoomActions.IsEmpty())
|
|
{
|
|
if (myUI.ZoomActions.ChangeLast().Point == theDelta.Point)
|
|
{
|
|
myUI.ZoomActions.ChangeLast().Delta += theDelta.Delta;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
myUI.ZoomActions.Append(theDelta);
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateZRotation(double theAngle)
|
|
{
|
|
if (!ToAllowTouchZRotation())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
myUI.ZRotate.Angle = myUI.ZRotate.ToRotate ? myUI.ZRotate.Angle + theAngle : theAngle;
|
|
if (myUI.ZRotate.ToRotate)
|
|
{
|
|
return false;
|
|
}
|
|
myUI.ZRotate.ToRotate = true;
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateMouseScroll(const Aspect_ScrollDelta& theDelta)
|
|
{
|
|
Aspect_ScrollDelta aDelta = theDelta;
|
|
aDelta.Delta *= myScrollZoomRatio;
|
|
return UpdateZoom(aDelta);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateMouseClick(const Graphic3d_Vec2i& thePoint,
|
|
Aspect_VKeyMouse theButton,
|
|
Aspect_VKeyFlags theModifiers,
|
|
bool theIsDoubleClick)
|
|
{
|
|
(void)theIsDoubleClick;
|
|
|
|
if (myToPauseObjAnimation && !myObjAnimation.IsNull() && !myObjAnimation->IsStopped())
|
|
{
|
|
myObjAnimation->Pause();
|
|
}
|
|
|
|
AIS_SelectionScheme aScheme = AIS_SelectionScheme_UNKNOWN;
|
|
if (myMouseSelectionSchemes.Find(theButton | theModifiers, aScheme))
|
|
{
|
|
SelectInViewer(thePoint, aScheme);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateMouseButtons(const Graphic3d_Vec2i& thePoint,
|
|
Aspect_VKeyMouse theButtons,
|
|
Aspect_VKeyFlags theModifiers,
|
|
bool theIsEmulated)
|
|
{
|
|
bool toUpdateView = false;
|
|
const double aTolClick = (theIsEmulated ? myTouchToleranceScale : 1.0) * myMouseClickThreshold;
|
|
if (theButtons == Aspect_VKeyMouse_NONE && myMouseSingleButton > 0)
|
|
{
|
|
const Graphic3d_Vec2i aDelta = thePoint - myMousePressPoint;
|
|
if (double(aDelta.cwiseAbs().maxComp()) < aTolClick)
|
|
{
|
|
++myMouseClickCounter;
|
|
|
|
const bool isCounterValid = myMouseClickCounter == 2;
|
|
const bool isTimerStarted = myMouseClickTimer.IsStarted();
|
|
const bool isTimerElapsed = myMouseClickTimer.ElapsedTime() > myMouseDoubleClickInt;
|
|
|
|
const bool isTimerValid = isTimerStarted && !isTimerElapsed;
|
|
|
|
const bool isDoubleClick = isCounterValid && isTimerValid;
|
|
|
|
if (!isTimerValid)
|
|
{
|
|
myMouseClickCounter = 1;
|
|
}
|
|
|
|
myMouseClickCounter %= 2;
|
|
|
|
myMouseClickTimer.Stop();
|
|
myMouseClickTimer.Reset();
|
|
myMouseClickTimer.Start();
|
|
if (isDoubleClick)
|
|
{
|
|
myMouseClickCounter = 0;
|
|
}
|
|
toUpdateView = UpdateMouseClick(thePoint,
|
|
(Aspect_VKeyMouse)myMouseSingleButton,
|
|
theModifiers,
|
|
isDoubleClick)
|
|
|| toUpdateView;
|
|
}
|
|
else
|
|
{
|
|
myMouseClickTimer.Stop();
|
|
myMouseClickCounter = 0;
|
|
myMouseStopDragOnUnclick = false;
|
|
myUI.Dragging.ToStop = true;
|
|
toUpdateView = true;
|
|
}
|
|
myMouseSingleButton = -1;
|
|
}
|
|
else if (theButtons == Aspect_VKeyMouse_NONE)
|
|
{
|
|
myMouseSingleButton = -1;
|
|
if (myMouseStopDragOnUnclick)
|
|
{
|
|
myMouseStopDragOnUnclick = false;
|
|
myUI.Dragging.ToStop = true;
|
|
toUpdateView = true;
|
|
}
|
|
}
|
|
else if (myMouseSingleButton == -1)
|
|
{
|
|
if ((theButtons & Aspect_VKeyMouse_LeftButton) == Aspect_VKeyMouse_LeftButton)
|
|
{
|
|
myMouseSingleButton = Aspect_VKeyMouse_LeftButton;
|
|
}
|
|
else if ((theButtons & Aspect_VKeyMouse_RightButton) == Aspect_VKeyMouse_RightButton)
|
|
{
|
|
myMouseSingleButton = Aspect_VKeyMouse_RightButton;
|
|
}
|
|
else if ((theButtons & Aspect_VKeyMouse_MiddleButton) == Aspect_VKeyMouse_MiddleButton)
|
|
{
|
|
myMouseSingleButton = Aspect_VKeyMouse_MiddleButton;
|
|
}
|
|
else
|
|
{
|
|
myMouseSingleButton = 0;
|
|
}
|
|
if (myMouseSingleButton != 0)
|
|
{
|
|
if (myMouseClickCounter == 1)
|
|
{
|
|
const Graphic3d_Vec2i aDelta = thePoint - myMousePressPoint;
|
|
if (double(aDelta.cwiseAbs().maxComp()) >= aTolClick)
|
|
{
|
|
myMouseClickTimer.Stop();
|
|
myMouseClickCounter = 0;
|
|
}
|
|
}
|
|
myMousePressPoint = thePoint;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
myMouseSingleButton = 0;
|
|
|
|
myUI.Dragging.ToAbort = true;
|
|
toUpdateView = true;
|
|
}
|
|
|
|
const AIS_MouseGesture aPrevGesture = myMouseActiveGesture;
|
|
const Aspect_VKeyMouse aPrevButtons = myMousePressed;
|
|
const Aspect_VKeyFlags aPrevModifiers = myMouseModifiers;
|
|
myMouseModifiers = theModifiers;
|
|
myMousePressed = theButtons;
|
|
if (theIsEmulated || myNavigationMode != AIS_NavigationMode_FirstPersonWalk)
|
|
{
|
|
myMouseActiveIdleRotation = false;
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
if (theButtons != 0)
|
|
{
|
|
myMousePressPoint = thePoint;
|
|
myMouseProgressPoint = myMousePressPoint;
|
|
}
|
|
|
|
if (myMouseGestureMap.Find(theButtons | theModifiers, myMouseActiveGesture))
|
|
{
|
|
switch (myMouseActiveGesture)
|
|
{
|
|
case AIS_MouseGesture_RotateView:
|
|
case AIS_MouseGesture_RotateOrbit: {
|
|
if (myToAllowRotation)
|
|
{
|
|
myUpdateStartPointRot = true;
|
|
}
|
|
else
|
|
{
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Pan: {
|
|
if (myToAllowPanning)
|
|
{
|
|
myUpdateStartPointPan = true;
|
|
}
|
|
else
|
|
{
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Zoom:
|
|
case AIS_MouseGesture_ZoomWindow:
|
|
case AIS_MouseGesture_ZoomVertical: {
|
|
if (!myToAllowZooming)
|
|
{
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_SelectRectangle: {
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_SelectLasso: {
|
|
UpdatePolySelection(thePoint, true);
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Drag: {
|
|
if (myToAllowDragging)
|
|
{
|
|
myUI.Dragging.ToStart = true;
|
|
myUI.Dragging.PointStart = thePoint;
|
|
}
|
|
else
|
|
{
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_NONE: {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
AIS_MouseGesture aSecGesture = AIS_MouseGesture_NONE;
|
|
if (myMouseGestureMapDrag.Find(theButtons | theModifiers, aSecGesture))
|
|
{
|
|
if (aSecGesture == AIS_MouseGesture_Drag && myToAllowDragging)
|
|
{
|
|
myUI.Dragging.ToStart = true;
|
|
myUI.Dragging.PointStart = thePoint;
|
|
if (myMouseActiveGesture == AIS_MouseGesture_NONE)
|
|
{
|
|
myMouseActiveGesture = aSecGesture;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aPrevGesture != myMouseActiveGesture)
|
|
{
|
|
if (aPrevGesture == AIS_MouseGesture_SelectRectangle
|
|
|| aPrevGesture == AIS_MouseGesture_SelectLasso
|
|
|| aPrevGesture == AIS_MouseGesture_ZoomWindow)
|
|
{
|
|
myUI.Selection.ToApplyTool = true;
|
|
myUI.Selection.Scheme = AIS_SelectionScheme_Replace;
|
|
myMouseSelectionSchemes.Find(aPrevButtons | aPrevModifiers, myUI.Selection.Scheme);
|
|
}
|
|
|
|
myUI.IsNewGesture = true;
|
|
toUpdateView = true;
|
|
}
|
|
|
|
return toUpdateView;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::UpdateMousePosition(const Graphic3d_Vec2i& thePoint,
|
|
Aspect_VKeyMouse theButtons,
|
|
Aspect_VKeyFlags theModifiers,
|
|
bool theIsEmulated)
|
|
{
|
|
myMousePositionLast = thePoint;
|
|
if (myMouseSingleButton > 0)
|
|
{
|
|
const double aTolClick = (theIsEmulated ? myTouchToleranceScale : 1.0) * myMouseClickThreshold;
|
|
const Graphic3d_Vec2i aPressDelta = thePoint - myMousePressPoint;
|
|
if (double(aPressDelta.cwiseAbs().maxComp()) >= aTolClick)
|
|
{
|
|
myMouseClickTimer.Stop();
|
|
myMouseClickCounter = 0;
|
|
myMouseSingleButton = -1;
|
|
myMouseStopDragOnUnclick = true;
|
|
myUI.Dragging.ToConfirm = true;
|
|
}
|
|
}
|
|
|
|
bool toUpdateView = false;
|
|
Graphic3d_Vec2i aDelta = thePoint - myMouseProgressPoint;
|
|
if (!theIsEmulated && myNavigationMode == AIS_NavigationMode_FirstPersonWalk)
|
|
{
|
|
if (!myMouseActiveIdleRotation || myMouseActiveGesture != AIS_MouseGesture_RotateView)
|
|
{
|
|
myMouseActiveIdleRotation = true;
|
|
myMouseActiveGesture = AIS_MouseGesture_RotateView;
|
|
myMousePressPoint = thePoint;
|
|
myMouseProgressPoint = thePoint;
|
|
myUpdateStartPointRot = false;
|
|
myUI.ViewRotation.ToStart = true;
|
|
myUI.ViewRotation.PointStart.SetValues(thePoint.x(), thePoint.y());
|
|
myUI.ViewRotation.ToRotate = false;
|
|
aDelta.SetValues(0, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (myMouseActiveIdleRotation && myMouseActiveGesture == AIS_MouseGesture_RotateView)
|
|
{
|
|
myMouseActiveGesture = AIS_MouseGesture_NONE;
|
|
}
|
|
myMouseActiveIdleRotation = false;
|
|
}
|
|
|
|
if (myMouseModifiers != theModifiers
|
|
&& UpdateMouseButtons(thePoint, theButtons, theModifiers, theIsEmulated))
|
|
{
|
|
toUpdateView = true;
|
|
}
|
|
|
|
switch (myMouseActiveGesture)
|
|
{
|
|
case AIS_MouseGesture_SelectRectangle:
|
|
case AIS_MouseGesture_ZoomWindow: {
|
|
UpdateRubberBand(myMousePressPoint, thePoint);
|
|
if (myMouseActiveGesture == AIS_MouseGesture_ZoomWindow)
|
|
{
|
|
myUI.Selection.Tool = AIS_ViewSelectionTool_ZoomWindow;
|
|
}
|
|
toUpdateView = true;
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_SelectLasso: {
|
|
UpdatePolySelection(thePoint, true);
|
|
toUpdateView = true;
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_RotateOrbit:
|
|
case AIS_MouseGesture_RotateView: {
|
|
if (!myToAllowRotation)
|
|
{
|
|
break;
|
|
}
|
|
if (myUpdateStartPointRot)
|
|
{
|
|
if (myMouseActiveGesture == AIS_MouseGesture_RotateOrbit)
|
|
{
|
|
myUI.OrbitRotation.ToStart = true;
|
|
myUI.OrbitRotation.PointStart.SetValues(myMousePressPoint.x(), myMousePressPoint.y());
|
|
}
|
|
else
|
|
{
|
|
myUI.ViewRotation.ToStart = true;
|
|
myUI.ViewRotation.PointStart.SetValues(myMousePressPoint.x(), myMousePressPoint.y());
|
|
}
|
|
myUpdateStartPointRot = false;
|
|
}
|
|
|
|
const double aRotTol =
|
|
theIsEmulated ? double(myTouchToleranceScale) * myTouchRotationThresholdPx : 0.0;
|
|
const Graphic3d_Vec2d aDeltaF(aDelta);
|
|
if (Abs(aDeltaF.x()) + Abs(aDeltaF.y()) > aRotTol)
|
|
{
|
|
const double aRotAccel =
|
|
myNavigationMode == AIS_NavigationMode_FirstPersonWalk ? myMouseAccel : myOrbitAccel;
|
|
const Graphic3d_Vec2i aRotDelta = thePoint - myMousePressPoint;
|
|
if (myMouseActiveGesture == AIS_MouseGesture_RotateOrbit)
|
|
{
|
|
myUI.OrbitRotation.ToRotate = true;
|
|
myUI.OrbitRotation.PointTo = Graphic3d_Vec2d(myMousePressPoint.x(), myMousePressPoint.y())
|
|
+ Graphic3d_Vec2d(aRotDelta.x(), aRotDelta.y()) * aRotAccel;
|
|
}
|
|
else
|
|
{
|
|
myUI.ViewRotation.ToRotate = true;
|
|
myUI.ViewRotation.PointTo = Graphic3d_Vec2d(myMousePressPoint.x(), myMousePressPoint.y())
|
|
+ Graphic3d_Vec2d(aRotDelta.x(), aRotDelta.y()) * aRotAccel;
|
|
}
|
|
|
|
myUI.Dragging.ToMove = true;
|
|
myUI.Dragging.PointTo = thePoint;
|
|
|
|
myMouseProgressPoint = thePoint;
|
|
toUpdateView = true;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Zoom:
|
|
case AIS_MouseGesture_ZoomVertical: {
|
|
if (!myToAllowZooming)
|
|
{
|
|
break;
|
|
}
|
|
const double aZoomTol =
|
|
theIsEmulated ? double(myTouchToleranceScale) * myTouchZoomThresholdPx : 0.0;
|
|
const double aScrollDelta =
|
|
myMouseActiveGesture == AIS_MouseGesture_Zoom ? aDelta.x() : aDelta.y();
|
|
if (Abs(aScrollDelta) > aZoomTol)
|
|
{
|
|
if (UpdateZoom(Aspect_ScrollDelta(aScrollDelta)))
|
|
{
|
|
toUpdateView = true;
|
|
}
|
|
|
|
myUI.Dragging.ToMove = true;
|
|
myUI.Dragging.PointTo = thePoint;
|
|
|
|
myMouseProgressPoint = thePoint;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Pan: {
|
|
if (!myToAllowPanning)
|
|
{
|
|
break;
|
|
}
|
|
const double aPanTol =
|
|
theIsEmulated ? double(myTouchToleranceScale) * myTouchPanThresholdPx : 0.0;
|
|
const Graphic3d_Vec2d aDeltaF(aDelta);
|
|
if (Abs(aDeltaF.x()) + Abs(aDeltaF.y()) > aPanTol)
|
|
{
|
|
if (myUpdateStartPointPan)
|
|
{
|
|
myUI.Panning.ToStart = true;
|
|
myUI.Panning.PointStart.SetValues(myMousePressPoint.x(), myMousePressPoint.y());
|
|
myUpdateStartPointPan = false;
|
|
}
|
|
|
|
aDelta.y() = -aDelta.y();
|
|
if (myUI.Panning.ToPan)
|
|
{
|
|
myUI.Panning.Delta += aDelta;
|
|
}
|
|
else
|
|
{
|
|
myUI.Panning.ToPan = true;
|
|
myUI.Panning.Delta = aDelta;
|
|
}
|
|
|
|
myUI.Dragging.ToMove = true;
|
|
myUI.Dragging.PointTo = thePoint;
|
|
|
|
myMouseProgressPoint = thePoint;
|
|
|
|
toUpdateView = true;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_MouseGesture_Drag: {
|
|
if (!myToAllowDragging)
|
|
{
|
|
break;
|
|
}
|
|
|
|
const double aDragTol =
|
|
theIsEmulated ? double(myTouchToleranceScale) * myTouchDraggingThresholdPx : 0.0;
|
|
if (double(Abs(aDelta.x()) + Abs(aDelta.y())) > aDragTol)
|
|
{
|
|
const double aRotAccel =
|
|
myNavigationMode == AIS_NavigationMode_FirstPersonWalk ? myMouseAccel : myOrbitAccel;
|
|
const Graphic3d_Vec2i aRotDelta = thePoint - myMousePressPoint;
|
|
myUI.ViewRotation.ToRotate = true;
|
|
myUI.ViewRotation.PointTo = Graphic3d_Vec2d(myMousePressPoint.x(), myMousePressPoint.y())
|
|
+ Graphic3d_Vec2d(aRotDelta.x(), aRotDelta.y()) * aRotAccel;
|
|
myUI.Dragging.ToMove = true;
|
|
myUI.Dragging.PointTo = thePoint;
|
|
|
|
myMouseProgressPoint = thePoint;
|
|
toUpdateView = true;
|
|
}
|
|
break;
|
|
}
|
|
default: {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (theButtons == Aspect_VKeyMouse_NONE && myNavigationMode != AIS_NavigationMode_FirstPersonWalk
|
|
&& !theIsEmulated && !HasTouchPoints() && myToAllowHighlight)
|
|
{
|
|
myUI.MoveTo.ToHilight = true;
|
|
myUI.MoveTo.Point = thePoint;
|
|
toUpdateView = true;
|
|
}
|
|
return toUpdateView;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::AddTouchPoint(Standard_Size theId,
|
|
const Graphic3d_Vec2d& thePnt,
|
|
Standard_Boolean theClearBefore)
|
|
{
|
|
myUI.MoveTo.ToHilight = false;
|
|
Aspect_WindowInputListener::AddTouchPoint(theId, thePnt, theClearBefore);
|
|
|
|
myTouchClick.From = Graphic3d_Vec2d(-1.0);
|
|
if (myTouchPoints.Extent() == 1)
|
|
{
|
|
myTouchClick.From = thePnt;
|
|
myUpdateStartPointRot = true;
|
|
myStartRotCoord = thePnt;
|
|
if (myToAllowDragging)
|
|
{
|
|
myUI.Dragging.ToStart = true;
|
|
myUI.Dragging.PointStart.SetValues((int)thePnt.x(), (int)thePnt.y());
|
|
}
|
|
}
|
|
else if (myTouchPoints.Extent() == 2)
|
|
{
|
|
myUI.Dragging.ToAbort = true;
|
|
|
|
myUpdateStartPointPan = true;
|
|
myStartPanCoord = thePnt;
|
|
}
|
|
myUI.IsNewGesture = true;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::RemoveTouchPoint(Standard_Size theId, Standard_Boolean theClearSelectPnts)
|
|
{
|
|
if (!Aspect_WindowInputListener::RemoveTouchPoint(theId, theClearSelectPnts))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (myTouchPoints.Extent() == 1)
|
|
{
|
|
// avoid incorrect transition from pinch to one finger
|
|
Aspect_Touch& aFirstTouch = myTouchPoints.ChangeFromIndex(1);
|
|
aFirstTouch.To = aFirstTouch.From;
|
|
|
|
myStartRotCoord = aFirstTouch.To;
|
|
myUpdateStartPointRot = true;
|
|
}
|
|
else if (myTouchPoints.Extent() == 2)
|
|
{
|
|
myStartPanCoord = myTouchPoints.FindFromIndex(1).To;
|
|
myUpdateStartPointPan = true;
|
|
}
|
|
else if (myTouchPoints.IsEmpty())
|
|
{
|
|
if (theClearSelectPnts)
|
|
{
|
|
myUI.Selection.ToApplyTool = true;
|
|
}
|
|
|
|
myUI.Dragging.ToStop = true;
|
|
|
|
if (theId == (Standard_Size)-1)
|
|
{
|
|
// abort clicking
|
|
myTouchClick.From = Graphic3d_Vec2d(-1);
|
|
}
|
|
else if (myTouchClick.From.minComp() >= 0.0)
|
|
{
|
|
bool isDoubleClick = false;
|
|
if (myTouchDoubleTapTimer.IsStarted()
|
|
&& myTouchDoubleTapTimer.ElapsedTime() <= myMouseDoubleClickInt)
|
|
{
|
|
isDoubleClick = true;
|
|
}
|
|
else
|
|
{
|
|
myTouchDoubleTapTimer.Stop();
|
|
myTouchDoubleTapTimer.Reset();
|
|
myTouchDoubleTapTimer.Start();
|
|
}
|
|
|
|
// emulate mouse click
|
|
UpdateMouseClick(Graphic3d_Vec2i(myTouchClick.From),
|
|
Aspect_VKeyMouse_LeftButton,
|
|
Aspect_VKeyFlags_NONE,
|
|
isDoubleClick);
|
|
}
|
|
}
|
|
myUI.IsNewGesture = true;
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::UpdateTouchPoint(Standard_Size theId, const Graphic3d_Vec2d& thePnt)
|
|
{
|
|
Aspect_WindowInputListener::UpdateTouchPoint(theId, thePnt);
|
|
|
|
const double aTouchTol = double(myTouchToleranceScale) * double(myTouchClickThresholdPx);
|
|
if (myTouchPoints.Extent() == 1 && (myTouchClick.From - thePnt).cwiseAbs().maxComp() > aTouchTol)
|
|
{
|
|
myTouchClick.From.SetValues(-1.0, -1.0);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::Update3dMouse(const WNT_HIDSpaceMouse& theEvent)
|
|
{
|
|
bool toUpdate = false;
|
|
toUpdate = update3dMouseTranslation(theEvent) || toUpdate;
|
|
toUpdate = (myToAllowRotation && update3dMouseRotation(theEvent)) || toUpdate;
|
|
toUpdate = update3dMouseKeys(theEvent) || toUpdate;
|
|
return toUpdate;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::SetNavigationMode(AIS_NavigationMode theMode)
|
|
{
|
|
myNavigationMode = theMode;
|
|
|
|
// abort rotation
|
|
myUI.OrbitRotation.ToStart = false;
|
|
myUI.OrbitRotation.ToRotate = false;
|
|
myUI.ViewRotation.ToStart = false;
|
|
myUI.ViewRotation.ToRotate = false;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::KeyDown(Aspect_VKey theKey, double theTime, double thePressure)
|
|
{
|
|
Aspect_WindowInputListener::KeyDown(theKey, theTime, thePressure);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::KeyUp(Aspect_VKey theKey, double theTime)
|
|
{
|
|
Aspect_WindowInputListener::KeyUp(theKey, theTime);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::KeyFromAxis(Aspect_VKey theNegative,
|
|
Aspect_VKey thePositive,
|
|
double theTime,
|
|
double thePressure)
|
|
{
|
|
Aspect_WindowInputListener::KeyFromAxis(theNegative, thePositive, theTime, thePressure);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
AIS_WalkDelta AIS_ViewController::FetchNavigationKeys(Standard_Real theCrouchRatio,
|
|
Standard_Real theRunRatio)
|
|
{
|
|
AIS_WalkDelta aWalk;
|
|
|
|
// navigation keys
|
|
double aPrevEventTime = 0.0, aNewEventTime = 0.0;
|
|
updateEventsTime(aPrevEventTime, aNewEventTime);
|
|
|
|
double aDuration = 0.0, aPressure = 1.0;
|
|
if (Abs(myThrustSpeed) > gp::Resolution())
|
|
{
|
|
if (myHasThrust)
|
|
{
|
|
aWalk[AIS_WalkTranslation_Forward].Value = myThrustSpeed * (aNewEventTime - aPrevEventTime);
|
|
}
|
|
myHasThrust = true;
|
|
myToAskNextFrame = true;
|
|
}
|
|
else
|
|
{
|
|
myHasThrust = false;
|
|
}
|
|
|
|
aWalk.SetRunning(theRunRatio > 1.0 && myKeys.IsKeyDown(Aspect_VKey_Shift));
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavJump, aNewEventTime, aDuration))
|
|
{
|
|
myKeys.KeyUp(Aspect_VKey_NavJump, aNewEventTime);
|
|
aWalk.SetDefined(true);
|
|
aWalk.SetJumping(true);
|
|
}
|
|
if (!aWalk.IsJumping() && theCrouchRatio < 1.0
|
|
&& myKeys.HoldDuration(Aspect_VKey_NavCrouch, aNewEventTime, aDuration))
|
|
{
|
|
aWalk.SetDefined(true);
|
|
aWalk.SetRunning(false);
|
|
aWalk.SetCrouching(true);
|
|
}
|
|
|
|
const double aMaxDuration = aNewEventTime - aPrevEventTime;
|
|
const double aRunRatio = aWalk.IsRunning() ? theRunRatio
|
|
: aWalk.IsCrouching() ? theCrouchRatio
|
|
: 1.0;
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavForward, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aProgress *= aRunRatio;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Forward].Value += aProgress;
|
|
aWalk[AIS_WalkTranslation_Forward].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Forward].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavBackward, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aProgress *= aRunRatio;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Forward].Value += -aProgress;
|
|
aWalk[AIS_WalkTranslation_Forward].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Forward].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavSlideLeft, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aProgress *= aRunRatio;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Side].Value = -aProgress;
|
|
aWalk[AIS_WalkTranslation_Side].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Side].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavSlideRight, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aProgress *= aRunRatio;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Side].Value = aProgress;
|
|
aWalk[AIS_WalkTranslation_Side].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Side].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavLookLeft, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Yaw].Value = aProgress;
|
|
aWalk[AIS_WalkRotation_Yaw].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Yaw].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavLookRight, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Yaw].Value = -aProgress;
|
|
aWalk[AIS_WalkRotation_Yaw].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Yaw].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavLookUp, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Pitch].Value = !myToInvertPitch ? -aProgress : aProgress;
|
|
aWalk[AIS_WalkRotation_Pitch].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Pitch].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavLookDown, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Pitch].Value = !myToInvertPitch ? aProgress : -aProgress;
|
|
aWalk[AIS_WalkRotation_Pitch].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Pitch].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavRollCCW, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Roll].Value = -aProgress;
|
|
aWalk[AIS_WalkRotation_Roll].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Roll].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavRollCW, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration)) * aPressure;
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkRotation_Roll].Value = aProgress;
|
|
aWalk[AIS_WalkRotation_Roll].Pressure = aPressure;
|
|
aWalk[AIS_WalkRotation_Roll].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavSlideUp, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Up].Value = aProgress;
|
|
aWalk[AIS_WalkTranslation_Up].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Up].Duration = aDuration;
|
|
}
|
|
if (myKeys.HoldDuration(Aspect_VKey_NavSlideDown, aNewEventTime, aDuration, aPressure))
|
|
{
|
|
double aProgress = Abs(Min(aMaxDuration, aDuration));
|
|
aWalk.SetDefined(true);
|
|
aWalk[AIS_WalkTranslation_Up].Value = -aProgress;
|
|
aWalk[AIS_WalkTranslation_Up].Pressure = aPressure;
|
|
aWalk[AIS_WalkTranslation_Up].Duration = aDuration;
|
|
}
|
|
return aWalk;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::AbortViewAnimation()
|
|
{
|
|
if (!myViewAnimation.IsNull() && !myViewAnimation->IsStopped())
|
|
{
|
|
myViewAnimation->Stop();
|
|
myViewAnimation->SetView(Handle(V3d_View)());
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handlePanning(const Handle(V3d_View)& theView)
|
|
{
|
|
if (!myGL.Panning.ToPan || !myToAllowPanning)
|
|
{
|
|
return;
|
|
}
|
|
|
|
AbortViewAnimation();
|
|
|
|
const Handle(Graphic3d_Camera)& aCam = theView->Camera();
|
|
if (aCam->IsOrthographic() || !hasPanningAnchorPoint())
|
|
{
|
|
theView->Pan(myGL.Panning.Delta.x(), myGL.Panning.Delta.y());
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRPosedToBaseCamera();
|
|
return;
|
|
}
|
|
|
|
Graphic3d_Vec2i aWinSize;
|
|
theView->Window()->Size(aWinSize.x(), aWinSize.y());
|
|
|
|
const gp_Dir& aDir = aCam->Direction();
|
|
const gp_Ax3 aCameraCS(aCam->Center(), aDir.Reversed(), aDir ^ aCam->Up());
|
|
const gp_XYZ anEyeToPnt = myPanPnt3d.XYZ() - aCam->Eye().XYZ();
|
|
// clang-format off
|
|
const gp_Pnt aViewDims = aCam->ViewDimensions (anEyeToPnt.Dot (aCam->Direction().XYZ())); // view dimensions at 3D point
|
|
// clang-format on
|
|
const Graphic3d_Vec2d aDxy(-aViewDims.X() * myGL.Panning.Delta.x() / double(aWinSize.x()),
|
|
-aViewDims.X() * myGL.Panning.Delta.y() / double(aWinSize.x()));
|
|
|
|
// theView->Translate (aCam, aDxy.x(), aDxy.y());
|
|
gp_Trsf aPanTrsf;
|
|
const gp_Vec aCameraPan =
|
|
gp_Vec(aCameraCS.XDirection()) * aDxy.x() + gp_Vec(aCameraCS.YDirection()) * aDxy.y();
|
|
aPanTrsf.SetTranslation(aCameraPan);
|
|
aCam->Transform(aPanTrsf);
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRPosedToBaseCamera();
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleZRotate(const Handle(V3d_View)& theView)
|
|
{
|
|
if (!myGL.ZRotate.ToRotate || !myToAllowRotation)
|
|
{
|
|
return;
|
|
}
|
|
|
|
AbortViewAnimation();
|
|
|
|
Graphic3d_Vec2i aViewPort;
|
|
theView->Window()->Size(aViewPort.x(), aViewPort.y());
|
|
Graphic3d_Vec2d aRotPnt(0.99 * aViewPort.x(), 0.5 * aViewPort.y());
|
|
theView->StartRotation(int(aRotPnt.x()), int(aRotPnt.y()), 0.4);
|
|
aRotPnt.y() += myGL.ZRotate.Angle * aViewPort.y();
|
|
theView->Rotation(int(aRotPnt.x()), int(aRotPnt.y()));
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRPosedToBaseCamera();
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleZoom(const Handle(V3d_View)& theView,
|
|
const Aspect_ScrollDelta& theParams,
|
|
const gp_Pnt* thePnt)
|
|
{
|
|
if (!myToAllowZooming)
|
|
{
|
|
return;
|
|
}
|
|
|
|
AbortViewAnimation();
|
|
|
|
const Handle(Graphic3d_Camera)& aCam = theView->Camera();
|
|
if (thePnt != NULL)
|
|
{
|
|
const double aViewDist = Max(myMinCamDistance, (thePnt->XYZ() - aCam->Eye().XYZ()).Modulus());
|
|
aCam->SetCenter(aCam->Eye().XYZ() + aCam->Direction().XYZ() * aViewDist);
|
|
}
|
|
|
|
if (!theParams.HasPoint())
|
|
{
|
|
Standard_Real aCoeff = Abs(theParams.Delta) / 100.0 + 1.0;
|
|
aCoeff = theParams.Delta > 0.0 ? aCoeff : 1.0 / aCoeff;
|
|
theView->SetZoom(aCoeff, true);
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRPosedToBaseCamera();
|
|
return;
|
|
}
|
|
|
|
// integer delta is too rough for small smooth increments
|
|
// theView->StartZoomAtPoint (theParams.Point.x(), theParams.Point.y());
|
|
// theView->ZoomAtPoint (0, 0, (int )theParams.Delta, (int )theParams.Delta);
|
|
|
|
double aDZoom = Abs(theParams.Delta) / 100.0 + 1.0;
|
|
aDZoom = (theParams.Delta > 0.0) ? aDZoom : 1.0 / aDZoom;
|
|
if (aDZoom <= 0.0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Graphic3d_Vec2d aViewDims(aCam->ViewDimensions().X(), aCam->ViewDimensions().Y());
|
|
|
|
// ensure that zoom will not be too small or too big
|
|
double aCoef = aDZoom;
|
|
if (aViewDims.x() < aCoef * Precision::Confusion())
|
|
{
|
|
aCoef = aViewDims.x() / Precision::Confusion();
|
|
}
|
|
else if (aViewDims.x() > aCoef * 1e12)
|
|
{
|
|
aCoef = aViewDims.x() / 1e12;
|
|
}
|
|
if (aViewDims.y() < aCoef * Precision::Confusion())
|
|
{
|
|
aCoef = aViewDims.y() / Precision::Confusion();
|
|
}
|
|
else if (aViewDims.y() > aCoef * 1e12)
|
|
{
|
|
aCoef = aViewDims.y() / 1e12;
|
|
}
|
|
|
|
Graphic3d_Vec2d aZoomAtPointXYv(0.0, 0.0);
|
|
theView->Convert(theParams.Point.x(),
|
|
theParams.Point.y(),
|
|
aZoomAtPointXYv.x(),
|
|
aZoomAtPointXYv.y());
|
|
Graphic3d_Vec2d aDxy = aZoomAtPointXYv / aCoef;
|
|
aCam->SetScale(aCam->Scale() / aCoef);
|
|
|
|
const gp_Dir& aDir = aCam->Direction();
|
|
const gp_Ax3 aCameraCS(aCam->Center(), aDir.Reversed(), aDir ^ aCam->Up());
|
|
|
|
// pan back to the point
|
|
aDxy = aZoomAtPointXYv - aDxy;
|
|
if (thePnt != NULL)
|
|
{
|
|
// zoom at 3D point with perspective projection
|
|
const gp_XYZ anEyeToPnt = thePnt->XYZ() - aCam->Eye().XYZ();
|
|
aDxy.SetValues(anEyeToPnt.Dot(aCameraCS.XDirection().XYZ()),
|
|
anEyeToPnt.Dot(aCameraCS.YDirection().XYZ()));
|
|
|
|
// view dimensions at 3D point
|
|
const gp_Pnt aViewDims1 = aCam->ViewDimensions(anEyeToPnt.Dot(aCam->Direction().XYZ()));
|
|
|
|
Graphic3d_Vec2i aWinSize;
|
|
theView->Window()->Size(aWinSize.x(), aWinSize.y());
|
|
const Graphic3d_Vec2d aWinSizeF(aWinSize);
|
|
const Graphic3d_Vec2d aPanFromCenterPx(double(theParams.Point.x()) - 0.5 * aWinSizeF.x(),
|
|
aWinSizeF.y() - double(theParams.Point.y()) - 1.0
|
|
- 0.5 * aWinSizeF.y());
|
|
aDxy.x() += -aViewDims1.X() * aPanFromCenterPx.x() / aWinSizeF.x();
|
|
aDxy.y() += -aViewDims1.Y() * aPanFromCenterPx.y() / aWinSizeF.y();
|
|
}
|
|
|
|
// theView->Translate (aCam, aDxy.x(), aDxy.y());
|
|
gp_Trsf aPanTrsf;
|
|
const gp_Vec aCameraPan =
|
|
gp_Vec(aCameraCS.XDirection()) * aDxy.x() + gp_Vec(aCameraCS.YDirection()) * aDxy.y();
|
|
aPanTrsf.SetTranslation(aCameraPan);
|
|
aCam->Transform(aPanTrsf);
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRPosedToBaseCamera();
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleZFocusScroll(const Handle(V3d_View)& theView,
|
|
const Aspect_ScrollDelta& theParams)
|
|
{
|
|
if (!myToAllowZFocus || !theView->Camera()->IsStereo())
|
|
{
|
|
return;
|
|
}
|
|
|
|
Standard_Real aFocus = theView->Camera()->ZFocus() + (theParams.Delta > 0.0 ? 0.05 : -0.05);
|
|
if (aFocus > 0.2 && aFocus < 2.0)
|
|
{
|
|
theView->Camera()->SetZFocus(theView->Camera()->ZFocusType(), aFocus);
|
|
theView->Invalidate();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleOrbitRotation(const Handle(V3d_View)& theView,
|
|
const gp_Pnt& thePnt,
|
|
bool theToLockZUp)
|
|
{
|
|
if (!myToAllowRotation)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Handle(Graphic3d_Camera)& aCam =
|
|
theView->View()->IsActiveXR() ? theView->View()->BaseXRCamera() : theView->Camera();
|
|
if (myGL.OrbitRotation.ToStart)
|
|
{
|
|
// default alternatives
|
|
// if (myRotationMode == AIS_RotationMode_BndBoxActive) theView->StartRotation
|
|
// (myGL.RotateAtPoint.x(), myGL.RotateAtPoint.y()); theView->Rotate (0.0, 0.0, 0.0, thePnt.X(),
|
|
// thePnt.Y(), thePnt.Z(), true);
|
|
|
|
myRotatePnt3d = thePnt;
|
|
myCamStartOpUp = aCam->Up();
|
|
myCamStartOpDir = aCam->Direction();
|
|
myCamStartOpEye = aCam->Eye();
|
|
myCamStartOpCenter = aCam->Center();
|
|
|
|
gp_Trsf aTrsf;
|
|
aTrsf.SetTransformation(gp_Ax3(myRotatePnt3d, aCam->OrthogonalizedUp(), aCam->Direction()),
|
|
gp_Ax3(myRotatePnt3d, gp::DZ(), gp::DX()));
|
|
const gp_Quaternion aRot = aTrsf.GetRotation();
|
|
aRot.GetEulerAngles(gp_YawPitchRoll,
|
|
myRotateStartYawPitchRoll[0],
|
|
myRotateStartYawPitchRoll[1],
|
|
myRotateStartYawPitchRoll[2]);
|
|
|
|
aTrsf.Invert();
|
|
myCamStartOpToEye = gp_Vec(myRotatePnt3d, aCam->Eye()).Transformed(aTrsf);
|
|
myCamStartOpToCenter = gp_Vec(myRotatePnt3d, aCam->Center()).Transformed(aTrsf);
|
|
|
|
theView->Invalidate();
|
|
}
|
|
|
|
if (!myGL.OrbitRotation.ToRotate)
|
|
{
|
|
return;
|
|
}
|
|
|
|
AbortViewAnimation();
|
|
if (theToLockZUp)
|
|
{
|
|
// amend camera to exclude roll angle (put camera Up vector to plane containing global Z and
|
|
// view direction)
|
|
Graphic3d_Vec2i aWinXY;
|
|
theView->Window()->Size(aWinXY.x(), aWinXY.y());
|
|
double aYawAngleDelta =
|
|
((myGL.OrbitRotation.PointStart.x() - myGL.OrbitRotation.PointTo.x()) / double(aWinXY.x()))
|
|
* (M_PI * 0.5);
|
|
double aPitchAngleDelta =
|
|
-((myGL.OrbitRotation.PointStart.y() - myGL.OrbitRotation.PointTo.y()) / double(aWinXY.y()))
|
|
* (M_PI * 0.5);
|
|
double aPitchAngleNew = 0.0, aRoll = 0.0;
|
|
const double aYawAngleNew = myRotateStartYawPitchRoll[0] + aYawAngleDelta;
|
|
if (!theView->View()->IsActiveXR())
|
|
{
|
|
aPitchAngleNew =
|
|
Max(Min(myRotateStartYawPitchRoll[1] + aPitchAngleDelta, M_PI * 0.5 - M_PI / 180.0),
|
|
-M_PI * 0.5 + M_PI / 180.0);
|
|
aRoll = 0.0;
|
|
}
|
|
|
|
gp_Quaternion aRot;
|
|
aRot.SetEulerAngles(gp_YawPitchRoll, aYawAngleNew, aPitchAngleNew, aRoll);
|
|
gp_Trsf aTrsfRot;
|
|
aTrsfRot.SetRotation(aRot);
|
|
|
|
const gp_Dir aNewUp = gp::DZ().Transformed(aTrsfRot);
|
|
aCam->SetUp(aNewUp);
|
|
aCam->SetEyeAndCenter(myRotatePnt3d.XYZ() + myCamStartOpToEye.Transformed(aTrsfRot).XYZ(),
|
|
myRotatePnt3d.XYZ() + myCamStartOpToCenter.Transformed(aTrsfRot).XYZ());
|
|
|
|
aCam->OrthogonalizeUp();
|
|
}
|
|
else
|
|
{
|
|
// default alternatives
|
|
// if (myRotationMode == AIS_RotationMode_BndBoxActive) theView->Rotation
|
|
// (myGL.RotateToPoint.x(), myGL.RotateToPoint.y()); theView->Rotate (aDX, aDY, aDZ,
|
|
// myRotatePnt3d.X(), myRotatePnt3d.Y(), myRotatePnt3d.Z(), false);
|
|
|
|
// restore previous camera state
|
|
aCam->SetEyeAndCenter(myCamStartOpEye, myCamStartOpCenter);
|
|
aCam->SetUp(myCamStartOpUp);
|
|
aCam->SetDirectionFromEye(myCamStartOpDir);
|
|
|
|
Graphic3d_Vec2d aWinXY;
|
|
theView->Size(aWinXY.x(), aWinXY.y());
|
|
const Standard_Real rx = (Standard_Real)theView->Convert(aWinXY.x());
|
|
const Standard_Real ry = (Standard_Real)theView->Convert(aWinXY.y());
|
|
|
|
const double THE_2PI = M_PI * 2.0;
|
|
double aDX = (myGL.OrbitRotation.PointTo.x() - myGL.OrbitRotation.PointStart.x()) * M_PI / rx;
|
|
double aDY = (myGL.OrbitRotation.PointStart.y() - myGL.OrbitRotation.PointTo.y()) * M_PI / ry;
|
|
|
|
if (aDX > 0.0)
|
|
{
|
|
while (aDX > THE_2PI)
|
|
{
|
|
aDX -= THE_2PI;
|
|
}
|
|
}
|
|
else if (aDX < 0.0)
|
|
{
|
|
while (aDX < -THE_2PI)
|
|
{
|
|
aDX += THE_2PI;
|
|
}
|
|
}
|
|
if (aDY > 0.0)
|
|
{
|
|
while (aDY > THE_2PI)
|
|
{
|
|
aDY -= THE_2PI;
|
|
}
|
|
}
|
|
else if (aDY < 0.0)
|
|
{
|
|
while (aDY < -THE_2PI)
|
|
{
|
|
aDY += THE_2PI;
|
|
}
|
|
}
|
|
|
|
// rotate camera around 3 initial axes
|
|
gp_Dir aCamDir(aCam->Direction().Reversed());
|
|
gp_Dir aCamUp(aCam->Up());
|
|
gp_Dir aCamSide(aCamUp.Crossed(aCamDir));
|
|
|
|
gp_Trsf aRot[2], aTrsf;
|
|
aRot[0].SetRotation(gp_Ax1(myRotatePnt3d, aCamUp), -aDX);
|
|
aRot[1].SetRotation(gp_Ax1(myRotatePnt3d, aCamSide), aDY);
|
|
aTrsf.Multiply(aRot[0]);
|
|
aTrsf.Multiply(aRot[1]);
|
|
|
|
aCam->Transform(aTrsf);
|
|
}
|
|
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRBaseToPosedCamera();
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleViewRotation(const Handle(V3d_View)& theView,
|
|
double theYawExtra,
|
|
double thePitchExtra,
|
|
double theRoll,
|
|
bool theToRestartOnIncrement)
|
|
{
|
|
if (!myToAllowRotation)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Handle(Graphic3d_Camera)& aCam = theView->Camera();
|
|
const bool toRotateAnyway = Abs(theYawExtra) > gp::Resolution()
|
|
|| Abs(thePitchExtra) > gp::Resolution()
|
|
|| Abs(theRoll - myRotateStartYawPitchRoll[2]) > gp::Resolution();
|
|
if (toRotateAnyway && theToRestartOnIncrement)
|
|
{
|
|
myGL.ViewRotation.ToStart = true;
|
|
myGL.ViewRotation.PointTo = myGL.ViewRotation.PointStart;
|
|
}
|
|
if (myGL.ViewRotation.ToStart)
|
|
{
|
|
gp_Trsf aTrsf;
|
|
aTrsf.SetTransformation(gp_Ax3(gp::Origin(), aCam->OrthogonalizedUp(), aCam->Direction()),
|
|
gp_Ax3(gp::Origin(), gp::DZ(), gp::DX()));
|
|
const gp_Quaternion aRot = aTrsf.GetRotation();
|
|
double aRollDummy = 0.0;
|
|
aRot.GetEulerAngles(gp_YawPitchRoll,
|
|
myRotateStartYawPitchRoll[0],
|
|
myRotateStartYawPitchRoll[1],
|
|
aRollDummy);
|
|
}
|
|
if (toRotateAnyway)
|
|
{
|
|
myRotateStartYawPitchRoll[0] += theYawExtra;
|
|
myRotateStartYawPitchRoll[1] += thePitchExtra;
|
|
myRotateStartYawPitchRoll[2] = theRoll;
|
|
myGL.ViewRotation.ToRotate = true;
|
|
}
|
|
|
|
if (!myGL.ViewRotation.ToRotate)
|
|
{
|
|
return;
|
|
}
|
|
|
|
AbortViewAnimation();
|
|
|
|
Graphic3d_Vec2i aWinXY;
|
|
theView->Window()->Size(aWinXY.x(), aWinXY.y());
|
|
double aYawAngleDelta =
|
|
((myGL.ViewRotation.PointStart.x() - myGL.ViewRotation.PointTo.x()) / double(aWinXY.x()))
|
|
* (M_PI * 0.5);
|
|
double aPitchAngleDelta =
|
|
-((myGL.ViewRotation.PointStart.y() - myGL.ViewRotation.PointTo.y()) / double(aWinXY.y()))
|
|
* (M_PI * 0.5);
|
|
const double aPitchAngleNew =
|
|
Max(Min(myRotateStartYawPitchRoll[1] + aPitchAngleDelta, M_PI * 0.5 - M_PI / 180.0),
|
|
-M_PI * 0.5 + M_PI / 180.0);
|
|
const double aYawAngleNew = myRotateStartYawPitchRoll[0] + aYawAngleDelta;
|
|
gp_Quaternion aRot;
|
|
aRot.SetEulerAngles(gp_YawPitchRoll, aYawAngleNew, aPitchAngleNew, theRoll);
|
|
gp_Trsf aTrsfRot;
|
|
aTrsfRot.SetRotation(aRot);
|
|
|
|
const gp_Dir aNewUp = gp::DZ().Transformed(aTrsfRot);
|
|
const gp_Dir aNewDir = gp::DX().Transformed(aTrsfRot);
|
|
aCam->SetUp(aNewUp);
|
|
aCam->SetDirectionFromEye(aNewDir);
|
|
aCam->OrthogonalizeUp();
|
|
theView->Invalidate();
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::PickPoint(gp_Pnt& thePnt,
|
|
const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const Graphic3d_Vec2i& theCursor,
|
|
bool theToStickToPickRay)
|
|
{
|
|
ResetPreviousMoveTo();
|
|
|
|
const Handle(StdSelect_ViewerSelector3d)& aSelector = theCtx->MainSelector();
|
|
aSelector->Pick(theCursor.x(), theCursor.y(), theView);
|
|
if (aSelector->NbPicked() < 1)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const SelectMgr_SortCriterion& aPicked = aSelector->PickedData(1);
|
|
if (theToStickToPickRay && !Precision::IsInfinite(aPicked.Depth))
|
|
{
|
|
thePnt = aSelector->GetManager().DetectedPoint(aPicked.Depth);
|
|
}
|
|
else
|
|
{
|
|
thePnt = aSelector->PickedPoint(1);
|
|
}
|
|
return !Precision::IsInfinite(thePnt.X()) && !Precision::IsInfinite(thePnt.Y())
|
|
&& !Precision::IsInfinite(thePnt.Z());
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
bool AIS_ViewController::PickAxis(gp_Pnt& theTopPnt,
|
|
const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const gp_Ax1& theAxis)
|
|
{
|
|
ResetPreviousMoveTo();
|
|
|
|
const Handle(StdSelect_ViewerSelector3d)& aSelector = theCtx->MainSelector();
|
|
aSelector->Pick(theAxis, theView);
|
|
if (aSelector->NbPicked() < 1)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const SelectMgr_SortCriterion& aPickedData = aSelector->PickedData(1);
|
|
theTopPnt = aPickedData.Point;
|
|
return !Precision::IsInfinite(theTopPnt.X()) && !Precision::IsInfinite(theTopPnt.Y())
|
|
&& !Precision::IsInfinite(theTopPnt.Z());
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
gp_Pnt AIS_ViewController::GravityPoint(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
switch (myRotationMode)
|
|
{
|
|
case AIS_RotationMode_PickLast:
|
|
case AIS_RotationMode_PickCenter: {
|
|
Graphic3d_Vec2i aCursor((int)myGL.OrbitRotation.PointStart.x(),
|
|
(int)myGL.OrbitRotation.PointStart.y());
|
|
if (myRotationMode == AIS_RotationMode_PickCenter)
|
|
{
|
|
Graphic3d_Vec2i aViewPort;
|
|
theView->Window()->Size(aViewPort.x(), aViewPort.y());
|
|
aCursor = aViewPort / 2;
|
|
}
|
|
|
|
gp_Pnt aPnt;
|
|
if (PickPoint(aPnt, theCtx, theView, aCursor, myToStickToRayOnRotation))
|
|
{
|
|
return aPnt;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_RotationMode_CameraAt: {
|
|
const Handle(Graphic3d_Camera)& aCam = theView->Camera();
|
|
return aCam->Center();
|
|
}
|
|
case AIS_RotationMode_BndBoxScene: {
|
|
Bnd_Box aBndBox = theView->View()->MinMaxValues(false);
|
|
if (!aBndBox.IsVoid())
|
|
{
|
|
return (aBndBox.CornerMin().XYZ() + aBndBox.CornerMax().XYZ()) * 0.5;
|
|
}
|
|
break;
|
|
}
|
|
case AIS_RotationMode_BndBoxActive:
|
|
break;
|
|
}
|
|
|
|
return theCtx->GravityPoint(theView);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::FitAllAuto(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
const Bnd_Box aBoxSel = theCtx->BoundingBoxOfSelection(theView);
|
|
const double aFitMargin = 0.01;
|
|
if (aBoxSel.IsVoid())
|
|
{
|
|
theView->FitAll(aFitMargin, false);
|
|
return;
|
|
}
|
|
|
|
// fit all algorithm is not 100% stable - so compute some precision to compare equal camera values
|
|
const double aFitTol =
|
|
(aBoxSel.CornerMax().XYZ() - aBoxSel.CornerMin().XYZ()).Modulus() * 0.000001;
|
|
const Bnd_Box aBoxAll = theView->View()->MinMaxValues();
|
|
|
|
const Handle(Graphic3d_Camera)& aCam = theView->Camera();
|
|
Handle(Graphic3d_Camera) aCameraSel = new Graphic3d_Camera(aCam);
|
|
Handle(Graphic3d_Camera) aCameraAll = new Graphic3d_Camera(aCam);
|
|
theView->FitMinMax(aCameraSel, aBoxSel, aFitMargin);
|
|
theView->FitMinMax(aCameraAll, aBoxAll, aFitMargin);
|
|
if (aCameraSel->Center().IsEqual(aCam->Center(), aFitTol)
|
|
&& Abs(aCameraSel->Scale() - aCam->Scale()) < aFitTol
|
|
&& Abs(aCameraSel->Distance() - aCam->Distance()) < aFitTol)
|
|
{
|
|
// fit all entire view on second FitALL request
|
|
aCam->Copy(aCameraAll);
|
|
}
|
|
else
|
|
{
|
|
aCam->Copy(aCameraSel);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleViewOrientationKeys(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (myNavigationMode == AIS_NavigationMode_FirstPersonWalk)
|
|
{
|
|
return;
|
|
}
|
|
|
|
Handle(Graphic3d_Camera) aCameraBack;
|
|
|
|
struct ViewKeyAction
|
|
{
|
|
Aspect_VKey Key;
|
|
V3d_TypeOfOrientation Orientation;
|
|
};
|
|
|
|
static const ViewKeyAction THE_VIEW_KEYS[] = {
|
|
{Aspect_VKey_ViewTop, V3d_TypeOfOrientation_Zup_Top},
|
|
{Aspect_VKey_ViewBottom, V3d_TypeOfOrientation_Zup_Bottom},
|
|
{Aspect_VKey_ViewLeft, V3d_TypeOfOrientation_Zup_Left},
|
|
{Aspect_VKey_ViewRight, V3d_TypeOfOrientation_Zup_Right},
|
|
{Aspect_VKey_ViewFront, V3d_TypeOfOrientation_Zup_Front},
|
|
{Aspect_VKey_ViewBack, V3d_TypeOfOrientation_Zup_Back},
|
|
{Aspect_VKey_ViewAxoLeftProj, V3d_TypeOfOrientation_Zup_AxoLeft},
|
|
{Aspect_VKey_ViewAxoRightProj, V3d_TypeOfOrientation_Zup_AxoRight},
|
|
{Aspect_VKey_ViewRoll90CW, (V3d_TypeOfOrientation)-1},
|
|
{Aspect_VKey_ViewRoll90CCW, (V3d_TypeOfOrientation)-1},
|
|
{Aspect_VKey_ViewFitAll, (V3d_TypeOfOrientation)-1}};
|
|
{
|
|
Standard_Mutex::Sentry aLock(myKeys.Mutex());
|
|
const size_t aNbKeys = sizeof(THE_VIEW_KEYS) / sizeof(*THE_VIEW_KEYS);
|
|
const double anEventTime = EventTime();
|
|
for (size_t aKeyIter = 0; aKeyIter < aNbKeys; ++aKeyIter)
|
|
{
|
|
const ViewKeyAction& aKeyAction = THE_VIEW_KEYS[aKeyIter];
|
|
if (!myKeys.IsKeyDown(aKeyAction.Key))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
myKeys.KeyUp(aKeyAction.Key, anEventTime);
|
|
if (aCameraBack.IsNull())
|
|
{
|
|
aCameraBack = theView->Camera();
|
|
theView->SetCamera(new Graphic3d_Camera(aCameraBack));
|
|
}
|
|
if (aKeyAction.Orientation != (V3d_TypeOfOrientation)-1)
|
|
{
|
|
theView->SetProj(aKeyAction.Orientation);
|
|
FitAllAuto(theCtx, theView);
|
|
}
|
|
else if (aKeyAction.Key == Aspect_VKey_ViewRoll90CW)
|
|
{
|
|
const double aTwist = theView->Twist() + M_PI / 2.0;
|
|
theView->SetTwist(aTwist);
|
|
}
|
|
else if (aKeyAction.Key == Aspect_VKey_ViewRoll90CCW)
|
|
{
|
|
const double aTwist = theView->Twist() - M_PI / 2.0;
|
|
theView->SetTwist(aTwist);
|
|
}
|
|
else if (aKeyAction.Key == Aspect_VKey_ViewFitAll)
|
|
{
|
|
FitAllAuto(theCtx, theView);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aCameraBack.IsNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
Handle(Graphic3d_Camera) aCameraNew = theView->Camera();
|
|
theView->SetCamera(aCameraBack);
|
|
const Graphic3d_Mat4d anOrientMat1 = aCameraBack->OrientationMatrix();
|
|
const Graphic3d_Mat4d anOrientMat2 = aCameraNew->OrientationMatrix();
|
|
if (anOrientMat1 != anOrientMat2)
|
|
{
|
|
const Handle(AIS_AnimationCamera)& aCamAnim = myViewAnimation;
|
|
aCamAnim->SetView(theView);
|
|
aCamAnim->SetStartPts(0.0);
|
|
aCamAnim->SetCameraStart(new Graphic3d_Camera(aCameraBack));
|
|
aCamAnim->SetCameraEnd(new Graphic3d_Camera(aCameraNew));
|
|
aCamAnim->StartTimer(0.0, 1.0, true, false);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
AIS_WalkDelta AIS_ViewController::handleNavigationKeys(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
// navigation keys
|
|
double aCrouchRatio = 1.0, aRunRatio = 1.0;
|
|
if (myNavigationMode == AIS_NavigationMode_FirstPersonFlight)
|
|
{
|
|
aRunRatio = 3.0;
|
|
}
|
|
|
|
const double aRotSpeed = 0.5;
|
|
const double aWalkSpeedCoef = WalkSpeedRelative();
|
|
AIS_WalkDelta aWalk = FetchNavigationKeys(aCrouchRatio, aRunRatio);
|
|
if (aWalk.IsJumping())
|
|
{
|
|
// ask more frames
|
|
setAskNextFrame();
|
|
theView->Invalidate();
|
|
}
|
|
if (aWalk.IsEmpty())
|
|
{
|
|
if (aWalk.IsDefined())
|
|
{
|
|
setAskNextFrame();
|
|
}
|
|
return aWalk;
|
|
}
|
|
else if (myGL.OrbitRotation.ToRotate || myGL.OrbitRotation.ToStart)
|
|
{
|
|
return aWalk;
|
|
}
|
|
|
|
gp_XYZ aMin, aMax;
|
|
const Bnd_Box aBndBox = theView->View()->MinMaxValues();
|
|
if (!aBndBox.IsVoid())
|
|
{
|
|
aMin = aBndBox.CornerMin().XYZ();
|
|
aMax = aBndBox.CornerMax().XYZ();
|
|
}
|
|
double aBndDiam = Max(Max(aMax.X() - aMin.X(), aMax.Y() - aMin.Y()), aMax.Z() - aMin.Z());
|
|
if (aBndDiam <= gp::Resolution())
|
|
{
|
|
aBndDiam = 0.001;
|
|
}
|
|
|
|
const double aWalkSpeed = myNavigationMode != AIS_NavigationMode_Orbit
|
|
&& myNavigationMode != AIS_NavigationMode_FirstPersonFlight
|
|
? theView->View()->UnitFactor() * WalkSpeedAbsolute()
|
|
: aWalkSpeedCoef * aBndDiam;
|
|
const Handle(Graphic3d_Camera)& aCam =
|
|
theView->View()->IsActiveXR() ? theView->View()->BaseXRCamera() : theView->Camera();
|
|
|
|
// move forward in plane XY and up along Z
|
|
const gp_Dir anUp = ToLockOrbitZUp() ? gp::DZ() : aCam->OrthogonalizedUp();
|
|
if (aWalk.ToMove() && myToAllowPanning)
|
|
{
|
|
const gp_Vec aSide = -aCam->SideRight();
|
|
gp_XYZ aFwd = aCam->Direction().XYZ();
|
|
aFwd -= anUp.XYZ() * (anUp.XYZ() * aFwd);
|
|
|
|
gp_XYZ aMoveVec;
|
|
if (!aWalk[AIS_WalkTranslation_Forward].IsEmpty())
|
|
{
|
|
if (!aCam->IsOrthographic())
|
|
{
|
|
aMoveVec += aFwd * aWalk[AIS_WalkTranslation_Forward].Value
|
|
* aWalk[AIS_WalkTranslation_Forward].Pressure * aWalkSpeed;
|
|
}
|
|
}
|
|
if (!aWalk[AIS_WalkTranslation_Side].IsEmpty())
|
|
{
|
|
aMoveVec += aSide.XYZ() * aWalk[AIS_WalkTranslation_Side].Value
|
|
* aWalk[AIS_WalkTranslation_Side].Pressure * aWalkSpeed;
|
|
}
|
|
if (!aWalk[AIS_WalkTranslation_Up].IsEmpty())
|
|
{
|
|
aMoveVec += anUp.XYZ() * aWalk[AIS_WalkTranslation_Up].Value
|
|
* aWalk[AIS_WalkTranslation_Up].Pressure * aWalkSpeed;
|
|
}
|
|
{
|
|
if (aCam->IsOrthographic())
|
|
{
|
|
if (!aWalk[AIS_WalkTranslation_Forward].IsEmpty())
|
|
{
|
|
const double aZoomDelta = aWalk[AIS_WalkTranslation_Forward].Value
|
|
* aWalk[AIS_WalkTranslation_Forward].Pressure * aWalkSpeedCoef;
|
|
handleZoom(theView, Aspect_ScrollDelta(aZoomDelta * 100.0), NULL);
|
|
}
|
|
}
|
|
|
|
gp_Trsf aTrsfTranslate;
|
|
aTrsfTranslate.SetTranslation(aMoveVec);
|
|
aCam->Transform(aTrsfTranslate);
|
|
}
|
|
}
|
|
|
|
if (myNavigationMode == AIS_NavigationMode_Orbit && myToAllowRotation)
|
|
{
|
|
if (!aWalk[AIS_WalkRotation_Yaw].IsEmpty())
|
|
{
|
|
gp_Trsf aTrsfRot;
|
|
aTrsfRot.SetRotation(gp_Ax1(aCam->Eye(), anUp),
|
|
aWalk[AIS_WalkRotation_Yaw].Value * aRotSpeed);
|
|
aCam->Transform(aTrsfRot);
|
|
}
|
|
if (!aWalk[AIS_WalkRotation_Pitch].IsEmpty())
|
|
{
|
|
const gp_Vec aSide = -aCam->SideRight();
|
|
gp_Trsf aTrsfRot;
|
|
aTrsfRot.SetRotation(gp_Ax1(aCam->Eye(), aSide),
|
|
-aWalk[AIS_WalkRotation_Pitch].Value * aRotSpeed);
|
|
aCam->Transform(aTrsfRot);
|
|
}
|
|
if (!aWalk[AIS_WalkRotation_Roll].IsEmpty() && !ToLockOrbitZUp())
|
|
{
|
|
gp_Trsf aTrsfRot;
|
|
aTrsfRot.SetRotation(gp_Ax1(aCam->Center(), aCam->Direction()),
|
|
aWalk[AIS_WalkRotation_Roll].Value * aRotSpeed);
|
|
aCam->Transform(aTrsfRot);
|
|
}
|
|
}
|
|
|
|
// ask more frames
|
|
setAskNextFrame();
|
|
theView->Invalidate();
|
|
theView->View()->SynchronizeXRBaseToPosedCamera();
|
|
return aWalk;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleCameraActions(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const AIS_WalkDelta& theWalk)
|
|
{
|
|
// apply view actions
|
|
if (myGL.Orientation.ToSetViewOrient)
|
|
{
|
|
theView->SetProj(myGL.Orientation.ViewOrient);
|
|
myGL.Orientation.ToFitAll = true;
|
|
}
|
|
|
|
// apply fit all
|
|
if (myGL.Orientation.ToFitAll)
|
|
{
|
|
const double aFitMargin = 0.01;
|
|
theView->FitAll(aFitMargin, false);
|
|
theView->Invalidate();
|
|
myGL.Orientation.ToFitAll = false;
|
|
}
|
|
|
|
AIS_ListOfInteractive anObjects;
|
|
theCtx->DisplayedObjects(anObjects);
|
|
for (AIS_ListIteratorOfListOfInteractive anObjIter(anObjects); anObjIter.More(); anObjIter.Next())
|
|
{
|
|
anObjIter.Value()->RecomputeTransformation(theView->Camera());
|
|
}
|
|
|
|
if (myGL.IsNewGesture)
|
|
{
|
|
if (myAnchorPointPrs1->HasInteractiveContext())
|
|
{
|
|
theCtx->Remove(myAnchorPointPrs1, false);
|
|
if (!theView->Viewer()->ZLayerSettings(myAnchorPointPrs1->ZLayer()).IsImmediate())
|
|
{
|
|
theView->Invalidate();
|
|
}
|
|
else
|
|
{
|
|
theView->InvalidateImmediate();
|
|
}
|
|
}
|
|
if (myAnchorPointPrs2->HasInteractiveContext())
|
|
{
|
|
theCtx->Remove(myAnchorPointPrs2, false);
|
|
if (!theView->Viewer()->ZLayerSettings(myAnchorPointPrs2->ZLayer()).IsImmediate())
|
|
{
|
|
theView->Invalidate();
|
|
}
|
|
else
|
|
{
|
|
theView->InvalidateImmediate();
|
|
}
|
|
}
|
|
|
|
if (myHasHlrOnBeforeRotation)
|
|
{
|
|
myHasHlrOnBeforeRotation = false;
|
|
theView->SetComputedMode(true);
|
|
theView->Invalidate();
|
|
}
|
|
}
|
|
|
|
if (myNavigationMode != AIS_NavigationMode_FirstPersonWalk)
|
|
{
|
|
if (myGL.Panning.ToStart && myToAllowPanning)
|
|
{
|
|
gp_Pnt aPanPnt(Precision::Infinite(), 0.0, 0.0);
|
|
if (!theView->Camera()->IsOrthographic())
|
|
{
|
|
bool toStickToRay = false;
|
|
if (myGL.Panning.PointStart.x() >= 0 && myGL.Panning.PointStart.y() >= 0)
|
|
{
|
|
PickPoint(aPanPnt, theCtx, theView, myGL.Panning.PointStart, toStickToRay);
|
|
}
|
|
if (Precision::IsInfinite(aPanPnt.X()))
|
|
{
|
|
Graphic3d_Vec2i aWinSize;
|
|
theView->Window()->Size(aWinSize.x(), aWinSize.y());
|
|
PickPoint(aPanPnt, theCtx, theView, aWinSize / 2, toStickToRay);
|
|
}
|
|
if (!Precision::IsInfinite(aPanPnt.X()) && myToShowPanAnchorPoint)
|
|
{
|
|
gp_Trsf aPntTrsf;
|
|
aPntTrsf.SetTranslation(gp_Vec(aPanPnt.XYZ()));
|
|
theCtx->SetLocation(myAnchorPointPrs2, aPntTrsf);
|
|
}
|
|
}
|
|
setPanningAnchorPoint(aPanPnt);
|
|
}
|
|
|
|
if (myToShowPanAnchorPoint && hasPanningAnchorPoint() && myGL.Panning.ToPan
|
|
&& !myGL.IsNewGesture && !myAnchorPointPrs2->HasInteractiveContext())
|
|
{
|
|
theCtx->Display(myAnchorPointPrs2, 0, -1, false, AIS_DS_Displayed);
|
|
}
|
|
|
|
handlePanning(theView);
|
|
handleZRotate(theView);
|
|
}
|
|
|
|
if ((myNavigationMode == AIS_NavigationMode_Orbit || myGL.OrbitRotation.ToStart
|
|
|| myGL.OrbitRotation.ToRotate)
|
|
&& myToAllowRotation)
|
|
{
|
|
if (myGL.OrbitRotation.ToStart && !myHasHlrOnBeforeRotation)
|
|
{
|
|
myHasHlrOnBeforeRotation = theView->ComputedMode();
|
|
if (myHasHlrOnBeforeRotation)
|
|
{
|
|
theView->SetComputedMode(false);
|
|
}
|
|
}
|
|
|
|
gp_Pnt aGravPnt;
|
|
if (myGL.OrbitRotation.ToStart)
|
|
{
|
|
aGravPnt = GravityPoint(theCtx, theView);
|
|
if (myToShowRotateCenter)
|
|
{
|
|
gp_Trsf aPntTrsf;
|
|
aPntTrsf.SetTranslation(gp_Vec(aGravPnt.XYZ()));
|
|
theCtx->SetLocation(myAnchorPointPrs1, aPntTrsf);
|
|
theCtx->SetLocation(myAnchorPointPrs2, aPntTrsf);
|
|
}
|
|
}
|
|
|
|
if (myToShowRotateCenter && myGL.OrbitRotation.ToRotate && !myGL.IsNewGesture
|
|
&& !myAnchorPointPrs1->HasInteractiveContext())
|
|
{
|
|
theCtx->Display(myAnchorPointPrs1, 0, -1, false, AIS_DS_Displayed);
|
|
theCtx->Display(myAnchorPointPrs2, 0, -1, false, AIS_DS_Displayed);
|
|
}
|
|
handleOrbitRotation(theView,
|
|
aGravPnt,
|
|
myToLockOrbitZUp || myNavigationMode != AIS_NavigationMode_Orbit);
|
|
}
|
|
|
|
if ((myNavigationMode != AIS_NavigationMode_Orbit || myGL.ViewRotation.ToStart
|
|
|| myGL.ViewRotation.ToRotate)
|
|
&& myToAllowRotation)
|
|
{
|
|
if (myGL.ViewRotation.ToStart && !myHasHlrOnBeforeRotation)
|
|
{
|
|
myHasHlrOnBeforeRotation = theView->ComputedMode();
|
|
if (myHasHlrOnBeforeRotation)
|
|
{
|
|
theView->SetComputedMode(false);
|
|
}
|
|
}
|
|
|
|
double aRoll = 0.0;
|
|
if (!theWalk[AIS_WalkRotation_Roll].IsEmpty() && !myToLockOrbitZUp)
|
|
{
|
|
aRoll = (M_PI / 12.0) * theWalk[AIS_WalkRotation_Roll].Pressure;
|
|
aRoll *= Min(1000.0 * theWalk[AIS_WalkRotation_Roll].Duration, 100.0) / 100.0;
|
|
if (theWalk[AIS_WalkRotation_Roll].Value < 0.0)
|
|
{
|
|
aRoll = -aRoll;
|
|
}
|
|
}
|
|
|
|
handleViewRotation(theView,
|
|
theWalk[AIS_WalkRotation_Yaw].Value,
|
|
theWalk[AIS_WalkRotation_Pitch].Value,
|
|
aRoll,
|
|
myNavigationMode == AIS_NavigationMode_FirstPersonFlight);
|
|
}
|
|
|
|
if (!myGL.ZoomActions.IsEmpty())
|
|
{
|
|
for (NCollection_Sequence<Aspect_ScrollDelta>::Iterator aZoomIter(myGL.ZoomActions);
|
|
aZoomIter.More();
|
|
aZoomIter.Next())
|
|
{
|
|
Aspect_ScrollDelta aZoomParams = aZoomIter.Value();
|
|
if (myToAllowZFocus && (aZoomParams.Flags & Aspect_VKeyFlags_CTRL) != 0
|
|
&& theView->Camera()->IsStereo())
|
|
{
|
|
handleZFocusScroll(theView, aZoomParams);
|
|
continue;
|
|
}
|
|
|
|
if (!myToAllowZooming)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!theView->Camera()->IsOrthographic())
|
|
{
|
|
gp_Pnt aPnt;
|
|
if (aZoomParams.HasPoint()
|
|
&& PickPoint(aPnt, theCtx, theView, aZoomParams.Point, myToStickToRayOnZoom))
|
|
{
|
|
handleZoom(theView, aZoomParams, &aPnt);
|
|
continue;
|
|
}
|
|
|
|
Graphic3d_Vec2i aWinSize;
|
|
theView->Window()->Size(aWinSize.x(), aWinSize.y());
|
|
if (PickPoint(aPnt, theCtx, theView, aWinSize / 2, myToStickToRayOnZoom))
|
|
{
|
|
aZoomParams.ResetPoint(); // do not pretend to zoom at 'nothing'
|
|
handleZoom(theView, aZoomParams, &aPnt);
|
|
continue;
|
|
}
|
|
}
|
|
handleZoom(theView, aZoomParams, NULL);
|
|
}
|
|
myGL.ZoomActions.Clear();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRInput(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const AIS_WalkDelta&)
|
|
{
|
|
theView->View()->ProcessXRInput();
|
|
if (!theView->View()->IsActiveXR())
|
|
{
|
|
return;
|
|
}
|
|
handleXRTurnPad(theCtx, theView);
|
|
handleXRTeleport(theCtx, theView);
|
|
handleXRPicking(theCtx, theView);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRTurnPad(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (myXRTurnAngle <= 0.0 || !theView->View()->IsActiveXR())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// turn left/right at 45 degrees on left/right trackpad clicks
|
|
for (int aHand = 0; aHand < 2; ++aHand)
|
|
{
|
|
const Aspect_XRTrackedDeviceRole aRole =
|
|
aHand == 0 ? Aspect_XRTrackedDeviceRole_RightHand : Aspect_XRTrackedDeviceRole_LeftHand;
|
|
const Handle(Aspect_XRAction)& aPadClickAct =
|
|
theView->View()->XRSession()->GenericAction(aRole, Aspect_XRGenericAction_InputTrackPadClick);
|
|
const Handle(Aspect_XRAction)& aPadPosAct =
|
|
theView->View()->XRSession()->GenericAction(aRole,
|
|
Aspect_XRGenericAction_InputTrackPadPosition);
|
|
if (aPadClickAct.IsNull() || aPadPosAct.IsNull())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const Aspect_XRDigitalActionData aPadClick =
|
|
theView->View()->XRSession()->GetDigitalActionData(aPadClickAct);
|
|
const Aspect_XRAnalogActionData aPadPos =
|
|
theView->View()->XRSession()->GetAnalogActionData(aPadPosAct);
|
|
if (aPadClick.IsActive && aPadClick.IsPressed && aPadClick.IsChanged && aPadPos.IsActive
|
|
&& Abs(aPadPos.VecXYZ.y()) < 0.5f && Abs(aPadPos.VecXYZ.x()) > 0.7f)
|
|
{
|
|
gp_Trsf aTrsfTurn;
|
|
aTrsfTurn.SetRotation(gp_Ax1(gp::Origin(), theView->View()->BaseXRCamera()->Up()),
|
|
aPadPos.VecXYZ.x() < 0.0f ? myXRTurnAngle : -myXRTurnAngle);
|
|
theView->View()->TurnViewXRCamera(aTrsfTurn);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRTeleport(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (!theView->View()->IsActiveXR())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// teleport on forward trackpad unclicks
|
|
const Aspect_XRTrackedDeviceRole aTeleOld = myXRLastTeleportHand;
|
|
myXRLastTeleportHand = Aspect_XRTrackedDeviceRole_Other;
|
|
for (int aHand = 0; aHand < 2; ++aHand)
|
|
{
|
|
const Aspect_XRTrackedDeviceRole aRole =
|
|
aHand == 0 ? Aspect_XRTrackedDeviceRole_RightHand : Aspect_XRTrackedDeviceRole_LeftHand;
|
|
const Standard_Integer aDeviceId = theView->View()->XRSession()->NamedTrackedDevice(aRole);
|
|
if (aDeviceId == -1)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const Handle(Aspect_XRAction)& aPadClickAct =
|
|
theView->View()->XRSession()->GenericAction(aRole, Aspect_XRGenericAction_InputTrackPadClick);
|
|
const Handle(Aspect_XRAction)& aPadPosAct =
|
|
theView->View()->XRSession()->GenericAction(aRole,
|
|
Aspect_XRGenericAction_InputTrackPadPosition);
|
|
if (aPadClickAct.IsNull() || aPadPosAct.IsNull())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const Aspect_XRDigitalActionData aPadClick =
|
|
theView->View()->XRSession()->GetDigitalActionData(aPadClickAct);
|
|
const Aspect_XRAnalogActionData aPadPos =
|
|
theView->View()->XRSession()->GetAnalogActionData(aPadPosAct);
|
|
const bool isPressed = aPadClick.IsPressed;
|
|
const bool isClicked = !aPadClick.IsPressed && aPadClick.IsChanged;
|
|
if (aPadClick.IsActive && (isPressed || isClicked) && aPadPos.IsActive
|
|
&& aPadPos.VecXYZ.y() > 0.6f && Abs(aPadPos.VecXYZ.x()) < 0.5f)
|
|
{
|
|
const Aspect_TrackedDevicePose& aPose =
|
|
theView->View()->XRSession()->TrackedPoses()[aDeviceId];
|
|
if (!aPose.IsValidPose)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
myXRLastTeleportHand = aRole;
|
|
Standard_Real& aPickDepth = aRole == Aspect_XRTrackedDeviceRole_LeftHand
|
|
? myXRLastPickDepthLeft
|
|
: myXRLastPickDepthRight;
|
|
aPickDepth = Precision::Infinite();
|
|
Graphic3d_Vec3 aPickNorm;
|
|
const gp_Trsf aHandBase = theView->View()->PoseXRToWorld(aPose.Orientation);
|
|
const Standard_Real aHeadHeight =
|
|
theView->View()->XRSession()->HeadPose().TranslationPart().Y();
|
|
{
|
|
const Standard_Integer aPickedId =
|
|
handleXRMoveTo(theCtx, theView, aPose.Orientation, false);
|
|
if (aPickedId >= 1)
|
|
{
|
|
const SelectMgr_SortCriterion& aPickedData =
|
|
theCtx->MainSelector()->PickedData(aPickedId);
|
|
aPickNorm = aPickedData.Normal;
|
|
if (aPickNorm.SquareModulus() > ShortRealEpsilon())
|
|
{
|
|
aPickDepth = aPickedData.Point.Distance(aHandBase.TranslationPart());
|
|
}
|
|
}
|
|
}
|
|
if (isClicked)
|
|
{
|
|
myXRLastTeleportHand = Aspect_XRTrackedDeviceRole_Other;
|
|
if (!Precision::IsInfinite(aPickDepth))
|
|
{
|
|
const gp_Dir aTeleDir = -gp::DZ().Transformed(aHandBase);
|
|
const gp_Dir anUpDir = theView->View()->BaseXRCamera()->Up();
|
|
|
|
bool isHorizontal = false;
|
|
gp_Dir aPickNormDir(aPickNorm.x(), aPickNorm.y(), aPickNorm.z());
|
|
if (anUpDir.IsEqual(aPickNormDir, M_PI_4) || anUpDir.IsEqual(-aPickNormDir, M_PI_4))
|
|
{
|
|
isHorizontal = true;
|
|
}
|
|
|
|
gp_Pnt aNewEye = aHandBase.TranslationPart();
|
|
if (isHorizontal)
|
|
{
|
|
aNewEye = aHandBase.TranslationPart() + aTeleDir.XYZ() * aPickDepth
|
|
+ anUpDir.XYZ() * aHeadHeight;
|
|
}
|
|
else
|
|
{
|
|
if (aPickNormDir.Dot(aTeleDir) < 0.0)
|
|
{
|
|
aPickNormDir.Reverse();
|
|
}
|
|
aNewEye = aHandBase.TranslationPart() + aTeleDir.XYZ() * aPickDepth
|
|
- aPickNormDir.XYZ() * aHeadHeight / 4;
|
|
}
|
|
|
|
theView->View()->PosedXRCamera()->MoveEyeTo(aNewEye);
|
|
theView->View()->ComputeXRBaseCameraFromPosed(theView->View()->PosedXRCamera(),
|
|
theView->View()->XRSession()->HeadPose());
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (myXRLastTeleportHand != aTeleOld)
|
|
{
|
|
if (aTeleOld != Aspect_XRTrackedDeviceRole_Other)
|
|
{
|
|
if (const Handle(Aspect_XRAction)& aHaptic =
|
|
theView->View()->XRSession()->GenericAction(aTeleOld,
|
|
Aspect_XRGenericAction_OutputHaptic))
|
|
{
|
|
theView->View()->XRSession()->AbortHapticVibrationAction(aHaptic);
|
|
}
|
|
}
|
|
if (myXRLastTeleportHand != Aspect_XRTrackedDeviceRole_Other)
|
|
{
|
|
if (const Handle(Aspect_XRAction)& aHaptic =
|
|
theView->View()->XRSession()->GenericAction(myXRLastTeleportHand,
|
|
Aspect_XRGenericAction_OutputHaptic))
|
|
{
|
|
theView->View()->XRSession()->TriggerHapticVibrationAction(aHaptic, myXRTeleportHaptic);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRPicking(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (!theView->View()->IsActiveXR())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// handle selection on trigger clicks
|
|
Aspect_XRTrackedDeviceRole aPickDevOld = myXRLastPickingHand;
|
|
myXRLastPickingHand = Aspect_XRTrackedDeviceRole_Other;
|
|
for (int aHand = 0; aHand < 2; ++aHand)
|
|
{
|
|
const Aspect_XRTrackedDeviceRole aRole =
|
|
aHand == 0 ? Aspect_XRTrackedDeviceRole_RightHand : Aspect_XRTrackedDeviceRole_LeftHand;
|
|
const Handle(Aspect_XRAction)& aTrigClickAct =
|
|
theView->View()->XRSession()->GenericAction(aRole, Aspect_XRGenericAction_InputTriggerClick);
|
|
const Handle(Aspect_XRAction)& aTrigPullAct =
|
|
theView->View()->XRSession()->GenericAction(aRole, Aspect_XRGenericAction_InputTriggerPull);
|
|
if (aTrigClickAct.IsNull() || aTrigPullAct.IsNull())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const Aspect_XRDigitalActionData aTrigClick =
|
|
theView->View()->XRSession()->GetDigitalActionData(aTrigClickAct);
|
|
const Aspect_XRAnalogActionData aTrigPos =
|
|
theView->View()->XRSession()->GetAnalogActionData(aTrigPullAct);
|
|
if (aTrigPos.IsActive && Abs(aTrigPos.VecXYZ.x()) > 0.1f)
|
|
{
|
|
myXRLastPickingHand = aRole;
|
|
handleXRHighlight(theCtx, theView);
|
|
if (aTrigClick.IsActive && aTrigClick.IsPressed && aTrigClick.IsChanged)
|
|
{
|
|
theCtx->SelectDetected();
|
|
OnSelectionChanged(theCtx, theView);
|
|
if (const Handle(Aspect_XRAction)& aHaptic =
|
|
theView->View()->XRSession()->GenericAction(myXRLastPickingHand,
|
|
Aspect_XRGenericAction_OutputHaptic))
|
|
{
|
|
theView->View()->XRSession()->TriggerHapticVibrationAction(aHaptic, myXRSelectHaptic);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (myXRLastPickingHand != aPickDevOld)
|
|
{
|
|
theCtx->ClearDetected();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::OnSelectionChanged(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)&)
|
|
{
|
|
//
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::OnSubviewChanged(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)&,
|
|
const Handle(V3d_View)&)
|
|
{
|
|
//
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::OnObjectDragged(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
AIS_DragAction theAction)
|
|
{
|
|
switch (theAction)
|
|
{
|
|
case AIS_DragAction_Start: {
|
|
myDragObject.Nullify();
|
|
myDragOwner.Nullify();
|
|
if (!theCtx->HasDetected())
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Handle(SelectMgr_EntityOwner)& aDetectedOwner = theCtx->DetectedOwner();
|
|
Handle(AIS_InteractiveObject) aDetectedPrs =
|
|
Handle(AIS_InteractiveObject)::DownCast(aDetectedOwner->Selectable());
|
|
|
|
if (aDetectedPrs->ProcessDragging(theCtx,
|
|
theView,
|
|
aDetectedOwner,
|
|
myGL.Dragging.PointStart,
|
|
myGL.Dragging.PointTo,
|
|
theAction))
|
|
{
|
|
myDragObject = aDetectedPrs;
|
|
myDragOwner = aDetectedOwner;
|
|
}
|
|
return;
|
|
}
|
|
case AIS_DragAction_Confirmed: {
|
|
if (myDragObject.IsNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
myDragObject->ProcessDragging(theCtx,
|
|
theView,
|
|
myDragOwner,
|
|
myGL.Dragging.PointStart,
|
|
myGL.Dragging.PointTo,
|
|
theAction);
|
|
return;
|
|
}
|
|
case AIS_DragAction_Update: {
|
|
if (myDragObject.IsNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (Handle(SelectMgr_EntityOwner) aGlobOwner = myDragObject->GlobalSelOwner())
|
|
{
|
|
theCtx->SetSelectedState(aGlobOwner, true);
|
|
}
|
|
|
|
myDragObject->ProcessDragging(theCtx,
|
|
theView,
|
|
myDragOwner,
|
|
myGL.Dragging.PointStart,
|
|
myGL.Dragging.PointTo,
|
|
theAction);
|
|
theView->Invalidate();
|
|
return;
|
|
}
|
|
case AIS_DragAction_Abort: {
|
|
if (myDragObject.IsNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
myGL.Dragging.PointTo = myGL.Dragging.PointStart;
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Update);
|
|
|
|
myDragObject->ProcessDragging(theCtx,
|
|
theView,
|
|
myDragOwner,
|
|
myGL.Dragging.PointStart,
|
|
myGL.Dragging.PointTo,
|
|
theAction);
|
|
Standard_FALLTHROUGH
|
|
}
|
|
case AIS_DragAction_Stop: {
|
|
if (myDragObject.IsNull())
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (Handle(SelectMgr_EntityOwner) aGlobOwner = myDragObject->GlobalSelOwner())
|
|
{
|
|
theCtx->SetSelectedState(aGlobOwner, false);
|
|
}
|
|
|
|
myDragObject->ProcessDragging(theCtx,
|
|
theView,
|
|
myDragOwner,
|
|
myGL.Dragging.PointStart,
|
|
myGL.Dragging.PointTo,
|
|
theAction);
|
|
theView->Invalidate();
|
|
myDragObject.Nullify();
|
|
myDragOwner.Nullify();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::contextLazyMoveTo(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const Graphic3d_Vec2i& thePnt)
|
|
{
|
|
if (myPrevMoveTo == thePnt
|
|
|| myHasHlrOnBeforeRotation) // ignore highlighting in-between rotation of HLR view
|
|
{
|
|
return;
|
|
}
|
|
|
|
myPrevMoveTo = thePnt;
|
|
|
|
Handle(SelectMgr_EntityOwner) aLastPicked = theCtx->DetectedOwner();
|
|
|
|
// Picking relies on the camera frustum (including Z-range) - so make temporary AutoZFit()
|
|
// and then restore previous frustum to avoid immediate layer rendering issues if View has not
|
|
// been invalidated.
|
|
const Standard_Real aZNear = theView->Camera()->ZNear(), aZFar = theView->Camera()->ZFar();
|
|
theView->AutoZFit();
|
|
theCtx->MoveTo(thePnt.x(), thePnt.y(), theView, false);
|
|
theView->Camera()->SetZRange(aZNear, aZFar);
|
|
|
|
Handle(SelectMgr_EntityOwner) aNewPicked = theCtx->DetectedOwner();
|
|
|
|
if (theView->Viewer()->IsGridActive() && theView->Viewer()->GridEcho())
|
|
{
|
|
if (aNewPicked.IsNull())
|
|
{
|
|
Graphic3d_Vec3d aPnt3d;
|
|
theView->ConvertToGrid(thePnt.x(), thePnt.y(), aPnt3d[0], aPnt3d[1], aPnt3d[2]);
|
|
theView->Viewer()->ShowGridEcho(theView, Graphic3d_Vertex(aPnt3d[0], aPnt3d[1], aPnt3d[2]));
|
|
theView->InvalidateImmediate();
|
|
}
|
|
else
|
|
{
|
|
theView->Viewer()->HideGridEcho(theView);
|
|
theView->InvalidateImmediate();
|
|
}
|
|
}
|
|
|
|
if (aLastPicked != aNewPicked || (!aNewPicked.IsNull() && aNewPicked->IsForcedHilight()))
|
|
{
|
|
// dynamic highlight affects all Views
|
|
for (V3d_ListOfViewIterator aViewIter(theView->Viewer()->ActiveViewIterator());
|
|
aViewIter.More();
|
|
aViewIter.Next())
|
|
{
|
|
const Handle(V3d_View)& aView = aViewIter.Value();
|
|
aView->InvalidateImmediate();
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleSelectionPick(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (myGL.Selection.Tool == AIS_ViewSelectionTool_Picking && !myGL.Selection.Points.IsEmpty())
|
|
{
|
|
for (NCollection_Sequence<Graphic3d_Vec2i>::Iterator aPntIter(myGL.Selection.Points);
|
|
aPntIter.More();
|
|
aPntIter.Next())
|
|
{
|
|
const bool hadPrevMoveTo = HasPreviousMoveTo();
|
|
contextLazyMoveTo(theCtx, theView, aPntIter.Value());
|
|
if (!hadPrevMoveTo)
|
|
{
|
|
ResetPreviousMoveTo();
|
|
}
|
|
|
|
theCtx->SelectDetected(myGL.Selection.Scheme);
|
|
|
|
// selection affects all Views
|
|
theView->Viewer()->Invalidate();
|
|
|
|
OnSelectionChanged(theCtx, theView);
|
|
}
|
|
|
|
myGL.Selection.Points.Clear();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleSelectionPoly(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
// rubber-band & window polygon selection
|
|
if (myGL.Selection.Tool == AIS_ViewSelectionTool_RubberBand
|
|
|| myGL.Selection.Tool == AIS_ViewSelectionTool_Polygon
|
|
|| myGL.Selection.Tool == AIS_ViewSelectionTool_ZoomWindow)
|
|
{
|
|
if (!myGL.Selection.Points.IsEmpty())
|
|
{
|
|
myRubberBand->ClearPoints();
|
|
myRubberBand->SetToUpdate();
|
|
|
|
const bool anIsRubber = myGL.Selection.Tool == AIS_ViewSelectionTool_RubberBand
|
|
|| myGL.Selection.Tool == AIS_ViewSelectionTool_ZoomWindow;
|
|
if (anIsRubber)
|
|
{
|
|
myRubberBand->SetRectangle(myGL.Selection.Points.First().x(),
|
|
-myGL.Selection.Points.First().y(),
|
|
myGL.Selection.Points.Last().x(),
|
|
-myGL.Selection.Points.Last().y());
|
|
}
|
|
else
|
|
{
|
|
Graphic3d_Vec2i aPrev(IntegerLast(), IntegerLast());
|
|
for (NCollection_Sequence<Graphic3d_Vec2i>::Iterator aSelIter(myGL.Selection.Points);
|
|
aSelIter.More();
|
|
aSelIter.Next())
|
|
{
|
|
Graphic3d_Vec2i aPntNew = Graphic3d_Vec2i(aSelIter.Value().x(), -aSelIter.Value().y());
|
|
if (aPntNew != aPrev)
|
|
{
|
|
aPrev = aPntNew;
|
|
myRubberBand->AddPoint(Graphic3d_Vec2i(aSelIter.Value().x(), -aSelIter.Value().y()));
|
|
}
|
|
}
|
|
}
|
|
|
|
myRubberBand->SetPolygonClosed(anIsRubber);
|
|
try
|
|
{
|
|
theCtx->Display(myRubberBand, 0, -1, false, AIS_DS_Displayed);
|
|
}
|
|
catch (const Standard_Failure& theEx)
|
|
{
|
|
Message::SendWarning(
|
|
TCollection_AsciiString("Internal error while displaying rubber-band: ")
|
|
+ theEx.DynamicType()->Name() + ", " + theEx.GetMessageString());
|
|
myRubberBand->ClearPoints();
|
|
}
|
|
if (!theView->Viewer()->ZLayerSettings(myRubberBand->ZLayer()).IsImmediate())
|
|
{
|
|
theView->Invalidate();
|
|
}
|
|
else
|
|
{
|
|
theView->InvalidateImmediate();
|
|
}
|
|
}
|
|
else if (!myRubberBand.IsNull() && myRubberBand->HasInteractiveContext())
|
|
{
|
|
theCtx->Remove(myRubberBand, false);
|
|
myRubberBand->ClearPoints();
|
|
}
|
|
}
|
|
|
|
if (myGL.Selection.ToApplyTool)
|
|
{
|
|
myGL.Selection.ToApplyTool = false;
|
|
if (theCtx->IsDisplayed(myRubberBand))
|
|
{
|
|
theCtx->Remove(myRubberBand, false);
|
|
{
|
|
const NCollection_Sequence<Graphic3d_Vec2i>& aPoints = myRubberBand->Points();
|
|
if (aPoints.Size() == 4 && aPoints.Value(1).x() == aPoints.Value(2).x()
|
|
&& aPoints.Value(3).x() == aPoints.Value(4).x()
|
|
&& aPoints.Value(1).y() == aPoints.Value(4).y()
|
|
&& aPoints.Value(2).y() == aPoints.Value(3).y())
|
|
{
|
|
const Graphic3d_Vec2i aPnt1(aPoints.Value(1).x(), -aPoints.Value(1).y());
|
|
const Graphic3d_Vec2i aPnt2(aPoints.Value(3).x(), -aPoints.Value(3).y());
|
|
if (myGL.Selection.Tool == AIS_ViewSelectionTool_ZoomWindow)
|
|
{
|
|
theView->WindowFitAll(aPnt1.x(), aPnt1.y(), aPnt2.x(), aPnt2.y());
|
|
theView->Invalidate();
|
|
}
|
|
else
|
|
{
|
|
theCtx->MainSelector()->AllowOverlapDetection(aPnt1.y() != Min(aPnt1.y(), aPnt2.y()));
|
|
theCtx->SelectRectangle(
|
|
Graphic3d_Vec2i(Min(aPnt1.x(), aPnt2.x()), Min(aPnt1.y(), aPnt2.y())),
|
|
Graphic3d_Vec2i(Max(aPnt1.x(), aPnt2.x()), Max(aPnt1.y(), aPnt2.y())),
|
|
theView,
|
|
myGL.Selection.Scheme);
|
|
theCtx->MainSelector()->AllowOverlapDetection(false);
|
|
}
|
|
}
|
|
else if (aPoints.Length() >= 3)
|
|
{
|
|
TColgp_Array1OfPnt2d aPolyline(1, aPoints.Length());
|
|
TColgp_Array1OfPnt2d::Iterator aPolyIter(aPolyline);
|
|
for (NCollection_Sequence<Graphic3d_Vec2i>::Iterator aSelIter(aPoints); aSelIter.More();
|
|
aSelIter.Next(), aPolyIter.Next())
|
|
{
|
|
const Graphic3d_Vec2i& aNewPnt = aSelIter.Value();
|
|
aPolyIter.ChangeValue() = gp_Pnt2d(aNewPnt.x(), -aNewPnt.y());
|
|
}
|
|
|
|
theCtx->SelectPolygon(aPolyline, theView, myGL.Selection.Scheme);
|
|
theCtx->MainSelector()->AllowOverlapDetection(false);
|
|
}
|
|
}
|
|
|
|
myRubberBand->ClearPoints();
|
|
if (myGL.Selection.Tool != AIS_ViewSelectionTool_ZoomWindow)
|
|
{
|
|
// selection affects all Views
|
|
theView->Viewer()->Invalidate();
|
|
OnSelectionChanged(theCtx, theView);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleDynamicHighlight(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if ((myGL.MoveTo.ToHilight || myGL.Dragging.ToStart)
|
|
&& myNavigationMode != AIS_NavigationMode_FirstPersonWalk)
|
|
{
|
|
const Graphic3d_Vec2i& aMoveToPnt =
|
|
myGL.MoveTo.ToHilight ? myGL.MoveTo.Point : myGL.Dragging.PointStart;
|
|
if (myGL.Dragging.ToStart && (!myGL.MoveTo.ToHilight || !myToAllowHighlight)
|
|
&& !HasPreviousMoveTo())
|
|
{
|
|
contextLazyMoveTo(theCtx, theView, aMoveToPnt);
|
|
ResetPreviousMoveTo();
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Start);
|
|
theCtx->ClearDetected();
|
|
}
|
|
else if (myToAllowHighlight)
|
|
{
|
|
if (myPrevMoveTo != aMoveToPnt
|
|
|| (!theView->View()->IsActiveXR()
|
|
&& (myGL.OrbitRotation.ToRotate || myGL.ViewRotation.ToRotate
|
|
|| theView->IsInvalidated())))
|
|
{
|
|
ResetPreviousMoveTo();
|
|
contextLazyMoveTo(theCtx, theView, aMoveToPnt);
|
|
}
|
|
if (myGL.Dragging.ToStart)
|
|
{
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Start);
|
|
}
|
|
}
|
|
|
|
myGL.MoveTo.ToHilight = false;
|
|
}
|
|
|
|
if (!myDragObject.IsNull())
|
|
{
|
|
if (myGL.Dragging.ToAbort)
|
|
{
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Abort);
|
|
myGL.OrbitRotation.ToRotate = false;
|
|
myGL.ViewRotation.ToRotate = false;
|
|
}
|
|
else if (myGL.Dragging.ToStop)
|
|
{
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Stop);
|
|
myGL.OrbitRotation.ToRotate = false;
|
|
myGL.ViewRotation.ToRotate = false;
|
|
}
|
|
else if (myGL.Dragging.ToMove)
|
|
{
|
|
if (myGL.Dragging.ToConfirm)
|
|
{
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Confirmed);
|
|
}
|
|
OnObjectDragged(theCtx, theView, AIS_DragAction_Update);
|
|
myGL.OrbitRotation.ToRotate = false;
|
|
myGL.ViewRotation.ToRotate = false;
|
|
myGL.Panning.ToPan = false;
|
|
myGL.ZoomActions.Clear();
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleMoveTo(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
handleSelectionPick(theCtx, theView);
|
|
handleDynamicHighlight(theCtx, theView);
|
|
handleSelectionPoly(theCtx, theView);
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleViewRedraw(const Handle(AIS_InteractiveContext)&,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
Handle(V3d_View) aParentView = theView->IsSubview() ? theView->ParentView() : theView;
|
|
|
|
// manage animation state
|
|
if (!myViewAnimation.IsNull() && !myViewAnimation->IsStopped())
|
|
{
|
|
myViewAnimation->UpdateTimer();
|
|
ResetPreviousMoveTo();
|
|
setAskNextFrame();
|
|
}
|
|
|
|
if (!myObjAnimation.IsNull() && !myObjAnimation->IsStopped())
|
|
{
|
|
myObjAnimation->UpdateTimer();
|
|
ResetPreviousMoveTo();
|
|
setAskNextFrame();
|
|
}
|
|
|
|
if (myIsContinuousRedraw)
|
|
{
|
|
myToAskNextFrame = true;
|
|
}
|
|
if (theView->View()->IsActiveXR())
|
|
{
|
|
// VR requires continuous rendering
|
|
myToAskNextFrame = true;
|
|
}
|
|
|
|
for (int aSubViewPass = 0; aSubViewPass < 2; ++aSubViewPass)
|
|
{
|
|
const bool isSubViewPass = (aSubViewPass == 0);
|
|
for (V3d_ListOfViewIterator aViewIter(theView->Viewer()->ActiveViewIterator());
|
|
aViewIter.More();
|
|
aViewIter.Next())
|
|
{
|
|
const Handle(V3d_View)& aView = aViewIter.Value();
|
|
if (isSubViewPass && !aView->IsSubview())
|
|
{
|
|
for (const Handle(V3d_View)& aSubviewIter : aView->Subviews())
|
|
{
|
|
if (aSubviewIter->Viewer() != theView->Viewer())
|
|
{
|
|
if (aSubviewIter->IsInvalidated())
|
|
{
|
|
if (aSubviewIter->ComputedMode())
|
|
{
|
|
aSubviewIter->Update();
|
|
}
|
|
else
|
|
{
|
|
aSubviewIter->Redraw();
|
|
}
|
|
}
|
|
else if (aSubviewIter->IsInvalidatedImmediate())
|
|
{
|
|
aSubviewIter->RedrawImmediate();
|
|
}
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
else if (!isSubViewPass && aView->IsSubview())
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (aView->IsInvalidated() || (myToAskNextFrame && aView == theView))
|
|
{
|
|
if (aView->ComputedMode())
|
|
{
|
|
aView->Update();
|
|
}
|
|
else
|
|
{
|
|
aView->Redraw();
|
|
}
|
|
|
|
if (aView->IsSubview())
|
|
{
|
|
aView->ParentView()->InvalidateImmediate();
|
|
}
|
|
}
|
|
else if (aView->IsInvalidatedImmediate())
|
|
{
|
|
if (aView->IsSubview())
|
|
{
|
|
aView->ParentView()->InvalidateImmediate();
|
|
}
|
|
|
|
aView->RedrawImmediate();
|
|
}
|
|
}
|
|
}
|
|
if (theView->IsSubview() && theView->Viewer() != aParentView->Viewer())
|
|
{
|
|
aParentView->RedrawImmediate();
|
|
}
|
|
|
|
if (myToAskNextFrame)
|
|
{
|
|
// ask more frames
|
|
aParentView->Window()->InvalidateContent(Handle(Aspect_DisplayConnection)());
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
Standard_Integer AIS_ViewController::handleXRMoveTo(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView,
|
|
const gp_Trsf& thePose,
|
|
const Standard_Boolean theToHighlight)
|
|
{
|
|
// ResetPreviousMoveTo();
|
|
const gp_Ax1 aViewAxis = theView->View()->ViewAxisInWorld(thePose);
|
|
Standard_Integer aPickResult = 0;
|
|
if (theToHighlight)
|
|
{
|
|
theCtx->MoveTo(aViewAxis, theView, false);
|
|
if (!theCtx->DetectedOwner().IsNull())
|
|
{
|
|
aPickResult = 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
theCtx->MainSelector()->Pick(aViewAxis, theView);
|
|
if (theCtx->MainSelector()->NbPicked() >= 1)
|
|
{
|
|
aPickResult = 1;
|
|
}
|
|
}
|
|
|
|
return aPickResult;
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRHighlight(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (myXRLastPickingHand != Aspect_XRTrackedDeviceRole_LeftHand
|
|
&& myXRLastPickingHand != Aspect_XRTrackedDeviceRole_RightHand)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Standard_Integer aDeviceId =
|
|
theView->View()->XRSession()->NamedTrackedDevice(myXRLastPickingHand);
|
|
if (aDeviceId == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const Aspect_TrackedDevicePose& aPose = theView->View()->XRSession()->TrackedPoses()[aDeviceId];
|
|
if (!aPose.IsValidPose)
|
|
{
|
|
return;
|
|
}
|
|
|
|
Handle(SelectMgr_EntityOwner) aDetOld = theCtx->DetectedOwner();
|
|
handleXRMoveTo(theCtx, theView, aPose.Orientation, true);
|
|
if (!theCtx->DetectedOwner().IsNull() && theCtx->DetectedOwner() != aDetOld)
|
|
{
|
|
if (const Handle(Aspect_XRAction)& aHaptic =
|
|
theView->View()->XRSession()->GenericAction(myXRLastPickingHand,
|
|
Aspect_XRGenericAction_OutputHaptic))
|
|
{
|
|
theView->View()->XRSession()->TriggerHapticVibrationAction(aHaptic, myXRPickingHaptic);
|
|
}
|
|
}
|
|
|
|
Standard_Real& aPickDepth = myXRLastPickingHand == Aspect_XRTrackedDeviceRole_LeftHand
|
|
? myXRLastPickDepthLeft
|
|
: myXRLastPickDepthRight;
|
|
aPickDepth = Precision::Infinite();
|
|
if (theCtx->MainSelector()->NbPicked() > 0)
|
|
{
|
|
const gp_Trsf aHandBase = theView->View()->PoseXRToWorld(aPose.Orientation);
|
|
const SelectMgr_SortCriterion& aPicked = theCtx->MainSelector()->PickedData(1);
|
|
aPickDepth = aPicked.Point.Distance(aHandBase.TranslationPart());
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::handleXRPresentations(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
if (!theView->View()->IsActiveXR() || (!myToDisplayXRAuxDevices && !myToDisplayXRHands))
|
|
{
|
|
for (NCollection_Array1<Handle(AIS_XRTrackedDevice)>::Iterator aPrsIter(myXRPrsDevices);
|
|
aPrsIter.More();
|
|
aPrsIter.Next())
|
|
{
|
|
if (!aPrsIter.Value().IsNull() && aPrsIter.Value()->HasInteractiveContext())
|
|
{
|
|
theCtx->Remove(aPrsIter.Value(), false);
|
|
}
|
|
aPrsIter.ChangeValue().Nullify();
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (myXRPrsDevices.Length() != theView->View()->XRSession()->TrackedPoses().Length())
|
|
{
|
|
for (NCollection_Array1<Handle(AIS_XRTrackedDevice)>::Iterator aPrsIter(myXRPrsDevices);
|
|
aPrsIter.More();
|
|
aPrsIter.Next())
|
|
{
|
|
if (!aPrsIter.Value().IsNull())
|
|
{
|
|
theCtx->Remove(aPrsIter.Value(), false);
|
|
}
|
|
}
|
|
myXRPrsDevices.Resize(theView->View()->XRSession()->TrackedPoses().Lower(),
|
|
theView->View()->XRSession()->TrackedPoses().Upper(),
|
|
false);
|
|
}
|
|
|
|
const Standard_Integer aHeadDevice =
|
|
theView->View()->XRSession()->NamedTrackedDevice(Aspect_XRTrackedDeviceRole_Head);
|
|
const Standard_Integer aLeftDevice =
|
|
theView->View()->XRSession()->NamedTrackedDevice(Aspect_XRTrackedDeviceRole_LeftHand);
|
|
const Standard_Integer aRightDevice =
|
|
theView->View()->XRSession()->NamedTrackedDevice(Aspect_XRTrackedDeviceRole_RightHand);
|
|
for (Standard_Integer aDeviceIter = theView->View()->XRSession()->TrackedPoses().Lower();
|
|
aDeviceIter <= theView->View()->XRSession()->TrackedPoses().Upper();
|
|
++aDeviceIter)
|
|
{
|
|
const Aspect_TrackedDevicePose& aPose =
|
|
theView->View()->XRSession()->TrackedPoses()[aDeviceIter];
|
|
Handle(AIS_XRTrackedDevice)& aPosePrs = myXRPrsDevices[aDeviceIter];
|
|
if (!aPose.IsValidPose)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const bool isHand = aDeviceIter == aLeftDevice || aDeviceIter == aRightDevice;
|
|
if ((!myToDisplayXRHands && isHand) || (!myToDisplayXRAuxDevices && !isHand))
|
|
{
|
|
if (!aPosePrs.IsNull() && aPosePrs->HasInteractiveContext())
|
|
{
|
|
theCtx->Remove(aPosePrs, false);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
Aspect_XRTrackedDeviceRole aRole = Aspect_XRTrackedDeviceRole_Other;
|
|
if (aDeviceIter == aLeftDevice)
|
|
{
|
|
aRole = Aspect_XRTrackedDeviceRole_LeftHand;
|
|
}
|
|
else if (aDeviceIter == aRightDevice)
|
|
{
|
|
aRole = Aspect_XRTrackedDeviceRole_RightHand;
|
|
}
|
|
|
|
if (!aPosePrs.IsNull() && aPosePrs->UnitFactor() != (float)theView->View()->UnitFactor())
|
|
{
|
|
theCtx->Remove(aPosePrs, false);
|
|
aPosePrs.Nullify();
|
|
}
|
|
|
|
if (aPosePrs.IsNull())
|
|
{
|
|
Handle(Image_Texture) aTexture;
|
|
Handle(Graphic3d_ArrayOfTriangles) aTris;
|
|
if (aDeviceIter != aHeadDevice)
|
|
{
|
|
aTris = theView->View()->XRSession()->LoadRenderModel(aDeviceIter, aTexture);
|
|
}
|
|
if (!aTris.IsNull())
|
|
{
|
|
aPosePrs = new AIS_XRTrackedDevice(aTris, aTexture);
|
|
}
|
|
else
|
|
{
|
|
aPosePrs = new AIS_XRTrackedDevice();
|
|
}
|
|
aPosePrs->SetUnitFactor((float)theView->View()->UnitFactor());
|
|
aPosePrs->SetMutable(true);
|
|
aPosePrs->SetInfiniteState(true);
|
|
}
|
|
aPosePrs->SetRole(aRole);
|
|
|
|
if (!aPosePrs->HasInteractiveContext())
|
|
{
|
|
theCtx->Display(aPosePrs, 0, -1, false);
|
|
}
|
|
|
|
gp_Trsf aPoseLocal = aPose.Orientation;
|
|
if (aDeviceIter == aHeadDevice)
|
|
{
|
|
// show headset position on floor level
|
|
aPoseLocal.SetTranslationPart(
|
|
gp_Vec(aPoseLocal.TranslationPart().X(), 0.0, aPoseLocal.TranslationPart().Z()));
|
|
}
|
|
const gp_Trsf aPoseWorld = theView->View()->PoseXRToWorld(aPoseLocal);
|
|
theCtx->SetLocation(aPosePrs, aPoseWorld);
|
|
|
|
Standard_Real aLaserLen = 0.0;
|
|
if (isHand && aPosePrs->Role() == myXRLastPickingHand)
|
|
{
|
|
aLaserLen = myXRLastPickingHand == Aspect_XRTrackedDeviceRole_LeftHand
|
|
? myXRLastPickDepthLeft
|
|
: myXRLastPickDepthRight;
|
|
if (Precision::IsInfinite(aLaserLen))
|
|
{
|
|
const Bnd_Box aViewBox = theView->View()->MinMaxValues(true);
|
|
if (!aViewBox.IsVoid())
|
|
{
|
|
aLaserLen = Sqrt(aViewBox.SquareExtent());
|
|
}
|
|
else
|
|
{
|
|
aLaserLen = 100.0;
|
|
}
|
|
}
|
|
aPosePrs->SetLaserColor(myXRLaserPickColor);
|
|
}
|
|
else if (isHand && aPosePrs->Role() == myXRLastTeleportHand)
|
|
{
|
|
aLaserLen = myXRLastTeleportHand == Aspect_XRTrackedDeviceRole_LeftHand
|
|
? myXRLastPickDepthLeft
|
|
: myXRLastPickDepthRight;
|
|
if (Precision::IsInfinite(aLaserLen))
|
|
{
|
|
const Bnd_Box aViewBox = theView->View()->MinMaxValues(true);
|
|
if (!aViewBox.IsVoid())
|
|
{
|
|
aLaserLen = Sqrt(aViewBox.SquareExtent());
|
|
}
|
|
else
|
|
{
|
|
aLaserLen = 100.0;
|
|
}
|
|
}
|
|
aPosePrs->SetLaserColor(myXRLaserTeleColor);
|
|
}
|
|
aPosePrs->SetLaserLength((float)aLaserLen);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================
|
|
|
|
void AIS_ViewController::HandleViewEvents(const Handle(AIS_InteractiveContext)& theCtx,
|
|
const Handle(V3d_View)& theView)
|
|
{
|
|
const bool wasImmediateUpdate = theView->SetImmediateUpdate(false);
|
|
|
|
Handle(V3d_View) aPickedView;
|
|
if (theView->IsSubview() || !theView->Subviews().IsEmpty())
|
|
{
|
|
// activate another subview on mouse click
|
|
bool toPickSubview = false;
|
|
Graphic3d_Vec2i aClickPoint;
|
|
if (myGL.Selection.Tool == AIS_ViewSelectionTool_Picking && !myGL.Selection.Points.IsEmpty())
|
|
{
|
|
aClickPoint = myGL.Selection.Points.Last();
|
|
toPickSubview = true;
|
|
}
|
|
else if (!myGL.ZoomActions.IsEmpty())
|
|
{
|
|
// aClickPoint = myGL.ZoomActions.Last().Point;
|
|
// toPickSubview = true;
|
|
}
|
|
|
|
if (toPickSubview)
|
|
{
|
|
if (theView->IsSubview())
|
|
{
|
|
aClickPoint += theView->View()->SubviewTopLeft();
|
|
}
|
|
Handle(V3d_View) aParent = !theView->IsSubview() ? theView : theView->ParentView();
|
|
aPickedView = aParent->PickSubview(aClickPoint);
|
|
}
|
|
}
|
|
|
|
handleViewOrientationKeys(theCtx, theView);
|
|
const AIS_WalkDelta aWalk = handleNavigationKeys(theCtx, theView);
|
|
handleXRInput(theCtx, theView, aWalk);
|
|
if (theView->View()->IsActiveXR())
|
|
{
|
|
theView->View()->SetupXRPosedCamera();
|
|
}
|
|
handleMoveTo(theCtx, theView);
|
|
handleCameraActions(theCtx, theView, aWalk);
|
|
// clang-format off
|
|
theView->View()->SynchronizeXRPosedToBaseCamera(); // handleCameraActions() may modify posed camera position - copy this modifications also to the base camera
|
|
// clang-format on
|
|
handleXRPresentations(theCtx, theView);
|
|
|
|
handleViewRedraw(theCtx, theView);
|
|
theView->View()->UnsetXRPosedCamera();
|
|
|
|
theView->SetImmediateUpdate(wasImmediateUpdate);
|
|
|
|
if (!aPickedView.IsNull() && aPickedView != theView)
|
|
{
|
|
OnSubviewChanged(theCtx, theView, aPickedView);
|
|
}
|
|
|
|
// make sure to not process the same events twice
|
|
myGL.Reset();
|
|
myToAskNextFrame = false;
|
|
}
|