hotshotiguana
hotshotiguana

Reputation: 1570

THREE.js Texture to Sphere with React

I am trying to load a flag (gif) texture to a sphere geometry in THREE.js, but the caveat is I am using React to do this.

const textureLoader = new THREE.TextureLoader();
const flag = getFlagForCountry(flags, x.id),
      texture = textureLoader.load(require(`../assets/images/flags/${flag.name}.gif`));

const mat = new THREE.MeshLambertMaterial({
        transparent: true,
        opacity: .5,
        map: texture
    });

const sphere = new THREE.Mesh(new THREE.SphereGeometry(1, 10, 10), mat);
    sphere.overdraw = true;

When I remove the map: texture property I am able to see the sphere in the scene, but then when I add back in the texture it is simply a black screen. I know the docs for TextureLoader say url is a string, but I am not getting any errors and in fact I am getting warnings that make it appear like something is working. Has anyone had success loading a texture onto a sphere using require() in React.

THREE.WebGLRenderer: image is not power of two (1181x788). Resized to 1024x512 
<img crossorigin="anonymous" src="/static/media/Argentina.4c3ff3da.gif">

Upvotes: 2

Views: 3011

Answers (2)

Hitesh Sahu
Hitesh Sahu

Reputation: 45140

import React, { Component } from "react";
import * as THREE from "three";

var earthMesh;

class ThreeScene extends Component {
  componentDidMount() {
    const width = this.mount.clientWidth;
    const height = this.mount.clientHeight;
    //ADD SCENE
    this.scene = new THREE.Scene();
    //ADD CAMERA
    this.camera = new THREE.PerspectiveCamera(75, width / height, 0.1, 1000);
    this.camera.position.z = 8;
    //ADD RENDERER
    this.renderer = new THREE.WebGLRenderer({ antialias: true });
    this.renderer.setClearColor("#263238");
    this.renderer.setSize(width, height);
    this.mount.appendChild(this.renderer.domElement);
    //ADD CUBE
    const geometry = new THREE.BoxGeometry(5, 5, 5);
    const material = new THREE.MeshBasicMaterial({
      color: "#0F0",
      wireframe: true
    });
    this.cube = new THREE.Mesh(geometry, material);
    this.scene.add(this.cube);

    //Add SPHERE
    //LOAD TEXTURE and on completion apply it on box
    var loader = new THREE.TextureLoader();
    loader.load(
      "",
      this.onLoad,
      this.onProgress,
      this.onError
    );

    //LIGHTS
    var lights = [];
    lights[0] = new THREE.PointLight(0x304ffe, 1, 0);
    lights[1] = new THREE.PointLight(0xffffff, 1, 0);
    lights[2] = new THREE.PointLight(0xffffff, 1, 0);
    lights[0].position.set(0, 200, 0);
    lights[1].position.set(100, 200, 100);
    lights[2].position.set(-100, -200, -100);
    this.scene.add(lights[0]);
    this.scene.add(lights[1]);
    this.scene.add(lights[2]);
  }

  componentWillUnmount() {
    this.stop();
    this.mount.removeChild(this.renderer.domElement);
  }
  start = () => {
    if (!this.frameId) {
      this.frameId = requestAnimationFrame(this.animate);
    }
  };
  stop = () => {
    cancelAnimationFrame(this.frameId);
  };
  animate = () => {
    this.earthMesh.rotation.x += 0.01;
    this.cube.rotation.y += 0.01;
    this.renderScene();
    this.frameId = window.requestAnimationFrame(this.animate);
  };
  renderScene = () => {
    this.renderer.render(this.scene, this.camera);
  };

  onLoad = texture => {
    var objGeometry = new THREE.SphereBufferGeometry(3, 35, 35);
    var objMaterial = new THREE.MeshPhongMaterial({
      map: texture,
      shading: THREE.FlatShading
    });

    this.earthMesh = new THREE.Mesh(objGeometry, objMaterial);
    this.scene.add(this.earthMesh);
    this.renderScene();
    //start animation
    this.start();
  };

  onProgress = xhr => {
    console.log((xhr.loaded / xhr.total) * 100 + "% loaded");
  };

  // Function called when download errors
  onError = error => {
    console.log("An error happened" + error);
  };

  render() {
    return (
      <div
        style={{ width: "400px", height: "400px" }}
        ref={mount => {
          this.mount = mount;
        }}
      />
    );
  }
}
export default ThreeScene;

https://codesandbox.io/embed/kw7l49nw1r

Upvotes: 0

Dacre Denny
Dacre Denny

Reputation: 30390

I would recommend passing the image path directly to the .load() method rather than passing it via require(). Also, I suggest using the TextureLoader callback, to ensure that your texture object is valid and fully loaded, before trying to make use of it.

You can make use of the callback in this way:

const textureLoader = new THREE.TextureLoader();
const flag = getFlagForCountry(flags, x.id)'

// Use the loaders callback
textureLoader.load(`../assets/images/flags/${flag.name}.gif`, function(texture) {

  // The texture object has loaded and is now avalible to be used
  const mat = new THREE.MeshLambertMaterial({
          transparent: true,
          opacity: .5,
          map: texture
      });

  const sphere = new THREE.Mesh(new THREE.SphereGeometry(1, 10, 10), mat);
  sphere.overdraw = true;

  // Add sphere to your scene ... scene.add(sphere); 
});

As a final note, consider adjusting your image filepath to an absolute path (by removing the ..) if your assets directory is located in the same directory that your webserver is running from.

Hope this helps!

Upvotes: 1

Related Questions