194 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
import TempNode from '../core/TempNode.js';
 | 
						|
import { addNodeClass } from '../core/Node.js';
 | 
						|
import { texture } from '../accessors/TextureNode.js';
 | 
						|
import { textureCubeUV } from './PMREMUtils.js';
 | 
						|
import { uniform } from '../core/UniformNode.js';
 | 
						|
import { NodeUpdateType } from '../core/constants.js';
 | 
						|
import { nodeProxy, vec3 } from '../shadernode/ShaderNode.js';
 | 
						|
import { WebGLCoordinateSystem } from 'three';
 | 
						|
 | 
						|
let _generator = null;
 | 
						|
 | 
						|
const _cache = new WeakMap();
 | 
						|
 | 
						|
function _generateCubeUVSize( imageHeight ) {
 | 
						|
 | 
						|
	const maxMip = Math.log2( imageHeight ) - 2;
 | 
						|
 | 
						|
	const texelHeight = 1.0 / imageHeight;
 | 
						|
 | 
						|
	const texelWidth = 1.0 / ( 3 * Math.max( Math.pow( 2, maxMip ), 7 * 16 ) );
 | 
						|
 | 
						|
	return { texelWidth, texelHeight, maxMip };
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
function _getPMREMFromTexture( texture ) {
 | 
						|
 | 
						|
	let cacheTexture = _cache.get( texture );
 | 
						|
 | 
						|
	const pmremVersion = cacheTexture !== undefined ? cacheTexture.pmremVersion : - 1;
 | 
						|
 | 
						|
	if ( pmremVersion !== texture.pmremVersion ) {
 | 
						|
 | 
						|
		if ( texture.isCubeTexture ) {
 | 
						|
 | 
						|
			if ( texture.source.data.some( ( texture ) => texture === undefined ) ) {
 | 
						|
 | 
						|
				throw new Error( 'PMREMNode: Undefined texture in CubeTexture. Use onLoad callback or async loader' );
 | 
						|
 | 
						|
			}
 | 
						|
 | 
						|
			cacheTexture = _generator.fromCubemap( texture, cacheTexture );
 | 
						|
 | 
						|
		} else {
 | 
						|
 | 
						|
			if ( texture.image === undefined ) {
 | 
						|
 | 
						|
				throw new Error( 'PMREMNode: Undefined image in Texture. Use onLoad callback or async loader' );
 | 
						|
 | 
						|
			}
 | 
						|
 | 
						|
			cacheTexture = _generator.fromEquirectangular( texture, cacheTexture );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		cacheTexture.pmremVersion = texture.pmremVersion;
 | 
						|
 | 
						|
		_cache.set( texture, cacheTexture );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	return cacheTexture.texture;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
class PMREMNode extends TempNode {
 | 
						|
 | 
						|
	constructor( value, uvNode = null, levelNode = null ) {
 | 
						|
 | 
						|
		super( 'vec3' );
 | 
						|
 | 
						|
		this._value = value;
 | 
						|
		this._pmrem = null;
 | 
						|
 | 
						|
		this.uvNode = uvNode;
 | 
						|
		this.levelNode = levelNode;
 | 
						|
 | 
						|
		this._generator = null;
 | 
						|
		this._texture = texture( null );
 | 
						|
		this._width = uniform( 0 );
 | 
						|
		this._height = uniform( 0 );
 | 
						|
		this._maxMip = uniform( 0 );
 | 
						|
 | 
						|
		this.updateBeforeType = NodeUpdateType.RENDER;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	set value( value ) {
 | 
						|
 | 
						|
		this._value = value;
 | 
						|
		this._pmrem = null;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	get value() {
 | 
						|
 | 
						|
		return this._value;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	updateFromTexture( texture ) {
 | 
						|
 | 
						|
		const cubeUVSize = _generateCubeUVSize( texture.image.height );
 | 
						|
 | 
						|
		this._texture.value = texture;
 | 
						|
		this._width.value = cubeUVSize.texelWidth;
 | 
						|
		this._height.value = cubeUVSize.texelHeight;
 | 
						|
		this._maxMip.value = cubeUVSize.maxMip;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	updateBefore() {
 | 
						|
 | 
						|
		let pmrem = this._pmrem;
 | 
						|
 | 
						|
		const pmremVersion = pmrem ? pmrem.pmremVersion : - 1;
 | 
						|
		const texture = this._value;
 | 
						|
 | 
						|
		if ( pmremVersion !== texture.pmremVersion ) {
 | 
						|
 | 
						|
			if ( texture.isPMREMTexture === true ) {
 | 
						|
 | 
						|
				pmrem = texture;
 | 
						|
 | 
						|
			} else {
 | 
						|
 | 
						|
				pmrem = _getPMREMFromTexture( texture );
 | 
						|
 | 
						|
			}
 | 
						|
 | 
						|
			this._pmrem = pmrem;
 | 
						|
 | 
						|
			this.updateFromTexture( pmrem );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	setup( builder ) {
 | 
						|
 | 
						|
		if ( _generator === null ) {
 | 
						|
 | 
						|
			_generator = builder.createPMREMGenerator();
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		//
 | 
						|
 | 
						|
		this.updateBefore( builder );
 | 
						|
 | 
						|
		//
 | 
						|
 | 
						|
		let uvNode = this.uvNode;
 | 
						|
 | 
						|
		if ( uvNode === null && builder.context.getUV ) {
 | 
						|
 | 
						|
			uvNode = builder.context.getUV( this );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		//
 | 
						|
 | 
						|
		const texture = this.value;
 | 
						|
 | 
						|
		if ( builder.renderer.coordinateSystem === WebGLCoordinateSystem && texture.isPMREMTexture !== true && texture.isRenderTargetTexture === true ) {
 | 
						|
 | 
						|
			uvNode = vec3( uvNode.x.negate(), uvNode.yz );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		//
 | 
						|
 | 
						|
		let levelNode = this.levelNode;
 | 
						|
 | 
						|
		if ( levelNode === null && builder.context.getTextureLevel ) {
 | 
						|
 | 
						|
			levelNode = builder.context.getTextureLevel( this );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		//
 | 
						|
 | 
						|
		return textureCubeUV( this._texture, uvNode, levelNode, this._width, this._height, this._maxMip );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
export const pmremTexture = nodeProxy( PMREMNode );
 | 
						|
 | 
						|
addNodeClass( 'PMREMNode', PMREMNode );
 | 
						|
 | 
						|
export default PMREMNode;
 |