135 lines
3.6 KiB
JavaScript
135 lines
3.6 KiB
JavaScript
|
import DataMap from './DataMap.js';
|
||
|
import Color4 from './Color4.js';
|
||
|
import { Mesh, SphereGeometry, BackSide, LinearSRGBColorSpace } from 'three';
|
||
|
import { vec4, context, normalWorld, backgroundBlurriness, backgroundIntensity, NodeMaterial, modelViewProjection } from '../../nodes/Nodes.js';
|
||
|
|
||
|
const _clearColor = new Color4();
|
||
|
|
||
|
class Background extends DataMap {
|
||
|
|
||
|
constructor( renderer, nodes ) {
|
||
|
|
||
|
super();
|
||
|
|
||
|
this.renderer = renderer;
|
||
|
this.nodes = nodes;
|
||
|
|
||
|
}
|
||
|
|
||
|
update( scene, renderList, renderContext ) {
|
||
|
|
||
|
const renderer = this.renderer;
|
||
|
const background = this.nodes.getBackgroundNode( scene ) || scene.background;
|
||
|
|
||
|
let forceClear = false;
|
||
|
|
||
|
if ( background === null ) {
|
||
|
|
||
|
// no background settings, use clear color configuration from the renderer
|
||
|
|
||
|
renderer._clearColor.getRGB( _clearColor, LinearSRGBColorSpace );
|
||
|
_clearColor.a = renderer._clearColor.a;
|
||
|
|
||
|
} else if ( background.isColor === true ) {
|
||
|
|
||
|
// background is an opaque color
|
||
|
|
||
|
background.getRGB( _clearColor, LinearSRGBColorSpace );
|
||
|
_clearColor.a = 1;
|
||
|
|
||
|
forceClear = true;
|
||
|
|
||
|
} else if ( background.isNode === true ) {
|
||
|
|
||
|
const sceneData = this.get( scene );
|
||
|
const backgroundNode = background;
|
||
|
|
||
|
_clearColor.copy( renderer._clearColor );
|
||
|
|
||
|
let backgroundMesh = sceneData.backgroundMesh;
|
||
|
|
||
|
if ( backgroundMesh === undefined ) {
|
||
|
|
||
|
const backgroundMeshNode = context( vec4( backgroundNode ).mul( backgroundIntensity ), {
|
||
|
// @TODO: Add Texture2D support using node context
|
||
|
getUV: () => normalWorld,
|
||
|
getTextureLevel: () => backgroundBlurriness
|
||
|
} );
|
||
|
|
||
|
let viewProj = modelViewProjection();
|
||
|
viewProj = viewProj.setZ( viewProj.w );
|
||
|
|
||
|
const nodeMaterial = new NodeMaterial();
|
||
|
nodeMaterial.side = BackSide;
|
||
|
nodeMaterial.depthTest = false;
|
||
|
nodeMaterial.depthWrite = false;
|
||
|
nodeMaterial.fog = false;
|
||
|
nodeMaterial.vertexNode = viewProj;
|
||
|
nodeMaterial.fragmentNode = backgroundMeshNode;
|
||
|
|
||
|
sceneData.backgroundMeshNode = backgroundMeshNode;
|
||
|
sceneData.backgroundMesh = backgroundMesh = new Mesh( new SphereGeometry( 1, 32, 32 ), nodeMaterial );
|
||
|
backgroundMesh.frustumCulled = false;
|
||
|
|
||
|
backgroundMesh.onBeforeRender = function ( renderer, scene, camera ) {
|
||
|
|
||
|
this.matrixWorld.copyPosition( camera.matrixWorld );
|
||
|
|
||
|
};
|
||
|
|
||
|
}
|
||
|
|
||
|
const backgroundCacheKey = backgroundNode.getCacheKey();
|
||
|
|
||
|
if ( sceneData.backgroundCacheKey !== backgroundCacheKey ) {
|
||
|
|
||
|
sceneData.backgroundMeshNode.node = vec4( backgroundNode ).mul( backgroundIntensity );
|
||
|
|
||
|
backgroundMesh.material.needsUpdate = true;
|
||
|
|
||
|
sceneData.backgroundCacheKey = backgroundCacheKey;
|
||
|
|
||
|
}
|
||
|
|
||
|
renderList.unshift( backgroundMesh, backgroundMesh.geometry, backgroundMesh.material, 0, 0, null );
|
||
|
|
||
|
} else {
|
||
|
|
||
|
console.error( 'THREE.Renderer: Unsupported background configuration.', background );
|
||
|
|
||
|
}
|
||
|
|
||
|
//
|
||
|
|
||
|
if ( renderer.autoClear === true || forceClear === true ) {
|
||
|
|
||
|
_clearColor.multiplyScalar( _clearColor.a );
|
||
|
|
||
|
const clearColorValue = renderContext.clearColorValue;
|
||
|
|
||
|
clearColorValue.r = _clearColor.r;
|
||
|
clearColorValue.g = _clearColor.g;
|
||
|
clearColorValue.b = _clearColor.b;
|
||
|
clearColorValue.a = _clearColor.a;
|
||
|
|
||
|
renderContext.depthClearValue = renderer._clearDepth;
|
||
|
renderContext.stencilClearValue = renderer._clearStencil;
|
||
|
|
||
|
renderContext.clearColor = renderer.autoClearColor === true;
|
||
|
renderContext.clearDepth = renderer.autoClearDepth === true;
|
||
|
renderContext.clearStencil = renderer.autoClearStencil === true;
|
||
|
|
||
|
} else {
|
||
|
|
||
|
renderContext.clearColor = false;
|
||
|
renderContext.clearDepth = false;
|
||
|
renderContext.clearStencil = false;
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
export default Background;
|