1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00

0032723: Draw Harness, ViewerTest - sloppy animation in WebGL

Fixed emscripten_async_call() calls to use requestAnimationFrame() instead of setTimeout().
ViewerTest_EventManager::handleViewRedraw() - fixed queuing new onWasmRedrawView()
before previous one has been processed (leading to multiplying of pending redraws executed at the same time).
RWGltf_TriangulationReader::readDracoBuffer - suppressed CLang compilation warnings.
This commit is contained in:
kgv 2021-12-07 20:07:04 +03:00 committed by smoskvin
parent f4f9ce4b77
commit e3dae4a9f3
6 changed files with 44 additions and 23 deletions

View File

@ -114,7 +114,7 @@ WasmOcctView& WasmOcctView::Instance()
// ================================================================ // ================================================================
WasmOcctView::WasmOcctView() WasmOcctView::WasmOcctView()
: myDevicePixelRatio (1.0f), : myDevicePixelRatio (1.0f),
myUpdateRequests (0) myNbUpdateRequests (0)
{ {
addActionHotKeys (Aspect_VKey_NavForward, Aspect_VKey_W, Aspect_VKey_W | Aspect_VKeyFlags_SHIFT); addActionHotKeys (Aspect_VKey_NavForward, Aspect_VKey_W, Aspect_VKey_W | Aspect_VKeyFlags_SHIFT);
addActionHotKeys (Aspect_VKey_NavBackward , Aspect_VKey_S, Aspect_VKey_S | Aspect_VKeyFlags_SHIFT); addActionHotKeys (Aspect_VKey_NavBackward , Aspect_VKey_S, Aspect_VKey_S | Aspect_VKeyFlags_SHIFT);
@ -395,9 +395,11 @@ void WasmOcctView::ProcessInput()
// Queue onRedrawView()/redrawView callback to redraw canvas after all user input is flushed by browser. // Queue onRedrawView()/redrawView callback to redraw canvas after all user input is flushed by browser.
// Redrawing viewer on every single message would be a pointless waste of resources, // Redrawing viewer on every single message would be a pointless waste of resources,
// as user will see only the last drawn frame due to WebGL implementation details. // as user will see only the last drawn frame due to WebGL implementation details.
if (++myUpdateRequests == 1) // -1 in emscripten_async_call() redirects to requestAnimationFrame();
// requestPostAnimationFrame() is a better under development alternative.
if (++myNbUpdateRequests == 1)
{ {
emscripten_async_call (onRedrawView, this, 0); emscripten_async_call (onRedrawView, this, -1);
} }
} }
} }
@ -424,6 +426,7 @@ void WasmOcctView::redrawView()
{ {
if (!myView.IsNull()) if (!myView.IsNull())
{ {
myNbUpdateRequests = 0;
FlushViewEvents (myContext, myView, true); FlushViewEvents (myContext, myView, true);
} }
} }
@ -435,13 +438,14 @@ void WasmOcctView::redrawView()
void WasmOcctView::handleViewRedraw (const Handle(AIS_InteractiveContext)& theCtx, void WasmOcctView::handleViewRedraw (const Handle(AIS_InteractiveContext)& theCtx,
const Handle(V3d_View)& theView) const Handle(V3d_View)& theView)
{ {
myUpdateRequests = 0;
AIS_ViewController::handleViewRedraw (theCtx, theView); AIS_ViewController::handleViewRedraw (theCtx, theView);
if (myToAskNextFrame) if (myToAskNextFrame)
{ {
// ask more frames // ask more frames
++myUpdateRequests; if (++myNbUpdateRequests == 1)
emscripten_async_call (onRedrawView, this, 0); {
emscripten_async_call (onRedrawView, this, -1);
}
} }
} }

View File

@ -249,7 +249,7 @@ private:
TCollection_AsciiString myCanvasId; //!< canvas element id on HTML page TCollection_AsciiString myCanvasId; //!< canvas element id on HTML page
Graphic3d_Vec2i myWinSizeOld; Graphic3d_Vec2i myWinSizeOld;
float myDevicePixelRatio; //!< device pixel ratio for handling high DPI displays float myDevicePixelRatio; //!< device pixel ratio for handling high DPI displays
unsigned int myUpdateRequests; //!< counter for unhandled update requests unsigned int myNbUpdateRequests; //!< counter for unhandled update requests
}; };

View File

@ -224,7 +224,7 @@ bool RWGltf_TriangulationReader::readDracoBuffer (const Handle(RWGltf_GltfLatePr
#ifdef HAVE_DRACO #ifdef HAVE_DRACO
std::vector<char> aReadData; std::vector<char> aReadData;
aReadData.resize (theGltfData.StreamLength); aReadData.resize ((size_t )theGltfData.StreamLength);
aSharedStream->read (aReadData.data(), (std::streamsize )theGltfData.StreamLength); aSharedStream->read (aReadData.data(), (std::streamsize )theGltfData.StreamLength);
if (!aSharedStream->good()) if (!aSharedStream->good())
{ {

View File

@ -31,6 +31,7 @@
#pragma clang diagnostic push #pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wall" #pragma clang diagnostic ignored "-Wall"
#pragma clang diagnostic ignored "-Wextra" #pragma clang diagnostic ignored "-Wextra"
#pragma clang diagnostic ignored "-Wshorten-64-to-32"
#elif defined(_MSC_VER) #elif defined(_MSC_VER)
#pragma warning(push, 0) #pragma warning(push, 0)
#elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)) #elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))

View File

@ -38,17 +38,25 @@
#include <emscripten.h> #include <emscripten.h>
#include <emscripten/html5.h> #include <emscripten/html5.h>
//! Callback flushing events and redrawing the WebGL canvas. //=======================================================================
static void onWasmRedrawView (void* ) //function : onWasmRedrawView
//purpose :
//=======================================================================
void ViewerTest_EventManager::onWasmRedrawView (void* )
{
Handle(ViewerTest_EventManager) aViewCtrl = ViewerTest::CurrentEventManager();
if (!aViewCtrl.IsNull())
{ {
Handle(ViewerTest_EventManager) aViewCtrl = ViewerTest::CurrentEventManager(); aViewCtrl->myNbUpdateRequests = 0;
const Handle(V3d_View)& aView = ViewerTest::CurrentView(); const Handle(V3d_View)& aView = ViewerTest::CurrentView();
const Handle(AIS_InteractiveContext)& aCtx = ViewerTest::GetAISContext(); const Handle(AIS_InteractiveContext)& aCtx = ViewerTest::GetAISContext();
if (!aViewCtrl.IsNull() && !aView.IsNull() && !aCtx.IsNull()) if (!aView.IsNull() && !aCtx.IsNull())
{ {
aViewCtrl->ProcessExpose(); aViewCtrl->ProcessExpose();
} }
} }
}
#endif #endif
Standard_IMPORT Standard_Boolean Draw_Interprete (const char* theCommand); Standard_IMPORT Standard_Boolean Draw_Interprete (const char* theCommand);
@ -76,7 +84,7 @@ ViewerTest_EventManager::ViewerTest_EventManager (const Handle(V3d_View)&
myView (theView), myView (theView),
myToPickPnt (Standard_False), myToPickPnt (Standard_False),
myIsTmpContRedraw (Standard_False), myIsTmpContRedraw (Standard_False),
myUpdateRequests (0) myNbUpdateRequests (0)
{ {
myViewAnimation = GlobalViewAnimation(); myViewAnimation = GlobalViewAnimation();
@ -183,7 +191,6 @@ void ViewerTest_EventManager::ProcessExpose()
void ViewerTest_EventManager::handleViewRedraw (const Handle(AIS_InteractiveContext)& theCtx, void ViewerTest_EventManager::handleViewRedraw (const Handle(AIS_InteractiveContext)& theCtx,
const Handle(V3d_View)& theView) const Handle(V3d_View)& theView)
{ {
myUpdateRequests = 0;
AIS_ViewController::handleViewRedraw (theCtx, theView); AIS_ViewController::handleViewRedraw (theCtx, theView);
// On non-Windows platforms Aspect_Window::InvalidateContent() from rendering thread does not work as expected // On non-Windows platforms Aspect_Window::InvalidateContent() from rendering thread does not work as expected
@ -202,10 +209,12 @@ void ViewerTest_EventManager::handleViewRedraw (const Handle(AIS_InteractiveCont
} }
// ask more frames // ask more frames
++myUpdateRequests; if (++myNbUpdateRequests == 1)
#if defined(__EMSCRIPTEN__) {
emscripten_async_call (onWasmRedrawView, this, 0); #if defined(__EMSCRIPTEN__)
#endif emscripten_async_call (onWasmRedrawView, this, -1);
#endif
}
} }
else if (myIsTmpContRedraw) else if (myIsTmpContRedraw)
{ {
@ -256,11 +265,11 @@ void ViewerTest_EventManager::ProcessInput()
// Queue onWasmRedrawView() callback to redraw canvas after all user input is flushed by browser. // Queue onWasmRedrawView() callback to redraw canvas after all user input is flushed by browser.
// Redrawing viewer on every single message would be a pointless waste of resources, // Redrawing viewer on every single message would be a pointless waste of resources,
// as user will see only the last drawn frame due to WebGL implementation details. // as user will see only the last drawn frame due to WebGL implementation details.
if (++myUpdateRequests == 1) // -1 in emscripten_async_call() redirects to requestAnimationFrame();
// requestPostAnimationFrame() is a better under development alternative.
if (++myNbUpdateRequests == 1)
{ {
#if defined(__EMSCRIPTEN__) emscripten_async_call (onWasmRedrawView, this, -1);
emscripten_async_call (onWasmRedrawView, this, 0);
#endif
} }
#else #else
// handle synchronously // handle synchronously

View File

@ -138,6 +138,13 @@ protected:
unsigned int theModifNew, unsigned int theModifNew,
double theTimeStamp); double theTimeStamp);
private:
#if defined(__EMSCRIPTEN__)
//! Callback flushing events and redrawing the WebGL canvas.
static void onWasmRedrawView (void* );
#endif
private: private:
Handle(AIS_InteractiveContext) myCtx; Handle(AIS_InteractiveContext) myCtx;
@ -148,7 +155,7 @@ private:
Standard_Boolean myToPickPnt; Standard_Boolean myToPickPnt;
Standard_Boolean myIsTmpContRedraw; Standard_Boolean myIsTmpContRedraw;
unsigned int myUpdateRequests; //!< counter for unhandled update requests unsigned int myNbUpdateRequests; //!< counter for unhandled update requests
}; };