Bootstrap

Three.js针对.gltf类型建模文件封装记录

记录Three.js代码组件封装片段 , 支持定制旋转位置大小配置

three.js官方连接:Three.js中文网

3D模型文件下载地址:3D模型可视化编辑器

完整效果图片

封装文件位置 : utils文件夹下 

依赖安装:

"dependencies": {
    "three": "^0.165.0",
    "three-obj-mtl-loader": "^1.0.3",
    "three-orbit-controls": "^82.1.0",
}

封装文件:

<!--THREEJS组件-->
<template>
  <div class="d3-container">
    <div
      class="camera-icon"
      v-show="
        this.camera &&
        (Math.round(this.camera.position.x) !== 1 ||
          Math.round(this.camera.position.z) !== 20) &&
        ishome
      "
    >
      <el-tooltip effect="light" content="复位" placement="top">
        <i class="el-icon-refresh" @click="onResetModelCamera"></i>
      </el-tooltip>
    </div>
    
    <div
      element-loading-text="模型加载中"
      element-loading-spinner="el-icon-loading"
      element-loading-background="rgba(0, 0, 0, 0)"
      id="d3Container"
      v-loading="loading"
      ref="mainContent"
    ></div>
  </div>
</template>
<script>
import * as THREE from "three";
import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js";
import { GLTFLoader } from "three/examples/jsm/loaders/GLTFLoader";
import { FBXLoader } from "three/examples/jsm/loaders/FBXLoader";
import { OBJLoader } from "three/examples/jsm/loaders/OBJLoader";

export default {
  name: "ThreePage",
  props: {
    /*模型资源地址*/
    ossPath: {
      type: String,
      default() {
        return "/scene.gltf";
      },
    },
    /*是否主页*/
    ishome: {
      type: Boolean,
      default() {
        return true;
      },
    },
    // 是否缩放
    isEnableZoom: {
      type: Boolean,
      default() {
        return true;
      },
    },
    // 是否拖拽
    isEnablePan: {
      type: Boolean,
      default() {
        return true;
      },
    },
    /*d3物件的初始位置*/
    d3Position: {
      type: Array,
      default() {
        return [0, -6, 0];
      },
    },
    /*d3物件的初始缩放*/
    d3Scale: {
      type: Array,
      default() {
        return [2, 2, 2];
      },
    },
    /*d3相机的位置*/
    cameraPosition: {
      type: Array,
      default() {
        return [1, 1, 20];
      },
    },
    /*d3相机的观测点*/
    cameraLookAt: {
      type: Array,
      default() {
        return [1, 10, 0];
      },
    },
    /*是否开启模型动画*/
    startModelAnimaion: {
      type: Boolean,
      default() {
        return true;
      },
    },
    /*模型动画运转速度*/
    mixerTimeScale: {
      type: Number,
      default() {
        return 1;
      },
    },
    /*文件类型*/
    suffix: {
      type: String,
      default() {
        return "gltf";
      },
    },
    AutoFresh: {
      type: Boolean,
      default() {
        return true;
      },
    },
    /*是否开启自动旋转*/
    autoAnimate: {
      type: Boolean,
      default() {
        return true;
      },
    },
    /*当前模型的颜色*/
    currentColor: {
      type: String,
      default() {
        return "";
      },
    },
    /*配准后的颜色*/
    matchedColor: {
      type: String,
      default() {
        return "";
      },
    },
    /*配准后的地址*/
    matchedOssPatch: {
      type: String,
      default() {
        return "";
      },
    },
    showMatchWatch: {
      type: Boolean,
      default() {
        return false;
      },
    },
  },
  data() {
    return {
      OricameraUp: null,
      OricamerPosition: null,
      OricameraQuaternion: null,
      OricontrolTarget: null,
      composer: null,
      renderPass: null,
      outlinePass: null,
      loading: false,
      publicPath: process.env.BASE_URL,
      mesh: null,
      camera: null,
      scene: null,
      originX: 10,
      originY: 20,
      originZ: 10,
      renderer: null,
      controls: null,
      fileLoaderMap: {
        glb: new GLTFLoader(),
        fbx: new FBXLoader(),
        gltf: new GLTFLoader(),
        obj: new OBJLoader(),
      },
      clock: new THREE.Clock(),
      mixer: null,
    };
  },
  mounted() {
    this.init();
  },
  watch: {
    //监听地址变化时需要更新地址,防止多次点击同一个渲染多次;
    ossPath(val, oldVal) {
      if (val != oldVal) {
        this.init();
      }
    },
    //监测是否更新整个场景
    AutoFresh(val, oldVal) {
      if (val) {
        this.init();
      } else {
        //自我清理
        this.destroyed();
      }
    },
    //监测是否展示配准,更新场景,该属性的变化只负责更新场景,具体业务交给按钮的最终展现结果,按钮勾中就展示配准,没有勾中就不展示配准,属性没变就是原来的状态。
    showMatchWatch(val, oldVal) {
      this.init();
    },
    //由于上传标签时,CAD会绕过。
    currentColor(val, oldVal) {
      if (val != oldVal) {
        this.init();
      }
    },
  },
  //组件被销毁时,干掉所有3D资源;
  methods: {
    // 重置相机位置
    onResetModelCamera() {
      // // 设置相机位置
      this.camera.position.set(1, 1, 20);
      // 初始化位置
      this.scene.up.copy(this.OricameraUp);
      this.scene.position.copy(this.OricamerPosition);
      this.scene.quaternion.copy(this.OricameraQuaternion);
      this.controls.target.copy(this.OricontrolTarget);
    },
    destroyed() {
      this.clear();
    },
    // 初始化
    init() {
      /*利用vue单项数据流的特性做最后的守卫,在最底层监听是否需要展示配准图,只影响该组件的内部数据而不影响外部的matchedOssPatch*/
      if (!this.showMatchWatch) {
        this.matchedOssPatch = "";
      }
      this.createScene(); // 创建场景
      this.loadLoader(); // 加载P模型
      this.createLight(); // 创建光源
      this.createCamera(); // 创建相机
      this.createRender(); // 创建渲染器
      this.createControls(); // 创建控件对象
      this.render(); // 渲染
      // 获取初始模型位置
      this.OricameraUp = this.scene.up.clone();
      this.OricamerPosition = this.scene.position.clone();
      this.OricameraQuaternion = this.scene.quaternion.clone();
      this.OricontrolTarget = this.controls.target.clone();
      // console.log( this.OricontrolTarget);
      // console.log(
      //   this.OricameraUp,
      //   this.OricamerPosition,
      //   this.OricameraQuaternion,
      //   this.OricontrolTarget
      // );
    },
    //清除当前所有场景
    clear() {
      this.mesh = null;
      this.camera = null;
      this.scene = null;
      this.renderer = null;
      this.controls = null;
      this.clock = null;
      this.mixer = null;
      cancelAnimationFrame(this.animationId);
      console.log("我要清除啦");
    },
    // 创建场景
    createScene() {
      this.loading = true;
      this.scene = new THREE.Scene();
      //开启 底部网格
      var grid = new THREE.GridHelper(24, 24, 0xff0000, 0x444444);
      grid.material.opacity = 0.4;
      grid.material.transparent = true;
      grid.rotation.x = Math.PI / 2.0;
      grid.rotation.z = Math.PI / 4;
      // makeRotationAxis(axis, angle)
      // this.scene.add()
      this.scene = new THREE.Scene();
      const texture = new THREE.TextureLoader().load(
        require("@/assets/images/view-4.png")
      );
      texture.mapping = THREE.EquirectangularReflectionMapping;
      texture.colorSpace = THREE.SRGBColorSpace;
      this.scene.background = texture;
      this.scene.environment = texture;
    },
    // 加载PLY模型
    loadLoader() {
      const THIS = this;
      let fileType = THIS.ossPath.substring(THIS.ossPath.lastIndexOf(".") + 1);
      const loader = this.fileLoaderMap[fileType];
      return new Promise((resolve, reject) => {
        loader.load(
          THIS.ossPath,
          (result) => {
            //加载不同类型的文件
            switch (fileType) {
              case "glb":
                this.mesh = result.scene;
                break;
              case "fbx":
                this.mesh = result;
                break;
              case "gltf":
                this.mesh = result.scene;

                break;
              case "obj":
                this.mesh = result;
                break;
              default:
                break;
            }
            this.mesh.traverse(() => {
            });

            // 新建一个AnimationMixer
            this.mixer = new THREE.AnimationMixer(this.mesh);
            this.mixer.clipAction(result.animations[0]).play();
            this.mixer.timeScale = this.mixerTimeScale; //默认1,可以调节播放速度
            this.loading = false; //关闭载入中效果

            // 设置模型大小
            this.mesh.scale.set(
              this.d3Scale[0],
              this.d3Scale[1],
              this.d3Scale[2]
            );
            // 设置模型位置
            this.mesh.position.set(
              this.d3Position[0],
              this.d3Position[1],
              this.d3Position[2]
            ); //设置3d物品的位置
            // 将模型添加到场景中去
            this.scene.add(this.mesh);
            resolve(true);
          },
          (xhr) => {
            const percent = Math.floor((xhr.loaded / xhr.total) * 100);
          },
          (err) => {
            console.log(err);
            reject(new Error());
          }
        );
      });
      //如果有配准结果,加载配准结果,配准结果未ply格式;
    },
    // 创建光源
    createLight() {
      // 环境光
      let pointColor = " #000";
      const ambientLight = new THREE.AmbientLight(pointColor, 0.35); // 创建环境光
      this.scene.add(ambientLight); // 将环境光添加到场景

      const spotLight = new THREE.SpotLight(0xffffff); // 创建聚光灯
      spotLight.position.set(1, 1, 1);
      // spotLight.castShadow = true //平行光开启阴影
      // spotLight.receiveShadow = true
      this.scene.add(spotLight);
    },

    // 创建相机
    createCamera() {
      const element = this.$refs.mainContent;
      const width = element.clientWidth; // 窗口宽度
      const height = element.clientHeight; //
      this.cWidth = width;
      this.cHeight = height;
      const k = width / height; // 窗口宽高比
      this.aspect = k;
      // PerspectiveCamera( fov, aspect, near, far )
      this.camera = new THREE.PerspectiveCamera(35, k, 1, 20000);
      this.camera.position.set(
        this.cameraPosition[0],
        this.cameraPosition[1],
        this.cameraPosition[2]
      ); // 设置相机位置
      this.camera.up.set(0, 1, 0);
      this.camera.lookAt(
        new THREE.Vector3(
          this.cameraLookAt[0],
          this.cameraLookAt[1],
          this.cameraLookAt[2]
        )
      ); // 设置相机方向
      this.scene.add(this.camera);
    },
    // 创建渲染器
    createRender() {
      const element = this.$refs.mainContent;
      this.renderer = new THREE.WebGLRenderer({
        antialias: true,
        alpha: true,
        preserveDrawingBuffer: true,
      });
      this.renderer.setSize(element.clientWidth, element.clientHeight); // 设置渲染区域尺寸
      this.renderer.shadowMap.enabled = true; // 显示阴影
      this.renderer.shadowMap.type = THREE.PCFSoftShadowMap;
      this.scene.background = null; //设置背景颜色,null为透明
      // this.renderer.setClearColor(new THREE.Color(0xeeeeee)) // 设置背景颜色
      // this.renderer.setClearColor(new THREE.Color(0x111111)) // 设置背景颜色
      this.renderer.toneMappingExposure = 1.5; //曝光
      this.renderer.toneMapping = THREE.ACESFilmicToneMapping; //色调映射
      element.innerHTML = "";
      element.appendChild(this.renderer.domElement);
    },
    render() {
      //TODO:更新模型动画
      if (this.mixer && this.startModelAnimaion) {
        const delta = this.clock.getDelta();
        this.mixer.update(delta);
      }

      this.animationId = requestAnimationFrame(this.render); //旋转动画;
      this.renderer.render(this.scene, this.camera);
      this.controls.update();
      // 点击事件影像处理 有问题
      // if (this.composer) {
      //   this.composer.render()
      // }
    },
    // 创建控件对象
    createControls() {
      this.controls = new OrbitControls(this.camera, this.renderer.domElement);
      // const newMesh = this.mesh.clone();
      // console.log(this.mesh.position);
      // console.log(this.mesh);
      // this.controls.minDistance = 1 //设置相机距离原点的最远距离
      // this.controls.maxDistance = 1000 //设置相机距离原点的最远距离
      // this.controls.maxPolarAngle = Math.PI / 1 //设置可旋转的范围
      this.controls.enableZoom = true; //是否可以缩放
      this.controls.enablePan = true; //是否可拖拽
      this.controls.enableDamping = true; // 开启阻尼效果
      this.controls.dampingFactor = 1; // 阻尼系数
      this.controls.autoRotate = this.autoAnimate; // 自动旋转
      this.controls.autoRotateSpeed = 5; // 自动旋转速度
      this.controls.minPolarAngle = 0; // 设置最小旋转角度
      this.controls.maxPolarAngle = Math.PI; // 设置最大旋转角度
    },
    onWindowResize() {
      this.camera.aspect = this.aspect;
      this.camera.updateProjectionMatrix();
    },
  },
  beforeDestroy() {
    this.destroyed();
  },
};
</script>

<style scoped lang="less">
.d3-container {
  width: 100%;
  height: 100%;
  .camera-icon {
    position: absolute;
    top: 20px;
    left: 50%;
    cursor: pointer;
  }
}
#d3Container {
  width: 100%;
  height: 100%;
}
.el-icon-refresh {
  font-size: 28px;
  color: #ccc;
  z-index: 99;
}
</style>

使用方法

<template> 
 <d3
            :cameraPosition="[1, 3, 20]"
            :d3Position="[0, 0, 0]"
            :d3Scale="[.85, .85, .85]"
            :isEnableZoom="true"
            :isEnableRotate="true"
            :autoAnimate="true"
            ossPath="HN.gltf"
            class="three-canvas"
          ></d3>
</template>
<script>
import d3 from "../../utils/index.vue";
</script>

;