@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.
274 lines • 15.7 kB
JavaScript
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");
this._style.type = "text/css";
const keyFrames = `@-webkit-keyframes spin1 {\
0% { -webkit-transform: rotate(0deg);}
100% { -webkit-transform: rotate(360deg);}
}\
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