ThreeJS通过制作渐变光效贴图方式实现光柱效果

这段代码是一个使用Three.js库创建的简单3D场景,其中包含一个可以动态缩放和旋转的柱子(Pillar)。以下是对代码的分析和解释:

1. 引入Three.js及其扩展

代码首先通过ES6模块语法引入了Three.js的核心模块和一些扩展模块,如OrbitControls用于相机控制。

2. 定义Pillar函数

Pillar函数用于创建一个柱子对象,该对象包含一个mesh(实际是一个包含两个平面的Group)和一个render方法用于动画效果。

  • 配置合并:函数接受一个可选配置对象,与默认配置合并。
  • 几何体创建:使用PlaneGeometry创建一个平面,根据给定的高度和图片的比例计算宽度。
  • 材质加载:使用TextureLoader加载贴图,并创建MeshBasicMaterial
  • 网格创建:创建一个Mesh对象,并复制一个旋转90度以形成柱子的两个面。
  • 动画逻辑render方法实现柱子的缩放和旋转动画。

3. HTML和JavaScript集成

HTML部分设置了基本的页面结构和样式,JavaScript部分(作为模块脚本)则负责创建Three.js场景。

  • 场景、相机和渲染器:初始化ScenePerspectiveCameraWebGLRenderer
  • 添加柱子:调用Pillar函数创建柱子,并将其添加到场景中。
  • 灯光和背景:注释掉的部分显示了如何添加环境光和聚光灯,以及设置场景背景色。
  • 相机控制:使用OrbitControls允许用户通过鼠标交互控制相机。
  • 动画循环animate函数通过requestAnimationFrame创建动画循环,调用render方法更新柱子状态,并渲染场景。
  • 窗口大小调整:监听resize事件,以在窗口大小变化时更新相机和渲染器的设置。

4. 动画和交互

  • 柱子的缩放和旋转是通过Pillar函数内的render方法实现的,该方法在动画循环中被调用。
  • 用户可以通过OrbitControls自由旋转、缩放和平移相机,以不同角度查看场景。

总结

这段代码展示了如何使用Three.js创建一个简单的3D动态场景,包括几何体的创建、材质的加载、动画的实现以及用户交互的添加。通过修改配置和取消注释的灯光部分,可以进一步定制和丰富场景效果。

<!DOCTYPE html>
<html lang="en">

<head>
	<title>three.js webgl - particles - columns</title>
	<meta charset="utf-8">
	<meta name="viewport" content="width=device-width, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0">
	<style>
		body {
			margin: 0;
		}

		canvas {
			display: block;
		}
	</style>
</head>

<body>

	<script type="module">

		import { Scene, PerspectiveCamera, WebGLRenderer, AmbientLight, Color, SpotLight } from "three";
		import { Pillar } from "./LightColumnEffect.ts";
		import { OrbitControls } from 'three/addons/controls/OrbitControls.js';

		// 初始化场景、相机和渲染器  
		const scene = new Scene();
		const camera = new PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
		const renderer = new WebGLRenderer();
		renderer.setSize(window.innerWidth, window.innerHeight);
		document.body.appendChild(renderer.domElement);

		// 创建柱子并添加到场景  
		const { mesh, render } = Pillar();
		scene.add(mesh);
		//
		// scene.background = new Color(0xf0f0f0);

		// scene.add(new AmbientLight(0xaaaaaa));

		// const light = new SpotLight(0xffffff, 10000);
		// light.position.set(0, 25, 50);
		// light.angle = Math.PI / 5;

		// light.castShadow = true;
		// light.shadow.camera.near = 10;
		// light.shadow.camera.far = 100;
		// light.shadow.mapSize.width = 1024;
		// light.shadow.mapSize.height = 1024;

		// scene.add(light);
		/
		// 设置相机位置  
		camera.position.z = 50;

		// 添加OrbitControls以便交互  
		const controls = new OrbitControls(camera, renderer.domElement);

		// 创建动画循环  
		const animate = function () {
			requestAnimationFrame(animate);
			render(); // 调用Pillar的渲染函数来旋转柱子  
			controls.update(); // 更新OrbitControls  
			renderer.render(scene, camera);
		};

		// 处理窗口大小调整  
		window.addEventListener('resize', () => {
			camera.aspect = window.innerWidth / window.innerHeight;
			camera.updateProjectionMatrix();
			renderer.setSize(window.innerWidth, window.innerHeight);
		});

		// 开始动画循环  
		animate();

	</script>
</body>

</html>
import * as THREE from 'three';
import { OrbitControls } from 'three/addons/controls/OrbitControls.js';
import { Group, PlaneGeometry, TextureLoader, MeshBasicMaterial, Mesh, DoubleSide } from 'three';

export const Pillar = (config?: Partial<{
    width: number;
    height: number;
    lightPillarUrl: string;
    color: number;
    minScale: number; // 最小缩放比例  
    maxScale: number; // 最大缩放比例  
    scaleSpeed: number; // 缩放速度  
}>) => {
    // 默认配置  
    const defaultConfig = {
        height: 20,
        lightPillarUrl: "./pillarTexture.png",
        color: 0x00ffff,
        minScale: 1, // 最小缩放比例  
        maxScale: 2, // 最大缩放比例  
        scaleSpeed: 0.01 // 缩放速度  
    };

    // 合并配置  
    const conf = {
        ...defaultConfig,
        ...config
    };

    // 图片的尺寸255*41  
    const ratios = 255 / 41;
    // 按给定的高度计算宽度  
    const width = conf.height / ratios;
    const group = new Group();

    // 使用PlaneGeometry生成一个平面  
    const geometry = new PlaneGeometry(width, conf.height);
    geometry.rotateX(Math.PI / 2);
    geometry.translate(0, 0, conf.height / 2);

    // 加载贴图  
    const textureLoader = new TextureLoader();
    const material = new MeshBasicMaterial({
        map: textureLoader.load(conf.lightPillarUrl),
        color: conf.color,
        transparent: true,
        depthWrite: false,
        side: DoubleSide
    });

    const mesh = new Mesh(geometry, material);
    // 复制一个,并旋转90度  
    const mesh2 = mesh.clone();
    mesh2.rotateZ(Math.PI / 2);

    group.add(mesh, mesh2);

    // 初始缩放比例  
    let currentScale = conf.minScale;
    // 缩放方向,初始为增大  
    let scalingUp = true;

    const render = () => {
        // 更新缩放比例  
        if (scalingUp) {
            currentScale += conf.scaleSpeed;
            if (currentScale >= conf.maxScale) {
                scalingUp = false; // 达到最大缩放比例,开始缩小  
            }
        } else {
            currentScale -= conf.scaleSpeed;
            if (currentScale <= conf.minScale) {
                scalingUp = true; // 达到最小缩放比例,开始增大  
            }
        }

        // 应用缩放比例到组  
        group.scale.set(currentScale, currentScale, currentScale);

        // 旋转组  
        group.rotation.z += 0.01;
    };

    return { mesh: group, render };
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/886417.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

从0到1深入浅出构建Nest.Js项目

Nest (NestJS) 是一个用于构建高效、可扩展的 Node.js 服务器端应用程序的开发框架。它利用JavaScript 的渐进增强的能力&#xff0c;使用并完全支持 TypeScript &#xff08;仍然允许开发者使用纯 JavaScript 进行开发&#xff09;&#xff0c;并结合了 OOP &#xff08;面向对…

Spring Boot技术栈:打造高效在线商城

2 相关技术 2.1 Springboot框架介绍 Spring Boot是由Pivotal团队提供的全新框架&#xff0c;其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置&#xff0c;从而使开发人员不再需要定义样板化的配置。通过这种方式&#xff0c;Spring…

sentinel原理源码分析系列(三)-启动和初始化

本文是sentinel原理源码分析系列第三篇&#xff0c;分析sentinel启动和初始化 启动/初始化 sentinel初始化分两块&#xff0c;静态初始和适配器(包括aop) 静态初始 1. Root EntranceNode 如果我们用一栋楼类比资源调用&#xff0c;root EntranceNode好比一栋楼的大门&…

vulnhub-unknowndevice64 2靶机

vulnhub&#xff1a;https://www.vulnhub.com/entry/unknowndevice64-2,297/ 导入靶机&#xff0c;放在kali同网段&#xff0c;扫描 靶机在192.168.81.9&#xff0c;扫描端口 啥啊这都是&#xff0c;详细扫描一下 5555是adb&#xff0c;6465是ssh&#xff0c;12345看样子应该是…

简单的springboot 编写Socket服务接口

简单的springboot 编写Socket服务接口 1.需求 我们项目中有部分老接口为票据接口&#xff0c;其中实现为java socket形式进行实现&#xff0c;但是其中大部分信息都是原始公司封装的包进行实现的&#xff0c;想要修改非常费劲&#xff0c;所以此处简单了解了一下socket&#…

洛谷 P11045 [蓝桥杯 2024 省 Java B] 最优分组

[Problem Discription] \color{blue}{\texttt{[Problem Discription]}} [Problem Discription] [Analysis] \color{blue}{\texttt{[Analysis]}} [Analysis] 首先得注意这么一点&#xff1a; k k k 必须得是 n n n 的因数&#xff08;这里的 n , k n,k n,k 对应于题目的 N ,…

量化交易backtrader实践(三)_指标与策略篇(2)_内置指标A开头

在第1节中&#xff0c;我们学习了移动平均线的原理&#xff0c;中位数以及正态分布的概念&#xff0c;并通过python手工做了一个双均线的策略回测。了解了怎么用pandas计算移动平均线&#xff08;rollingmean)&#xff0c;怎么得到某一列上1个的值&#xff08;shift)&#xff0…

【算法与图】通向高效解决方案的钥匙

文章目录 遍历算法BFS&#xff08;广度优先遍历&#xff09;1. 什么是 BFS&#xff1f;2. 特点和应用3. BFS 示例 DFS&#xff08;深度优先搜索&#xff09;1. 什么是 DFS&#xff1f;2. DFS 的基本步骤3. 特点4. DFS 的应用5. DFS 示例 最小生成树问题1. 什么是最小生成树&…

【mmengine】优化器封装(OptimWrapper)(入门)优化器封装 vs 优化器

MMEngine 实现了优化器封装&#xff0c;为用户提供了统一的优化器访问接口。优化器封装支持不同的训练策略&#xff0c;包括混合精度训练、梯度累加和梯度截断。用户可以根据需求选择合适的训练策略。优化器封装还定义了一套标准的参数更新流程&#xff0c;用户可以基于这一套流…

虚拟机三种网络模式详解

在电脑里开一台虚拟机&#xff0c;是再常见不过的操作了。无论是用虚拟机玩只有旧版本系统能运行的游戏&#xff0c;还是用来学习Linux、跑跑应用程序都是很好的。而这其中&#xff0c;虚拟机网络是绝对绕不过去的。本篇文章通俗易懂的介绍了常见的虚拟网络提供的三种网络链接模…

数据结构-3.5.队列的顺序实现

一.队列的顺序实现&#xff0c;初始化操作以及判断队列是否为空&#xff1a; 1.图解&#xff1a; 2.代码&#xff1a; #include<stdio.h> #define MaxSize 10 //定义一个队列最多存储的元素个数 ​ typedef struct {int data[MaxSize]; //用静态数组存放队列元素int f…

【springboot】整合沙箱支付

目录 1. 配置沙箱应用环境2. 配置springboot项目1. 引入依赖2. 配置文件注册下载ngrok 3. 创建支付宝支付服务类4. 支付界面模板5. 控制类实现支付6. 测试 1. 配置沙箱应用环境 使用支付宝账号登录到开放平台控制台。 使用支付宝登录后&#xff0c;看到以下页面&#xff0c;下…

MFC工控项目实例二十二主界面计数背景颜色改变

承接专栏《MFC工控项目实例二十一型号选择界面删除参数按钮禁用切换》 1、在SEAL_PRESSUREDlg.h文件中添加代码 class CSEAL_PRESSUREDlg : public CDialog { public: CBrush m_brush1;CBrush m_brush2;CBrush m_brush3;... } 2、在SEAL_PRESSUREDlg.cpp文件中添加代码 BO…

在2核2G服务器安装部署MySQL数据库可以稳定运行吗?

阿里云2核2G服务器可以安装MySQL数据库吗&#xff1f;当然可以&#xff0c;并且可以稳定运行MySQL数据库&#xff0c;目前阿里云服务器网aliyunfuwuqi.com使用的就是阿里云2核2G服务器&#xff0c;在云服务器上安装MySQL数据库&#xff0c;可以稳定运行。 目前阿腾云用于运行M…

查看 git log的过程中看到 :说明日志输出可能超出屏幕大小,系统进入了分页模式

在命令行提示符中&#xff0c;通常 : 表示系统等待进一步的输入。如果你在查看 git log 的过程中看到 :&#xff0c;说明日志输出可能超出屏幕大小&#xff0c;系统进入了分页模式&#xff0c;默认使用 less 命令查看内容。 此时你可以&#xff1a; 按 q 退出日志查看。按 En…

算法笔记(五)——分治

文章目录 算法笔记&#xff08;五&#xff09;——分治快排颜色分类排序数组数组中的第K个最大元素库存管理 III 归并排序数组交易逆序对的总数计算右侧小于当前元素的个数翻转对 算法笔记&#xff08;五&#xff09;——分治 分治算法字面上的解释是“分而治之”&#xff0c;就…

Python 从入门到实战32(数据库MySQL)

我们的目标是&#xff1a;通过这一套资料学习下来&#xff0c;通过熟练掌握python基础&#xff0c;然后结合经典实例、实践相结合&#xff0c;使我们完全掌握python&#xff0c;并做到独立完成项目开发的能力。 上篇文章我们讨论了数据库编程接口操作的相关知识。今天我们将学习…

【框架篇】过滤器和拦截器的区别以及使用场景

在项目开发中&#xff0c;常常会同时配置拦截器&#xff08;Interceptor&#xff09;和过滤器&#xff08;Filter&#xff09;&#xff0c;以下就是它们两个主要的区别&#xff1a; 过滤器&#xff08;Filter&#xff09; 配置和实现 Filter的实现还是很简单的&#xff0c;可…

【微服务】组件、基础工程构建(day2)

组件 服务注册和发现 微服务模块中&#xff0c;一般是以集群的方式进行部署的&#xff0c;如果我们调用的时候以硬编码的方式&#xff0c;那么当服务出现问题、服务扩缩容等就需要对代码进行修改&#xff0c;这是非常不好的。所以微服务模块中就出现了服务注册和发现组件&…

计算机毕业设计 基于Python的广东旅游数据分析系统的设计与实现 Python+Django+Vue Python爬虫 附源码 讲解 文档

&#x1f34a;作者&#xff1a;计算机编程-吉哥 &#x1f34a;简介&#xff1a;专业从事JavaWeb程序开发&#xff0c;微信小程序开发&#xff0c;定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事&#xff0c;生活就是快乐的。 &#x1f34a;心愿&#xff1a;点…