Текстура THREE.js для сферы с React

Я пытаюсь загрузить текстуру флага (gif) в геометрию сферы в файле THREE.js, но предостережение заключается в том, что я использую 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 это строка, но я не получаю никаких ошибок, и на самом деле я получаю предупреждения, которые заставляют его казаться, что что-то работает. Кто-нибудь успешно загрузил текстуру на сферу, используя require() в реакции.

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

2 ответа

Я бы порекомендовал передать путь изображения непосредственно к .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); 
});

В завершение рассмотрите возможность настройки пути к файлу изображения по абсолютному пути (удалив ..) если твой assets Каталог находится в том же каталоге, с которого работает ваш веб-сервер.

Надеюсь это поможет!

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

Другие вопросы по тегам