Home Reference Source Repository

src/lib/descriptors/Material/MeshStandardMaterialDescriptor.js

import * as THREE from 'three';

import PropTypes from 'react/lib/ReactPropTypes';

import MaterialDescriptorBase from './MaterialDescriptorBase';
import propTypeInstanceOf from '../../utils/propTypeInstanceOf';

class MeshStandardMaterialDescriptor extends MaterialDescriptorBase {
  constructor(react3RendererInstance) {
    super(react3RendererInstance);

    this.hasColor();
    this.hasColor('emissive', 0x000000);
    this.hasWireframe();

    [
      'roughness',
      'metalness',
    ]
      .forEach((propName) => {
        this.hasProp(propName, {
          type: PropTypes.number,
          simple: true,
          default: 0.5,
        });
      });

    [
      'lightMapIntensity',
      'aoMapIntensity',
      'emissiveIntensity',
      'bumpScale',
      'displacementScale',
    ]
      .forEach((propName) => {
        this.hasProp(propName, {
          type: PropTypes.number,
          update(threeObject, propValue) {
            threeObject[propName] = propValue;
            threeObject.needsUpdate = true;
          },
          default: 1,
        });
      });

    [
      'displacementBias',
    ]
      .forEach((propName) => {
        this.hasProp(propName, {
          type: PropTypes.number,
          update(threeObject, propValue) {
            threeObject[propName] = propValue;
            threeObject.needsUpdate = true;
          },
          default: 0,
        });
      });

    [
      'refractionRatio',
    ]
      .forEach((propName) => {
        this.hasProp(propName, {
          type: PropTypes.number,
          update(threeObject, propValue) {
            threeObject[propName] = propValue;
            threeObject.needsUpdate = true;
          },
          default: 0.98,
        });
      });

    this.hasProp('normalScale', {
      type: propTypeInstanceOf(THREE.Vector2),
      update(threeObject, normalScale) {
        threeObject.normalScale.copy(normalScale);
        threeObject.needsUpdate = true;
      },
      default: new THREE.Vector2(1, 1),
    });

    this.hasProp('shading', {
      type: PropTypes.oneOf([THREE.FlatShading, THREE.SmoothShading]),
      update(threeObject, shading) {
        threeObject.shading = shading;
        threeObject.needsUpdate = true;
      },
      default: THREE.SmoothShading,
    });

    [
      'skinning',
      'morphTargets',
      'morphNormals',
    ].forEach((propName) => {
      this.hasProp(propName, {
        type: PropTypes.bool,
        update(threeObject, propValue) {
          threeObject[propName] = propValue;
          threeObject.needsUpdate = true;
        },
        default: false,
      });
    });

    this.hasMap();
    this.hasMap('lightMap');
    this.hasMap('aoMap');
    this.hasMap('emissiveMap');
    this.hasMap('bumpMap');
    this.hasMap('normalMap');
    this.hasMap('displacementMap');
    this.hasMap('roughnessMap');
    this.hasMap('metalnessMap');
    this.hasMap('alphaMap');
    this.hasMap('envMap');
  }

  getMaterialDescription(props) {
    const materialDescription = super.getMaterialDescription(props);

    [
      'roughness',
      'metalness',

      'lightMapIntensity',
      'aoMapIntensity',
      'emissiveIntensity',
      'bumpScale',
      'displacementScale',

      'displacementBias',

      'refractionRatio',

      'normalScale',

      'shading',

      'skinning',
      'morphTargets',
      'morphNormals',
    ].forEach((propName) => {
      if (props.hasOwnProperty(propName)) {
        materialDescription[propName] = props[propName];
      }
    });

    return materialDescription;
  }

  construct(props) {
    const materialDescription = this.getMaterialDescription(props);

    return new THREE.MeshStandardMaterial(materialDescription);
  }
}

module.exports = MeshStandardMaterialDescriptor;