• WebGL-Vue3-TS-Threejs:基础练习 / Javascript 3D library / demo


    一、理解Three.js

    Three.js是一个用于WebGL渲染的JavaScript库。它提供了一组工具和类,用于创建和渲染3D图形和动画。简单理解(并不十分准确),Three.js之于WebGL,好比,jQuery.js之于JavaScript。

    OpenGL 是一个跨平台3D/2D的绘图标准,WebGL则是OpenGL 在浏览器上的一个实现。
    web前端开发人员可以直接用WebGL接口进行编程,但 WebGL只是非常基础的绘图API,需要编程人员有很多的数学知识、绘图知识才能完成3D编程任务,而且代码量巨大。Threejs 对 WebGL 进行了封装,让前端开发人员在不需要掌握很多数学知识和绘图知识的情况下,也能够轻松进行web 3D开发,降低了门槛,同时大大提升了效率。

    WebGL:开始学习 / 理解 WebGL / WebGL 需要掌握哪些知识 / 应用领域 / 前端值得学WebGL吗_webgl培训-CSDN博客

    Three.js的主要特点

    序号特点描述
    1简单易用它的API非常友好,易于理解和使用。
    2兼容性强支持多种浏览器和设备,不需要其他插件和软件。
    3功能强大提供了广泛的3D渲染功能,包括材质、灯光、相机等多种元素。
    4社区活跃拥有庞大的开发者社区,可以找到大量的示例代码和教程。

    通过Three.js,开发者可以快速创建并展示3D模型、场景、动画等内容。它支持多种文件格式,如obj、fbx、glb等,可以轻松导入和使用。同时,它还支持VR和AR等技术,可以创建更加沉浸式的体验。

    关键词:场景、相机、光源、材质、贴图、建模、着色

    二、文档

    ThingJS 文档中心

    https://threejs.org/

    Three.js中文网

    三、项目介绍

    vue3 + ts + vite + three.js

    四、安装

    
    <code class="language-plaintext hljs">pnpm add three
    pnpm add @types/three</code>

    "three": "^0.158.0", 

    "@types/three": "^0.154.0", 

    五、导入核心库,获取场景

    1. <template>
    2. <div class="container">
    3. <span>threejs</span>
    4. </div>
    5. </template>
    6. <script setup lang="ts">
    7. import * as THREE from 'three'
    8. const scene = new THREE.Scene()
    9. console.log('scene:', scene)
    10. </script>
    11. <style scoped lang="less">
    12. </style>

    六、基础练习

    6.1、绘制一条直线

    1. <template>
    2. <div ref="container" id="container" style="width: 200px;height: 200px;"></div>
    3. </template>
    4. <script setup lang="ts">
    5. import * as THREE from 'three'
    6. const container:any = ref(null)
    7. onMounted(()=>{
    8. // 创建场景
    9. const scene = new THREE.Scene();
    10. scene.background = new THREE.Color(0xadacad);
    11. // 创建相机
    12. const camera = new THREE.PerspectiveCamera(
    13. 45,
    14. container.value.clientWidth / container.value.clientHeight,
    15. 0.1,
    16. 100
    17. );
    18. camera.position.set(0, 0, 3);
    19. // 创建渲染器
    20. const renderer = new THREE.WebGLRenderer();
    21. console.log('24', container)
    22. renderer.setSize(container.value.clientWidth, container.value.clientHeight);
    23. container.value.appendChild(renderer.domElement);
    24. // 定义顶点数据
    25. const positions = [
    26. -1, 0, 0,
    27. 0, 1, 0,
    28. ];
    29. // 创建一个 BufferGeometry 对象
    30. const geometry = new THREE.BufferGeometry();
    31. const positionAttribute = new THREE.Float32BufferAttribute(positions, 3);
    32. geometry.setAttribute('position', positionAttribute);
    33. // 创建一个 LineLoop 对象
    34. const material = new THREE.LineBasicMaterial({ color: 0xf00, linewidth: 5 });
    35. const line = new THREE.LineLoop(geometry, material);
    36. // 将线条添加到场景中
    37. scene.add(line);
    38. // 开始渲染循环
    39. function render() {
    40. renderer.render(scene, camera);
    41. requestAnimationFrame(render);
    42. }
    43. render();
    44. })
    45. </script>
    46. <style scoped lang="less">
    47. </style>

    6.2、通过直线,绘制画一个正方形

    本例中,我们首先创建了一个 div 元素引用,它将用作 three.js 的渲染器容器。然后创建了场景、相机和渲染器,并将渲染器的输出添加到页面中。

    接着,我们定义了闭合线条的顶点数据,并使用 THREE.BufferGeometry 创建了几何体。然后我们使用 THREE.LineLoop 创建了线条对象,并将其添加到场景中。

    1. <template>
    2. <div ref="container" id="container" style="width: 200px;height: 200px;"></div>
    3. </template>
    4. <script setup lang="ts">
    5. import * as THREE from 'three'
    6. const container:any = ref(null)
    7. onMounted(()=>{
    8. // 创建场景
    9. const scene = new THREE.Scene();
    10. scene.background = new THREE.Color(0xadacad);
    11. // 创建相机
    12. const camera = new THREE.PerspectiveCamera(
    13. 45,
    14. container.value.clientWidth / container.value.clientHeight,
    15. 0.1,
    16. 100
    17. );
    18. camera.position.set(0, 0, 3);
    19. // 创建渲染器
    20. const renderer = new THREE.WebGLRenderer();
    21. console.log('24', container)
    22. renderer.setSize(container.value.clientWidth, container.value.clientHeight);
    23. container.value.appendChild(renderer.domElement);
    24. // 定义顶点数据
    25. const positions = [
    26. -1, 0, 0,
    27. 0, 1, 0,
    28. 1, 0, 0,
    29. 0, -1, 0,
    30. -1, 0, 0,
    31. ];
    32. // 创建一个 BufferGeometry 对象
    33. const geometry = new THREE.BufferGeometry();
    34. const positionAttribute = new THREE.Float32BufferAttribute(positions, 3);
    35. geometry.setAttribute('position', positionAttribute);
    36. // 创建一个 LineLoop 对象
    37. const material = new THREE.LineBasicMaterial({ color: 0xf00, linewidth: 5 });
    38. const line = new THREE.LineLoop(geometry, material);
    39. // 将线条添加到场景中
    40. scene.add(line);
    41. // 开始渲染循环
    42. function render() {
    43. renderer.render(scene, camera);
    44. requestAnimationFrame(render);
    45. }
    46. render();
    47. })
    48. </script>
    49. <style scoped lang="less">
    50. </style>

    6.3、 vue3 typescript项目 在script使用setup语法糖的方式下,使用最新 threejs api 创建文字

    待补充

    使用THREE.TextGeometry创建文字的模型,并使用THREE.MeshBasicMaterialTHREE.Mesh将其渲染到场景中

    6.4、绘制一个圆形

    本例中,首先创建了一个Three.js场景、相机和渲染器。然后创建了一个圆形几何体和一个材质,并使用它们创建了一个网格。最后将圆形添加到场景中,并创建一个animate函数用于实现动画效果。

    1. <script lang="ts">
    2. import { defineComponent } from 'vue';
    3. import * as THREE from 'three';
    4. export default defineComponent({
    5. name: 'HelloWorld',
    6. setup() {
    7. const scene = new THREE.Scene();
    8. const camera = new THREE.PerspectiveCamera(
    9. 75,
    10. window.innerWidth / window.innerHeight,
    11. 0.1,
    12. 1000
    13. );
    14. const renderer = new THREE.WebGLRenderer();
    15. renderer.setSize(window.innerWidth, window.innerHeight);
    16. document.getElementById('app')?.appendChild(renderer.domElement);
    17. const geometry = new THREE.CircleGeometry(50, 32);
    18. const material = new THREE.MeshBasicMaterial({
    19. color: 0xff0000,
    20. wireframe: true,
    21. });
    22. const circle = new THREE.Mesh(geometry, material);
    23. scene.add(circle);
    24. camera.position.z = 100;
    25. const animate = function () {
    26. requestAnimationFrame(animate);
    27. circle.rotation.x += 0.01;
    28. circle.rotation.y += 0.01;
    29. renderer.render(scene, camera);
    30. };
    31. animate();
    32. return { scene, camera, renderer };
    33. },
    34. });
    35. script>

    6.5、绘制一个圆锥

    1. <script lang="ts">
    2. import { defineComponent } from 'vue';
    3. import * as THREE from 'three';
    4. export default defineComponent({
    5. name: 'HelloWorld',
    6. setup() {
    7. const scene = new THREE.Scene();
    8. const camera = new THREE.PerspectiveCamera(
    9. 75,
    10. window.innerWidth / window.innerHeight,
    11. 0.1,
    12. 1000
    13. );
    14. const renderer = new THREE.WebGLRenderer();
    15. renderer.setSize(window.innerWidth, window.innerHeight);
    16. document.getElementById('app')?.appendChild(renderer.domElement);
    17. const geometry = new THREE.ConeGeometry(50, 100, 32);
    18. const material = new THREE.MeshBasicMaterial({
    19. color: 0xff0000,
    20. wireframe: true,
    21. });
    22. const cone = new THREE.Mesh(geometry, material);
    23. scene.add(cone);
    24. camera.position.z = 100;
    25. const animate = function () {
    26. requestAnimationFrame(animate);
    27. cone.rotation.x += 0.01;
    28. cone.rotation.y += 0.01;
    29. renderer.render(scene, camera);
    30. };
    31. animate();
    32. return { scene, camera, renderer };
    33. },
    34. });
    35. script>

    6.6、绘制一个立方体

    1. <template>
    2. <div id="app"></div>
    3. </template>
    4. <script lang="ts">
    5. import { defineComponent } from 'vue';
    6. import * as THREE from 'three';
    7. export default defineComponent({
    8. name: 'HelloWorld',
    9. setup() {
    10. const scene = new THREE.Scene();
    11. const camera = new THREE.PerspectiveCamera(
    12. 75,
    13. window.innerWidth / window.innerHeight,
    14. 0.1,
    15. 1000
    16. );
    17. const renderer = new THREE.WebGLRenderer();
    18. renderer.setSize(window.innerWidth, window.innerHeight);
    19. document.getElementById('app')?.appendChild(renderer.domElement);
    20. const geometry = new THREE.BoxGeometry(1, 1, 1);
    21. const material = new THREE.MeshBasicMaterial({ color: 0x00ff00 });
    22. const cube = new THREE.Mesh(geometry, material);
    23. scene.add(cube);
    24. camera.position.z = 5;
    25. const animate = function () {
    26. requestAnimationFrame(animate);
    27. cube.rotation.x += 0.01;
    28. cube.rotation.y += 0.01;
    29. renderer.render(scene, camera);
    30. };
    31. animate();
    32. return { scene, camera, renderer };
    33. },
    34. });
    35. </script>

    6.7、绘制一个球体

    1. <script lang="ts">
    2. import { defineComponent, onMounted, ref } from 'vue';
    3. import * as THREE from 'three';
    4. export default defineComponent({
    5. setup() {
    6. const container = ref(null);
    7. onMounted(() => {
    8. // 创建渲染器
    9. const renderer = new THREE.WebGLRenderer();
    10. renderer.setSize(window.innerWidth, window.innerHeight);
    11. // 将渲染器添加到DOM中
    12. if(container.value) {
    13. container.value.appendChild(renderer.domElement);
    14. }
    15. // 创建场景
    16. const scene = new THREE.Scene();
    17. // 创建相机
    18. const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
    19. camera.position.z = 5;
    20. // 添加环境光
    21. const ambientLight = new THREE.AmbientLight(0xffffff, 0.5);
    22. scene.add(ambientLight);
    23. // 添加点光源
    24. const pointLight = new THREE.PointLight(0xffffff, 1, 100);
    25. pointLight.position.set(0, 0, 5);
    26. scene.add(pointLight);
    27. // 创建红色球体
    28. const geometry = new THREE.SphereGeometry(1, 32, 32);
    29. const material = new THREE.MeshPhongMaterial({ color: 0xff0000, shininess: 100 });
    30. const sphere = new THREE.Mesh(geometry, material);
    31. scene.add(sphere);
    32. // 渲染场景
    33. const render = () => {
    34. requestAnimationFrame(render);
    35. sphere.rotation.x += 0.01;
    36. sphere.rotation.y += 0.01;
    37. renderer.render(scene, camera);
    38. };
    39. render();
    40. });
    41. return {
    42. container
    43. }
    44. }
    45. });
    46. script>

    在这个示例中,我们添加了一个点光源,并将球体的材质参数shininess设置为100,这使球体的表面更加光滑。

    你还可以尝试调整环境光和点光源的强度和颜色,或者使用其他类型的光源和材质来改变球体的外观。

     

    七、相关内容

    canvas、canvas3D、playcanvas、tween、pixi、three.js、webGL、OpenGL

    八、过程记录

    8.1、学习目标

    1、了解基础的3D数学和计算机图形学概念。学习向量、矩阵、坐标系、光照和着色器等基础知识。
    2、熟悉WebGL技术,理解网页版3D渲染的基本原理,包括顶点着色器和片段着色器。
    3、掌握Three.js的API,包括场景、相机、灯光、材质和几何体等基础概念,以及如何使用它们来创建3D图形。
    4、学习如何使用Three.js创建复杂的3D场景,包括如何向场景添加3D模型、纹理、动画和粒子效果。
    5、探索Three.js的高级功能,如阴影、反射、抗锯齿和后期处理。
    6、将Three.js与其他技术集成,如HTML、CSS和JavaScript,以创建功能丰富且交互性强的3D应用程序。
    7、学习如何与其他Web技术集成。将Three.js集成到Web应用程序中,并与其他技术,如HTML、CSS和JavaScript一起使用。
    8、练习并参与开源项目,在实践中提高技能水平。
    学习Three.js需要一定数学和图形学基础,需要耐心和实践。建议先从基础开始,逐步学习和实践,掌握基本概念和技能后再进行高级的应用。

    8.2、Three.js之于WebGL,好比,jQuery.js之于JavaScript,,这样理解有合理性吗

    这个比喻在某些方面是有合理性的,但不是完全准确的。 

    类比中的相似之处在于,jQuery.js在JavaScript世界中扮演了许多网站开发人员的角色。Three.js也在WebGL世界中扮演着相似的角色。jQuery.js使用DOM元素并操作它们,类似地,Three.js使用WebGL上下文并操作它们。这是比喻的相似之处。

    然而,WebGL在功能和范围上远远超过了DOM操作,因此Three.js提供了比jQuery.js更加复杂的功能和API。Three.js的目的是演示和创建3D图形和交互式应用程序,并且具有原生优化的性能。与jQuery.js不同,Three.js是专注于3D图形编程的库。

    因此,比喻可以作为一种简单的概括,但它并不完全准确,并且可以误导人们对Three.js和WebGL的理解。

    8.3、使用Three.js创建一个简单的立方体场景的基本步骤

    1、设置照相机 camera = new THREE.Camera(); 用于观察物体
    2、设置场景 scene = new THREE.Scene() 用于承载物体
    3、建立物体 geometry = new THREE.CubeGeometry(200, 200, 200); 
    4、创建网格 mesh = new THREE.Mesh(geometry, material) 5、渲染呈现 renderer.render(scene, camera) 这是啥

    8.4、npm插件'three'和'@types/three'的区别如下

    `npm`插件`three`是`Three.js`的`JavaScript`实现,是用于在`JavaScript`中创建和渲染`3D`图形的框架。

    `@types/three`是TypeScript类型定义的`npm`包,它提供了`TypeScript`开发环境中`Three.js`库的类型定义文件。这些类型定义文件可以帮助`TypeScript`编译器检查代码类型,并提供提示和自动完成。

    因此,`npm`插件`three`是`Three.js`库本身,而`@types/three`则提供了`TypeScript`开发环境所需的类型定义文件。如果你在使用`TypeScript`开发`Three.js`应用程序,则需要安装`npm`包`@types/three`,以获取`Three.js`库的类型定义文件,从而可以在开发过程中获得更好的类型安全支持和代码提示。如果你只是在使用标准的`JavaScript`进行开发,则只需要安装`npm`包`three`即可。

    8.5、THREE.Geometry方法不存在新版本了吗

    是的,从版本 r125 开始,three.js 废弃了 THREE.Geometry,并引入了 THREE.BufferGeometry 作为建模的首选选择。这是为了提高性能,并使开发人员更容易控制顶点数据。

    因此,如果您使用的是最新版本的 three.js,建议使用 THREE.BufferGeometry 来代替 THREE.Geometry

    8.6、需要在onmounted声明周期,确保DOM元素已经加载完成

    需要注意的是,在使用Three.js时,需要确保DOM元素已经加载完成,否则会出现错误。上述示例中使用了安全访问运算符(?.)和类型断言(!),以确保能够正确获取DOM元素。

    8.7、playcanvas.js 和 three.js 有什么相同异同

    PlayCanvas和Three.js都是用于创建和显示3D图形的JavaScript库,但它们有一些关键的区别。

    序号相同
    1两者都是用于创建和显示3D图形的JavaScript库
    2两者都提供了一系列的API来方便开发者创建和控制3D图形
    3两者都支持在浏览器中运行,而且都能很好地支持各种主流的浏览器
    4两者都提供了一些基本的3D功能,如3D模型的加载和呈现、材质和纹理的应用、动画和物理效果等
    5两者都是开源的项目,并且都有一个活跃的社区在支持和维护
    序号不同
    1PlayCanvas.js的定位更多是在线多人游戏的开发,它在多人游戏的网络性能优化方面有更好的表现。而Three.js的应用范围更广泛,包括但不限于游戏开发,定位更多是用于3D呈现和交互式场景的创建。
    2PlayCanvas提供了一套完整的游戏开发框架,包括物理引擎、碰撞检测、声音系统等,而Three.js主要专注于图形渲染。
    3PlayCanvas提供了实时渲染功能,可以在浏览器中实现流畅的3D游戏体验,而Three.js虽然也支持实时渲染,但通常需要配合其他库(如WebGL或WebGPU)来实现更好的效果。
    4PlayCanvas的文档和社区比Three.js更加完善,API更加简洁易用,对于开发者来说更容易上手和使用。而Three.js的API更加底层,需要开发者有一定的3D编程经验。
    5PlayCanvas支持组件化开发,可以通过组合不同的组件来快速构建复杂的3D场景,而Three.js则需要更多的手动设置和调整。
    6PlayCanvas支持多种输入方式(如键盘、鼠标、触摸等),适合开发各种类型的交互式应用,而Three.js则更侧重于图形渲染方面的输入控制。

    九、老语法 --》新语法

    序号老语法新语法
    1

    new THREE.FontLoader()

    TextureLoader
    THREE.TypefaceLoader
    2

    new THREE.TextGeometry()

    THREE.TextBufferGeometry???
    THREE.Mesh???
    待补充待补充待补充

    8.8、threejs可以用来建模吗

    是的,three.js可以用来建模,你可以使用three.js中的几何体来创建基本图形,也可以使用模型文件导入器加载已经建好的3D模型文件,然后在场景中进行操作和渲染。同时,three.js还提供了一些工具和插件可以帮助制作高质量的模型,如线框渲染,材质编辑器,灯光编辑器等。

    8.9、three.js 和 Blender 用来建模的区别

    1、使用难度:Blender是一款完整的3D建模软件,使用起来相对来说比较复杂,需要掌握一些专业的操作技巧;而three.js则是一个基于Web的3D库,使用起来相对简单,只需要了解一些基本的Web开发知识即可。

    2、功能差异:Blender作为一款专业的3D建模软件,功能非常强大,可以进行复杂的建模、渲染、动画等操作;而three.js主要是用来在Web中展示和操作3D场景,其建模功能相对简单。

    3、输出格式:Blender支持多种输出格式,可以将模型输出为OBJ、FBX、3DS等格式,方便在其他3D软件中使用;而three.js则主要使用JSON等Web格式进行输出,方便在Web中使用。

    如果需要进行复杂的3D建模操作,可以考虑使用Blender;如果只需要在Web中展示和操作简单的3D场景,可以选择使用three.js。

    十、参考链接

    补间动画tween.js_tween补间动画-CSDN博客

    和我一起学 Three.js【初级篇】:2. 掌握几何体 - 知乎

    关于javascript:Three.js新的THREE.TextBufferGeometry()无法读取未定义错误的属性’yMax’ | 码农家园

    Three.js API手册 / LineBasicMaterial - 汇智网

    Caught error TypeError: THREE.Geometry is not a constructor

    Three.js中文网

    https://www.cnblogs.com/tiandi/p/17053774.html

    vue3项目中使用three.js的操作步骤_vue.js_脚本之家

    开始第一个Hello world! | 码上动力

    Lamborghini Huracán STO 2020

    three.js全网最全最新入门课程(2023年10月更新)【搞定前端前沿技术】_哔哩哔哩_bilibili

    threejs-vite-vue实战课程_哔哩哔哩_bilibili

    带你入门three.js——从0到1实现一个3d可视化地图 - 知乎

  • 相关阅读:
    使用tkinter开发GUI程序4 -- tkinter常见控件的特征属性(第二部分)
    静态static
    python安装imblearn一直找不到包的解决方法
    基于STM32设计的便携式温度自动记录仪
    linux驱动开发:驱动中的并发控制
    Jmter接口网站压力测试工具
    【RocketMQ】主从同步实现原理
    对抗生成网络GAN系列——GANomaly原理及源码解析
    1543_AURIX_TC275_CPU子系统_CPU内核实现特性
    8+肿瘤+多组机器学习+分型。
  • 原文地址:https://blog.csdn.net/snowball_li/article/details/128677686