添加关照、全局等高线、修改图层问题
This commit is contained in:
145
static/sdk/three/jsm/nodes/accessors/ClippingNode.js
Normal file
145
static/sdk/three/jsm/nodes/accessors/ClippingNode.js
Normal file
@ -0,0 +1,145 @@
|
||||
|
||||
import Node from '../core/Node.js';
|
||||
import { nodeObject } from '../shadernode/ShaderNode.js';
|
||||
import { positionView } from './PositionNode.js';
|
||||
import { diffuseColor, property } from '../core/PropertyNode.js';
|
||||
import { tslFn } from '../shadernode/ShaderNode.js';
|
||||
import { loop } from '../utils/LoopNode.js';
|
||||
import { smoothstep } from '../math/MathNode.js';
|
||||
import { uniforms } from './UniformsNode.js';
|
||||
|
||||
class ClippingNode extends Node {
|
||||
|
||||
constructor( scope = ClippingNode.DEFAULT ) {
|
||||
|
||||
super();
|
||||
|
||||
this.scope = scope;
|
||||
|
||||
}
|
||||
|
||||
setup( builder ) {
|
||||
|
||||
super.setup( builder );
|
||||
|
||||
const clippingContext = builder.clippingContext;
|
||||
const { localClipIntersection, localClippingCount, globalClippingCount } = clippingContext;
|
||||
|
||||
const numClippingPlanes = globalClippingCount + localClippingCount;
|
||||
const numUnionClippingPlanes = localClipIntersection ? numClippingPlanes - localClippingCount : numClippingPlanes;
|
||||
|
||||
if ( this.scope === ClippingNode.ALPHA_TO_COVERAGE ) {
|
||||
|
||||
return this.setupAlphaToCoverage( clippingContext.planes, numClippingPlanes, numUnionClippingPlanes );
|
||||
|
||||
} else {
|
||||
|
||||
return this.setupDefault( clippingContext.planes, numClippingPlanes, numUnionClippingPlanes );
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
setupAlphaToCoverage( planes, numClippingPlanes, numUnionClippingPlanes ) {
|
||||
|
||||
return tslFn( () => {
|
||||
|
||||
const clippingPlanes = uniforms( planes );
|
||||
|
||||
const distanceToPlane = property( 'float', 'distanceToPlane' );
|
||||
const distanceGradient = property( 'float', 'distanceToGradient' );
|
||||
|
||||
const clipOpacity = property( 'float', 'clipOpacity' );
|
||||
|
||||
clipOpacity.assign( 1 );
|
||||
|
||||
let plane;
|
||||
|
||||
loop( numUnionClippingPlanes, ( { i } ) => {
|
||||
|
||||
plane = clippingPlanes.element( i );
|
||||
|
||||
distanceToPlane.assign( positionView.dot( plane.xyz ).negate().add( plane.w ) );
|
||||
distanceGradient.assign( distanceToPlane.fwidth().div( 2.0 ) );
|
||||
|
||||
clipOpacity.mulAssign( smoothstep( distanceGradient.negate(), distanceGradient, distanceToPlane ) );
|
||||
|
||||
clipOpacity.equal( 0.0 ).discard();
|
||||
|
||||
} );
|
||||
|
||||
if ( numUnionClippingPlanes < numClippingPlanes ) {
|
||||
|
||||
const unionClipOpacity = property( 'float', 'unionclipOpacity' );
|
||||
|
||||
unionClipOpacity.assign( 1 );
|
||||
|
||||
loop( { start: numUnionClippingPlanes, end: numClippingPlanes }, ( { i } ) => {
|
||||
|
||||
plane = clippingPlanes.element( i );
|
||||
|
||||
distanceToPlane.assign( positionView.dot( plane.xyz ).negate().add( plane.w ) );
|
||||
distanceGradient.assign( distanceToPlane.fwidth().div( 2.0 ) );
|
||||
|
||||
unionClipOpacity.mulAssign( smoothstep( distanceGradient.negate(), distanceGradient, distanceToPlane ).oneMinus() );
|
||||
|
||||
} );
|
||||
|
||||
clipOpacity.mulAssign( unionClipOpacity.oneMinus() );
|
||||
|
||||
}
|
||||
|
||||
diffuseColor.a.mulAssign( clipOpacity );
|
||||
|
||||
diffuseColor.a.equal( 0.0 ).discard();
|
||||
|
||||
} )();
|
||||
|
||||
}
|
||||
|
||||
setupDefault( planes, numClippingPlanes, numUnionClippingPlanes ) {
|
||||
|
||||
return tslFn( () => {
|
||||
|
||||
const clippingPlanes = uniforms( planes );
|
||||
|
||||
let plane;
|
||||
|
||||
loop( numUnionClippingPlanes, ( { i } ) => {
|
||||
|
||||
plane = clippingPlanes.element( i );
|
||||
positionView.dot( plane.xyz ).greaterThan( plane.w ).discard();
|
||||
|
||||
} );
|
||||
|
||||
if ( numUnionClippingPlanes < numClippingPlanes ) {
|
||||
|
||||
const clipped = property( 'bool', 'clipped' );
|
||||
|
||||
clipped.assign( true );
|
||||
|
||||
loop( { start: numUnionClippingPlanes, end: numClippingPlanes }, ( { i } ) => {
|
||||
|
||||
plane = clippingPlanes.element( i );
|
||||
clipped.assign( positionView.dot( plane.xyz ).greaterThan( plane.w ).and( clipped ) );
|
||||
|
||||
} );
|
||||
|
||||
clipped.discard();
|
||||
|
||||
}
|
||||
|
||||
} )();
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
ClippingNode.ALPHA_TO_COVERAGE = 'alphaToCoverage';
|
||||
ClippingNode.DEFAULT = 'default';
|
||||
|
||||
export default ClippingNode;
|
||||
|
||||
export const clipping = () => nodeObject( new ClippingNode() );
|
||||
|
||||
export const clippingAlpha = () => nodeObject( new ClippingNode( ClippingNode.ALPHA_TO_COVERAGE ) );
|
Reference in New Issue
Block a user