feat: device performance optimization
All checks were successful
Build Demo / build (push) Successful in 36s

This commit is contained in:
Cdm2883 2024-08-10 23:03:46 +08:00
parent 8c031efee0
commit 23cff070e2
6 changed files with 86 additions and 4 deletions

View File

@ -7,7 +7,7 @@
<option name="description" /> <option name="description" />
<option name="exitCodeBehavior" value="ERROR" /> <option name="exitCodeBehavior" value="ERROR" />
<option name="fileExtension" value="*" /> <option name="fileExtension" value="*" />
<option name="immediateSync" value="true" /> <option name="immediateSync" value="false" />
<option name="name" value="dev-server-rebuild" /> <option name="name" value="dev-server-rebuild" />
<option name="output" value="$ContentRoot$/site/" /> <option name="output" value="$ContentRoot$/site/" />
<option name="outputFilters"> <option name="outputFilters">

View File

@ -16,3 +16,11 @@
height: 100%; height: 100%;
pointer-events: none; pointer-events: none;
} }
#fullscreen-loading {
position: fixed;
width: 100%;
height: 100%;
/*pointer-events: fill;*/
pointer-events: none;
}

View File

@ -16,6 +16,10 @@
<div id="pseudo-3d-layer"></div> <div id="pseudo-3d-layer"></div>
<div id="fullscreen-loading">
</div>
<script src="assets/javascripts/home.js"></script> <script src="assets/javascripts/home.js"></script>
</body> </body>
</html> </html>

View File

@ -1,14 +1,20 @@
import WebGL from "three/examples/jsm/capabilities/WebGL.js"; import WebGL from "three/examples/jsm/capabilities/WebGL.js";
import WorldViewport from "./world-viewport.js"; import WorldViewport from "./world-viewport.js";
import Pseudo3DLayer from "./pseudo-3d-layer.js"; import Pseudo3DLayer from "./pseudo-3d-layer.js";
import FullscreenLoading from "./loading.js";
if (!WebGL.isWebGL2Available()) { if (!WebGL.isWebGL2Available()) {
location.replace('./webgl-less.html'); location.replace('./webgl-less.html');
throw 0; throw 0;
} }
const loading = new FullscreenLoading();
// setTimeout(() => loading.loading(), 1000);
// setTimeout(() => loading.loaded(), 10000);
const canvas = document.getElementById('three-viewport'); const canvas = document.getElementById('three-viewport');
const world = new WorldViewport(canvas); const world = new WorldViewport(canvas);
world.loadingIndicator = loading;
world.setup(); world.setup();
console.log(world); console.log(world);

28
src/loading.js Normal file
View File

@ -0,0 +1,28 @@
export default class FullscreenLoading {
counter = 0;
isLoading = false;
loading() {
if (++this.counter > 0 && !this.isLoading) {
this.isLoading = true;
this.onLoading();
}
}
loaded() {
if (--this.counter <= 0 && this.isLoading) {
this.isLoading = false;
this.onLoaded();
}
}
constructor() {
this.container = document.getElementById('fullscreen-loading');
}
onLoading() {
console.log('onLoading');
}
onLoaded() {
console.log('onLoaded');
}
}

View File

@ -34,7 +34,28 @@ const motionBlurShader = {
`, `,
}; };
class FpsProfiler {
count;
static NowProvider = performance || Date;
static Now = () => FpsProfiler.NowProvider.now();
prevTime = FpsProfiler.Now();
frames = 0;
end() {
this.frames++;
const time = FpsProfiler.Now();
if (time >= this.prevTime + 1000) {
this.count = (this.frames * 1000) / (time - this.prevTime);
this.prevTime = time;
this.frames = 0;
}
console.log(`fps: ${this.count}`);
}
}
export default class WorldViewport { export default class WorldViewport {
fps = new FpsProfiler();
highQuality = true;
/** @param {HTMLCanvasElement} canvas */ /** @param {HTMLCanvasElement} canvas */
constructor(canvas) { constructor(canvas) {
this.canvas = canvas; this.canvas = canvas;
@ -53,9 +74,10 @@ export default class WorldViewport {
} }
resize() { resize() {
const { innerWidth: width, innerHeight: height } = window; const { innerWidth: width, innerHeight: height } = window;
this.renderer.setPixelRatio(window.devicePixelRatio); const dpi = this.highQuality ? window.devicePixelRatio : 1;
this.renderer.setPixelRatio(dpi);
this.renderer.setSize(width, height); this.renderer.setSize(width, height);
this.composer.setPixelRatio(window.devicePixelRatio); this.composer.setPixelRatio(dpi);
this.composer.setSize(width, height); this.composer.setSize(width, height);
this.camera.aspect = width / height; this.camera.aspect = width / height;
this.camera.updateProjectionMatrix(); this.camera.updateProjectionMatrix();
@ -72,16 +94,23 @@ export default class WorldViewport {
this.composer.addPass(this.bloomPass); this.composer.addPass(this.bloomPass);
} }
/** @type FullscreenLoading */
loadingIndicator;
setup() { setup() {
const loader = new GLTFLoader(); const loader = new GLTFLoader();
this.loadingIndicator.loading();
loader.load("assets/models/website_viewport_1.glb", data => { loader.load("assets/models/website_viewport_1.glb", data => {
const scene = data.scene; const scene = data.scene;
this.scene.add(scene); this.scene.add(scene);
this.loadingIndicator.loaded();
}); });
this.loadingIndicator.loading();
loader.load("assets/models/website_viewport_2.glb", data => { loader.load("assets/models/website_viewport_2.glb", data => {
const scene = data.scene; const scene = data.scene;
scene.position.set(0, 0, 64); scene.position.set(0, 0, 64);
this.scene.add(scene); this.scene.add(scene);
this.loadingIndicator.loaded();
}); });
{ {
@ -255,7 +284,11 @@ export default class WorldViewport {
renderPseudo; renderPseudo;
animate() { animate() {
requestAnimationFrame(this.animate.bind(this)); const highQuality = this.highQuality;
if (highQuality && this.fps.count < 25) this.highQuality = false;
else if (!highQuality && this.fps.count >= 40) this.highQuality = true;
// this.highQuality = this.fps.count >= 25;
if (highQuality !== this.highQuality) this.resize();
this.velocity.set(this.camera.position.z - this.previousCameraPosition.z, 0); this.velocity.set(this.camera.position.z - this.previousCameraPosition.z, 0);
this.previousCameraPosition.copy(this.camera.position); this.previousCameraPosition.copy(this.camera.position);
@ -277,6 +310,9 @@ export default class WorldViewport {
if (this.renderPseudo) this.renderPseudo(); if (this.renderPseudo) this.renderPseudo();
this.composer.render(); this.composer.render();
this.fps.end();
requestAnimationFrame(this.animate.bind(this));
} }
} }