import { GetFileExtension, TransformFileHostUrls } from '../engine/io/fileutils.js'; import { InputFilesFromFileObjects, InputFilesFromUrls } from '../engine/import/importerfiles.js'; import { ImportErrorCode, ImportSettings } from '../engine/import/importer.js'; import { CameraMode, Viewer } from '../engine/viewer/viewer.js'; import { AddDiv, AddDomElement, ShowDomElement, SetDomElementOuterHeight, CreateDomElement, GetDomElementOuterWidth } from '../engine/viewer/domutils.js'; import { CalculatePopupPositionToScreen, ShowListPopup } from './dialogs.js'; import { HandleEvent } from './eventhandler.js'; import { HashHandler } from './hashhandler.js'; import { Navigator, Selection, SelectionType } from './navigator.js'; import { Settings, Theme } from './settings.js'; import { Sidebar } from './sidebar.js'; import { ThemeHandler } from './themehandler.js'; import { ThreeModelLoaderUI } from './threemodelloaderui.js'; import { Toolbar } from './toolbar.js'; import { DownloadModel, ShowExportDialog } from './exportdialog.js'; import { ShowSnapshotDialog } from './snapshotdialog.js'; import { AddSvgIconElement, GetFilesFromDataTransfer, InstallTooltip, IsSmallWidth } from './utils.js'; import { ShowOpenUrlDialog } from './openurldialog.js'; import { ShowSharingDialog } from './sharingdialog.js'; import { HasDefaultMaterial, ReplaceDefaultMaterialColor } from '../engine/model/modelutils.js'; import { Direction } from '../engine/geometry/geometry.js'; import { CookieGetBoolVal, CookieSetBoolVal } from './cookiehandler.js'; import { MeasureTool } from './measuretool.js'; import { CloseAllDialogs } from './dialog.js'; import { CreateVerticalSplitter } from './splitter.js'; import { EnumeratePlugins, PluginType } from './pluginregistry.js'; import * as THREE from 'three'; const WebsiteUIState = { Undefined : 0, Intro : 1, Model : 2, Loading : 3 }; class WebsiteLayouter { constructor (parameters, navigator, sidebar, viewer, measureTool) { this.parameters = parameters; this.navigator = navigator; this.sidebar = sidebar; this.viewer = viewer; this.measureTool = measureTool; this.limits = { minPanelWidth : 290, minCanvasWidth : 100 }; } Init () { this.InstallSplitter (this.parameters.navigatorSplitterDiv, this.parameters.navigatorDiv, (originalWidth, xDiff) => { let newWidth = originalWidth + xDiff; this.OnSplitterDragged (newWidth - this.navigator.GetWidth (), 0); }); this.InstallSplitter (this.parameters.sidebarSplitterDiv, this.parameters.sidebarDiv, (originalWidth, xDiff) => { let newWidth = originalWidth - xDiff; this.OnSplitterDragged (0, newWidth - this.sidebar.GetWidth ()); }); this.Resize (); } InstallSplitter (splitterDiv, resizedDiv, onSplit) { let originalWidth = null; CreateVerticalSplitter (splitterDiv, { onSplitStart : () => { originalWidth = GetDomElementOuterWidth (resizedDiv); }, onSplit : (xDiff) => { onSplit (originalWidth, xDiff); } }); } OnSplitterDragged (leftDiff, rightDiff) { let windowWidth = window.innerWidth; let navigatorWidth = this.navigator.GetWidth (); let sidebarWidth = this.sidebar.GetWidth (); let leftWidth = GetDomElementOuterWidth (this.parameters.leftContainerDiv); let rightWidth = GetDomElementOuterWidth (this.parameters.rightContainerDiv); let newLeftWidth = leftWidth + leftDiff; let newRightWidth = rightWidth + rightDiff; let contentNewWidth = windowWidth - newLeftWidth - newRightWidth; let isNavigatorVisible = this.navigator.IsPanelsVisible (); let isSidebarVisible = this.sidebar.IsPanelsVisible (); if (isNavigatorVisible && newLeftWidth < this.limits.minPanelWidth) { newLeftWidth = this.limits.minPanelWidth; } if (isSidebarVisible && newRightWidth < this.limits.minPanelWidth) { newRightWidth = this.limits.minPanelWidth; } if (contentNewWidth < this.limits.minCanvasWidth) { if (leftDiff > 0) { newLeftWidth = windowWidth - newRightWidth - this.limits.minCanvasWidth; } else if (rightDiff > 0) { newRightWidth = windowWidth - newLeftWidth - this.limits.minCanvasWidth; } } if (isNavigatorVisible) { let newNavigatorWidth = navigatorWidth + (newLeftWidth - leftWidth); this.navigator.SetWidth (newNavigatorWidth); } if (isSidebarVisible) { let newSidebarWidth = sidebarWidth + (newRightWidth - rightWidth); this.sidebar.SetWidth (newSidebarWidth); } this.Resize (); } Resize () { let windowWidth = window.innerWidth; let windowHeight = window.innerHeight; let headerHeight = this.parameters.headerDiv.offsetHeight; let leftWidth = 0; let rightWidth = 0; let safetyMargin = 0; if (!IsSmallWidth ()) { leftWidth = GetDomElementOuterWidth (this.parameters.leftContainerDiv); rightWidth = GetDomElementOuterWidth (this.parameters.rightContainerDiv); safetyMargin = 1; } let contentWidth = windowWidth - leftWidth - rightWidth; let contentHeight = windowHeight - headerHeight; if (contentWidth < this.limits.minCanvasWidth) { let neededIncrease = this.limits.minCanvasWidth - contentWidth; let isNavigatorVisible = this.navigator.IsPanelsVisible (); let isSidebarVisible = this.sidebar.IsPanelsVisible (); if (neededIncrease > 0 && isNavigatorVisible) { let navigatorDecrease = Math.min (neededIncrease, leftWidth - this.limits.minPanelWidth); this.navigator.SetWidth (this.navigator.GetWidth () - navigatorDecrease); neededIncrease = neededIncrease - navigatorDecrease; } if (neededIncrease > 0 && isSidebarVisible) { let sidebarDecrease = Math.min (neededIncrease, rightWidth - this.limits.minPanelWidth); this.sidebar.SetWidth (this.sidebar.GetWidth () - sidebarDecrease); } leftWidth = GetDomElementOuterWidth (this.parameters.leftContainerDiv); rightWidth = GetDomElementOuterWidth (this.parameters.rightContainerDiv); contentWidth = windowWidth - leftWidth - rightWidth; } this.navigator.Resize (contentHeight); SetDomElementOuterHeight (this.parameters.navigatorSplitterDiv, contentHeight); this.sidebar.Resize (contentHeight); SetDomElementOuterHeight (this.parameters.sidebarSplitterDiv, contentHeight); SetDomElementOuterHeight (this.parameters.introDiv, contentHeight); this.viewer.Resize (contentWidth - safetyMargin, contentHeight); this.measureTool.Resize (); } } export class Website { constructor (parameters) { this.parameters = parameters; this.settings = new Settings (); this.viewer = new Viewer (); this.measureTool = new MeasureTool (this.viewer, this.settings); this.hashHandler = new HashHandler (); this.toolbar = new Toolbar (this.parameters.toolbarDiv); this.navigator = new Navigator (this.parameters.navigatorDiv); this.sidebar = new Sidebar (this.parameters.sidebarDiv, this.settings); this.modelLoaderUI = new ThreeModelLoaderUI (); this.themeHandler = new ThemeHandler (); this.highlightColor = new THREE.Color (0x8ec9f0); this.uiState = WebsiteUIState.Undefined; this.layouter = new WebsiteLayouter (this.parameters, this.navigator, this.sidebar, this.viewer, this.measureTool); this.model = null; } Load () { this.settings.LoadFromCookies (); this.SwitchTheme (this.settings.themeId, false); HandleEvent ('theme_on_load', this.settings.themeId === Theme.Light ? 'light' : 'dark'); EnumeratePlugins (PluginType.Header, (plugin) => { plugin.registerButtons ({ createHeaderButton : (icon, title, link) => { this.CreateHeaderButton (icon, title, link); } }); }); this.InitViewer (); this.InitToolbar (); this.InitDragAndDrop (); this.InitSidebar (); this.InitNavigator (); this.InitCookieConsent (); this.viewer.SetMouseClickHandler (this.OnModelClicked.bind (this)); this.viewer.SetMouseMoveHandler (this.OnModelMouseMoved.bind (this)); this.viewer.SetContextMenuHandler (this.OnModelContextMenu.bind (this)); this.layouter.Init (); this.SetUIState (WebsiteUIState.Intro); this.hashHandler.SetEventListener (this.OnHashChange.bind (this)); this.OnHashChange (); window.addEventListener ('resize', () => { this.layouter.Resize (); }); } HasLoadedModel () { return this.model !== null; } SetUIState (uiState) { function ShowOnlyOnModelElements (show) { let root = document.querySelector (':root'); root.style.setProperty ('--ov_only_on_model_display', show ? 'inherit' : 'none'); } if (this.uiState === uiState) { return; } this.uiState = uiState; if (this.uiState === WebsiteUIState.Intro) { ShowDomElement (this.parameters.introDiv, true); ShowDomElement (this.parameters.mainDiv, false); ShowOnlyOnModelElements (false); } else if (this.uiState === WebsiteUIState.Model) { ShowDomElement (this.parameters.introDiv, false); ShowDomElement (this.parameters.mainDiv, true); ShowOnlyOnModelElements (true); this.UpdatePanelsVisibility (); } else if (this.uiState === WebsiteUIState.Loading) { ShowDomElement (this.parameters.introDiv, false); ShowDomElement (this.parameters.mainDiv, false); ShowOnlyOnModelElements (false); } this.layouter.Resize (); } ClearModel () { CloseAllDialogs (); this.model = null; this.viewer.Clear (); this.parameters.fileNameDiv.innerHTML = ''; this.navigator.Clear (); this.sidebar.Clear (); this.measureTool.SetActive (false); } OnModelLoaded (importResult, threeObject) { this.model = importResult.model; this.parameters.fileNameDiv.innerHTML = importResult.mainFile; this.viewer.SetMainObject (threeObject); this.viewer.SetUpVector (Direction.Y, false); this.navigator.FillTree (importResult); this.sidebar.UpdateControlsVisibility (); this.FitModelToWindow (true); } OnModelClicked (button, mouseCoordinates) { if (button !== 1) { return; } if (this.measureTool.IsActive ()) { this.measureTool.Click (mouseCoordinates); return; } let meshUserData = this.viewer.GetMeshUserDataUnderMouse (mouseCoordinates); if (meshUserData === null) { this.navigator.SetSelection (null); } else { this.navigator.SetSelection (new Selection (SelectionType.Mesh, meshUserData.originalMeshId)); } } OnModelMouseMoved (mouseCoordinates) { if (this.measureTool.IsActive ()) { this.measureTool.MouseMove (mouseCoordinates); } } OnModelContextMenu (globalMouseCoordinates, mouseCoordinates) { let meshUserData = this.viewer.GetMeshUserDataUnderMouse (mouseCoordinates); let items = []; if (meshUserData === null) { items.push ({ name : 'Fit model to window', icon : 'fit', onClick : () => { this.FitModelToWindow (false); } }); if (this.navigator.HasHiddenMesh ()) { items.push ({ name : 'Show all meshes', icon : 'visible', onClick : () => { this.navigator.ShowAllMeshes (true); } }); } } else { items.push ({ name : 'Hide mesh', icon : 'hidden', onClick : () => { this.navigator.ToggleMeshVisibility (meshUserData.originalMeshId); } }); items.push ({ name : 'Fit mesh to window', icon : 'fit', onClick : () => { this.navigator.FitMeshToWindow (meshUserData.originalMeshId); } }); if (this.navigator.MeshItemCount () > 1) { let isMeshIsolated = this.navigator.IsMeshIsolated (meshUserData.originalMeshId); items.push ({ name : isMeshIsolated ? 'Remove isolation' : 'Isolate mesh', icon : isMeshIsolated ? 'deisolate' : 'isolate', onClick : () => { if (isMeshIsolated) { this.navigator.ShowAllMeshes (true); } else { this.navigator.IsolateMesh (meshUserData.originalMeshId); } } }); } } ShowListPopup (items, { calculatePosition : (contentDiv) => { return CalculatePopupPositionToScreen (globalMouseCoordinates, contentDiv); }, onClick : (index) => { let clickedItem = items[index]; clickedItem.onClick (); } }); } OnHashChange () { if (this.hashHandler.HasHash ()) { let urls = this.hashHandler.GetModelFilesFromHash (); if (urls === null) { return; } TransformFileHostUrls (urls); let importSettings = new ImportSettings (); importSettings.defaultColor = this.settings.defaultColor; let defaultColor = this.hashHandler.GetDefaultColorFromHash (); if (defaultColor !== null) { importSettings.defaultColor = defaultColor; } HandleEvent ('model_load_started', 'hash'); this.LoadModelFromUrlList (urls, importSettings); } else { this.ClearModel (); this.SetUIState (WebsiteUIState.Intro); } } OpenFileBrowserDialog () { this.parameters.fileInput.click (); } FitModelToWindow (onLoad) { let animation = !onLoad; let boundingSphere = this.viewer.GetBoundingSphere ((meshUserData) => { return this.navigator.IsMeshVisible (meshUserData.originalMeshId); }); if (onLoad) { this.viewer.AdjustClippingPlanesToSphere (boundingSphere); } this.viewer.FitSphereToWindow (boundingSphere, animation); } FitMeshToWindow (meshInstanceId) { let boundingSphere = this.viewer.GetBoundingSphere ((meshUserData) => { return meshUserData.originalMeshId.IsEqual (meshInstanceId); }); this.viewer.FitSphereToWindow (boundingSphere, true); } FitMeshesToWindow (meshInstanceIdSet) { let meshInstanceIdKeys = new Set (); for (let meshInstanceId of meshInstanceIdSet) { meshInstanceIdKeys.add (meshInstanceId.GetKey ()); } let boundingSphere = this.viewer.GetBoundingSphere ((meshUserData) => { return meshInstanceIdKeys.has (meshUserData.originalMeshId.GetKey ()); }); this.viewer.FitSphereToWindow (boundingSphere, true); } UpdateMeshesVisibility () { this.viewer.SetMeshesVisibility ((meshUserData) => { return this.navigator.IsMeshVisible (meshUserData.originalMeshId); }); } UpdateMeshesSelection () { let selectedMeshId = this.navigator.GetSelectedMeshId (); this.viewer.SetMeshesHighlight (this.highlightColor, (meshUserData) => { if (selectedMeshId !== null && meshUserData.originalMeshId.IsEqual (selectedMeshId)) { return true; } return false; }); } LoadModelFromUrlList (urls, settings) { let inputFiles = InputFilesFromUrls (urls); this.LoadModelFromInputFiles (inputFiles, settings); this.ClearHashIfNotOnlyUrlList (); } LoadModelFromFileList (files) { let importSettings = new ImportSettings (); importSettings.defaultColor = this.settings.defaultColor; let inputFiles = InputFilesFromFileObjects (files); this.LoadModelFromInputFiles (inputFiles, importSettings); this.ClearHashIfNotOnlyUrlList (); } LoadModelFromInputFiles (files, settings) { this.modelLoaderUI.LoadModel (files, settings, { onStart : () => { this.SetUIState (WebsiteUIState.Loading); this.ClearModel (); }, onFinish : (importResult, threeObject) => { this.SetUIState (WebsiteUIState.Model); this.OnModelLoaded (importResult, threeObject); let importedExtension = GetFileExtension (importResult.mainFile); HandleEvent ('model_loaded', importedExtension); }, onRender : () => { this.viewer.Render (); }, onError : (importError) => { this.SetUIState (WebsiteUIState.Intro); let extensionStr = null; if (importError.mainFile !== null) { extensionStr = GetFileExtension (importError.mainFile); } else { let extensions = []; let importer = this.modelLoaderUI.GetImporter (); let fileList = importer.GetFileList ().GetFiles (); for (let i = 0; i < fileList.length; i++) { let extension = fileList[i].extension; extensions.push (extension); } extensionStr = extensions.join (','); } if (importError.code === ImportErrorCode.NoImportableFile) { HandleEvent ('no_importable_file', extensionStr); } else if (importError.code === ImportErrorCode.FailedToLoadFile) { HandleEvent ('failed_to_load_file', extensionStr); } else if (importError.code === ImportErrorCode.ImportFailed) { HandleEvent ('import_failed', extensionStr, { error_message : importError.message }); } } }); } ClearHashIfNotOnlyUrlList () { let importer = this.modelLoaderUI.GetImporter (); let isOnlyUrl = importer.GetFileList ().IsOnlyUrlSource (); if (!isOnlyUrl && this.hashHandler.HasHash ()) { this.hashHandler.SkipNextEventHandler (); this.hashHandler.ClearHash (); } } UpdateEdgeDisplay () { this.settings.SaveToCookies (); this.viewer.SetEdgeSettings (this.settings.edgeSettings); } UpdateEnvironmentMap () { let envMapPath = 'assets/envmaps/' + this.settings.environmentMapName + '/'; let envMapTextures = [ envMapPath + 'posx.jpg', envMapPath + 'negx.jpg', envMapPath + 'posy.jpg', envMapPath + 'negy.jpg', envMapPath + 'posz.jpg', envMapPath + 'negz.jpg' ]; this.viewer.SetEnvironmentMapSettings (envMapTextures, this.settings.backgroundIsEnvMap); } SwitchTheme (newThemeId, resetColors) { this.settings.themeId = newThemeId; this.themeHandler.SwitchTheme (this.settings.themeId); this.settings.SaveToCookies (); if (resetColors) { this.viewer.SetBackgroundColor (this.settings.backgroundColor); let modelLoader = this.modelLoaderUI.GetModelLoader (); if (modelLoader.GetDefaultMaterial () !== null) { ReplaceDefaultMaterialColor (this.model, this.settings.defaultColor); modelLoader.ReplaceDefaultMaterialColor (this.settings.defaultColor); } } } InitViewer () { let canvas = AddDomElement (this.parameters.viewerDiv, 'canvas'); this.viewer.Init (canvas); this.viewer.SetEdgeSettings (this.settings.edgeSettings); this.viewer.SetBackgroundColor (this.settings.backgroundColor); this.UpdateEnvironmentMap (); } InitToolbar () { function AddButton (toolbar, imageName, imageTitle, classNames, onClick) { let button = toolbar.AddImageButton (imageName, imageTitle, () => { onClick (); }); for (let className of classNames) { button.AddClass (className); } return button; } function AddPushButton (toolbar, imageName, imageTitle, classNames, onClick) { let button = toolbar.AddImagePushButton (imageName, imageTitle, false, (isSelected) => { onClick (isSelected); }); for (let className of classNames) { button.AddClass (className); } return button; } function AddRadioButton (toolbar, imageNames, imageTitles, selectedIndex, classNames, onClick) { let imageData = []; for (let i = 0; i < imageNames.length; i++) { let imageName = imageNames[i]; let imageTitle = imageTitles[i]; imageData.push ({ image : imageName, title : imageTitle }); } let buttons = toolbar.AddImageRadioButton (imageData, selectedIndex, (buttonIndex) => { onClick (buttonIndex); }); for (let className of classNames) { for (let button of buttons) { button.AddClass (className); } } } function AddSeparator (toolbar, classNames) { let separator = toolbar.AddSeparator (); if (classNames !== null) { for (let className of classNames) { separator.classList.add (className); } } } let importer = this.modelLoaderUI.GetImporter (); AddButton (this.toolbar, 'open', 'Open from your device', [], () => { this.OpenFileBrowserDialog (); }); AddButton (this.toolbar, 'open_url', 'Open from url', [], () => { ShowOpenUrlDialog ((urls) => { if (urls.length > 0) { this.hashHandler.SetModelFilesToHash (urls); } }); }); AddSeparator (this.toolbar, ['only_on_model']); AddButton (this.toolbar, 'fit', 'Fit model to window', ['only_on_model'], () => { this.FitModelToWindow (false); }); AddButton (this.toolbar, 'up_y', 'Set Y axis as up vector', ['only_on_model'], () => { this.viewer.SetUpVector (Direction.Y, true); }); AddButton (this.toolbar, 'up_z', 'Set Z axis as up vector', ['only_on_model'], () => { this.viewer.SetUpVector (Direction.Z, true); }); AddButton (this.toolbar, 'flip', 'Flip up vector', ['only_on_model'], () => { this.viewer.FlipUpVector (); }); AddSeparator (this.toolbar, ['only_on_model']); AddRadioButton (this.toolbar, ['fix_up_on', 'fix_up_off'], ['Fixed up vector', 'Free orbit'], 0, ['only_full_width', 'only_on_model'], (buttonIndex) => { if (buttonIndex === 0) { this.viewer.SetFixUpVector (true); } else if (buttonIndex === 1) { this.viewer.SetFixUpVector (false); } }); AddSeparator (this.toolbar, ['only_full_width', 'only_on_model']); AddRadioButton (this.toolbar, ['camera_perspective', 'camera_orthographic'], ['Perspective camera', 'Orthographic camera'], 0, ['only_on_model'], (buttonIndex) => { if (buttonIndex === 0) { this.viewer.SetCameraMode (CameraMode.Perspective); } else if (buttonIndex === 1) { this.viewer.SetCameraMode (CameraMode.Orthographic); } this.sidebar.UpdateControlsVisibility (); }); AddSeparator (this.toolbar, ['only_full_width', 'only_on_model']); let measureToolButton = AddPushButton (this.toolbar, 'measure', 'Measure', ['only_full_width', 'only_on_model'], (isSelected) => { HandleEvent ('measure_tool_activated', isSelected ? 'on' : 'off'); this.navigator.SetSelection (null); this.measureTool.SetActive (isSelected); }); this.measureTool.SetButton (measureToolButton); AddSeparator (this.toolbar, ['only_full_width', 'only_on_model']); AddButton (this.toolbar, 'download', 'Download', ['only_full_width', 'only_on_model'], () => { HandleEvent ('model_downloaded', ''); let importer = this.modelLoaderUI.GetImporter (); DownloadModel (importer); }); AddButton (this.toolbar, 'export', 'Export', ['only_full_width', 'only_on_model'], () => { ShowExportDialog (this.model, this.viewer, { isMeshVisible : (meshInstanceId) => { return this.navigator.IsMeshVisible (meshInstanceId); } }); }); AddButton (this.toolbar, 'share', 'Share', ['only_full_width', 'only_on_model'], () => { ShowSharingDialog (importer.GetFileList (), this.settings, this.viewer); }); AddSeparator (this.toolbar, ['only_full_width', 'only_on_model']); AddButton (this.toolbar, 'snapshot', 'Create snapshot', ['only_full_width', 'only_on_model'], () => { ShowSnapshotDialog (this.viewer); }); EnumeratePlugins (PluginType.Toolbar, (plugin) => { plugin.registerButtons ({ createSeparator : (classNames) => { AddSeparator (this.toolbar, classNames); }, createButton : (icon, title, classNames, onClick) => { AddButton (this.toolbar, icon, title, classNames, onClick); }, getModel : () => { return this.model; } }); }); this.parameters.fileInput.addEventListener ('change', (ev) => { if (ev.target.files.length > 0) { HandleEvent ('model_load_started', 'open_file'); this.LoadModelFromFileList (ev.target.files); } }); } InitDragAndDrop () { window.addEventListener ('dragstart', (ev) => { ev.preventDefault (); }, false); window.addEventListener ('dragover', (ev) => { ev.stopPropagation (); ev.preventDefault (); ev.dataTransfer.dropEffect = 'copy'; }, false); window.addEventListener ('drop', (ev) => { ev.stopPropagation (); ev.preventDefault (); GetFilesFromDataTransfer (ev.dataTransfer, (files) => { if (files.length > 0) { HandleEvent ('model_load_started', 'drop'); this.LoadModelFromFileList (files); } }); }, false); } InitSidebar () { this.sidebar.Init ({ getShadingType : () => { return this.viewer.GetShadingType (); }, getCameraMode : () => { return this.viewer.GetCameraMode (); }, hasDefaultMaterial : () => { return HasDefaultMaterial (this.model); }, onEnvironmentMapChanged : () => { this.settings.SaveToCookies (); this.UpdateEnvironmentMap (); if (this.measureTool.IsActive ()) { this.measureTool.UpdatePanel (); } }, onBackgroundColorChanged : () => { this.settings.SaveToCookies (); this.viewer.SetBackgroundColor (this.settings.backgroundColor); if (this.measureTool.IsActive ()) { this.measureTool.UpdatePanel (); } }, onDefaultColorChanged : () => { this.settings.SaveToCookies (); let modelLoader = this.modelLoaderUI.GetModelLoader (); if (modelLoader.GetDefaultMaterial () !== null) { ReplaceDefaultMaterialColor (this.model, this.settings.defaultColor); modelLoader.ReplaceDefaultMaterialColor (this.settings.defaultColor); } this.viewer.Render (); }, onEdgeDisplayChanged : () => { HandleEvent ('edge_display_changed', this.settings.showEdges ? 'on' : 'off'); this.UpdateEdgeDisplay (); }, onThemeChanged : () => { HandleEvent ('theme_changed', this.settings.themeId === Theme.Light ? 'light' : 'dark'); this.SwitchTheme (this.settings.themeId, true); }, onResizeRequested : () => { this.layouter.Resize (); }, onShowHidePanels : (show) => { ShowDomElement (this.parameters.sidebarSplitterDiv, show); CookieSetBoolVal ('ov_show_sidebar', show); } }); } InitNavigator () { function GetMeshUserData (viewer, meshInstanceId) { let userData = null; viewer.EnumerateMeshesUserData ((meshUserData) => { if (meshUserData.originalMeshId.IsEqual (meshInstanceId)) { userData = meshUserData; } }); return userData; } function GetMeshesForMaterial (viewer, model, materialIndex) { let usedByMeshes = []; viewer.EnumerateMeshesUserData ((meshUserData) => { if (materialIndex === null || meshUserData.originalMaterials.indexOf (materialIndex) !== -1) { const mesh = model.GetMesh (meshUserData.originalMeshId.meshIndex); usedByMeshes.push ({ meshId : meshUserData.originalMeshId, name : mesh.GetName () }); } }); return usedByMeshes; } function GetMaterialReferenceInfo (model, materialIndex) { const material = model.GetMaterial (materialIndex); return { index : materialIndex, name : material.name, color : material.color.Clone () }; } function GetMaterialsForMesh (viewer, model, meshInstanceId) { let usedMaterials = []; if (meshInstanceId === null) { for (let materialIndex = 0; materialIndex < model.MaterialCount (); materialIndex++) { usedMaterials.push (GetMaterialReferenceInfo (model, materialIndex)); } } else { let userData = GetMeshUserData (viewer, meshInstanceId); for (let i = 0; i < userData.originalMaterials.length; i++) { const materialIndex = userData.originalMaterials[i]; usedMaterials.push (GetMaterialReferenceInfo (model, materialIndex)); } } usedMaterials.sort ((a, b) => { return a.index - b.index; }); return usedMaterials; } this.navigator.Init ({ openFileBrowserDialog : () => { this.OpenFileBrowserDialog (); }, fitMeshToWindow : (meshInstanceId) => { this.FitMeshToWindow (meshInstanceId); }, fitMeshesToWindow : (meshInstanceIdSet) => { this.FitMeshesToWindow (meshInstanceIdSet); }, getMeshesForMaterial : (materialIndex) => { return GetMeshesForMaterial (this.viewer, this.model, materialIndex); }, getMaterialsForMesh : (meshInstanceId) => { return GetMaterialsForMesh (this.viewer, this.model, meshInstanceId); }, onMeshVisibilityChanged : () => { this.UpdateMeshesVisibility (); }, onMeshSelectionChanged : () => { this.UpdateMeshesSelection (); }, onSelectionCleared : () => { this.sidebar.AddObject3DProperties (this.model); }, onMeshSelected : (meshInstanceId) => { let meshInstance = this.model.GetMeshInstance (meshInstanceId); this.sidebar.AddObject3DProperties (meshInstance); }, onMaterialSelected : (materialIndex) => { this.sidebar.AddMaterialProperties (this.model.GetMaterial (materialIndex)); }, onResizeRequested : () => { this.layouter.Resize (); }, onShowHidePanels : (show) => { ShowDomElement (this.parameters.navigatorSplitterDiv, show); CookieSetBoolVal ('ov_show_navigator', show); } }); } UpdatePanelsVisibility () { let showNavigator = CookieGetBoolVal ('ov_show_navigator', true); let showSidebar = CookieGetBoolVal ('ov_show_sidebar', true); this.navigator.ShowPanels (showNavigator); this.sidebar.ShowPanels (showSidebar); } CreateHeaderButton (icon, title, link) { let buttonLink = CreateDomElement ('a'); buttonLink.setAttribute ('href', link); buttonLink.setAttribute ('target', '_blank'); buttonLink.setAttribute ('rel', 'noopener noreferrer'); InstallTooltip (buttonLink, title); AddSvgIconElement (buttonLink, icon, 'header_button'); this.parameters.headerButtonsDiv.appendChild (buttonLink); return buttonLink; } InitCookieConsent () { let accepted = CookieGetBoolVal ('ov_cookie_consent', false); if (accepted) { return; } let text = 'This website uses cookies to offer you better user experience. See the details at the Cookies Policy page.'; let popupDiv = AddDiv (document.body, 'ov_bottom_floating_panel'); AddDiv (popupDiv, 'ov_floating_panel_text', text); let acceptButton = AddDiv (popupDiv, 'ov_button ov_floating_panel_button', 'Accept'); acceptButton.addEventListener ('click', () => { CookieSetBoolVal ('ov_cookie_consent', true); popupDiv.remove (); }); } }