From 5de4b704fef08f565012f3e087afa9b514eb4eed Mon Sep 17 00:00:00 2001 From: kgv Date: Mon, 18 Jan 2021 20:50:52 +0300 Subject: [PATCH] 0032065: Samples - use MODULARIZE within WebGL sample Fixed multitouch input. Module is now exported with global functions hidden via MODULARIZE as global object OccViewerModule created by createOccViewerModule(). Global Module setup has been moved to occt-webgl-viewer.js. Use EMSCRIPTEN_KEEPALIVE attribute istead of listing C functions via EXTRA_EXPORTED_RUNTIME_METHODS. WasmOcctView now exports static methods as Module functions using EMSCRIPTEN_BINDINGS. Standard_ASSERT_DBGBREAK_() is now defined using emscripten_debugger(). --- adm/scripts/ios_build.sh | 2 +- adm/scripts/wasm_build.bat | 30 ++- adm/scripts/wasm_custom.bat.template | 8 +- adm/scripts/wasm_sample_build.bat | 22 +- samples/webgl/CMakeLists.txt | 23 +- samples/webgl/WasmOcctPixMap.cpp | 75 ++++++ samples/webgl/WasmOcctPixMap.h | 47 ++++ samples/webgl/WasmOcctView.cpp | 373 ++++++++++++++++++++++++++- samples/webgl/WasmOcctView.h | 71 +++++ samples/webgl/main.cpp | 49 +--- samples/webgl/occt-webgl-sample.html | 49 ++-- samples/webgl/occt-webgl-viewer.js | 28 ++ src/DrawResources/OCC_logo.png | Bin 7532 -> 4728 bytes src/Standard/Standard_Assert.hxx | 3 + 14 files changed, 675 insertions(+), 105 deletions(-) create mode 100644 samples/webgl/WasmOcctPixMap.cpp create mode 100644 samples/webgl/WasmOcctPixMap.h create mode 100644 samples/webgl/occt-webgl-viewer.js diff --git a/adm/scripts/ios_build.sh b/adm/scripts/ios_build.sh index 840da17d22..7de7c13d79 100755 --- a/adm/scripts/ios_build.sh +++ b/adm/scripts/ios_build.sh @@ -110,7 +110,7 @@ if [[ $toCMake == 1 ]]; then -D ENABLE_VISIBILITY:BOOL="TRUE" \ -D CMAKE_C_USE_RESPONSE_FILE_FOR_OBJECTS:BOOL="OFF" \ -D CMAKE_CXX_USE_RESPONSE_FILE_FOR_OBJECTS:BOOL="OFF" \ - -D CMAKE_BUILD_TYPE:STRING="Release" \ + -D CMAKE_BUILD_TYPE:STRING="$aBuildType" \ -D BUILD_LIBRARY_TYPE:STRING="$aLibType" \ -D INSTALL_DIR:PATH="$aDestDir" \ -D INSTALL_DIR_INCLUDE:STRING="inc" \ diff --git a/adm/scripts/wasm_build.bat b/adm/scripts/wasm_build.bat index c3ee490a66..6581b7eaef 100644 --- a/adm/scripts/wasm_build.bat +++ b/adm/scripts/wasm_build.bat @@ -20,7 +20,9 @@ set "toClean=0" set "toMake=1" set "toInstall=1" set "toPack=0" +set "toDebug=0" set "toBuildSample=0" +set "sourceMapBase=" rem OCCT Modules to build set "BUILD_ModelingData=ON" @@ -50,6 +52,13 @@ for /f tokens^=2^ delims^=^" %%i in ('findstr /b /c:"#define OCC_VERSION_DEVELOP for /f tokens^=2^ delims^=^" %%i in ('findstr /b /c:"#define OCC_VERSION_COMPLETE" "%aCasSrc%\src\Standard\Standard_Version.hxx"') do ( set "anOcctVersion=%%i" ) for /f %%i in ('git symbolic-ref --short HEAD') do ( set "aGitBranch=%%i" ) +set "aBuildType=Release" +set "aBuildTypePrefix=" +if ["%toDebug%"] == ["1"] ( + set "aBuildType=Debug" + set "aBuildTypePrefix=-debug" +) + call :cmakeGenerate if errorlevel 1 ( if not ["%1"] == ["-nopause"] ( @@ -72,7 +81,7 @@ set DAY00=%DAY00:~-2% set MONTH00=%MONTH00:~-2% set "aRevision=-%YEAR%-%MONTH00%-%DAY00%" rem set "aRevision=-%aGitBranch%" -set "anArchName=occt-%anOcctVersion%%anOcctVerSuffix%%aRevision%-wasm32" +set "anArchName=occt-%anOcctVersion%%anOcctVerSuffix%%aRevision%-wasm32%aBuildTypePrefix%" set "aTarget=%aBuildRoot%\%anArchName%" if ["%toPack%"] == ["1"] ( echo Creating archive %anArchName%.7z @@ -95,10 +104,10 @@ if not ["%1"] == ["-nopause"] ( goto :eof :cmakeGenerate -set "aPlatformAndCompiler=wasm32" -set "aWorkDir=%aBuildRoot%\%aPlatformAndCompiler%-make" -set "aDestDir=%aBuildRoot%\%aPlatformAndCompiler%" -set "aLogFile=%aBuildRoot%\build-%aPlatformAndCompiler%.log" +set "aPlatformAndCompiler=wasm32%aBuildTypePrefix%" +set "aWorkDir=%aBuildRoot%\occt-%aPlatformAndCompiler%-make" +set "aDestDir=%aBuildRoot%\occt-%aPlatformAndCompiler%" +set "aLogFile=%aBuildRoot%\occt-%aPlatformAndCompiler%-build.log" if ["%toCMake%"] == ["1"] ( if ["%toClean%"] == ["1"] ( rmdir /S /Q %aWorkDir%" @@ -109,9 +118,9 @@ if not exist "%aWorkDir%" ( mkdir "%aWorkDir%" ) if exist "%aLogFile%" ( del "%aLogFile%" ) set "aSrcRootSmpl=%aCasSrc%\samples\webgl" -set "aWorkDirSmpl=%aBuildRoot%\%aPlatformAndCompiler%-sample-make" -set "aDestDirSmpl=%aBuildRoot%\%aPlatformAndCompiler%-sample" -set "aLogFileSmpl=%aBuildRoot%\build-%aPlatformAndCompiler%-sample.log" +set "aWorkDirSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%-make" +set "aDestDirSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%" +set "aLogFileSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%-build.log" if ["%toBuildSample%"] == ["1"] ( if ["%toCMake%"] == ["1"] ( if ["%toClean%"] == ["1"] ( @@ -139,7 +148,7 @@ if ["%toCMake%"] == ["1"] ( echo Configuring OCCT for WASM... cmake -G "MinGW Makefiles" ^ -D CMAKE_TOOLCHAIN_FILE:FILEPATH="%aToolchain%" ^ - -D CMAKE_BUILD_TYPE:STRING="Release" ^ + -D CMAKE_BUILD_TYPE:STRING="%aBuildType%" ^ -D BUILD_LIBRARY_TYPE:STRING="Static" ^ -D INSTALL_DIR:PATH="%aDestDir%" ^ -D INSTALL_DIR_INCLUDE:STRING="inc" ^ @@ -226,8 +235,9 @@ pushd "%aWorkDirSmpl%" if ["%toCMake%"] == ["1"] ( cmake -G "MinGW Makefiles" ^ -D CMAKE_TOOLCHAIN_FILE:FILEPATH="%aToolchain%" ^ - -D CMAKE_BUILD_TYPE:STRING="Release" ^ + -D CMAKE_BUILD_TYPE:STRING="%aBuildType%" ^ -D CMAKE_INSTALL_PREFIX:PATH="%aDestDirSmpl%" ^ + -D SOURCE_MAP_BASE:STRING="%sourceMapBase%" ^ -D OpenCASCADE_DIR:PATH="%aDestDir%/lib/cmake/opencascade" ^ -D freetype_DIR:PATH="%aFreeType%/lib/cmake/freetype" ^ "%aSrcRootSmpl%" diff --git a/adm/scripts/wasm_custom.bat.template b/adm/scripts/wasm_custom.bat.template index a9ef56f60d..1b07aa7144 100644 --- a/adm/scripts/wasm_custom.bat.template +++ b/adm/scripts/wasm_custom.bat.template @@ -7,11 +7,15 @@ rem set "aCmakeBin=%ProgramW6432%\CMake\bin" rem Uncomment to customize building steps rem set "aBuildRoot=work" rem set "toCMake=1" -rem set "toClean=0" +rem set "toClean=1" rem set "toMake=1" rem set "toInstall=1" -rem set "toPack=0" +rem set "toPack=1" +rem set "toDebug=1" rem set "toBuildSample=1" +rem Source map base (should point to server where C++ sources will be copied) +rem enables -g4 debug building option for WebGL sample and allows navigating C++ source code within JavaScript debugger. +rem set "sourceMapBase=http://localhost:9090/" rem set "BUILD_ModelingData=ON" rem set "BUILD_ModelingAlgorithms=ON" diff --git a/adm/scripts/wasm_sample_build.bat b/adm/scripts/wasm_sample_build.bat index 84c917e33f..fd9a33b301 100644 --- a/adm/scripts/wasm_sample_build.bat +++ b/adm/scripts/wasm_sample_build.bat @@ -18,6 +18,8 @@ set "toCMake=1" set "toClean=0" set "toMake=1" set "toInstall=1" +set "toDebug=0" +set "sourceMapBase=" rem Configuration file if exist "%~dp0wasm_custom.bat" call "%~dp0wasm_custom.bat" @@ -26,6 +28,13 @@ call "%EMSDK_ROOT%\emsdk_env.bat" set "aToolchain=%EMSDK%/upstream/emscripten/cmake/Modules/Platform/Emscripten.cmake" if not ["%aCmakeBin%"] == [""] ( set "PATH=%aCmakeBin%;%PATH%" ) +set "aBuildType=Release" +set "aBuildTypePrefix=" +if ["%toDebug%"] == ["1"] ( + set "aBuildType=Debug" + set "aBuildTypePrefix=-debug" +) + call :cmakeGenerate if not ["%1"] == ["-nopause"] ( pause @@ -34,12 +43,12 @@ if not ["%1"] == ["-nopause"] ( goto :eof :cmakeGenerate -set "aPlatformAndCompiler=wasm32" -set "aDestDirOcct=%aBuildRoot%\%aPlatformAndCompiler%" +set "aPlatformAndCompiler=wasm32%aBuildTypePrefix%" +set "aDestDirOcct=%aBuildRoot%\occt-%aPlatformAndCompiler%" set "aSrcRootSmpl=%aCasSrc%\samples\webgl" -set "aWorkDirSmpl=%aBuildRoot%\%aPlatformAndCompiler%-sample-make" -set "aDestDirSmpl=%aBuildRoot%\%aPlatformAndCompiler%-sample" -set "aLogFileSmpl=%aBuildRoot%\build-%aPlatformAndCompiler%-sample.log" +set "aWorkDirSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%-make" +set "aDestDirSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%" +set "aLogFileSmpl=%aBuildRoot%\sample-%aPlatformAndCompiler%-build.log" if ["%toCMake%"] == ["1"] ( if ["%toClean%"] == ["1"] ( rmdir /S /Q %aWorkDirSmpl%" @@ -54,8 +63,9 @@ pushd "%aWorkDirSmpl%" if ["%toCMake%"] == ["1"] ( cmake -G "MinGW Makefiles" ^ -D CMAKE_TOOLCHAIN_FILE:FILEPATH="%aToolchain%" ^ - -D CMAKE_BUILD_TYPE:STRING="Release" ^ + -D CMAKE_BUILD_TYPE:STRING="%aBuildType%" ^ -D CMAKE_INSTALL_PREFIX:PATH="%aDestDirSmpl%" ^ + -D SOURCE_MAP_BASE:STRING="%sourceMapBase%" ^ -D OpenCASCADE_DIR:PATH="%aDestDirOcct%/lib/cmake/opencascade" ^ -D freetype_DIR:PATH="%aFreeType%/lib/cmake/freetype" ^ "%aSrcRootSmpl%" diff --git a/samples/webgl/CMakeLists.txt b/samples/webgl/CMakeLists.txt index ee35d9d828..bdcb29aa7b 100644 --- a/samples/webgl/CMakeLists.txt +++ b/samples/webgl/CMakeLists.txt @@ -7,16 +7,29 @@ set(APP_VERSION_MAJOR 1) set(APP_VERSION_MINOR 0) set(APP_TARGET occt-webgl-sample) +# option to enable or disable use of precompiled headers +if (NOT DEFINED SOURCE_MAP_BASE) + set (SOURCE_MAP_BASE "" CACHE STRING "Path to source map server for debugging C++ code; e.g. http://localhost:9090/") +endif() + # customize build set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s WASM=1") set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s USE_WEBGL2=1") set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s ALLOW_MEMORY_GROWTH=1") +set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} --bind") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s SAFE_HEAP=1") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s NO_EXIT_RUNTIME=1") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s TOTAL_MEMORY=16MB") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s ABORTING_MALLOC=0") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -s FORCE_FILESYSTEM=1") #set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} --preload-file myFile") +if (NOT "${SOURCE_MAP_BASE}" STREQUAL "") + set(CMAKE_CXX_FLAGS_DEBUG "-g4 --source-map-base ${SOURCE_MAP_BASE}") +endif() +set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -s MODULARIZE=1") +set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -s EXPORT_NAME='createOccViewerModule'") +set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -s EXTRA_EXPORTED_RUNTIME_METHODS=['ccall','cwrap']") +set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} --extern-post-js ${CMAKE_CURRENT_SOURCE_DIR}/occt-webgl-viewer.js") INCLUDE_DIRECTORIES(${PROJECT_SOURCE_DIR}) file(GLOB SOURCES @@ -24,9 +37,11 @@ file(GLOB SOURCES *.cpp ) source_group ("Headers" FILES - WasmOcctView.h) + WasmOcctView.h + WasmOcctPixMap.h) source_group ("Sources" FILES WasmOcctView.cpp + WasmOcctPixMap.cpp main.cpp) # FreeType @@ -57,10 +72,14 @@ target_link_libraries( ${OpenCASCADE_LIBS} freetype ) -set_target_properties(${APP_TARGET} PROPERTIES LINK_FLAGS "-s EXPORTED_FUNCTIONS=['_main','_onFileDataRead'] -s EXTRA_EXPORTED_RUNTIME_METHODS=['ccall','cwrap']") install(TARGETS ${PROJECT_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" LIBRARY DESTINATION "${CMAKE_INSTALL_PREFIX}") install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_NAME}.wasm DESTINATION ${CMAKE_INSTALL_PREFIX}) +if (NOT "${SOURCE_MAP_BASE}" STREQUAL "") + if ("${CMAKE_BUILD_TYPE}" STREQUAL "Debug") + install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_NAME}.wasm.map DESTINATION ${CMAKE_INSTALL_PREFIX}) + endif() +endif() install(FILES occt-webgl-sample.html DESTINATION ${CMAKE_INSTALL_PREFIX}) install(FILES ${OpenCASCADE_RESOURCE_DIR}/DrawResources/OCC_logo.png DESTINATION ${CMAKE_INSTALL_PREFIX}) install(FILES ${OpenCASCADE_RESOURCE_DIR}/DrawResources/lamp.ico DESTINATION ${CMAKE_INSTALL_PREFIX}) diff --git a/samples/webgl/WasmOcctPixMap.cpp b/samples/webgl/WasmOcctPixMap.cpp new file mode 100644 index 0000000000..7a39a26741 --- /dev/null +++ b/samples/webgl/WasmOcctPixMap.cpp @@ -0,0 +1,75 @@ +// Copyright (c) 2021 OPEN CASCADE SAS +// +// This file is part of the examples of the Open CASCADE Technology software library. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE + +#include "WasmOcctPixMap.h" + +#include + +#include + +// ================================================================ +// Function : WasmOcctPixMap +// Purpose : +// ================================================================ +WasmOcctPixMap::WasmOcctPixMap() +: myRawDataPtr (nullptr) {} + +// ================================================================ +// Function : ~WasmOcctPixMap +// Purpose : +// ================================================================ +WasmOcctPixMap::~WasmOcctPixMap() +{ + Clear(); +} + +// ================================================================ +// Function : Clear +// Purpose : +// ================================================================ +void WasmOcctPixMap::Clear() +{ + if (myRawDataPtr != nullptr) { free (myRawDataPtr); } + myRawDataPtr = nullptr; + Image_PixMap::Clear(); +} + +// ================================================================ +// Function : Init +// Purpose : +// ================================================================ +bool WasmOcctPixMap::Init (const char* theFilePath) +{ + Clear(); + int aSizeX = 0, aSizeY = 0; + char* anImgData = emscripten_get_preloaded_image_data (theFilePath, &aSizeX, &aSizeY); + if (anImgData == nullptr) + { + Message::DefaultMessenger()->Send (TCollection_AsciiString("Error: invalid image ") + theFilePath, Message_Fail); + return false; + } + + Message::DefaultMessenger()->Send (TCollection_AsciiString("Loaded image ") + theFilePath + "@" + aSizeX + "x" + aSizeY, Message_Info); + InitWrapper (Image_Format_RGBA, (Standard_Byte* )anImgData, aSizeX, aSizeY); + SetTopDown (true); + myRawDataPtr = anImgData; + return true; +} diff --git a/samples/webgl/WasmOcctPixMap.h b/samples/webgl/WasmOcctPixMap.h new file mode 100644 index 0000000000..6cd0faa994 --- /dev/null +++ b/samples/webgl/WasmOcctPixMap.h @@ -0,0 +1,47 @@ +// Copyright (c) 2021 OPEN CASCADE SAS +// +// This file is part of the examples of the Open CASCADE Technology software library. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE + +#ifndef _WasmOcctPixMap_HeaderFile +#define _WasmOcctPixMap_HeaderFile + +#include + +//! Image pixmap loading image using Emscripten. +class WasmOcctPixMap : public Image_PixMap +{ +public: + //! Empty constructor. + WasmOcctPixMap(); + + //! Destructor. + virtual ~WasmOcctPixMap(); + + //! Load RGBA pixmap using emscripten_get_preloaded_image_data() from the given path. + bool Init (const char* theFilePath); + + //! Release memory. + virtual void Clear() override; + +private: + char* myRawDataPtr; +}; + +#endif // _WasmOcctPixMap_HeaderFile diff --git a/samples/webgl/WasmOcctView.cpp b/samples/webgl/WasmOcctView.cpp index db4c6ac773..ec33303300 100644 --- a/samples/webgl/WasmOcctView.cpp +++ b/samples/webgl/WasmOcctView.cpp @@ -22,6 +22,7 @@ #include "WasmOcctView.h" #include "WasmVKeys.h" +#include "WasmOcctPixMap.h" #include #include @@ -30,8 +31,18 @@ #include #include #include +#include #include #include +#include +#include + +#include +#include +#include +#include + +#include #include @@ -40,11 +51,11 @@ namespace { EM_JS(int, jsCanvasGetWidth, (), { - return canvas.width; + return Module.canvas.width; }); EM_JS(int, jsCanvasGetHeight, (), { - return canvas.height; + return Module.canvas.height; }); EM_JS(float, jsDevicePixelRatio, (), { @@ -57,6 +68,65 @@ namespace { return Graphic3d_Vec2i (jsCanvasGetWidth(), jsCanvasGetHeight()); } + + //! Auxiliary wrapper for loading model. + struct ModelAsyncLoader + { + std::string Name; + std::string Path; + + ModelAsyncLoader (const char* theName, const char* thePath) + : Name (theName), Path (thePath) {} + + //! File data read event. + static void onDataRead (void* theOpaque, void* theBuffer, int theDataLen) + { + const ModelAsyncLoader* aTask = (ModelAsyncLoader* )theOpaque; + WasmOcctView::openFromMemory (aTask->Name, reinterpret_cast(theBuffer), theDataLen, false); + delete aTask; + } + + //! File read error event. + static void onReadFailed (void* theOpaque) + { + const ModelAsyncLoader* aTask = (ModelAsyncLoader* )theOpaque; + Message::DefaultMessenger()->Send (TCollection_AsciiString("Error: unable to load file ") + aTask->Path.c_str(), Message_Fail); + delete aTask; + } + }; + + //! Auxiliary wrapper for loading cubemap. + struct CubemapAsyncLoader + { + //! Image file read event. + static void onImageRead (const char* theFilePath) + { + Handle(Graphic3d_CubeMapPacked) aCubemap; + Handle(WasmOcctPixMap) anImage = new WasmOcctPixMap(); + if (anImage->Init (theFilePath)) + { + aCubemap = new Graphic3d_CubeMapPacked (anImage); + } + WasmOcctView::Instance().View()->SetBackgroundCubeMap (aCubemap, true, false); + WasmOcctView::Instance().UpdateView(); + } + + //! Image file failed read event. + static void onImageFailed (const char* theFilePath) + { + Message::DefaultMessenger()->Send (TCollection_AsciiString("Error: unable to load image ") + theFilePath, Message_Fail); + } + }; +} + +// ================================================================ +// Function : Instance +// Purpose : +// ================================================================ +WasmOcctView& WasmOcctView::Instance() +{ + static WasmOcctView aViewer; + return aViewer; } // ================================================================ @@ -239,6 +309,14 @@ bool WasmOcctView::initViewer() aViewer->SetDefaultShadingModel (Graphic3d_TOSM_FRAGMENT); aViewer->SetDefaultLights(); aViewer->SetLightOn(); + for (V3d_ListOfLight::Iterator aLightIter (aViewer->ActiveLights()); aLightIter.More(); aLightIter.Next()) + { + const Handle(V3d_Light)& aLight = aLightIter.Value(); + if (aLight->Type() == Graphic3d_TOLS_DIRECTIONAL) + { + aLight->SetCastShadows (true); + } + } Handle(Aspect_NeutralWindow) aWindow = new Aspect_NeutralWindow(); Graphic3d_Vec2i aWinSize = jsCanvasSize(); @@ -260,7 +338,9 @@ bool WasmOcctView::initViewer() myTextStyle->SetVerticalJustification (Graphic3d_VTA_BOTTOM); myView = new V3d_View (aViewer); + myView->Camera()->SetProjectionType (Graphic3d_Camera::Projection_Perspective); myView->SetImmediateUpdate (false); + myView->ChangeRenderingParams().IsShadowEnabled = false; myView->ChangeRenderingParams().Resolution = (unsigned int )(96.0 * myDevicePixelRatio + 0.5); myView->ChangeRenderingParams().ToShowStats = true; myView->ChangeRenderingParams().StatsTextAspect = myTextStyle->Aspect(); @@ -301,6 +381,18 @@ void WasmOcctView::initDemoScene() // Build with "--preload-file MySampleFile.brep" option to load some shapes here. } +// ================================================================ +// Function : UpdateView +// Purpose : +// ================================================================ +void WasmOcctView::UpdateView() +{ + if (!myView.IsNull()) + { + myView->Invalidate(); + updateView(); + } +} // ================================================================ // Function : updateView @@ -536,7 +628,7 @@ EM_BOOL WasmOcctView::onTouchEvent (int theEventType, const EmscriptenTouchEvent } const Standard_Size aTouchId = (Standard_Size )aTouch.identifier; - const Graphic3d_Vec2i aNewPos = convertPointToBacking (Graphic3d_Vec2i (aTouch.canvasX, aTouch.canvasY)); + const Graphic3d_Vec2i aNewPos = convertPointToBacking (Graphic3d_Vec2i (aTouch.targetX, aTouch.targetY)); switch (theEventType) { case EMSCRIPTEN_EVENT_TOUCHSTART: @@ -677,3 +769,278 @@ EM_BOOL WasmOcctView::onKeyUpEvent (int theEventType, const EmscriptenKeyboardEv } return EM_FALSE; } + +// ================================================================ +// Function : setCubemapBackground +// Purpose : +// ================================================================ +void WasmOcctView::setCubemapBackground (const std::string& theImagePath) +{ + if (!theImagePath.empty()) + { + emscripten_async_wget (theImagePath.c_str(), "/emulated/cubemap.jpg", CubemapAsyncLoader::onImageRead, CubemapAsyncLoader::onImageFailed); + } + else + { + WasmOcctView::Instance().View()->SetBackgroundCubeMap (Handle(Graphic3d_CubeMapPacked)(), true, false); + WasmOcctView::Instance().UpdateView(); + } +} + +// ================================================================ +// Function : fitAllObjects +// Purpose : +// ================================================================ +void WasmOcctView::fitAllObjects (bool theAuto) +{ + WasmOcctView& aViewer = Instance(); + if (theAuto) + { + aViewer.FitAllAuto (aViewer.Context(), aViewer.View()); + } + else + { + aViewer.View()->FitAll (0.01, false); + } + aViewer.UpdateView(); +} + +// ================================================================ +// Function : removeAllObjects +// Purpose : +// ================================================================ +void WasmOcctView::removeAllObjects() +{ + WasmOcctView& aViewer = Instance(); + for (NCollection_IndexedDataMap::Iterator anObjIter (aViewer.myObjects); + anObjIter.More(); anObjIter.Next()) + { + aViewer.Context()->Remove (anObjIter.Value(), false); + } + aViewer.myObjects.Clear(); + aViewer.UpdateView(); +} + +// ================================================================ +// Function : removeObject +// Purpose : +// ================================================================ +bool WasmOcctView::removeObject (const std::string& theName) +{ + WasmOcctView& aViewer = Instance(); + Handle(AIS_InteractiveObject) anObj; + if (!theName.empty() + && !aViewer.myObjects.FindFromKey (theName.c_str(), anObj)) + { + return false; + } + + aViewer.Context()->Remove (anObj, false); + aViewer.myObjects.RemoveKey (theName.c_str()); + aViewer.UpdateView(); + return true; +} + +// ================================================================ +// Function : eraseObject +// Purpose : +// ================================================================ +bool WasmOcctView::eraseObject (const std::string& theName) +{ + WasmOcctView& aViewer = Instance(); + Handle(AIS_InteractiveObject) anObj; + if (!theName.empty() + && !aViewer.myObjects.FindFromKey (theName.c_str(), anObj)) + { + return false; + } + + aViewer.Context()->Erase (anObj, false); + aViewer.UpdateView(); + return true; +} + +// ================================================================ +// Function : displayObject +// Purpose : +// ================================================================ +bool WasmOcctView::displayObject (const std::string& theName) +{ + WasmOcctView& aViewer = Instance(); + Handle(AIS_InteractiveObject) anObj; + if (!theName.empty() + && !aViewer.myObjects.FindFromKey (theName.c_str(), anObj)) + { + return false; + } + + aViewer.Context()->Display (anObj, false); + aViewer.UpdateView(); + return true; +} + +// ================================================================ +// Function : openFromUrl +// Purpose : +// ================================================================ +void WasmOcctView::openFromUrl (const std::string& theName, + const std::string& theModelPath) +{ + ModelAsyncLoader* aTask = new ModelAsyncLoader (theName.c_str(), theModelPath.c_str()); + emscripten_async_wget_data (theModelPath.c_str(), (void* )aTask, ModelAsyncLoader::onDataRead, ModelAsyncLoader::onReadFailed); +} + +// ================================================================ +// Function : openFromMemory +// Purpose : +// ================================================================ +bool WasmOcctView::openFromMemory (const std::string& theName, + uintptr_t theBuffer, int theDataLen, + bool theToFree) +{ + removeObject (theName); + char* aBytes = reinterpret_cast(theBuffer); + if (aBytes == nullptr + || theDataLen <= 0) + { + return false; + } + + // Function to check if specified data stream starts with specified header. + #define dataStartsWithHeader(theData, theHeader) (::strncmp(theData, theHeader, sizeof(theHeader) - 1) == 0) + + if (dataStartsWithHeader(aBytes, "DBRep_DrawableShape")) + { + return openBRepFromMemory (theName, theBuffer, theDataLen, theToFree); + } + else if (dataStartsWithHeader(aBytes, "glTF")) + { + //return openGltfFromMemory (theName, theBuffer, theDataLen, theToFree); + } + if (theToFree) + { + free (aBytes); + } + + Message::SendFail() << "Error: file '" << theName.c_str() << "' has unsupported format"; + return false; +} + +// ================================================================ +// Function : openBRepFromMemory +// Purpose : +// ================================================================ +bool WasmOcctView::openBRepFromMemory (const std::string& theName, + uintptr_t theBuffer, int theDataLen, + bool theToFree) +{ + removeObject (theName); + + WasmOcctView& aViewer = Instance(); + TopoDS_Shape aShape; + BRep_Builder aBuilder; + bool isLoaded = false; + { + char* aRawData = reinterpret_cast(theBuffer); + Standard_ArrayStreamBuffer aStreamBuffer (aRawData, theDataLen); + std::istream aStream (&aStreamBuffer); + BRepTools::Read (aShape, aStream, aBuilder); + if (theToFree) + { + free (aRawData); + } + isLoaded = true; + } + if (!isLoaded) + { + return false; + } + + Handle(AIS_Shape) aShapePrs = new AIS_Shape (aShape); + if (!theName.empty()) + { + aViewer.myObjects.Add (theName.c_str(), aShapePrs); + } + aShapePrs->SetMaterial (Graphic3d_NameOfMaterial_Silver); + aViewer.Context()->Display (aShapePrs, AIS_Shaded, 0, false); + aViewer.View()->FitAll (0.01, false); + aViewer.UpdateView(); + + Message::DefaultMessenger()->Send (TCollection_AsciiString("Loaded file ") + theName.c_str(), Message_Info); + Message::DefaultMessenger()->Send (OSD_MemInfo::PrintInfo(), Message_Trace); + return true; +} + +// ================================================================ +// Function : displayGround +// Purpose : +// ================================================================ +void WasmOcctView::displayGround (bool theToShow) +{ + static Handle(AIS_Shape) aGroundPrs = new AIS_Shape (TopoDS_Shape()); + + WasmOcctView& aViewer = Instance(); + Bnd_Box aBox; + if (theToShow) + { + aViewer.Context()->Remove (aGroundPrs, false); + aBox = aViewer.View()->View()->MinMaxValues(); + } + if (aBox.IsVoid() + || aBox.IsZThin (Precision::Confusion())) + { + if (!aGroundPrs.IsNull() + && aGroundPrs->HasInteractiveContext()) + { + aViewer.Context()->Remove (aGroundPrs, false); + aViewer.UpdateView(); + } + return; + } + + const gp_XYZ aSize = aBox.CornerMax().XYZ() - aBox.CornerMin().XYZ(); + const double aRadius = Max (aSize.X(), aSize.Y()); + const double aZValue = aBox.CornerMin().Z() - Min (10.0, aSize.Z() * 0.01); + const double aZSize = aRadius * 0.01; + gp_XYZ aGroundCenter ((aBox.CornerMin().X() + aBox.CornerMax().X()) * 0.5, + (aBox.CornerMin().Y() + aBox.CornerMax().Y()) * 0.5, + aZValue); + + TopoDS_Compound aGround; + gp_Trsf aTrsf1, aTrsf2; + aTrsf1.SetTranslation (gp_Vec (aGroundCenter - gp_XYZ(0.0, 0.0, aZSize))); + aTrsf2.SetTranslation (gp_Vec (aGroundCenter)); + Prs3d_ToolCylinder aCylTool (aRadius, aRadius, aZSize, 50, 1); + Prs3d_ToolDisk aDiskTool (0.0, aRadius, 50, 1); + TopoDS_Face aCylFace, aDiskFace1, aDiskFace2; + BRep_Builder().MakeFace (aCylFace, aCylTool .CreatePolyTriangulation (aTrsf1)); + BRep_Builder().MakeFace (aDiskFace1, aDiskTool.CreatePolyTriangulation (aTrsf1)); + BRep_Builder().MakeFace (aDiskFace2, aDiskTool.CreatePolyTriangulation (aTrsf2)); + + BRep_Builder().MakeCompound (aGround); + BRep_Builder().Add (aGround, aCylFace); + BRep_Builder().Add (aGround, aDiskFace1); + BRep_Builder().Add (aGround, aDiskFace2); + + aGroundPrs->SetShape (aGround); + aGroundPrs->SetToUpdate(); + aGroundPrs->SetMaterial (Graphic3d_NameOfMaterial_Stone); + aGroundPrs->SetInfiniteState (false); + aViewer.Context()->Display (aGroundPrs, AIS_Shaded, -1, false); + aGroundPrs->SetInfiniteState (true); + aViewer.UpdateView(); +} + +// Module exports +EMSCRIPTEN_BINDINGS(OccViewerModule) { + emscripten::function("setCubemapBackground", &WasmOcctView::setCubemapBackground); + emscripten::function("fitAllObjects", &WasmOcctView::fitAllObjects); + emscripten::function("removeAllObjects", &WasmOcctView::removeAllObjects); + emscripten::function("removeObject", &WasmOcctView::removeObject); + emscripten::function("eraseObject", &WasmOcctView::eraseObject); + emscripten::function("displayObject", &WasmOcctView::displayObject); + emscripten::function("displayGround", &WasmOcctView::displayGround); + emscripten::function("openFromUrl", &WasmOcctView::openFromUrl); + emscripten::function("openFromMemory", &WasmOcctView::openFromMemory, emscripten::allow_raw_pointers()); + emscripten::function("openBRepFromMemory", &WasmOcctView::openBRepFromMemory, emscripten::allow_raw_pointers()); +} diff --git a/samples/webgl/WasmOcctView.h b/samples/webgl/WasmOcctView.h index 0715e7cbac..c05ce179a3 100644 --- a/samples/webgl/WasmOcctView.h +++ b/samples/webgl/WasmOcctView.h @@ -35,6 +35,72 @@ class AIS_ViewCube; class WasmOcctView : protected AIS_ViewController { public: + + //! Return global viewer instance. + static WasmOcctView& Instance(); + +public: //! @name methods exported by Module + + //! Set cubemap background. + //! File will be loaded asynchronously. + //! @param theImagePath [in] image path to load + static void setCubemapBackground (const std::string& theImagePath); + + //! Clear all named objects from viewer. + static void removeAllObjects(); + + //! Fit all/selected objects into view. + //! @param theAuto [in] fit selected objects (TRUE) or all objects (FALSE) + static void fitAllObjects (bool theAuto); + + //! Remove named object from viewer. + //! @param theName [in] object name + //! @return FALSE if object was not found + static bool removeObject (const std::string& theName); + + //! Temporarily hide named object. + //! @param theName [in] object name + //! @return FALSE if object was not found + static bool eraseObject (const std::string& theName); + + //! Display temporarily hidden object. + //! @param theName [in] object name + //! @return FALSE if object was not found + static bool displayObject (const std::string& theName); + + //! Show/hide ground. + //! @param theToShow [in] show or hide flag + static void displayGround (bool theToShow); + + //! Open object from the given URL. + //! File will be loaded asynchronously. + //! @param theName [in] object name + //! @param theModelPath [in] model path + static void openFromUrl (const std::string& theName, + const std::string& theModelPath); + + //! Open object from memory. + //! @param theName [in] object name + //! @param theBuffer [in] pointer to data + //! @param theDataLen [in] data length + //! @param theToFree [in] free theBuffer if set to TRUE + //! @return FALSE on reading error + static bool openFromMemory (const std::string& theName, + uintptr_t theBuffer, int theDataLen, + bool theToFree); + + //! Open BRep object from memory. + //! @param theName [in] object name + //! @param theBuffer [in] pointer to data + //! @param theDataLen [in] data length + //! @param theToFree [in] free theBuffer if set to TRUE + //! @return FALSE on reading error + static bool openBRepFromMemory (const std::string& theName, + uintptr_t theBuffer, int theDataLen, + bool theToFree); + +public: + //! Default constructor. WasmOcctView(); @@ -53,6 +119,9 @@ public: //! Return device pixel ratio for handling high DPI displays. float DevicePixelRatio() const { return myDevicePixelRatio; } + //! Request view redrawing. + void UpdateView(); + private: //! Create window. @@ -142,6 +211,8 @@ private: private: + NCollection_IndexedDataMap myObjects; //!< map of named objects + Handle(AIS_InteractiveContext) myContext; //!< interactive context Handle(V3d_View) myView; //!< 3D view Handle(Prs3d_TextAspect) myTextStyle; //!< text style for OSD elements diff --git a/samples/webgl/main.cpp b/samples/webgl/main.cpp index 306a42f517..d39f55e261 100644 --- a/samples/webgl/main.cpp +++ b/samples/webgl/main.cpp @@ -8,17 +8,9 @@ #include #include -#include -#include -#include -#include - #include #include -//! Global viewer instance. -static WasmOcctView aViewer; - //! Dummy main loop callback for a single shot. extern "C" void onMainLoop() { @@ -26,42 +18,7 @@ extern "C" void onMainLoop() emscripten_cancel_main_loop(); } -//! File data read event. -extern "C" void onFileDataRead (void* theOpaque, void* theBuffer, int theDataLen) -{ - const char* aName = theOpaque != NULL ? (const char* )theOpaque : ""; - { - AIS_ListOfInteractive aShapes; - aViewer.Context()->DisplayedObjects (AIS_KOI_Shape, -1, aShapes); - for (AIS_ListOfInteractive::Iterator aShapeIter (aShapes); aShapeIter.More(); aShapeIter.Next()) - { - aViewer.Context()->Remove (aShapeIter.Value(), false); - } - } - - Standard_ArrayStreamBuffer aStreamBuffer ((const char* )theBuffer, theDataLen); - std::istream aStream (&aStreamBuffer); - TopoDS_Shape aShape; - BRep_Builder aBuilder; - BRepTools::Read (aShape, aStream, aBuilder); - - Handle(AIS_Shape) aShapePrs = new AIS_Shape (aShape); - aShapePrs->SetMaterial (Graphic3d_NameOfMaterial_Silver); - aViewer.Context()->Display (aShapePrs, AIS_Shaded, 0, false); - aViewer.View()->FitAll (0.01, false); - aViewer.View()->Redraw(); - Message::DefaultMessenger()->Send (TCollection_AsciiString("Loaded file ") + aName, Message_Info); - Message::DefaultMessenger()->Send (OSD_MemInfo::PrintInfo(), Message_Trace); -} - -//! File read error event. -static void onFileReadFailed (void* theOpaque) -{ - const char* aName = (const char* )theOpaque; - Message::DefaultMessenger()->Send (TCollection_AsciiString("Error: unable to load file ") + aName, Message_Fail); -} - -int main() +EMSCRIPTEN_KEEPALIVE int main() { Message::DefaultMessenger()->Printers().First()->SetTraceLevel (Message_Trace); Handle(Message_PrinterSystemLog) aJSConsolePrinter = new Message_PrinterSystemLog ("webgl-sample", Message_Trace); @@ -71,10 +28,8 @@ int main() // setup a dummy single-shot main loop callback just to shut up a useless Emscripten error message on calling eglSwapInterval() emscripten_set_main_loop (onMainLoop, -1, 0); + WasmOcctView& aViewer = WasmOcctView::Instance(); aViewer.run(); Message::DefaultMessenger()->Send (OSD_MemInfo::PrintInfo(), Message_Trace); - - // load some file - emscripten_async_wget_data ("samples/Ball.brep", (void* )"samples/Ball.brep", onFileDataRead, onFileReadFailed); return 0; } diff --git a/samples/webgl/occt-webgl-sample.html b/samples/webgl/occt-webgl-sample.html index efec7dba0e..c04add8b47 100644 --- a/samples/webgl/occt-webgl-sample.html +++ b/samples/webgl/occt-webgl-sample.html @@ -9,11 +9,15 @@

OCCT WebGL Viewer Sample

- +
-
+
+ + + +

Console output: