React篇之three渲染

需求:拖拽右侧面板,里面的three模型能够自适应

import { useEffect, useState, useRef } from 'react'
import './App.css'
import * as THREE from 'three';
import { GLTFLoader } from 'three/addons/loaders/GLTFLoader.js';
import { debounce } from 'lodash';
const CanvasDemo = () => {let camera, scene, renderer, model, face;const [leftWidth, setLeftWidth] = useState(300);const leftWidthRef = useRef(leftWidth);const containerRef = useRef(null);const rightRef = useRef(null);const isDragging = useRef(false);// 更新 leftWidth 时同步更新 leftWidthRefuseEffect(() => {leftWidthRef.current = leftWidth;}, [leftWidth]);const handleMouseDown = () => {isDragging.current = true;};const handleMouseMove = (e) => {if (!isDragging.current) return;const containerRect = containerRef?.current?.getBoundingClientRect();const newLeftWidth = e.clientX - containerRect.left;setLeftWidth(newLeftWidth);};const handleMouseUp = () => {isDragging.current = false;};const init = () => {camera = new THREE.PerspectiveCamera(45, (window.innerWidth - leftWidthRef.current) / window.innerHeight, 0.25, 100);camera.position.set(- 5, 3, 10);camera.lookAt(0, 2, 0);scene = new THREE.Scene();scene.background = new THREE.Color(0xe0e0e0);scene.fog = new THREE.Fog(0xe0e0e0, 20, 100);const hemiLight = new THREE.HemisphereLight(0xffffff, 0x8d8d8d, 3);hemiLight.position.set(0, 20, 0);scene.add(hemiLight);const dirLight = new THREE.DirectionalLight(0xffffff, 3);dirLight.position.set(0, 20, 10);scene.add(dirLight);const mesh = new THREE.Mesh(new THREE.PlaneGeometry(2000, 2000), new THREE.MeshPhongMaterial({ color: 0xcbcbcb, depthWrite: false }));mesh.rotation.x = - Math.PI / 2;scene.add(mesh);const grid = new THREE.GridHelper(200, 40, 0x000000, 0x000000);grid.material.opacity = 0.2;grid.material.transparent = true;grid.position.set(0, 0, 0); // 将网格放置在场景中心scene.add(grid);const loader = new GLTFLoader();loader.load('https://threejs.org/examples/models/gltf/RobotExpressive/RobotExpressive.glb', function (gltf) {model = gltf.scene;scene.add(model);// 打印模型信息,调试用console.log('Model loaded:', model);}, undefined, function (e) {console.error(e);});renderer = new THREE.WebGLRenderer({ antialias: true });renderer.setPixelRatio(window.devicePixelRatio);renderer.setSize(window.innerWidth - leftWidthRef.current, window.innerHeight);rightRef.current?.appendChild(renderer.domElement);}const debouncedResize = debounce(() => {onWindowResize();}, 10); // 100ms 防抖const onWindowResize = () => {if (!camera || !renderer) return;camera.aspect = (window.innerWidth - leftWidthRef.current) / window.innerHeight;camera.updateProjectionMatrix();renderer.setSize(window.innerWidth - leftWidthRef.current, window.innerHeight);// 确保动画持续运行requestAnimationFrame(animate);}useEffect(() => {init();animate(); // 启动渲染循环if (rightRef.current) {const resizeObserver = new ResizeObserver(() => {if (rightRef.current) {debouncedResize();}});resizeObserver.observe(rightRef.current);return () => resizeObserver.disconnect();}}, [])const animate = () => {requestAnimationFrame(animate);renderer.render(scene, camera);};return (<divref={containerRef}className="container"onMouseMove={handleMouseMove}onMouseUp={handleMouseUp}onMouseLeave={handleMouseUp}><div className="left-pane" style={{ width: leftWidth }}>左侧内容</div><div className="divider" onMouseDown={handleMouseDown}></div><div ref={rightRef} className="right-pane"></div></div>);
}export default CanvasDemo
.container {display: flex;height: 100vh;user-select: none;
}
.left-pane {background-color: #f0f0f0;overflow: auto;
}
.divider {width: 5px;cursor: ew-resize;background-color: #ccc;
}
.right-pane {flex-grow: 1;background-color: #e0e0e0;overflow: auto;
}

问题1:页面宽度变化第一时间都是window.onresize的事件,然而,resize 事件只在 window 对象(即由 document.defaultView 返回)上触发。只有在 window 对象上注册的处理器才能接收 resize 事件。

所以替换方法为:

const resizeObserver = new ResizeObserver(() => {});

            resizeObserver.observe(dom);

            return () => resizeObserver.disconnect();

问题2:拖拽的时候,渲染模型会白屏闪烁,==>解决:加个防抖

 

 const debouncedResize = debounce(() => {

        onWindowResize();

    }, 100); // 100ms 防抖

以上就是解决思路 

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI本地部署

文档加载&#xff08;Document Loading&#xff09;&#xff1a;从多种不同来源加载文档。LangChain提供了100多种不同的文档加载器&#xff0c;包括PDF在内的非结构化的数据、SQL在内的结构化的数据&#xff0c;以及Python、Java之类的代码等​ •文本分割&#xff08;Splitti…

计算机视觉算法实战——驾驶员分心检测(主页有源码)

✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连✨ ​ ​​​ 1. 领域简介&#xff1a;驾驶员分心检测的意义与挑战 驾驶员分心检测是智能驾驶安全领域的重要研究方向。据统计&#xff0c;全球每…

2025因果机器学习好中高区idea汇总

机器学习变天了&#xff01;近来因果机器学习大热&#xff0c;这便意味着机器学习已经从“预测”向“理解”的范式转变。同时&#xff0c;这也是我们发论文、找创新的好时机。 因果机器学习可谓是&#xff0c;提高模型决策科学性和可靠性的“仙丹”&#xff01;其核心就在于&a…

Linux笔记---文件系统硬件部分

1. 文件系统 文件系统是操作系统用于明确存储设备&#xff08;常见的是磁盘&#xff0c;也有基于NAND Flash的固态硬盘&#xff09;或分区上的文件的方法和数据结构&#xff0c;即在存储设备上组织文件的方法。 1.1 基本组成 索引节点&#xff08;inode&#xff09;&#xff…

AutoSar架构-----XCP模块与协议介绍

1、XCP 模块定义 XCP 一般要求如下图&#xff1a; XCP 导入的类型需要如下表这些头文件&#xff1a; 2、ETAS 工具配置 2.1、XcpGeneral 配置 3、XCP 协议 ASAM-MCD-1MC&#xff1a;ECU 和标定测量系统接口 ASAM-MCD-2MC&#xff1a;即 A2L 文件&#xff0c;是控制器内部信息…

江科大51单片机笔记【10】DS1302时钟可调时钟(下)

写在前言 此为博主自学江科大51单片机&#xff08;B站&#xff09;的笔记&#xff0c;方便后续重温知识 在后面的章节中&#xff0c;为了防止篇幅过长和易于查找&#xff0c;我把一个小节分成两部分来发&#xff0c;上章节主要是关于本节课的硬件介绍、电路图、原理图等理论知识…

字节流 InputStream/OutputStream

一、java的I/O java的I/O(输入/输出)流是用于处理数据输入和输出的抽象类。 java的I/O流主要分为两大类&#xff0c;字节流跟字符流。 字节流&#xff1a;用于处理二进制数据&#xff0c;包括InputStream和OutputStream两个主要类及其子类。 字符流&#xff1a;用于处理字符…

使用位运算如何找到数组中只出现一次的数?

题目链接&#xff1a;137. 只出现一次的数字 II - 力扣&#xff08;LeetCode&#xff09; 算法解析 位运算是用于二进制的运算符号。而对于多次出现的数字&#xff0c;其二进制都是一模一样的&#xff0c;这里是3次重复的出现是数字。由此我们可以想到&#xff0c;如果我们由低…

最节省服务器,手搓电子证书查询系统

用户预算150元&#xff0c;想要一个最简单证书查询系统。前台能查询证书、后台管理员能登录能修改密码&#xff0c;证书能够手动输入修改删除、批量导入导出删除数据、查询搜索。能够兼容苹果、安卓、PC三端浏览器&#xff0c;最后帮忙部署到云服务器上。 用户预算不多&#xf…

.net 6程序在IIS中部署后点击IIS设置报错“执行此操作时出错”

.net 6写的程序&#xff0c;需要在Windows服务器的IIS中部署&#xff0c;由于是刚装的系统&#xff0c;先安装.net 6运行时&#xff0c;装了才发现没有IIS&#xff0c;于是又通过“添加角色和功能”添加与IIS相关的功能。安装完毕后&#xff0c;在IIS中添加网站&#xff0c;并将…

探针泄露(WEB)

##解题思路 题目提示是探针泄露&#xff0c;未及时删除的探针可能造成严重的数据泄露 探针的文件常见命名为tz.php&#xff0c;访问它 对于php相关参数&#xff0c;我们是可以点击的&#xff0c;点击phpinfo访问 跳转后搜索flag&#xff0c;得到flag

考研复试c语言常见问答题汇总2

11. 关键字和一般标识符有什么不同&#xff1f; C语言中关键字与一般标识符区别&#xff1a; 定义&#xff1a;关键字是C语言预定义的特殊单词&#xff08;如int、for&#xff09;&#xff0c;有固定含义&#xff1b;标识符是自定义的名称&#xff08;如变量名、函数名&#xf…

贝壳找房:以 OceanBase 为 JuiceFS 元数据引擎,构建 AI 存储底座

本文作者&#xff1a;王天庆&#xff0c;贝壳计算存储方向容器引擎团队负责人&#xff0c;他专注于云原生技术和AI基础设施的架构设计与实践&#xff0c;在为公司搭建高效、可靠的基础设施的同时&#xff0c;促进了大模型技术在企业内部的快速落地与应用。 导语&#xff1a;随着…

人工智能-周志华ML版|系列习题参考答案与综合测试目录

YI时间&#xff5c;松子茶碎碎念&#xff5c;MM-DFW&#xff5c;LAMBDA系列 星标&#x1f31f;松子茶 更新不掉队&#x1f31f; 作者 | 松子茶 © 原创内容(除图片外) 未经作者授权&#xff0c;严禁转载或镜像 机器学习是人工智能领域的核心课程之一。机器学习的基本概念…

OSPF-单区域的配置

一、单区域概念&#xff1a; 单区域OSPF中&#xff0c;整个网络被视为一个区域&#xff0c;区域ID通常为0&#xff08;骨干区域&#xff09;。所有的路由器都在这个区域内交换链路状态信息。 补充知识点&#xff1a; OSPF为何需要loopback接口&#xff1a; 1.Loopback接口的…

基于Bert模型的增量微调3-使用csv文件训练

我们使用weibo评价数据&#xff0c;8分类的csv格式数据集。 一、创建数据集合 使用csv格式的数据作为数据集。 1、创建MydataCSV.py from torch.utils.data import Dataset from datasets import load_datasetclass MyDataset(Dataset):#初始化数据集def __init__(self, s…

C语言 —— 此去经年梦浪荡魂音 - 深入理解指针(卷一)

目录 1. 内存和地址 2. 指针变量和地址 2.1 取地址操作符&#xff08;&&#xff09; 2.2 指针变量 2.3 解引用操作符 &#xff08;*&#xff09; 3. 指针的解引用 3.1 指针 - 整数 3.2 void* 指针 4. const修饰指针 4.1 const修饰变量 4.2 const修饰指针变量 5…

【Linux】线程

文章目录 线程&#xff08;Thread&#xff09;1. 什么是线程&#xff1f; 创建线程多线程中的重入问题线程异常线程等待总结 线程&#xff08;Thread&#xff09; 1. 什么是线程&#xff1f; 线程是进程中的一个执行单元&#xff0c;它是 CPU 调度的基本单位。线程依赖于进程…

SpringBoot第二天

目录 1.Web开发 1.1简介 1.2SpringBoot对静态资源的映射规则 1.3模板引擎 1.3.1引入thymeleaf&#xff1b; 1.3.2Thymeleaf语法 1.3.2.1标准表达式语法 1.变量表达式 1.3.2.2表达式支持的语法 1.3.2.3常用的thymeleaf标签 1.4Springboot整合springmvc 1.4.1Springmvc…

如何接入DeepSeek布局企业AI系统开发技术

在当今科技飞速发展的时代&#xff0c;人工智能&#xff08;AI&#xff09;已成为企业提升竞争力、实现创新突破的关键驱动力。DeepSeek作为一款强大的AI工具&#xff0c;为企业开发自身AI系统提供了有力支持。那么&#xff0c;企业该如何接入DeepSeek进行AI系统开发呢&#xf…