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

news2024/10/3 16:25:54

这段代码是一个使用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.coloradmin.cn/o/2186524.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

算法与数据结构--二分查找

原理 简单说就是每次二分取中间值&#xff0c;然后将中间值与我们要找的值比较&#xff0c;如果比它大就移动左边界到其右侧&#xff0c;如果比它小就移动右边界到其左侧。直到中间值等于该值或者经过移动直至右边界在左边界左侧为止。 二.具体实现 public int searchInsert(…

数据结构 ——— 单链表oj题:反转链表

目录 题目要求 手搓一个简易链表 代码实现 题目要求 给你单链表的头节点 head &#xff0c;请你反转链表&#xff0c;并返回反转后的链表 手搓一个简易链表 代码演示&#xff1a; struct ListNode* n1 (struct ListNode*)malloc(sizeof(struct ListNode)); assert(n1);…

【重学 MySQL】五十三、MySQL数据类型概述和字符集设置

【重学 MySQL】五十三、MySQL数据类型概述和字符集设置 MySQL数据类型概述MySQL字符集设置注意事项 MySQL数据类型概述 MySQL是一个流行的关系型数据库管理系统&#xff0c;它支持多种数据类型&#xff0c;以满足不同数据处理和存储的需求。理解并正确使用这些数据类型对于提高…

已解决:org.springframework.web.HttpMediaTypeNotAcceptableException

文章目录 写在前面问题描述报错原因分析&#xff1a; 解决思路解决办法1. 确保客户端请求的 Accept 头正确2. 修改 Controller 方法的 produces 参数3. 配置合适的消息转换器4. 检查 Spring 配置中的媒体类型5. 其他解决方案 总结 写在前面 在开发过程中&#xff0c;Spring 框…

实验1 集成开发环境的使用及程序设计入门

1、求两点之间的距离 【问题描述】从键盘输入平面上已知两点的坐标A(x1,y1)、B(x2,y2)&#xff0c;计算两点之间的距离。结果保留4位小数。 【输入形式】两个点的坐标 【输出形式】两点之间的距离 【样例输入】 0 0 3 4 【样例输出】 5.0000 #define _CRT_SECURE_NO_WARNI…

(C语言贪吃蛇)12.Linux线程概念引入及编程实现

目录 前言 Linux线程概念 解决方法 注意事项⚠️ 解决问题&#x1f525; 总结 前言 我们上节提出了两个死循环不能同时运行&#xff0c;导致我们无法控制贪吃蛇的运动方向&#xff0c;本节我们便来解决这个问题。 Linux线程概念 线程是一个进程内部的控制序列&#xff0c…

鸿蒙应用开发前置学习-TypeScript

注意&#xff1a;博主有个鸿蒙专栏&#xff0c;里面从上到下有关于鸿蒙next的教学文档&#xff0c;大家感兴趣可以学习下 如果大家觉得博主文章写的好的话&#xff0c;可以点下关注&#xff0c;博主会一直更新鸿蒙next相关知识 专栏地址: https://blog.csdn.net/qq_56760790/…

杭州电子科技大学《2019年+2023年861自动控制原理真题》 (完整版)

本文内容&#xff0c;全部选自自动化考研联盟的&#xff1a;《杭州电子科技大学861自控考研资料》的真题篇。后续会持续更新更多学校&#xff0c;更多年份的真题&#xff0c;记得关注哦~ 目录 2019年真题 2023年真题 Part1&#xff1a;2019年2023年完整版真题 2019年真题 2…

【Linux私房菜(九)之进程间通信】

文章目录 进程间通信介绍进程间通信的概念进程间通信的目的进程间通信的本质进程间通信的分类 管道什么是管道匿名管道匿名管道的原理pipe函数匿名管道使用步骤管道读写规则管道的特点管道的四种特殊情况管道的大小 命名管道命名管道的原理使用命令创建命名管道创建一个命名管道…

会声会影导出视频mp4格式哪个最高清,会声会影输出格式哪个清晰

调高分辨率后&#xff0c;mp4视频还是不清晰。哪怕全部使用4K级素材&#xff0c;仍然剪不出理想中的高画质作品。不是你的操作有问题&#xff0c;而是剪辑软件没选对。Corel公司拥有全球顶尖的图像处理技术&#xff0c;该公司研发的会声会影视频剪辑软件&#xff0c;在过去的20…

详解CSS中的伪元素

4.3 伪元素 可以把样式应用到文档树中根本不存在的元素上。 ::first-line 文本中的第一行 ::first-letter 文本中的第一个字母 ::after 元素之后添加 ::before 元素之前 代码&#xff1a; <!DOCTYPE html> <html> <head><meta charset"utf-8&q…

netty之Netty传输Java对象

前言 Netty在实际应用级开发中&#xff0c;有时候某些特定场景下会需要使用Java对象类型进行传输&#xff0c;但是如果使用Java本身序列化进行传输&#xff0c;那么对性能的损耗比较大。为此我们需要借助protostuff-core的工具包将对象以二进制形式传输并做编码解码处理。与直接…

无线物联网通信与智能家居

无线物联网通信技术与智能家居之间存在着密不可分的关系。无线物联网通信技术作为智能家居系统的核心支撑&#xff0c;为智能家居设备之间的互联互通提供了可能&#xff0c;从而实现了家居生活的智能化、便捷化和舒适化。 一、无线物联网通信技术在智能家居中的应用 1、传感器…

LabVIEW程序员从幼稚到成熟的标志是什么?

LabVIEW程序员从“幼稚”到“成熟”的转变标志主要体现在以下几个方面&#xff1a; 1. 系统性思维与架构设计 初学者通常会注重功能实现&#xff0c;常常直接编写功能模块&#xff0c;而忽略系统整体的架构设计。成熟的LabVIEW程序员则具备系统性思维&#xff0c;在开始编写代…

Ericsson EPSFB 通话掉话现象优化案例

Ericsson EPSFB 通话掉话现象优化案例 在移动通信网络中&#xff0c;用户体验的稳定性和通话质量至关重要。随着通信语言通话技术的发展&#xff0c;部分用户在通话时会遇到掉话现象&#xff0c;尤其是在采用EPSFB&#xff08;EvolvedPacket>System Fallback&#xff09;技术…

【WRF工具】cmip6-to-wrfinterm工具概述:生成WRF中间文件

cmip6-to-wrfinterm工具概述 cmip6-to-wrfinterm工具安装cmip6-to-wrfinterm工具使用快速启动&#xff08;Quick start&#xff09;情景1&#xff1a;MPI-ESM-1-2-HR&#xff08;默认&#xff09;&#xff1a;情景2&#xff1a;BCMM情景3&#xff1a;EC-Earth3 更改使用&#x…

SpringBoot框架下美容院管理系统的设计与实现

第二章 开发环境 对美容院管理系统进行开发&#xff0c;需要了解开发技术的理论与实际运用&#xff0c;对开发工具&#xff0c;尤其是数据库的使用方法需要进行掌握。 2.1 SpringBoot框架 SpringBoot框架的诞生是出于需要优化Spring框架的前提下&#xff0c;Spring框架随着时间…

【Simulink仿真】电池储能系统(BESS)与太阳能电站的SPS微电网模型

摘要 本文介绍了一个基于电池储能系统&#xff08;BESS&#xff09;与太阳能电站的微电网仿真模型。该模型使用SPS&#xff08;特殊保护系统&#xff09;进行模拟&#xff0c;能够在跟随电网和形成电网的两种模式下运行。微电网通过太阳能电站为主要能源&#xff0c;并结合了1…

ROS2功能包、节点、可执行程序是怎么来的、之间的关系是什么?详解!

在一个简单的ros2工作空间中&#xff0c;一般有一下几个文件夹&#xff0c; 1、src 2、install 3、build 4、log 其中&#xff0c;存放功能包项目的文件夹在src中&#xff0c;一般情况下&#xff0c;当我们要创建一个功能包项目的时候&#xff0c;要在src文件夹下输入以下…

安卓真机调试“no target device found“以及“ INSTALL_FAILED_USER_RESTRICTED“两个问题的解决办法

目录 1 no target device found问题解决办法 2 “INSTALL_FAILED_USER_RESTRICTED”解决办法 使用android studio 2023.2.1.23windows版本。手机为小米K70 Pro 1 no target device found问题解决办法 参考小米手机如何开启usb调试功能&#xff1f; (baidu.com) 1 联接手机…