UNPKG

three

Version:

JavaScript 3D library

192 lines (114 loc) 5.09 kB
import Node from '../core/Node.js'; import { NodeUpdateType } from '../core/constants.js'; import { nodeObject } from '../tsl/TSLBase.js'; import { attribute } from '../core/AttributeNode.js'; import { reference, referenceBuffer } from './ReferenceNode.js'; import { add } from '../math/OperatorNode.js'; import { normalLocal } from './Normal.js'; import { positionLocal, positionPrevious } from './Position.js'; import { tangentLocal } from './Tangent.js'; import { uniform } from '../core/UniformNode.js'; import { buffer } from './BufferNode.js'; const _frameId = new WeakMap(); class SkinningNode extends Node { static get type() { return 'SkinningNode'; } constructor( skinnedMesh, useReference = false ) { super( 'void' ); this.skinnedMesh = skinnedMesh; this.useReference = useReference; this.updateType = NodeUpdateType.OBJECT; // this.skinIndexNode = attribute( 'skinIndex', 'uvec4' ); this.skinWeightNode = attribute( 'skinWeight', 'vec4' ); let bindMatrixNode, bindMatrixInverseNode, boneMatricesNode; if ( useReference ) { bindMatrixNode = reference( 'bindMatrix', 'mat4' ); bindMatrixInverseNode = reference( 'bindMatrixInverse', 'mat4' ); boneMatricesNode = referenceBuffer( 'skeleton.boneMatrices', 'mat4', skinnedMesh.skeleton.bones.length ); } else { bindMatrixNode = uniform( skinnedMesh.bindMatrix, 'mat4' ); bindMatrixInverseNode = uniform( skinnedMesh.bindMatrixInverse, 'mat4' ); boneMatricesNode = buffer( skinnedMesh.skeleton.boneMatrices, 'mat4', skinnedMesh.skeleton.bones.length ); } this.bindMatrixNode = bindMatrixNode; this.bindMatrixInverseNode = bindMatrixInverseNode; this.boneMatricesNode = boneMatricesNode; this.previousBoneMatricesNode = null; } getSkinnedPosition( boneMatrices = this.boneMatricesNode, position = positionLocal ) { const { skinIndexNode, skinWeightNode, bindMatrixNode, bindMatrixInverseNode } = this; const boneMatX = boneMatrices.element( skinIndexNode.x ); const boneMatY = boneMatrices.element( skinIndexNode.y ); const boneMatZ = boneMatrices.element( skinIndexNode.z ); const boneMatW = boneMatrices.element( skinIndexNode.w ); // POSITION const skinVertex = bindMatrixNode.mul( position ); const skinned = add( boneMatX.mul( skinWeightNode.x ).mul( skinVertex ), boneMatY.mul( skinWeightNode.y ).mul( skinVertex ), boneMatZ.mul( skinWeightNode.z ).mul( skinVertex ), boneMatW.mul( skinWeightNode.w ).mul( skinVertex ) ); return bindMatrixInverseNode.mul( skinned ).xyz; } getSkinnedNormal( boneMatrices = this.boneMatricesNode, normal = normalLocal ) { const { skinIndexNode, skinWeightNode, bindMatrixNode, bindMatrixInverseNode } = this; const boneMatX = boneMatrices.element( skinIndexNode.x ); const boneMatY = boneMatrices.element( skinIndexNode.y ); const boneMatZ = boneMatrices.element( skinIndexNode.z ); const boneMatW = boneMatrices.element( skinIndexNode.w ); // NORMAL let skinMatrix = add( skinWeightNode.x.mul( boneMatX ), skinWeightNode.y.mul( boneMatY ), skinWeightNode.z.mul( boneMatZ ), skinWeightNode.w.mul( boneMatW ) ); skinMatrix = bindMatrixInverseNode.mul( skinMatrix ).mul( bindMatrixNode ); return skinMatrix.transformDirection( normal ).xyz; } getPreviousSkinnedPosition( builder ) { const skinnedMesh = builder.object; if ( this.previousBoneMatricesNode === null ) { skinnedMesh.skeleton.previousBoneMatrices = new Float32Array( skinnedMesh.skeleton.boneMatrices ); this.previousBoneMatricesNode = referenceBuffer( 'skeleton.previousBoneMatrices', 'mat4', skinnedMesh.skeleton.bones.length ); } return this.getSkinnedPosition( this.previousBoneMatricesNode, positionPrevious ); } needsPreviousBoneMatrices( builder ) { const mrt = builder.renderer.getMRT(); return mrt && mrt.has( 'velocity' ); } setup( builder ) { if ( this.needsPreviousBoneMatrices( builder ) ) { positionPrevious.assign( this.getPreviousSkinnedPosition( builder ) ); } const skinPosition = this.getSkinnedPosition(); positionLocal.assign( skinPosition ); if ( builder.hasGeometryAttribute( 'normal' ) ) { const skinNormal = this.getSkinnedNormal(); normalLocal.assign( skinNormal ); if ( builder.hasGeometryAttribute( 'tangent' ) ) { tangentLocal.assign( skinNormal ); } } } generate( builder, output ) { if ( output !== 'void' ) { return positionLocal.build( builder, output ); } } update( frame ) { const object = this.useReference ? frame.object : this.skinnedMesh; const skeleton = object.skeleton; if ( _frameId.get( skeleton ) === frame.frameId ) return; _frameId.set( skeleton, frame.frameId ); if ( this.previousBoneMatricesNode !== null ) skeleton.previousBoneMatrices.set( skeleton.boneMatrices ); skeleton.update(); } } export default SkinningNode; export const skinning = ( skinnedMesh ) => nodeObject( new SkinningNode( skinnedMesh ) ); export const skinningReference = ( skinnedMesh ) => nodeObject( new SkinningNode( skinnedMesh, true ) );