three.js如何实现简易3D机房?(四)点击事件+呼吸灯效果

news2025/2/26 2:19:01

接上一篇:

three.js如何实现简易3D机房?(三)显示信息弹框/标签:http://t.csdnimg.cn/5W2wA

目录

八、点击事件

1.实现效果

2.获取相交点

3.呼吸灯效果

4.添加点击事件

5.问题解决


八、点击事件

1.实现效果

2.获取相交点

官方射线拾取的原理:由相机位置为射线起点,鼠标点击位置为射线方向发射射线,所有被射线穿过的所有几何体都会被捕捉到,距离越近捕捉到的几何体越靠前

在threeD/init.js中


// 获取鼠标和射线的相交点
export function getIntersectPoint (event) {
  // 鼠标控制对象
  const mouse = new THREE.Vector2();
  // 初始化射线辅助器
  const raycaster = new THREE.Raycaster();
  // 将鼠标位置归一化为设备坐标。x 和 y 方向的取值范围是 (-1 to +1)
  let rect = renderer.domElement.getBoundingClientRect()
  mouse.x = ((event.clientX - rect.left) / rect.width) * 2 - 1;
  mouse.y = -((event.clientY - rect.top) / rect.height) * 2 + 1;
  //通过鼠标点击的位置(二维坐标)和当前相机的矩阵计算出射线位置
  raycaster.setFromCamera(mouse, camera);
  // 获取与射线相交的对象数组,其中的元素按照距离排序,越近的越靠前
  return raycaster.intersectObjects(scene.children, true);
}
3.呼吸灯效果

在threeD/init.js中

import { EffectComposer } from 'three/addons/postprocessing/EffectComposer.js';
import { RenderPass } from "three/addons/postprocessing/RenderPass"
import { OutlinePass } from "three/addons/postprocessing/OutlinePass"
import { ShaderPass } from "three/addons/postprocessing/ShaderPass"
// SMAA抗锯齿通道
import { SMAAPass } from 'three/addons/postprocessing/SMAAPass.js';
// 颜色修正
import { GammaCorrectionShader } from 'three/addons/shaders/GammaCorrectionShader.js';

export let composer, renderPass, outlinePass

export function addOutlineEffect (selectedObjects, color) {
  composer = new EffectComposer(renderer);
  renderPass = new RenderPass(scene, camera);
  composer.addPass(renderPass);

  outlinePass = new OutlinePass(new THREE.Vector2(width, height), scene, camera);
  outlinePass.selectedObjects = selectedObjects
  outlinePass.edgeStrength = 8; // 发光的强度
  outlinePass.edgeGlow = 1; // 光晕
  outlinePass.usePatternTexture = false // 是否使用父级的材质
  outlinePass.edgeThickness = 8; // 边框的宽度
  outlinePass.downSampleRatio = 1 // 边框弯曲度
  outlinePass.pulsePeriod = 3; // 呼吸灯闪烁的速度
  outlinePass.visibleEdgeColor.set(color); // 呼吸显示的颜色
  outlinePass.hiddenEdgeColor.set(color); // 呼吸消失的颜色
  outlinePass.clear = true
  composer.addPass(outlinePass);

  //获取.setPixelRatio()设置的设备像素比
  const pixelRatio = renderer.getPixelRatio();
  // 抗锯齿
  const smaaPass = new SMAAPass(width * pixelRatio, height * pixelRatio);
  composer.addPass(smaaPass);

  // 模型颜色修正
  const gammaCorrectionShader = new ShaderPass(GammaCorrectionShader);
  composer.addPass(gammaCorrectionShader);
}

4.添加点击事件

业务逻辑:

(1)获取鼠标和射线的交点

(2)判断点击的几何体是否为设备,是则停止当前随机信息展示,清除前一个信息弹框;不是则清除呼吸灯和弹框,并继续开启随机显示正常设备信息的定时任务;

(3)在点击的几何体是设备的情况下,再次判断是否为报警设备,是只添加呼吸灯效果,否添加呼吸灯和信息弹框

在index.vue中

import {
	scene,
	composer,
	outlinePass,
	getDomInfo,
	init,
	createControls,
	initLight,
	createCSS3DRenderer,
	getIntersectPoint,
	addOutlineEffect,
	watchDom,
	renderResize,
	renderLoop,
} from './component/threeD/init.js';

onMounted(async () => {
	init(threeDemoRef.value);
	importModel();
	createControls();
	initLight();
	createCSS3DRenderer(threeDemoRef.value);
	watchDom(threeDemoRef.value);
	renderResize(threeDemoRef.value);
	renderLoop();
});

// 重置(清除呼吸灯和弹框,并继续随机显示正常设备的信息)
const resetRandomDialog = () => {
	if (outlinePass) {
		composer.removePass(outlinePass);
		clearDialog();
		createNormalDialog();
	}
};

window.addEventListener('click', onClick, false);

const onClick = (event: any) => {
	event.preventDefault();
	const intersects = getIntersectPoint(event);
	if (intersects.length) {
		const selectedDevice = intersects[0].object.parent;
		if (selectedDevice.name && selectedDevice.name.includes('AU')) {
			// 1.停止当前随机信息展示
			state.intervalId ? clearInterval(state.intervalId) : '';
			// 2.清除前一个弹框
			clearDialog();
			// 3.报警设备只添加呼吸灯效果,正常设备添加呼吸灯+弹框
			state.selectedDevice = { ...selectedDevice };
			const alarmName = state.alarmInfo.map((item: any) => item.name);
			if (alarmName.includes(selectedDevice.name)) {
				addOutlineEffect([selectedDevice], 0x8b1616);
			} else {
				addOutlineEffect([selectedDevice], 0x21793b);
				model.traverse((obj: any) => {
					if (obj.name.includes('AU') && obj.name == selectedDevice.name) {
						state.normalInfo.forEach((item: any) => {
							if (item.name == selectedDevice.name) {
								state.randomObject = { ...selectedDevice };
								insertDialogHtml(obj, item);
							}
						});
					}
				});
			}
		} else {
			resetRandomDialog();
		}
	} else {
		resetRandomDialog();
	}
};

到这儿差不多功能就已经全部实现了,但是问题来了,因为随机事件还在继续,切换到其他页面的时候会报错,最后就是一些小问题的解决了

5.问题解决
onUnmounted(() => {
	// 停止定时器
	clearInterval(state.intervalId);
	// 从场景中移除模型
	scene.remove(model.value);
	// 释放模型资源
	model.traverse((child: any) => {
		if (child instanceof THREE.Mesh) {
			child.geometry.dispose();
			child.material.dispose();
		}
	});
    // 停止点击事件
	window.removeEventListener('click', onClick);
});

完结,撒花

✿✿ヽ(°▽°)ノ✿                                        ✿✿ヽ(°▽°)ノ✿                                          ✿✿ヽ(°▽°)ノ✿

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

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

相关文章

ChatGPT发不出消息?GPT发不出消息怎么办?

前言 今天发现,很多人的ChatGPT无法发送信息,我就登陆看一下自己的GPT的情况,结果还真的无法发送消息,ChatGPT 无法发送消息,但是能查看历史的对话,不过通过下面的方法解决了。 第一时间先打开官方的网站&a…

Mint_21.3 drawing-area和goocanvas的FB笔记(七)

FreeBASIC gfx 基本 graphics 绘图 8、ScreenControl与屏幕窗口位置设置 FreeBASIC通过自建屏幕窗口摆脱了原来的屏幕模式限制,既然是窗口,在屏幕坐标中就有它的位置。ScreenControl GET_WINDOW_POS x, y 获取窗口左上角的x, y位置;ScreenC…

【REST2SQL】11 基于jwt-go生成token与验证

【REST2SQL】01RDB关系型数据库REST初设计 【REST2SQL】02 GO连接Oracle数据库 【REST2SQL】03 GO读取JSON文件 【REST2SQL】04 REST2SQL第一版Oracle版实现 【REST2SQL】05 GO 操作 达梦 数据库 【REST2SQL】06 GO 跨包接口重构代码 【REST2SQL】07 GO 操作 Mysql 数据库 【RE…

设计模式学习系列 -- 随记

文章目录 前言 一、设计模式是什么? 二、设计模式的历史 三、为什么以及如何学习设计模式? 四、关于模式的争议 一种针对不完善编程语言的蹩脚解决方案 低效的解决方案 不当使用 五、设计模式分类 总结 前言 最近可能工作生活上的稳定慢慢感觉自己丢失…

掌握 Vue3、Vite 和 SCSS 实现一键换肤的魔法步骤

前言 一个网站的换肤效果算是一个比较常见的功能,尤其是在后台管理系统中,我们几乎都能看到他的身影,这里给大家提供一个实现思路。 搭建项目 vitevue3搭建项目这里就不演示了,vite官网里面讲得很清楚。 注:这里使…

浅析开源内存数据库Fastdb

介绍: Fastdb是免费开源内存数据库,其优秀的性能,和简洁的C代码,让我学习使用过程中收益颇多,但是国内中文相关研究的文章相当稀少,外文我查询相当不便。有兴趣的朋友可以通过以下网站访问:Mai…

java-ssm-jsp基于ssm的冰淇淋在线购买网站

java-ssm-jsp基于ssm的冰淇淋在线购买网站 获取源码——》公主号:计算机专业毕设大全

【STM32】HAL库 CubeMX 教程 --- 通用定时器 TIM2 定时

实验目标: 通过CUbeMXHAL,配置TIM2,1s中断一次,闪烁LED。 一、常用型号的TIM时钟频率 1. STM32F103系列: 所有 TIM 的时钟频率都是72MHz;F103C8不带基本定时器,F103RC及以上才带基本定时器。…

react实战——react旅游网

慕课网react实战 搭建项目问题1.按照官网在index.tsx中引入antd出错?2.typescript中如何使用react-router3.react-router3.1 V63.2 V53.3V6实现私有路由 4.函数式组件接收props参数时定义数据接口?5.使用TypeScript开发react项目:6.要使一个组…

探索stable diffusion的奇妙世界--01

目录 1. 理解prompt提示词: 2. Prompt中的技术参数: 3. Prompt中的Negative提示词: 4. Prompt中的特殊元素: 5. Prompt在stable diffusion中的应用: 6. 作品展示: 在AI艺术领域,stable di…

数据结构——线性表顺序表示详解

目录 1.线性表的类型定义 2.基本操作 3.线性表的存储结构 4.补充 1.元素类型说明 2.数组定义​编辑 3.c语言的内存动态分配 4.c的动态存储分配 5.c中的参数传递 引用类型作参数 6.顺序表基本操作的实现 1.线性表的初始化 代码示例: 2.销毁线性表&…

远程连接Linux系统

图形化、命令行 对于操作系统的使用,有2种使用形式: 图形化页面使用操作系统 图形化:使用操作系统提供的图形化页面,以获得图形化反馈的形式去使用操作系统。 以命令的形式使用操作系统 命令行:使用操作系统提供的各…

腾讯云轻量服务器Windows系统使用IIS实现公网直链访问文件

windows方便所以服务器装的windows系统,windows默认不能分享文件直链,只要用IIS建个站点就行了 先弄一台有公网ip的windows系统服务器打开服务器管理器,添加这个 打开IIS右键添加网站 程序池默认,路径选个文件夹作为网站根目录 …

tomcat通过service.bat install方式安装,内存不够了怎么办?

1.通过service.bat安装 安装命令再tomcat的bin目录下,执行命令 .\service.bat install Apache Tomcat 8.5 Tomcat8之后就会在服务里面有个tomcat服务 2. 如何增加tomcat内存呢? 通过service.bat安装肯定再service.bat中配置啊。 再service.bat文件中…

RHCE——一、OpenEuler22.03安装部署及例行性任务

RHCE 一、OpenEuler22.03安装部署及例行性任务 一、网络服务1.准备工作2、RHEL9操作系统的安装部署3、配置并优化OpenEuler22.034、网络配置实验:修改网络配置 二、例行性工作1、 单一执行的例行性任务:at(一次性)at命令详解 2、循…

MQTT连接阿里云物联网上报物模型数据

目录 1. 创建产品(物联网平台 -> 产品 -> 创建产品) 2. 为产品添加设备 3. 添加物模型 4. mqtt.fx连接测试 5. 调试物模型 6. 使用mqtt.fx上报温度数据 1. 创建产品(物联网平台 -> 产品 -> 创建产品) 我这里再新…

企业内部培训考试系统首页自定义版块说明

企业内部培训考试系统首页自定义版块说明,企业内部培训考试系统手机端首页设计太灵活。 1、整站主题色自定义,更换主题色后,重要的文字和按钮颜色都自动使用主题色渲染,相当于一键换皮肤。 2、首页背景图自定义,想换…

【蓝牙协议栈】【经典蓝牙】【BLE蓝牙】蓝牙技术特点

目录 1. 蓝牙技术特点 2. 经典蓝牙与BLE蓝牙对比 2.1 BT/BLE技术区分 2.2 支持的profile不同 2.3 核心架构,不同的controler 3. Bluetooth的系统构成 4. 蓝牙协议规范 4.1 传输协议 4.2 中介协议 4.3 应用协议 5. 蓝牙硬件接口 1. 蓝牙技术特点 简单地说…

论文笔记:Compact Multi-Party Confidential Transactions

https://link.springer.com/chapter/10.1007/978-3-030-65411-5_21 A compact, private, Multi-Party Confidential Transactions (MCT) 紧凑型多方机密交易(Compact MCT):MCT的长度与常规的单一所有者交易一样短;换句话说&…

【STM32+HAL】GY25倾斜度角度模块

一、前言 有关MPU6050模块读取六轴传感器数值的详细内容,详见【STM32HAL】姿态传感器陀螺仪MPU6050模块 二、所用工具 1、芯片:STM32F103C8T6 2、配置软件:CUBEMX 3、编译器:KEIL5 4、产品型号:GY-25 5、使用芯片…