AssignNode.js 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. import { addNodeClass } from '../core/Node.js';
  2. import TempNode from '../core/TempNode.js';
  3. import { addNodeElement, nodeProxy } from '../shadernode/ShaderNode.js';
  4. import { vectorComponents } from '../core/constants.js';
  5. class AssignNode extends TempNode {
  6. constructor( targetNode, sourceNode ) {
  7. super();
  8. this.targetNode = targetNode;
  9. this.sourceNode = sourceNode;
  10. }
  11. hasDependencies() {
  12. return false;
  13. }
  14. getNodeType( builder, output ) {
  15. return output !== 'void' ? this.targetNode.getNodeType( builder ) : 'void';
  16. }
  17. needsSplitAssign( builder ) {
  18. const { targetNode } = this;
  19. if ( builder.isAvailable( 'swizzleAssign' ) === false && targetNode.isSplitNode && targetNode.components.length > 1 ) {
  20. const targetLength = builder.getTypeLength( targetNode.node.getNodeType( builder ) );
  21. const assignDiferentVector = vectorComponents.join( '' ).slice( 0, targetLength ) !== targetNode.components;
  22. return assignDiferentVector;
  23. }
  24. return false;
  25. }
  26. generate( builder, output ) {
  27. const { targetNode, sourceNode } = this;
  28. const needsSplitAssign = this.needsSplitAssign( builder );
  29. const targetType = targetNode.getNodeType( builder );
  30. const target = targetNode.context( { assign: true } ).build( builder );
  31. const source = sourceNode.build( builder, targetType );
  32. const sourceType = sourceNode.getNodeType( builder );
  33. //
  34. let snippet;
  35. if ( needsSplitAssign ) {
  36. const sourceVar = builder.getVarFromNode( this, null, targetType );
  37. const sourceProperty = builder.getPropertyName( sourceVar );
  38. builder.addLineFlowCode( `${ sourceProperty } = ${ source }` );
  39. const targetRoot = targetNode.node.context( { assign: true } ).build( builder );
  40. for ( let i = 0; i < targetNode.components.length; i ++ ) {
  41. const component = targetNode.components[ i ];
  42. builder.addLineFlowCode( `${ targetRoot }.${ component } = ${ sourceProperty }[ ${ i } ]` );
  43. }
  44. if ( output !== 'void' ) {
  45. snippet = target;
  46. }
  47. } else {
  48. snippet = `${ target } = ${ source }`;
  49. if ( output === 'void' || sourceType === 'void' ) {
  50. builder.addLineFlowCode( snippet );
  51. if ( output !== 'void' ) {
  52. snippet = target;
  53. }
  54. }
  55. }
  56. return builder.format( snippet, targetType, output );
  57. }
  58. }
  59. export default AssignNode;
  60. export const assign = nodeProxy( AssignNode );
  61. addNodeClass( 'AssignNode', AssignNode );
  62. addNodeElement( 'assign', assign );