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

接上一篇:

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.mfbz.cn/a/435115.html

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

相关文章

postman登录鉴权之接口测试

一.背景 在做接口测试的时候,有些接口向后台请求数据的时候,是需要用户在登录情况下才有数据返回。 以电商平台为例,用户的个人中心,用户的订单列表,用户的支付信息等等,所有用户维度的数据都是需要登录态…

网络原理TCP_IP

文章目录 应用层自定义协议 传输层udp协议TCP协议1.确认应答2.超时重传3.连接管理建立连接, 三次握手断开连接, 四次挥手tcp的状态 4.滑动窗口5.流量控制6.拥塞控制7.延时应答8.携带应答9.面向字节流10.异常情况 网络层IP协议地址管理路由选择 数据链路层以太网 应用层 自定义…

okHttp MediaType MIME格式详解

一、介绍 我们在做数据上传时,经常会用到Okhttp的开源库,okhttp开源库也遵循html提交的MIME数据格式。 所以我们经常会看到applicaiton/json这样的格式在传。 但是如果涉及到其他文件等就需要详细的数据格式,否则服务端无法解析 二、okHt…

【信息系统项目管理师】--【信息技术发展】--【新一代信息技术及应用】--【虚拟现实】

文章目录 第二章 信息技术发展2.2 新一代信息技术及应用2.2.6 虚拟现实1.技术基础2.关键技术3.应用和发展 第二章 信息技术发展 信息技术是在信息科学的基本原理和方法下,获取信息、处理信息、传输信息和使用信息的应用技术总称。从信息技术的发展过程来看&#xff…

3.5 力扣 交错字符串

97. 交错字符串 给定三个字符串 s1、s2、s3,请你帮忙验证 s3 是否是由 s1 和 s2 交错 组成的。 两个字符串 s 和 t 交错 的定义与过程如下,其中每个字符串都会被分割成若干 非空 子字符串: s s1 s2 ... snt t1 t2 ... tm|n - m| &…

PDF处理控件aspose.PDF功能演示:将 PDF 转换为 Word 文档

在 Web 应用程序中处理文档时,将 PDF 文件无缝转换为 Word 文档的能力是一项宝贵的资产。此任务不仅常见,而且对于文档转换器和编辑器、从编辑和协作到内容提取的各种应用程序来说也是必不可少的。在这篇博文中,我们将探讨如何使用 JavaScrip…

【vue3之组合式API】

组合式API 一、setup1.写法2.如何访问3.语法糖4.同步返回对象 二、reactive()和ref()1.reactive()2.ref() 三、computed四、watch函数1侦听单个数据2.侦听多个数据3. immediate4. deep5.精确侦听对象的某个属性 五、生命周期函数六、组件通信1.父传子2. 子传父 七、模版引用1. …

数字创新的风口:创业者如何在Web3时代抢占先机

随着区块链技术的不断发展,Web3正成为数字创新的新风口,为创业者们带来了前所未有的机遇和挑战。本文将从另一个角度探讨Web3对创业者的影响,并提出创业者在Web3时代抢占先机的策略和方法。 1. Web3重新定义了商业模式 Web3不仅仅是一种技术…

Java设计模式:建造者模式之经典与流式的三种实现(四)

本文将深入探讨Java中建造者模式的两种实现方式:经典建造者与流式建造者。建造者模式是一种创建型设计模式,它允许你构建复杂对象的步骤分解,使得对象的创建过程更加清晰和灵活。我们将通过示例代码详细解释这两种实现方式,并分析…

重塑Android通信新格局:探秘Android 8.0之后的Binder架构革新

重塑Android通信新格局:探秘Android 8.0 之后的Binder架构革新 1. 引言 Android作为全球主流移动操作系统,在移动设备领域扮演着举足轻重的角色。其开放性、灵活性和广泛的应用生态系统使得无数用户和开发者受益。作为一个基于Linux内核的操作系统,Android的核心架构设计至…

Spring Webflux 详解

目录 0、组件对比 1、WebFlux 1、引入 2、Reactor Core 1、HttpHandler、HttpServer 3、DispatcherHandler 1、请求处理流程 4、注解开发 1、目标方法传参 2.返回值写法 5、文件上传 6、错误处理 7、RequestContext 8、自定义Flux配置 9、Filter WebFlux&am…

Elasticsearch模拟网络丢包

背景 Elasticsearch一旦遇到网络抖动就可能节点(单个或者多个)掉出集群。从而集群出现red/yellow状态,理论情况下ES会自愈,但某些情况下可能非预期,此时就需要我们模拟各种case了,比如网络丢包。 操作 1…

postman Unable to load data as you are offline解决办法 重新登陆无效

postman Unable to load data as you are offline解决办法 重新登陆无效 也能如下一直打不开 - 重新登陆试过了 没有效果 - 刚刚代理切换到了全局,软件内的开关开启了 修改后

鸿蒙NEXT开发实战:【网络管理-数据请求】

概述 本示例仿postman输入API接口地址,获取相应数据,介绍数据请求接口的用法。 样例展示 基础信息 Http 介绍 本示例通过[ohos.net.http]等接口,实现了根据URL地址和相关配置项发起http请求的功能。 效果预览 首页结果页 使用说明 1.…

java上传本地文件到服务器共享

在Windows系统中,将本地文件夹中的某个文件上传到另一台Windows服务器电脑上,前提:两台电脑网络互通,要接收文件的Windows服务器文件夹开启了共享,可以被本机用如下方式进行写入和读取: 如何配置服务器共享请自行百度查找。 所需要的maven依赖如下: <dependency>…

备战蓝桥杯————二分查找(二)

引言 在上一篇博客中&#xff0c;我们深入探讨了二分搜索算法及其在寻找数组左侧边界的应用。二分搜索作为一种高效的查找方法&#xff0c;其核心思想在于通过不断缩小搜索范围来定位目标值。在本文中&#xff0c;我们将继续这一主题&#xff0c;不仅会回顾二分搜索的基本原理&…

leetcode 热题 100_最小覆盖子串

题解一&#xff1a; 双指针滑动窗口&#xff1a;暴力解法——用双指针来表示字符串s中的子串首尾&#xff0c;遍历所有子串并与字符串t判断是否符合条件。我们可以对遍历和判断的过程进行优化&#xff0c;首先是遍历&#xff0c;右指针先移动直到涵盖所以需要的字母&#xff0c…

弹性地基梁matlab有限元编程 | 双排桩支护结构 | Matlab源码 | 理论文本

专栏导读 作者简介&#xff1a;工学博士&#xff0c;高级工程师&#xff0c;专注于工业软件算法研究本文已收录于专栏&#xff1a;《有限元编程从入门到精通》本专栏旨在提供 1.以案例的形式讲解各类有限元问题的程序实现&#xff0c;并提供所有案例完整源码&#xff1b;2.单元…

记一次systemd服务启动找不到Java命令

首先systemd服务文件 /etc/systemd/system/test.service(文件简化处理了) [Unit] Descriptiontest Afternetwork.target [Service] ExecStart/opt/test/bin/test_start.sh [Install] WantedBymulti-user.target其中启动命令ExecStart指向的是一个sh启动脚本&#xff0c; 脚本内…

Zabbix(三)

监控Nginx服务 nginx配置 增加location{} [rootwenzi ~]#vim /etc/nginx/sites-enabled/defaultserver_name _; #_是通配符。服务器将响应任何域名的请求 ...location /status { stub_status;} ...访问 http://IP/status 即可 zabbix配置 Nginx by HTTP&#xff1a;无…
最新文章