three.js第一个3D案例

news/2024/7/19 14:50:31 标签: 前端, vue.js, webgl, javascript, js

在正式学习Three.js之前,先做一些必要的准备工作,具体说就是下载threejs官方文件包,threejs官方文件包提供了很多有用的学习资源

threejs官方文件包所有版本:https://github.com/mrdoob/three.js/releases

js文件资源目录介绍">threejs文件资源目录介绍

对于开发者而言,大家经常接触的是文档docs案例examples两个文件夹,平时查看文档,可以打开文档docs里面html文件,案例examples里面提供了海量threejs功能案例。

js">three.js-文件包
└───build——three.js相关库,可以引入你的.html文件中。
    │
└───docs——Three.js API文档文件
    │───index.html——打开该文件,本地离线方式预览threejs文档
└───examples——大量的3D案例,是你平时开发参考学习的最佳资源
    │───jsm——threejs各种功能扩展库
└───src——Three.js引擎的源码,有兴趣可以阅读。
    │
└───editor——Three.js的可视化编辑器,可以编辑3D场景
    │───index.html——打开应用程序  

工欲善其事,必先利其器

Web3D开发的代码编辑器和平时web前端开发一样,你可以根据自己的喜好选择,本课程选择的代码编辑器是  vscode

本地静态服务器

如果你想预览代码3D效果,咱们需要提供一个本地静态服务器的开发环境,正式的web项目开发,往往会用webpack或vite或其它方式配置一个开发环境。

如果只是学习threejs的话,可通过代码编辑器快速创建本地静态服务器,比如vsocde,安装live-server插件即可。或者使用phpstudy, 下载下来直接给下载的three.js包放在www目录里面,创建自己的HTML文件,引入three.js,就可以学习了

vscode配置live-server插件

  • 安装: vscode软件界面左侧,点击扩展,输入live-server关键词查询安装。
  • 使用:如果你想预览代码3D效果,打开对应.html文件,右键点击Open with Live Server即可。
预览3D案例和文档

打开课件案例,注意把Three.js视频教程源码文件作为根目录,使用vscode创建本地静态服务就可以预览。 

js">three.js-文件包
...
└───docs——Three.js API文档文件
    │───index.html——打开该文件,本地离线方式预览threejs文档
└───examples——大量的3D案例,是你平时开发参考学习的最佳资源
    │───.html——各种3D案例
...    

js">script标签方式引入three.js

通过script标签把three.js当做一个js库引入你的项目。three.js库可以在threejs官方文件包下面的build目录获取到。

javascript"><script src="./build/three.js"></script>

console.log(THREE.Scene); 

ES6 import方式引入

给script标签设置type="module",也可以在.html文件中使用import方式引入three.js

javascript"><script type="module">
// 现在浏览器支持ES6语法,自然包括import方式引入js文件
import * as THREE from './build/three.module.js';
</script>

type="importmap"配置路径

通过配置<script type="importmap">,实现学习环境.html文件和vue或reaact脚手架开发环境一样的写法。这样你实际项目的开发环境复制课程源码,不用改变threejs引入代码。

下面配置的type="importmap"代码具体写法不用掌握记忆,复制粘贴后,能修改目录就行,

javascript"><!-- 具体路径配置,你根据自己文件目录设置,我的是课件中源码形式 -->
<script type="importmap">
    {
		"imports": {
			"three": "../../../three.js/build/three.module.js"
		}
	}
</script>

<!-- 配置type="importmap",.html文件也能和项目开发环境一样方式引入threejs -->
<script type="module">
    import * as THREE from 'three';
    // 浏览器控制台测试,是否引入成功
    console.log(THREE.Scene);
</script>

type="importmap"配置——扩展库引入

通过配置<script type="importmap">,让学习环境.html文件,也能和vue或react开发环境中一样方式方式引入threejs扩展库。

javascript"><script type="importmap">
    {
		"imports": {
			"three": "./three.js/build/three.module.js",
            "three/addons/": "./three.js/examples/jsm/"
		}
	}
</script>

<script type="module">
    // three/addons/路径之后对应的是three.js官方文件包`/examples/jsm/`中的js库
    // 扩展库OrbitControls.js
    import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
    // 扩展库GLTFLoader.js
    import { GLTFLoader } from 'three/addons/loaders/GLTFLoader.js';
    console.log(OrbitControls);
    console.log(GLTFLoader);
</script>

配置addons/等价于examples/jsm/

项目的开发环境引入threejs

比如你采用的是Vue + threejsReact + threejs技术栈,这很简单,threejs就是一个js库,直接通过npm命令行安装就行。

javascript">// 比如安装148版本
npm install three@0.148.0 --save

//执行import * as THREE from 'three';ES6语法引入three.js核心。

// 引入three.js
import * as THREE from 'three';

js其他扩展库">npm安装后,如何引入three.js其他扩展库

了three.js核心库以外,在threejs文件包中examples/jsm目录下,你还可以看到各种不同功能的扩展库。一般来说,你项目用到那个扩展库,就引入那个,用不到就不需要引入。

javascript">// 引入扩展库OrbitControls.js
import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
// 引入扩展库GLTFLoader.js
import { GLTFLoader } from 'three/addons/loaders/GLTFLoader.js';

前期准备工作准备好后,我们就可以开始创建自己的第一个3D场景案例了。

入门Three.js的第一步,就是认识场景Scene相机Camera渲染器Renderer三个基本概念

三维场景Scene 

你可以把三维场景Scene (opens new window)对象理解为虚拟的3D场景,用来表示模拟生活中的真实三维场景,或者说三维世界。

javascript">// 创建3D场景对象Scene
const scene = new THREE.Scene();

物体形状:几何体Geometry

Three.js提供了各种各样的几何体API,用来表示三维物体的几何形状。

javascript">//创建一个长方体几何对象Geometry
const geometry = new THREE.BoxGeometry(200, 200, 200); 

物体外观:材质Material

如果你想定义物体的外观效果,比如颜色,就需要通过材质Material相关的API实现。

threejs不同材质渲染效果不同,下面就以threejs最简单的网格基础材质MeshBasicMaterial为例实现一个蓝色材质效果。

javascript">//创建一个材质对象Material
const material = new THREE.MeshBasicMaterial({
    color: 0x00ff00,//0x00ff00设置材质颜色为蓝色
}); 

物体:网格模型Mesh

在threejs中可以通过网格模型Mesh表示一个虚拟的物体,比如一个电脑等。

javascript">// 两个参数分别为几何体geometry、材质material
const mesh = new THREE.Mesh(geometry, material); //网格模型对象Mesh

模型位置.position

实际生活中,一个物体往往是有位置的,对于threejs而言也是一样的,你可以通过位置属性.position定义网格模型Mesh在三维场景Scene中的位置。

javascript">const mesh = new THREE.Mesh(geometry, material); //网格模型对象Mesh
//设置网格模型在三维空间中的位置坐标,默认是坐标原点
mesh.position.set(0,10,0);

在threejs中你创建了一个表示物体的虚拟对象Mesh,需要通过.add()方法,把网格模型mesh添加到三维场景scene中。

javascript">scene.add(mesh); 

透视投影相机PerspectiveCamera

Threejs如果想把三维场景Scene渲染到web网页上,还需要定义一个虚拟相机Camera,就像你生活中想获得一张照片,需要一台用来拍照的相机。

Threejs提供了正投影相机OrthographicCamera 和透视投影相机PerspectiveCamera 

透视投影相机PerspectiveCamera 本质上就是在模拟人眼观察这个世界的规律

javascript">// 实例化一个透视投影相机对象
const camera = new THREE.PerspectiveCamera();

相机位置.position 

相机对象Camera具有位置属性.position,通过位置属性.position可以设置相机的位置。

javascript">//相机在Three.js三维坐标系中的位置
// 根据需要设置相机位置具体值
camera.position.set(200, 100, 400); 

相机观察目标.lookAt()

你用相机拍照你需要控制相机的拍照目标,具体说相机镜头对准哪个物体或说哪个坐标。对于threejs相机而言,就是设置.lookAt()方法的参数,指定一个3D坐标。

javascript">//相机观察目标指向Threejs 3D空间中某个位置
camera.lookAt(0, 0, 0); //坐标原点

camera.lookAt(0, 10, 0);  //y轴上位置10


camera.lookAt(mesh.position);//指向mesh对应的位置

 判断相机相对三维场景中长方体位置

你可以把三维场景中长方体mesh想象为一个房间,然后根据相机位置和长方体位置尺寸对比,判断两者相对位置。你可以发现设置相机坐标(200, 200, 200),位于长方体外面一处位置。

javascript">// 长方体尺寸100, 100, 100
const geometry = new THREE.BoxGeometry( 100, 100, 100 );
const mesh = new THREE.Mesh(geometry,material);
// 网格模型位置xyz坐标:0,10,0
mesh.position.set(0,10,0);
// 相机位置xyz坐标:200, 200, 200
camera.position.set(200, 200, 200); 

定义相机渲染输出的画布尺寸 

Canvas画布:课程中会把threejs虚拟相机渲染三维场景在浏览器网页上呈现的结果称为Canvas画布

javascript">// 定义相机输出画布的尺寸(单位:像素px)
const width = 800; //宽度
const height = 500; //高度

透视投影相机PerspectiveCamera:视锥体 

透视投影相机的四个参数fov, aspect, near, far构成一个四棱台3D空间,被称为视锥体,只有视锥体之内的物体,才会渲染出来,视锥体范围之外的物体不会显示在Canvas画布上。

javascript">// width和height用来设置Three.js输出的Canvas画布尺寸(像素px)
const width = 800; //宽度
const height = 500; //高度
// 30:视场角度, width / height:Canvas画布宽高比, 1:近裁截面, 3000:远裁截面
const camera = new THREE.PerspectiveCamera(30, width / height, 1, 3000);
javascript">PerspectiveCamera( fov, aspect, near, far )

//参数	含义	                            默认值
fov	    相机视锥体竖直方向视野角度。	        50

aspect	相机视锥体水平方向和竖直方向长度比,一般设置为Canvas画布宽高比。 width / height	1

near	相机视锥体近裁截面相对相机距离	。        0.1

far	    相机视锥体远裁截面相对相机距离,far-near构成了视锥体高度方向。	2000

 WebGL渲染器WebGLRenderer

通过WebGL渲染器WebGLRenderer 可以实例化一个WebGL渲染器对象

javascript">// 创建渲染器对象
const renderer = new THREE.WebGLRenderer();

设置Canvas画布尺寸.setSize()

javascript">// 定义threejs输出画布的尺寸(单位:像素px)
const width = 1000; //宽度
const height = 500; //高度
renderer.setSize(width, height); //设置three.js渲染区域的尺寸(像素px)

渲染器渲染方法.render()

渲染器WebGLRenderer执行渲染方法.render()就可以生成一个Canvas画布(照片),并把三维场景Scene呈现在canvas画布上面,你可以把.render()

javascript">renderer.render(scene, camera); //执行渲染操作

渲染器Canvas画布属性.domElement

渲染器WebGLRenderer通过属性.domElement可以获得渲染方法.render()生成的Canvas画布,.domElement本质上就是一个HTML元素:Canvas画布。

javascript">document.body.appendChild(renderer.domElement);

到这里我们就可以创建了一个自己的3D场景第一个案例了,Canvas画布插入到任意HTML元素中

javascript"><div id="webgl" style="margin-top: 200px;margin-left: 100px;"></div>

document.getElementById('webgl').appendChild(renderer.domElement);

 轨道控制器

在 ThreeJS 中使用轨道控制器后,你可以通过长按鼠标左键并拖动鼠标来控制摄像头所处的位置。

javascript">import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls';

// 用法
const orbitControls = new OrbitControls(camera, renderer.domElement);


javascript"><div class="first" id="myFirst"></div>
  <script type="importmap">
    {
      "imports": {
        "three": "./js/three.module.js",
        "three/jsm/": "./js/jsm/"
      }
    }
  </script>
  <script type="module">
    import * as THREE from 'three';
    import { OrbitControls } from './js/jsm/controls/OrbitControls.js'; // 引入相机控件
    // three内置了 gui工具 我们先进行一个导入 允许我们动态的对于一些参数进行设置方便我们预览效果
    import { GUI } from './js/jsm/libs/lil-gui.module.min.js';
    //3D场景对象Scene
    const scene = new THREE.Scene();
    //创建一个长方体几何体
    const geometry = new THREE.BoxGeometry(2,4,1);
    //创建一个材质对象Material MeshBasicMaterial  不受光照影响
    const material = new THREE.MeshBasicMaterial({
      color: 0xff0000, //0xff0000设置材质颜色为红色
      transparent: true,
      opacity: 0.5,
    });
    //网格模型对象Mesh
    const mesh = new THREE.Mesh(geometry, material);
    // 添加到场景中去
    scene.add(mesh);

    let myFirst = document.getElementById("myFirst");
    // 实例化一个透视投影相机对象 PerspectiveCamera
    const camera = new THREE.PerspectiveCamera(75, myFirst.offsetWidth / myFirst.offsetHeight, 0.1, 1000);
    camera.position.z = 15;
    camera.position.x = 5;
    camera.position.y = 2;
    //camera.lookAt(-100, 100, 10); //坐标原点

    // 创建渲染器 WebGLRenderer
    const renderer = new THREE.WebGLRenderer({
      //lpha: true, // 背景是否可以设置透明色
      antialias: true, // 表示是否开启反锯齿
    });
    // AxesHelper:辅助观察的坐标系
    const axesHelper = new THREE.AxesHelper(150)
    scene.add(axesHelper)

    // 网格辅助线  第一个值是长度   第二个值是分割的段数
    const gridHelper = new THREE.GridHelper(10,10);
    scene.add(gridHelper)

    // 输出画布的尺寸
    
    renderer.setSize(myFirst.offsetWidth, myFirst.offsetHeight); // 设置渲染大小
    // 渲染到元素中
    myFirst.appendChild(renderer.domElement);
    //  相机控制控制的就是相机的postion的位置而在界面当中出现不同的几何体的观察形态
    const controls = new OrbitControls(camera, renderer.domElement);
    // addEventListener
    controls.addEventListener('change', ()=>{
      renderer.render(scene, camera)
    })

    // 创建一个GUI实例
    const gui = new GUI();
    // cube的位置
    const originPositon = {
      x: 0,
      y: 0,
      z: 0
    };
    // camera的位置
    const cameraPosition = {
      x: 0,
      y: 0,
      z: 5
    };
    // 创建第一个 gui的参数分组   几何体位置
    const cubeParams = gui.addFolder('几何体位置');
    cubeParams.add(originPositon, 'x', 0, 100).onChange(val => {
      cube.position.x = val;
      renderer.render(scene, camera)
    });
    cubeParams.add(originPositon, 'y', 0, 100).onChange(val => {
      cube.position.y = val;
      renderer.render(scene, camera)
    });

    //  相机位置的分组
    const cameraParams = gui.addFolder('相机的位置')
    // add(对象,属性名,数值)  生成的是一个数字输入框的交互
    // add(对象,属性名,数值1,数值2) 生成的是一个有可选范围的数字选择框
    cameraParams.add(cameraPosition, 'x', -5, 50).onChange(val => {
      camera.position.x = val;
      renderer.render(scene, camera)
    })
    cameraParams.add(cameraPosition, 'y', -5, 50).onChange(val => {
      camera.position.y = val;
      renderer.render(scene, camera)
    })
    // add(对象,属性名,数组)  会自动生成一个下拉菜单的交互界面
    //   cameraParams.add(cameraPosition,'x' ,[1,3,6]).onChange(val=>{
    //     camera.position.x = val;
    //     renderer.render(scene, camera) 
    //  })
    // add( 对象,属性名, 对象 ) 也会生成一下下拉菜单的交互方式
    // cameraParams.add(cameraPosition,'x' ,{
    //     param1: 1,
    //     param2: 5
    // }).onChange(val=>{
    //     camera.position.x = val;
    //     renderer.render(scene, camera) 
    //  })
    const testObj = {
        color: 0xff6600,
        val1: 10
    }
    // addColor 在gui当中添加一个 颜色选择器的交互 
    gui.addColor(testObj, 'color').onChange(val => {
      console.log('颜色的值', val)
    })
    // add()的其他后续的方法
    // .name(直接展示为参数的指定名称 方便理解)
    // .step(步进的数量) 每一个变化的最小单位
    gui.add(testObj, 'val1', 0, 100).name('某个值的大小').step(1)
    renderer.render(scene, camera)
  </script>

案例实际效果:


http://www.niftyadmin.cn/n/5390213.html

相关文章

Vue实现打印功能(vue-print-nb)

1、安装依赖 npm install vue-print-nb --save2、在main.js中引入 import Print from vue-print-nb Vue.use(Print)3、在组件的打印区域标签上加 id“printArea” <div id"printArea"> 打印区域 </div>4、在组件的打印按钮标签上使用指令 v-print“pr…

深入探索pdfplumber:从PDF中提取信息到实际项目应用【第94篇—pdfplumbe】

深入探索pdfplumber&#xff1a;从PDF中提取信息到实际项目应用 在数据处理和信息提取的过程中&#xff0c;PDF文档是一种常见的格式。然而&#xff0c;要从PDF中提取信息并进行进一步的分析&#xff0c;我们需要使用适当的工具。本文将介绍如何使用Python库中的pdfplumber库来…

计算机毕业设计 | SSM 学生信息管理 教务管理系统(附源码)

1&#xff0c;绪论 随着我国高等教育的发展&#xff0c;数字化校园将成为一种必然的趋势&#xff0c;国内高校迫切需要提高教育工作的质量与效率&#xff0c;学生成绩管理工作是高校信息管理工作的重要组成部分&#xff0c;与国外高校不同&#xff0c;他们一般具有较大规模的稳…

【区块链】联盟链

区块链中的联盟链 写在最前面**FAQs** 联盟链&#xff1a;区块链技术的新兴力量**联盟链的定义****联盟链的技术架构**共识机制智能合约加密技术身份认证 **联盟链的特点**高效性安全性可控性隐私保护 **联盟链的应用场景****金融服务****供应链管理****身份验证****跨境支付**…

AI文生图网站测评

主要测评文章配图生成效果、绘制logo等效果 测评关键点&#xff1a;生成效果、网站易用度、是否免费 测评prompt&#xff1a;请生成一个文章内容配图&#xff0c;图片比例是3&#xff1a;2&#xff0c;文章主旨是AI既是机遇&#xff0c;也存在挑战和风险&#xff0c;要求图片…

【论文阅读】ICASSP 2023 针对目标检测的无目标后门攻击

文章目录 一.论文信息二.论文内容1.摘要2.引言3.作者贡献4.主要图表5.结论 一.论文信息 论文题目&#xff1a; Untargeted backdoor attack against object detection&#xff08;针对目标检测的无目标后门攻击&#xff09; 论文来源&#xff1a; 2023-ICASSP&#xff08;CCF…

Github 2024-02-23 开源项目日报 Top10

根据Github Trendings的统计&#xff0c;今日(2024-02-23统计)共有10个项目上榜。根据开发语言中项目的数量&#xff0c;汇总情况如下&#xff1a; 开发语言项目数量非开发语言项目4Python项目3TypeScript项目1HTML项目1Dart项目1Rust项目1 从零开始构建你喜爱的技术 创建周…

Flutter 如何启动新的页面时给页面传递参数

前言 前台开发&#xff0c;我们常有启动页面同时传递一些参数的需求&#xff0c;在android里面是通过Intent实现&#xff0c;本文探讨flutter的实现方式 正文 在Flutter中&#xff0c;给一个新的界面传递参数通常通过构造函数来实现 以主页面&#xff08;HomePage&#xff…