当前位置: 首页 > news >正文

计算机图形学:(三)MVP变换扩展

Three.js

         WebGL允许把JavaScript和OpenGL 结合在一起运用,但使用WebGL原生的API来写3D程序非常的复杂,同时需要相对较多的数学知识,对于前端开发者来说学习成本非常高。

        Three.js是基于webGL的封装的一个易于使用且轻量级的3D库,Three.js对WebGL提供的接口进行了非常好的封装,简化了很多细节,大大降低了学习成本,极大地提高了性能,功能也非常强大。

        官网:Three.js – JavaScript 3D Library (threejs.org)

    # 下载

    GitHub:GitHub - mrdoob/three.js: JavaScript 3D Library. 

    # 基础

        基于Three.js写页面的三要素是:场景(Scene)相机(Camera)渲染器(Renderer)

  • 场景是一个容器,主要用于保存、跟踪所要渲染的物体和使用的光源
  • 摄像机对象,摄像机决定了能够在场景看到什么。
  • 渲染器对象,该对象会基于摄像机的角度来计算场景对象在浏览器中会渲染成什么样子。

        将一个Scene对象Camera对象传递给一个Renderer,它将在相机视锥体内的3D场景的部分作为2D图像呈现(绘制)到Canvas。 

        SceneGraph是一个树状结构,由各种对象组成,如一个Scene对象、多个Mesh对象、Light对象、Group、Object3D和Camera对象

        在图中,Camera对象有一半是不在SceneGraph中的。这是为了表示在three.js中,不像其他对象,Camera不需要在场景中才能起作用。

    每个添加到Three.js场景的对象,甚至包括THREE.Scene本身,都是继承自一个名为THREE.Object3D的对象。

        # 入门

        直接找了其他博主的教程:https://blog.csdn.net/sd1sd2/category_12898367.html

示例一:初始化

        这里我们参照计算机图形学:(二)MVP变换示例用Three.js初始化一个立方体,相机看向它: 

<html><head><title>视图变换基础案例</title><style>body {margin: 0;overflow: hidden;}canvas {width: 100%;height: 100%;}</style>
</head><body><script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/r83/three.min.js"></script><script>// 【0】创建一个场景var scene = new THREE.Scene();// 【1】立方体几何var geometry = new THREE.BoxGeometry(4, 4, 2);// 渲染的基础材料var materialBasic = new THREE.MeshBasicMaterial({color: 0xffffff, wireframe: true });// 这是一种简单的材质,根据物体表面的法向量计算颜色var materialNormal = new THREE.MeshNormalMaterial();// 创建多材料对象var cube = THREE.SceneUtils.createMultiMaterialObject(geometry, [materialBasic,materialNormal]);// 位置上添加渲染的东西scene.add(cube); // 默认情况下,使用 scene.add() 会将物体加载到(0,0,0) // 【2】做一个相机模拟真实人的视角// 注意,在three.js中的大多数角度都是弧度,但由于某些原因,透视相机需要角度。var camera = new THREE.PerspectiveCamera(30, window.innerWidth / window.innerHeight, 0.1, 1000);camera.position.set(0, 0, 20);// 观察位置camera.lookAt(cube.position);// 【3】渲染器var renderer = new THREE.WebGLRenderer();renderer.setSize(window.innerWidth, window.innerHeight);document.body.appendChild(renderer.domElement);// 【4】添加聚光灯并且添加到页面当中var spotLight = new THREE.SpotLight(0xffffff);spotLight.position.set(0, 20, 20);spotLight.intensity = 5;scene.add(spotLight);// 【5】添加三维坐标系var axes = new THREE.AxisHelper(6);scene.add(axes);// 最终的渲染renderer.render(scene, camera);</script>
</body></html>

示例二:绕点旋转

① 以固定角度旋转相机

        同计算机图形学:(二)MVP变换示例一样,我们需要通过计算相机旋转后的新位置来更新视图矩阵,从而达到让相机以一个固定的旋转度数绕立方体中心旋转的效果:

let angle = 0;
let radius = camera.position.z;
const rotationValue = 0.01;var tick = function () {angle += rotationValue;// 更新相机位置(圆形路径)camera.position.x = radius * Math.sin(angle);camera.position.z = radius * Math.cos(angle);camera.lookAt(cube.position); // 确保相机一直看向物体// 场景改变时,重新执行渲染操作渲染三维场景renderer.render(scene, camera);requestAnimationFrame(tick);
};
tick();

        轨道控制器(OrbitControls)是 Three.js 中的一个常用工具,用于控制相机的运动。如果目标是实现“相机绕物体旋转”,直接使用 OrbitControls更简单:

        注:这里为了导入OrbitControls模块,变更了有些API(下述代码删减了未做修改的部分)。

<script type="importmap">{"imports": {"three": "https://unpkg.com/three@0.148.0/build/three.module.js","three/examples/jsm/controls/OrbitControls": "https://unpkg.com/three@0.148.0/examples/jsm/controls/OrbitControls.js","three/examples/jsm/utils/SceneUtils": "https://unpkg.com/three@0.148.0/examples/jsm/utils/SceneUtils.js"}}
</script><script type="module">import * as THREE from 'three';import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls';import { createMultiMaterialObject } from 'three/examples/jsm/utils/SceneUtils';// 创建多材料对象var cube = createMultiMaterialObject(geometry, [ // 修改前:THREE.SceneUtils.createMultiMaterialObjectmaterialBasic,materialNormal]);// 添加三维坐标系var axes = new THREE.AxesHelper(6); // 修改前:THREE.AxisHelper// 添加轨道控制器const controls = new OrbitControls(camera, renderer.domElement);controls.target.set(cube.position.x, cube.position.y, cube.position.z); // 设置目标点为物体中心controls.autoRotate = true;   // 启用自动旋转controls.autoRotateSpeed = 1.0; // 旋转速度var tick = function () {controls.update();renderer.render(scene, camera);requestAnimationFrame(tick);};tick();</script>

        其他:当对相机绕Y轴进行旋转时,看起来会是“物体绕相机旋转”:

var tick = function () {camera.rotateY(0.01);renderer.render(scene, camera);requestAnimationFrame(tick);
};
tick();

② 鼠标拖拽旋转相机

        OrbitControls 会监听鼠标事件(如鼠标移动、鼠标滚轮滚动、鼠标按键点击等),根据用户的操作实时计算相机应该移动到的新位置和角度,然后更新相机的相关属性,从而改变相机在 3D 场景中的视角。

const controls = new OrbitControls(camera, renderer.domElement);// 监听控制器,每次拖动后重新渲染画面
controls.addEventListener('change', function () {renderer.render(scene, camera); //执行渲染操作
});

        比起之前做的简陋版,OrbitControls实现的效果更加丝滑了: 

        想知道其具体实现可以直接看源码:THREE:controls/OrbitControls.js(部分源码解读:https://zhuanlan.zhihu.com/p/447471816),以及也可以手写控制器,如three.js鼠标控制物体旋转

相关文章:

  • 初识 Flask 框架
  • 详解MySQL锁机制
  • 七彩喜适老化改造:让每个空间成为长者尊严的守护者
  • 广东省省考备考(第十八天5.23)—言语:语句排序题(听课后强化训练)
  • 基于大模型的股骨干骨折全周期预测与诊疗方案研究报告
  • 【MySQL成神之路】MySQL索引相关介绍
  • C++性能测试工具——sysprof的使用
  • FRP技术概览
  • 注册并创建一个微信小程序
  • git初始化及操作指南
  • 《边缘算力困局突破:智能体模型动态调度全解析》
  • 2025年电工杯数学建模竞赛A题的 数据基于Kaggle 的AMS 2013-2014 太阳能预测大赛 (2.8G 数据)
  • 腾讯地图WebServiceAPI提供基于HTTPS/HTTP协议的数据接口
  • JVM——Java 虚拟机的监控及诊断工具(GUI 篇)
  • 信息系统项目管理师考前练习9
  • 关系数据库基础入门
  • 代码随想录算法训练营第四十八天
  • Golang 之 Context 源码解析(1.20+)
  • 贪心算法套路模板+详细适用场景+经典题目清单
  • arcgis js统计FeatureLayer的椭球面积、平面面积
  • 培训web网站开发/如何建立电商平台
  • 专业网站建设出售/线上营销推广方案有哪些
  • 网店logo设计/seo公司后付费
  • 百度网站优化培训/今日最新消息新闻报道
  • 徐州网站建设找哪家好/优化人员配置
  • 网络系统部/seo网站管理