From 5f69cfa70c65f1120e963b54a216b8625268ba22 Mon Sep 17 00:00:00 2001 From: kgv Date: Fri, 22 Oct 2021 15:59:19 +0300 Subject: [PATCH] 0032638: Draw Harness, ViewerTest - HTML input range misbehavior in WebAssembly ViewerTest_EventManager - added tracking of EMSCRIPTEN_EVENT_FOCUSOUT event. onWasmMouseCallback() has been adjusted to return FALSE for EMSCRIPTEN_EVENT_TARGET_WINDOW target to avoid misbehavior of other HTML controls. WNT_Window::ProcessMessage() now handles WM_SETFOCUS/WM_KILLFOCUS instead of WM_ACTIVATE to track focus changes. AIS_ViewController::ProcessFocus() now redirects to AIS_ViewController::ResetViewInput() on focus loss. This fixes issues when key action (like WASD navigation) keep working even after releasing key if window has been switched. --- samples/webgl/WasmOcctView.cpp | 30 +++++++++++++++++++--- samples/webgl/WasmOcctView.h | 6 +++++ samples/webgl/occt-webgl-sample.html | 6 +++++ src/AIS/AIS_ViewController.hxx | 7 +++-- src/ViewerTest/ViewerTest_EventManager.cxx | 19 +++++++++++++- src/WNT/WNT_Window.cxx | 6 ++--- src/Wasm/Wasm_Window.cxx | 29 +++++++++++++++++++++ src/Wasm/Wasm_Window.hxx | 9 +++++++ 8 files changed, 102 insertions(+), 10 deletions(-) diff --git a/samples/webgl/WasmOcctView.cpp b/samples/webgl/WasmOcctView.cpp index e47f395acb..53e0096030 100644 --- a/samples/webgl/WasmOcctView.cpp +++ b/samples/webgl/WasmOcctView.cpp @@ -202,9 +202,12 @@ void WasmOcctView::initWindow() emscripten_set_touchmove_callback (aTargetId, this, toUseCapture, onTouchCallback); emscripten_set_touchcancel_callback(aTargetId, this, toUseCapture, onTouchCallback); - //emscripten_set_keypress_callback (EMSCRIPTEN_EVENT_TARGET_WINDOW, this, toUseCapture, onKeyCallback); - emscripten_set_keydown_callback (EMSCRIPTEN_EVENT_TARGET_WINDOW, this, toUseCapture, onKeyDownCallback); - emscripten_set_keyup_callback (EMSCRIPTEN_EVENT_TARGET_WINDOW, this, toUseCapture, onKeyUpCallback); + //emscripten_set_keypress_callback (aTargetId, this, toUseCapture, onKeyCallback); + emscripten_set_keydown_callback (aTargetId, this, toUseCapture, onKeyDownCallback); + emscripten_set_keyup_callback (aTargetId, this, toUseCapture, onKeyUpCallback); + //emscripten_set_focus_callback (aTargetId, this, toUseCapture, onFocusCallback); + //emscripten_set_focusin_callback (aTargetId, this, toUseCapture, onFocusCallback); + emscripten_set_focusout_callback (aTargetId, this, toUseCapture, onFocusCallback); } // ================================================================ @@ -513,7 +516,8 @@ EM_BOOL WasmOcctView::onMouseEvent (int theEventType, const EmscriptenMouseEvent EmscriptenMouseEvent anEvent = *theEvent; anEvent.targetX -= jsGetBoundingClientLeft(); anEvent.targetY -= jsGetBoundingClientTop(); - return aWindow->ProcessMouseEvent (*this, theEventType, &anEvent) ? EM_TRUE : EM_FALSE; + aWindow->ProcessMouseEvent (*this, theEventType, &anEvent); + return EM_FALSE; } return aWindow->ProcessMouseEvent (*this, theEventType, theEvent) ? EM_TRUE : EM_FALSE; @@ -587,6 +591,24 @@ bool WasmOcctView::navigationKeyModifierSwitch (unsigned int theModifOld, return hasActions; } +// ================================================================ +// Function : onFocusEvent +// Purpose : +// ================================================================ +EM_BOOL WasmOcctView::onFocusEvent (int theEventType, const EmscriptenFocusEvent* theEvent) +{ + if (myView.IsNull() + || (theEventType != EMSCRIPTEN_EVENT_FOCUS + && theEventType != EMSCRIPTEN_EVENT_FOCUSIN // about to receive focus + && theEventType != EMSCRIPTEN_EVENT_FOCUSOUT)) + { + return EM_FALSE; + } + + Handle(Wasm_Window) aWindow = Handle(Wasm_Window)::DownCast (myView->Window()); + return aWindow->ProcessFocusEvent (*this, theEventType, theEvent) ? EM_TRUE : EM_FALSE; +} + // ================================================================ // Function : onKeyDownEvent // Purpose : diff --git a/samples/webgl/WasmOcctView.h b/samples/webgl/WasmOcctView.h index 626ba00d8a..58921a6bed 100644 --- a/samples/webgl/WasmOcctView.h +++ b/samples/webgl/WasmOcctView.h @@ -181,6 +181,9 @@ private: //! Key up event. EM_BOOL onKeyUpEvent (int theEventType, const EmscriptenKeyboardEvent* theEvent); + //! Focus change event. + EM_BOOL onFocusEvent (int theEventType, const EmscriptenFocusEvent* theEvent); + //! @name Emscripten callbacks (static functions) private: @@ -205,6 +208,9 @@ private: static EM_BOOL onKeyUpCallback (int theEventType, const EmscriptenKeyboardEvent* theEvent, void* theView) { return ((WasmOcctView* )theView)->onKeyUpEvent (theEventType, theEvent); } + static EM_BOOL onFocusCallback (int theEventType, const EmscriptenFocusEvent* theEvent, void* theView) + { return ((WasmOcctView* )theView)->onFocusEvent (theEventType, theEvent); } + private: //! Register hot-keys for specified Action. diff --git a/samples/webgl/occt-webgl-sample.html b/samples/webgl/occt-webgl-sample.html index a831e3e1f0..7b48a41c23 100644 --- a/samples/webgl/occt-webgl-sample.html +++ b/samples/webgl/occt-webgl-sample.html @@ -42,6 +42,11 @@ function updateCanvasSize() window.onresize = updateCanvasSize; updateCanvasSize(); +// capture keyboard input on mouse click +occViewerCanvas.tabIndex = -1; +occViewerCanvas.onclick = (theEvent) => { occViewerCanvas.focus() }; +occViewerCanvas.focus(); + //! Check browser support. function isWasmSupported() { @@ -78,6 +83,7 @@ fileInput.onchange = function() OccViewerModule.openFromMemory (aFile.name, aDataBuffer, aDataArray.length, true); //OccViewerModule._free (aDataBuffer); will be freed by called method OccViewerModule.displayGround (true); + occViewerCanvas.focus(); }; aReader.readAsArrayBuffer(aFile); }; diff --git a/src/AIS/AIS_ViewController.hxx b/src/AIS/AIS_ViewController.hxx index 1437b20fda..8ab8fd23a8 100644 --- a/src/AIS/AIS_ViewController.hxx +++ b/src/AIS/AIS_ViewController.hxx @@ -441,10 +441,13 @@ public: //! @name resize events virtual void ProcessInput() Standard_OVERRIDE {} //! Handle focus event. - //! Default implementation does nothing. + //! Default implementation resets cached input state (pressed keys). virtual void ProcessFocus (bool theIsActivated) Standard_OVERRIDE { - (void )theIsActivated; + if (!theIsActivated) + { + ResetViewInput(); + } } //! Handle window close event. diff --git a/src/ViewerTest/ViewerTest_EventManager.cxx b/src/ViewerTest/ViewerTest_EventManager.cxx index 7ae3a25c74..3335af811f 100644 --- a/src/ViewerTest/ViewerTest_EventManager.cxx +++ b/src/ViewerTest/ViewerTest_EventManager.cxx @@ -655,7 +655,8 @@ static EM_BOOL onWasmMouseCallback (int theEventType, const EmscriptenMouseEvent EmscriptenMouseEvent anEvent = *theEvent; anEvent.targetX -= occJSGetBoundingClientLeft(); anEvent.targetY -= occJSGetBoundingClientTop(); - return aWindow->ProcessMouseEvent (*aViewCtrl, theEventType, &anEvent) ? EM_TRUE : EM_FALSE; + aWindow->ProcessMouseEvent (*aViewCtrl, theEventType, &anEvent); + return EM_FALSE; } return aWindow->ProcessMouseEvent (*aViewCtrl, theEventType, theEvent) ? EM_TRUE : EM_FALSE; @@ -702,6 +703,19 @@ static EM_BOOL onWasmKeyCallback (int theEventType, const EmscriptenKeyboardEven } return EM_FALSE; } + +//! Handle focus change event. +static EM_BOOL onWasmFocusCallback (int theEventType, const EmscriptenFocusEvent* theEvent, void*) +{ + Handle(ViewerTest_EventManager) aViewCtrl = ViewerTest::CurrentEventManager(); + if (!aViewCtrl.IsNull() + && !ViewerTest::CurrentView().IsNull()) + { + Handle(Wasm_Window) aWindow = Handle(Wasm_Window)::DownCast (ViewerTest::CurrentView()->Window()); + return aWindow->ProcessFocusEvent (*aViewCtrl, theEventType, theEvent) ? EM_TRUE : EM_FALSE; + } + return EM_FALSE; +} #endif // ============================================================================== @@ -777,6 +791,9 @@ void ViewerTest_EventManager::SetupWindowCallbacks (const Handle(Aspect_Window)& // keyboard input requires a focusable element or EMSCRIPTEN_EVENT_TARGET_WINDOW emscripten_set_keydown_callback (aTargetId, anOpaque, toUseCapture, onWasmKeyCallback); emscripten_set_keyup_callback (aTargetId, anOpaque, toUseCapture, onWasmKeyCallback); + //emscripten_set_focus_callback (aTargetId, anOpaque, toUseCapture, onWasmFocusCallback); + //emscripten_set_focusin_callback (aTargetId, anOpaque, toUseCapture, onWasmFocusCallback); + emscripten_set_focusout_callback (aTargetId, anOpaque, toUseCapture, onWasmFocusCallback); #else (void )theWin; #endif diff --git a/src/WNT/WNT_Window.cxx b/src/WNT/WNT_Window.cxx index 73ebe522ef..10edc7a242 100644 --- a/src/WNT/WNT_Window.cxx +++ b/src/WNT/WNT_Window.cxx @@ -815,12 +815,12 @@ bool WNT_Window::ProcessMessage (Aspect_WindowInputListener& theListener, } return false; } - case WM_ACTIVATE: + case WM_SETFOCUS: + case WM_KILLFOCUS: { if (theMsg.hwnd == (HWND )myHWindow) { - theListener.ProcessFocus (LOWORD(theMsg.wParam) == WA_CLICKACTIVE - || LOWORD(theMsg.wParam) == WA_ACTIVE); + theListener.ProcessFocus (theMsg.message == WM_SETFOCUS); return true; } return false; diff --git a/src/Wasm/Wasm_Window.cxx b/src/Wasm/Wasm_Window.cxx index 19100a4b7c..985333b634 100644 --- a/src/Wasm/Wasm_Window.cxx +++ b/src/Wasm/Wasm_Window.cxx @@ -243,6 +243,12 @@ bool Wasm_Window::ProcessMessage (Aspect_WindowInputListener& theListener, { return ProcessUiEvent (theListener, theEventType, (const EmscriptenUiEvent* )theEvent); } + case EMSCRIPTEN_EVENT_FOCUS: + case EMSCRIPTEN_EVENT_FOCUSIN: + case EMSCRIPTEN_EVENT_FOCUSOUT: + { + return ProcessFocusEvent (theListener, theEventType, (const EmscriptenFocusEvent* )theEvent); + } } return false; #else @@ -536,6 +542,29 @@ bool Wasm_Window::ProcessUiEvent (Aspect_WindowInputListener& theListener, return true; } +// ======================================================================= +// function : ProcessFocusEvent +// purpose : +// ======================================================================= +bool Wasm_Window::ProcessFocusEvent (Aspect_WindowInputListener& theListener, + int theEventType, const EmscriptenFocusEvent* ) +{ + bool isActivated = false; +#if defined(__EMSCRIPTEN__) + if (theEventType != EMSCRIPTEN_EVENT_FOCUS + && theEventType != EMSCRIPTEN_EVENT_FOCUSIN // about to receive focus + && theEventType != EMSCRIPTEN_EVENT_FOCUSOUT) + { + return false; + } + isActivated = theEventType == EMSCRIPTEN_EVENT_FOCUS; +#else + (void )theEventType; +#endif + theListener.ProcessFocus (isActivated); + return true; +} + // ======================================================================= // function : MouseButtonsFromNative // purpose : diff --git a/src/Wasm/Wasm_Window.hxx b/src/Wasm/Wasm_Window.hxx index 62b00b3752..5043f83cdd 100644 --- a/src/Wasm/Wasm_Window.hxx +++ b/src/Wasm/Wasm_Window.hxx @@ -28,6 +28,7 @@ struct EmscriptenWheelEvent; struct EmscriptenTouchEvent; struct EmscriptenKeyboardEvent; struct EmscriptenUiEvent; +struct EmscriptenFocusEvent; //! This class defines WebAssembly window (HTML5 canvas) intended for creation of OpenGL (WebGL) context. //! @@ -175,6 +176,14 @@ public: Standard_EXPORT virtual bool ProcessUiEvent (Aspect_WindowInputListener& theListener, int theEventType, const EmscriptenUiEvent* theEvent); + //! Process a focus input change message. + //! @param[in,out] theListener listener to redirect message + //! @param[in] theEventType message type to process + //! @param[in] theEvent message to process + //! @return TRUE if message has been processed + Standard_EXPORT virtual bool ProcessFocusEvent (Aspect_WindowInputListener& theListener, + int theEventType, const EmscriptenFocusEvent* theEvent); + protected: TCollection_AsciiString myCanvasId;