THREE.js使用React制作纹理到球体

时间:2018-09-06 01:47:12

标签: reactjs three.js texture-mapping

我正在尝试在THREE.js中将标志(gif)纹理加载到球体几何中,但是要注意的是,我正在使用React来做到这一点。

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;

当我删除map: texture属性时,我可以看到场景中的球体,但是当我重新添加纹理时,它只是一个黑屏。我知道TextureLoader的文档说url是一个字符串,但是我没有收到任何错误,实际上我收到的警告是使它看起来像是在工作。有谁成功使用React中的require()将纹理加载到球体上。

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

2 个答案:

答案 0 :(得分:1)

我建议将图像路径直接传递给.load()方法,而不是通过require()传递。另外,我建议使用TextureLoader回调,以确保您的纹理对象有效且已完全加载,然后再尝试使用它。

您可以通过以下方式使用回调:

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); 
});

最后一点,如果您的..目录位于运行Web服务器的同一目录中,请考虑将图像文件路径调整为绝对路径(通过删除assets)。

希望这会有所帮助!

答案 1 :(得分:0)

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