I want to use THREEjs with Angular and want to create a good structure. Therefore I am trying to put my render loop into a service.
requestAnimationFrame(this.requestRenderer.bind(this));
but I get this following error:
ERROR Error: Uncaught (in promise): TypeError: Cannot read properties of undefined (reading 'add')
I am uncertain if this is even possible:
my page script:
import {
AfterViewInit,
Component,
ElementRef,
Input,
OnInit,
ViewChild,
} from '@angular/core';
import * as THREE from 'three';
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls';
import { RenderingServiceService } from '../shared/services/rendering-service/rendering-service.service';
@Component({
selector: 'app-tab1',
templateUrl: 'tab1.page.html',
styleUrls: ['tab1.page.scss'],
})
export class Tab1Page implements AfterViewInit {
@ViewChild('canvas') private canvasRef: ElementRef;
scene!: THREE.Scene;
controls!: OrbitControls;
// Set Light
light: THREE.DirectionalLight;
lightProps = { intensity: 1, color: 0xffffff };
// Set Cube
geometry = new THREE.BoxGeometry(1, 1, 1);
material = new THREE.MeshPhongMaterial({ color: 0x44aa88 });
cubes: THREE.Mesh[] = [];
camera: THREE.PerspectiveCamera;
renderer!: THREE.WebGLRenderer;
constructor(private myrenderService: RenderingServiceService) {}
ngAfterViewInit(): void {
// Set Scene
this.camera = this.myrenderService.setCamerara();
this.renderer = this.myrenderService.setRenderer(this.canvas);
// Add to scene
this.cubes = [
this.makeInstance(this.geometry, 0x44aa88, 0),
this.makeInstance(this.geometry, 0x8844aa, -2),
this.makeInstance(this.geometry, 0xaa8844, 2),
];
this.cubes.forEach((cube) => this.scene.add(cube));
// Set Light
this.light = new THREE.DirectionalLight(
this.lightProps.color,
this.lightProps.intensity
);
this.light.position.set(-1, 2, 4);
this.scene.add(this.light);
// Controls
this.controls = new OrbitControls(this.camera, this.renderer.domElement);
// this.animate();
// requestAnimationFrame(this.render);
this.myrenderService.requestRenderer(this.scene, this.camera, this.cubes);
}
get canvas(): HTMLCanvasElement {
return this.canvasRef.nativeElement;
}
makeInstance(geometry, color, x) {
const material = new THREE.MeshPhongMaterial({ color });
const cube = new THREE.Mesh(geometry, material);
this.scene.add(cube);
cube.position.x = x;
return cube;
}
}
my rendering service:
import { Injectable } from '@angular/core';
import * as THREE from 'three';
@Injectable({
providedIn: 'root',
})
export class RenderingServiceService {
//? Helper Properties (Private Properties);
camera!: THREE.PerspectiveCamera;
renderer!: THREE.WebGLRenderer;
rotationSpeedX = 0.05;
rotationSpeedY = 0.01;
// Set Camera Property
cameraProps = {
fov: 75,
aspect: 2,
near: 0.1,
far: 5,
};
constructor() {}
setRenderer(canvas) {
// Set Canvas
this.renderer = new THREE.WebGLRenderer({ canvas: canvas });
return this.renderer;
}
setCamerara() {
// Set Camera
this.camera = new THREE.PerspectiveCamera(
this.cameraProps.fov,
this.cameraProps.aspect,
this.cameraProps.near,
this.cameraProps.far
);
this.camera.position.z = 2;
return this.camera;
}
requestRenderer(scene, camera, cubes) {
this.renderer.render(scene, camera);
requestAnimationFrame(this.requestRenderer.bind(this));
const canvas = this.renderer.domElement;
this.camera.aspect = canvas.clientWidth / canvas.clientHeight;
this.camera.updateProjectionMatrix();
if (this.resizeRendererToDisplaySize(this.renderer)) {
this.camera.aspect = canvas.clientWidth / canvas.clientHeight;
this.camera.updateProjectionMatrix();
}
this.animateCube(cubes);
}
renderrender() {}
resizeRendererToDisplaySize(renderer) {
const canvas = renderer.domElement;
const pixelRatio = window.devicePixelRatio;
const width = canvas.clientWidth * pixelRatio || 0;
const height = canvas.clientHeight * pixelRatio || 0;
const needResize = canvas.width !== width || canvas.height !== height;
if (needResize) {
renderer.setSize(width, height, false);
}
return needResize;
}
animateCube(cubes) {
const time = new Date();
const getTime = time.getTime() * 0.001;
cubes.forEach((cube) => {
cube.rotation.x = getTime * this.rotationSpeedX * 100;
cube.rotation.y = getTime * this.rotationSpeedY * 100;
});
}
}