UNPKG

@babylonjs/core

Version:

Getting started? Play directly with the Babylon.js API using our [playground](https://playground.babylonjs.com/). It also contains a lot of samples to learn how to use it.

273 lines 15.6 kB
import { AbstractEngine } from "../Engines/abstractEngine.js"; import { EngineStore } from "../Engines/engineStore.js"; /** * Class used for the default loading screen * @see https://doc.babylonjs.com/features/featuresDeepDive/scene/customLoadingScreen */ export class DefaultLoadingScreen { /** * Creates a new default loading screen * @param _renderingCanvas defines the canvas used to render the scene * @param _loadingText defines the default text to display * @param _loadingDivBackgroundColor defines the default background color */ constructor(_renderingCanvas, _loadingText = "", _loadingDivBackgroundColor = "black") { this._renderingCanvas = _renderingCanvas; this._loadingText = _loadingText; this._loadingDivBackgroundColor = _loadingDivBackgroundColor; /** * Maps a loading `HTMLDivElement` to a tuple containing the associated `HTMLCanvasElement` * and its `DOMRect` (or `null` if not yet available). */ this._loadingDivToRenderingCanvasMap = new Map(); // Resize this._resizeLoadingUI = () => { if (!this._isLoading) { return; } this._loadingDivToRenderingCanvasMap.forEach(([canvas, previousCanvasRect], loadingDiv) => { const currentCanvasRect = canvas.getBoundingClientRect(); if (this._isCanvasLayoutChanged(previousCanvasRect, currentCanvasRect)) { const canvasPositioning = window.getComputedStyle(canvas).position; loadingDiv.style.position = canvasPositioning === "fixed" ? "fixed" : "absolute"; loadingDiv.style.left = currentCanvasRect.left + window.scrollX + "px"; loadingDiv.style.top = currentCanvasRect.top + window.scrollY + "px"; loadingDiv.style.width = currentCanvasRect.width + "px"; loadingDiv.style.height = currentCanvasRect.height + "px"; this._loadingDivToRenderingCanvasMap.set(loadingDiv, [canvas, currentCanvasRect]); } }); }; } /** * Function called to display the loading screen */ displayLoadingUI() { if (this._isLoading) { // Do not add a loading screen if it is already loading return; } this._isLoading = true; // get current engine by rendering canvas this._engine = EngineStore.Instances.find((engine) => engine.getRenderingCanvas() === this._renderingCanvas); const loadingDiv = document.createElement("div"); loadingDiv.id = "babylonjsLoadingDiv"; loadingDiv.style.opacity = "0"; loadingDiv.style.transition = "opacity 1.5s ease"; loadingDiv.style.pointerEvents = "none"; loadingDiv.style.display = "grid"; loadingDiv.style.gridTemplateRows = "100%"; loadingDiv.style.gridTemplateColumns = "100%"; loadingDiv.style.justifyItems = "center"; loadingDiv.style.alignItems = "center"; // Loading text this._loadingTextDiv = document.createElement("div"); this._loadingTextDiv.style.position = "absolute"; this._loadingTextDiv.style.left = "0"; this._loadingTextDiv.style.top = "50%"; this._loadingTextDiv.style.marginTop = "80px"; this._loadingTextDiv.style.width = "100%"; this._loadingTextDiv.style.height = "20px"; this._loadingTextDiv.style.fontFamily = "Arial"; this._loadingTextDiv.style.fontSize = "14px"; this._loadingTextDiv.style.color = "white"; this._loadingTextDiv.style.textAlign = "center"; this._loadingTextDiv.style.zIndex = "1"; this._loadingTextDiv.innerHTML = "Loading"; loadingDiv.appendChild(this._loadingTextDiv); //set the predefined text this._loadingTextDiv.innerHTML = this._loadingText; // Generating keyframes this._style = document.createElement("style"); const keyFrames = `@-webkit-keyframes spin1 {\ 0% { -webkit-transform: rotate(0deg);} 100% { -webkit-transform: rotate(360deg);} }\ @keyframes spin1 {\ 0% { transform: rotate(0deg);} 100% { transform: rotate(360deg);} }`; this._style.innerHTML = keyFrames; document.getElementsByTagName("head")[0].appendChild(this._style); const svgSupport = !!window.SVGSVGElement; // Loading img const imgBack = new Image(); if (!DefaultLoadingScreen.DefaultLogoUrl) { imgBack.src = !svgSupport ? "https://cdn.babylonjs.com/Assets/babylonLogo.png" : ``; } else { imgBack.src = DefaultLoadingScreen.DefaultLogoUrl; } imgBack.style.width = "150px"; imgBack.style.gridColumn = "1"; imgBack.style.gridRow = "1"; imgBack.style.top = "50%"; imgBack.style.left = "50%"; imgBack.style.transform = "translate(-50%, -50%)"; imgBack.style.position = "absolute"; const imageSpinnerContainer = document.createElement("div"); imageSpinnerContainer.style.width = "300px"; imageSpinnerContainer.style.gridColumn = "1"; imageSpinnerContainer.style.gridRow = "1"; imageSpinnerContainer.style.top = "50%"; imageSpinnerContainer.style.left = "50%"; imageSpinnerContainer.style.transform = "translate(-50%, -50%)"; imageSpinnerContainer.style.position = "absolute"; // Loading spinner const imgSpinner = new Image(); if (!DefaultLoadingScreen.DefaultSpinnerUrl) { imgSpinner.src = !svgSupport ? "https://cdn.babylonjs.com/Assets/loadingIcon.png" : ``; } else { imgSpinner.src = DefaultLoadingScreen.DefaultSpinnerUrl; } imgSpinner.style.animation = "spin1 0.75s infinite linear"; imgSpinner.style.transformOrigin = "50% 50%"; if (!svgSupport) { const logoSize = { w: 16, h: 18.5 }; const loadingSize = { w: 30, h: 30 }; // set styling correctly imgBack.style.width = `${logoSize.w}vh`; imgBack.style.height = `${logoSize.h}vh`; imgBack.style.left = `calc(50% - ${logoSize.w / 2}vh)`; imgBack.style.top = `calc(50% - ${logoSize.h / 2}vh)`; imgSpinner.style.width = `${loadingSize.w}vh`; imgSpinner.style.height = `${loadingSize.h}vh`; imgSpinner.style.left = `calc(50% - ${loadingSize.w / 2}vh)`; imgSpinner.style.top = `calc(50% - ${loadingSize.h / 2}vh)`; } imageSpinnerContainer.appendChild(imgSpinner); loadingDiv.appendChild(imgBack); loadingDiv.appendChild(imageSpinnerContainer); loadingDiv.style.backgroundColor = this._loadingDivBackgroundColor; loadingDiv.style.opacity = "1"; const canvases = []; const views = this._engine.views; if (views?.length) { for (const view of views) { if (view.enabled) { canvases.push(view.target); } } } else { canvases.push(this._renderingCanvas); } for (let i = 0; i < canvases.length; i++) { const canvas = canvases[i]; const clonedLoadingDiv = loadingDiv.cloneNode(true); clonedLoadingDiv.id += `-${i}`; this._loadingDivToRenderingCanvasMap.set(clonedLoadingDiv, [canvas, null]); } this._resizeLoadingUI(); this._resizeObserver = this._engine.onResizeObservable.add(() => { this._resizeLoadingUI(); }); this._loadingDivToRenderingCanvasMap.forEach((_, loadingDiv) => { document.body.appendChild(loadingDiv); }); } /** * Function called to hide the loading screen */ hideLoadingUI() { if (!this._isLoading) { return; } let completedTransitions = 0; const onTransitionEnd = (event) => { const loadingDiv = event.target; // ensure that ending transition event is generated by one of the current loadingDivs const isTransitionEndOnLoadingDiv = this._loadingDivToRenderingCanvasMap.has(loadingDiv); if (isTransitionEndOnLoadingDiv) { completedTransitions++; loadingDiv.remove(); const allTransitionsCompleted = completedTransitions === this._loadingDivToRenderingCanvasMap.size; if (allTransitionsCompleted) { if (this._loadingTextDiv) { this._loadingTextDiv.remove(); this._loadingTextDiv = null; } if (this._style) { this._style.remove(); this._style = null; } window.removeEventListener("transitionend", onTransitionEnd); this._engine.onResizeObservable.remove(this._resizeObserver); this._loadingDivToRenderingCanvasMap.clear(); this._engine = null; this._isLoading = false; } } }; this._loadingDivToRenderingCanvasMap.forEach((_, loadingDiv) => { loadingDiv.style.opacity = "0"; }); window.addEventListener("transitionend", onTransitionEnd); } /** * Gets or sets the text to display while loading */ set loadingUIText(text) { this._loadingText = text; if (this._loadingTextDiv) { this._loadingDivToRenderingCanvasMap.forEach((_, loadingDiv) => { // set loadingTextDiv of current loadingDiv loadingDiv.children[0].innerHTML = this._loadingText; }); } } get loadingUIText() { return this._loadingText; } /** * Gets or sets the color to use for the background */ get loadingUIBackgroundColor() { return this._loadingDivBackgroundColor; } set loadingUIBackgroundColor(color) { this._loadingDivBackgroundColor = color; if (!this._isLoading) { return; } this._loadingDivToRenderingCanvasMap.forEach((_, loadingDiv) => { loadingDiv.style.backgroundColor = this._loadingDivBackgroundColor; }); } /** * Checks if the layout of the canvas has changed by comparing the current layout * rectangle with the previous one. * * This function compares of the two `DOMRect` objects to determine if any of the layout dimensions have changed. * If the layout has changed or if there is no previous layout (i.e., `previousCanvasRect` is `null`), * it returns `true`. Otherwise, it returns `false`. * * @param previousCanvasRect defines the previously recorded `DOMRect` of the canvas, or `null` if no previous state exists. * @param currentCanvasRect defines the current `DOMRect` of the canvas to compare against the previous layout. * @returns `true` if the layout has changed, otherwise `false`. */ _isCanvasLayoutChanged(previousCanvasRect, currentCanvasRect) { return (!previousCanvasRect || previousCanvasRect.left !== currentCanvasRect.left || previousCanvasRect.top !== currentCanvasRect.top || previousCanvasRect.right !== currentCanvasRect.right || previousCanvasRect.bottom !== currentCanvasRect.bottom || previousCanvasRect.width !== currentCanvasRect.width || previousCanvasRect.height !== currentCanvasRect.height || previousCanvasRect.x !== currentCanvasRect.x || previousCanvasRect.y !== currentCanvasRect.y); } } /** Gets or sets the logo url to use for the default loading screen */ DefaultLoadingScreen.DefaultLogoUrl = ""; /** Gets or sets the spinner url to use for the default loading screen */ DefaultLoadingScreen.DefaultSpinnerUrl = ""; AbstractEngine.DefaultLoadingScreenFactory = (canvas) => { return new DefaultLoadingScreen(canvas); }; //# sourceMappingURL=loadingScreen.js.map