85 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			85 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
import { addNodeMaterial } from './NodeMaterial.js';
 | 
						|
import { transformedNormalView } from '../accessors/NormalNode.js';
 | 
						|
import { positionViewDirection } from '../accessors/PositionNode.js';
 | 
						|
import PhysicalLightingModel from '../functions/PhysicalLightingModel.js';
 | 
						|
import MeshPhysicalNodeMaterial from './MeshPhysicalNodeMaterial.js';
 | 
						|
import { float, vec3 } from '../shadernode/ShaderNode.js';
 | 
						|
 | 
						|
class SSSLightingModel extends PhysicalLightingModel {
 | 
						|
 | 
						|
	constructor( useClearcoat, useSheen, useIridescence, useSSS ) {
 | 
						|
 | 
						|
		super( useClearcoat, useSheen, useIridescence );
 | 
						|
 | 
						|
		this.useSSS = useSSS;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	direct( { lightDirection, lightColor, reflectedLight }, stack, builder ) {
 | 
						|
 | 
						|
		if ( this.useSSS === true ) {
 | 
						|
 | 
						|
			const material = builder.material;
 | 
						|
 | 
						|
			const { thicknessColorNode, thicknessDistortionNode, thicknessAmbientNode, thicknessAttenuationNode, thicknessPowerNode, thicknessScaleNode } = material;
 | 
						|
 | 
						|
			const scatteringHalf = lightDirection.add( transformedNormalView.mul( thicknessDistortionNode ) ).normalize();
 | 
						|
			const scatteringDot = float( positionViewDirection.dot( scatteringHalf.negate() ).saturate().pow( thicknessPowerNode ).mul( thicknessScaleNode ) );
 | 
						|
			const scatteringIllu = vec3( scatteringDot.add( thicknessAmbientNode ).mul( thicknessColorNode ) );
 | 
						|
 | 
						|
			reflectedLight.directDiffuse.addAssign( scatteringIllu.mul( thicknessAttenuationNode.mul( lightColor ) ) );
 | 
						|
 | 
						|
		}
 | 
						|
 | 
						|
		super.direct( { lightDirection, lightColor, reflectedLight }, stack, builder );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
class MeshSSSNodeMaterial extends MeshPhysicalNodeMaterial {
 | 
						|
 | 
						|
	constructor( parameters ) {
 | 
						|
 | 
						|
		super( parameters );
 | 
						|
 | 
						|
		this.thicknessColorNode = null;
 | 
						|
		this.thicknessDistortionNode = float( 0.1 );
 | 
						|
		this.thicknessAmbientNode = float( 0.0 );
 | 
						|
		this.thicknessAttenuationNode = float( .1 );
 | 
						|
		this.thicknessPowerNode = float( 2.0 );
 | 
						|
		this.thicknessScaleNode = float( 10.0 );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	get useSSS() {
 | 
						|
 | 
						|
		return this.thicknessColorNode !== null;
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	setupLightingModel( /*builder*/ ) {
 | 
						|
 | 
						|
		return new SSSLightingModel( this.useClearcoat, this.useSheen, this.useIridescence, this.useSSS );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
	copy( source ) {
 | 
						|
 | 
						|
		this.thicknessColorNode = source.thicknessColorNode;
 | 
						|
		this.thicknessDistortionNode = source.thicknessDistortionNode;
 | 
						|
		this.thicknessAmbientNode = source.thicknessAmbientNode;
 | 
						|
		this.thicknessAttenuationNode = source.thicknessAttenuationNode;
 | 
						|
		this.thicknessPowerNode = source.thicknessPowerNode;
 | 
						|
		this.thicknessScaleNode = source.thicknessScaleNode;
 | 
						|
 | 
						|
		return super.copy( source );
 | 
						|
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
export default MeshSSSNodeMaterial;
 | 
						|
 | 
						|
addNodeMaterial( 'MeshSSSNodeMaterial', MeshSSSNodeMaterial );
 |