当前位置: 首页 > article >正文

解决 Three.js Raycaster 点击位置与实际交点偏差问题

当使用 Three.js 的 Raycaster 时,如果发现点击位置与显示的碰撞点之间存在较大偏差,这通常是由于坐标系统不匹配或参数设置不正确导致的。以下是系统性的排查和解决方案:

1. 检查鼠标坐标转换

最常见的偏差原因是鼠标坐标到标准化设备坐标(NDC)的转换不正确。

function onMouseClick(event) {// 正确的鼠标坐标转换步骤:// 1. 获取鼠标在canvas中的像素坐标const mouse = {x: (event.clientX / window.innerWidth) * 2 - 1,y: -(event.clientY / window.innerHeight) * 2 + 1};// 2. 对于非全屏canvas的情况:const rect = renderer.domElement.getBoundingClientRect();const correctMouse = {x: ((event.clientX - rect.left) / rect.width) * 2 - 1,y: -((event.clientY - rect.top) / rect.height) * 2 + 1};raycaster.setFromCamera(correctMouse, camera);const intersects = raycaster.intersectObjects(scene.children);// 调试输出console.log("Canvas尺寸:", rect.width, rect.height);console.log("鼠标位置:", event.clientX, event.clientY);console.log("相对位置:", event.clientX - rect.left, event.clientY - rect.top);console.log("NDC坐标:", correctMouse.x, correctMouse.y);
}

2. 验证相机参数

不正确的相机参数会导致投影错误:

// 检查相机参数
console.log("相机位置:", camera.position);
console.log("相机朝向:", camera.getWorldDirection(new THREE.Vector3()));
console.log("相机投影矩阵:", camera.projectionMatrix);// 对于透视相机
if (camera instanceof THREE.PerspectiveCamera) {console.log("FOV:", camera.fov);console.log("宽高比:", camera.aspect);console.log("近平面:", camera.near);console.log("远平面:", camera.far);
}// 对于正交相机
if (camera instanceof THREE.OrthographicCamera) {console.log("正交相机参数:", camera.left, camera.right, camera.top, camera.bottom, camera.near, camera.far);
}

3. 检查场景缩放和单位

// 检查场景和物体的缩放
console.log("场景缩放:", scene.scale);
yourObject.traverse(obj => {console.log(`物体 ${obj.name || obj.uuid} 位置:`, obj.position);console.log(`物体 ${obj.name || obj.uuid} 缩放:`, obj.scale);
});// 确保物体矩阵已更新
scene.updateMatrixWorld(true);

4. 高级调试技巧

 4.1 创建可视化调试场景

function setupDebugScene() {// 创建网格地面参考const gridHelper = new THREE.GridHelper(100, 100);scene.add(gridHelper);// 创建坐标轴参考const axesHelper = new THREE.AxesHelper(5);scene.add(axesHelper);// 创建简单测试物体const testGeometry = new THREE.BoxGeometry(1, 1, 1);const testMaterial = new THREE.MeshBasicMaterial({ color: 0x00ff00, wireframe: true });const testCube = new THREE.Mesh(testGeometry, testMaterial);testCube.position.set(0, 0.5, 0);scene.add(testCube);
}

4.2 使用屏幕空间标记

// 在屏幕空间标记点击位置(2D覆盖)
const clickMarker = document.createElement('div');
clickMarker.style.position = 'absolute';
clickMarker.style.width = '10px';
clickMarker.style.height = '10px';
clickMarker.style.backgroundColor = 'red';
clickMarker.style.borderRadius = '50%';
clickMarker.style.transform = 'translate(-50%, -50%)';
document.body.appendChild(clickMarker);function updateClickMarker(event) {clickMarker.style.left = `${event.clientX}px`;clickMarker.style.top = `${event.clientY}px`;
}

5. 常见问题解决方案

问题1:点击位置整体偏移

原因:Canvas元素有边框、边距或CSS变换
解决

// 确保canvas的CSS样式没有影响
renderer.domElement.style.position = 'absolute';
renderer.domElement.style.top = '0';
renderer.domElement.style.left = '0';
renderer.domElement.style.width = '100%';
renderer.domElement.style.height = '100%';
renderer.domElement.style.outline = 'none';

问题2:只在屏幕部分区域有效

原因:Canvas尺寸与显示尺寸不匹配
解决

function handleResize() {const width = renderer.domElement.clientWidth;const height = renderer.domElement.clientHeight;// 检查是否需要调整if (width !== renderer.domElement.width || height !== renderer.domElement.height) {renderer.setSize(width, height, false);camera.aspect = width / height;camera.updateProjectionMatrix();}
}window.addEventListener('resize', handleResize);

问题3:透视变形导致远处点击不准

原因:透视相机FOV过大或物体距离过远
解决

// 调整相机FOV
camera.fov = 45; // 典型值在30-60之间
camera.updateProjectionMatrix();// 或者使用正交相机
// camera = new THREE.OrthographicCamera(width / -2, width / 2, height / 2, height / -2, 1, 1000);

6. 完整调试流程

  1. 简化场景到最基本的物体

  2. 添加坐标轴和网格辅助

  3. 验证鼠标坐标转换

  4. 检查相机参数和投影矩阵

  5. 逐步添加复杂物体,观察何时出现偏差

  6. 检查所有物体的位置、旋转和缩放

通过以上系统性的调试方法,你应该能够准确定位并解决Raycaster点击位置偏差的问题。

相关文章:

解决 Three.js Raycaster 点击位置与实际交点偏差问题

当使用 Three.js 的 Raycaster 时,如果发现点击位置与显示的碰撞点之间存在较大偏差,这通常是由于坐标系统不匹配或参数设置不正确导致的。以下是系统性的排查和解决方案: 1. 检查鼠标坐标转换 最常见的偏差原因是鼠标坐标到标准化设备坐标…...

25、DeepSeek-R1论文笔记

DeepSeek-R1论文笔记 1、研究背景与核心目标2、核心模型与技术路线3、蒸馏技术与小模型优化4、训练过程简介5、COT思维链(Chain of Thought)6、强化学习算法(GRPO)7、冷启动**1. 冷启动的目的****2. 冷启动的实现步骤****3. 冷启动…...

LeetCode --- 156双周赛

题目列表 3541. 找到频率最高的元音和辅音 3542. 将所有元素变为 0 的最少操作次数 3543. K 条边路径的最大边权和 3544. 子树反转和 一、找到频率最高的元音和辅音 分别统计元音和辅音的出现次数最大值,然后相加即可,代码如下 // C class Solution {…...

模型量化AWQ和GPTQ哪种效果好?

环境: AWQ GPTQ 问题描述: 模型量化AWQ和GPTQ哪种效果好? 解决方案: 关于AWQ(Adaptive Weight Quantization)和GPTQ(Generative Pre-trained Transformer Quantization)这两种量化方法的…...

npm 报错 gyp verb `which` failed Error: not found: python2 解决方案

一、背景 npm 安装依赖报如下错: gyp verb check python checking for Python executable "python2" in the PATH gyp verb which failed Error: not found: python2 一眼看过去都觉得是Python环境问题,其实并不是你python环境问题&#xf…...

初识Linux · IP协议· 下

目录 前言: 内网IP和公网IP 内网IP 公网IP 路由 前言: 前文我们介绍了IP协议的协议头,通过源码等方式我们理解了IP协议中的字段,比如8位协议,比如通过环回问题引出的8位最大生存时间,比如8位协议&…...

5.27本日总结

一、英语 复习list2list29 二、数学 学习14讲部分内容 三、408 学习计组1.2内容 四、总结 高数和计网明天结束当前章节,计网内容学完之后主要学习计组和操作系统 五、明日计划 英语:复习lsit3list28,完成07年第二篇阅读 数学&#…...

JavaScript基础-创建对象的三种方式

在JavaScript中,对象是构建复杂数据结构和实现面向对象编程的核心。掌握如何创建对象对于每个开发者来说都是必不可少的技能。本文将介绍创建JavaScript对象的三种主要方式:对象字面量、构造函数以及类(ES6引入),并探讨…...

JAVA的常见API文档(上)

游戏打包 注意API文档中的方法不需要记忆!! 了解之后如果需要可以查询API文档 对Math的方法总结: 运用刚学的Math方法加快代码的运行效率 可以减少循环次数 找规律: 发现因子有规律: 必定一个大于平方根,…...

JavaScript 中的 for...in 和 for...of 循环详解

在 JavaScript 中,for...in 和 for...of 是两种常用的循环结构,但它们有着不同的用途和行为。很多初学者容易混淆这两者,本文将详细解析它们的区别、适用场景以及注意事项。 目录 for…in 循环 基本用法遍历对象属性注意事项 for…of 循环 …...

AtCoder AT_abc406_c [ABC406C] ~

前言 除了 A 题,唯一一道一遍过的题。 题目大意 我们定义满足以下所有条件的一个长度为 N N N 的序列 A ( A 1 , A 2 , … , A N ) A(A_1,A_2,\dots,A_N) A(A1​,A2​,…,AN​) 为波浪序列: N ≥ 4 N\ge4 N≥4(其实满足后面就必须满足这…...

Spark,连接MySQL数据库,添加数据,读取数据

连接数据库 可以看到shell中我们读取出的数据 在IDEA中打代码如果能输出跟shell中一样的结果即证明连接成功 【出错反思】 像我前面出错的原因就是在打代码时将密码输入错误 添加数据 读取数据就是在上面代码中一起展示了,这里我就不单独说了...

Linux容器技术详解

容器技术基础 什么是容器 容器是一种轻量级的虚拟化技术,它将应用程序及其依赖(库、二进制文件、配置文件等)打包在一个独立的单元中,可以在任何支持容器运行时的环境中一致地运行。 Docker官网:https://www.docker…...

【EDA软件】【联合Modelsim仿真使用方法】

背景 业界EDA工具仿真功能是必备的,例如Vivado自带仿真工具,且无需联合外部仿真工具,例如MoodelSim。 FUXI工具仿真功能需要联合Modelsim,才能实现仿真功能。 方法一:FUXI联合ModelSim 1 添加testbench文件 新建to…...

STM32 __main

STM32开发中__main与用户main()函数的本质区别及工作机制 在STM32开发中,__main和用户定义的main()函数是启动过程中的两个关键节点,分别承担运行时初始化和用户程序入口的职责。以下是它们的核心差异及协作机制: 一、定义与层级差异 ​__ma…...

【离散化 线段树】P3740 [HAOI2014] 贴海报|普及+

本文涉及知识点 C线段树 [HAOI2014] 贴海报 题目描述 Bytetown 城市要进行市长竞选,所有的选民可以畅所欲言地对竞选市长的候选人发表言论。为了统一管理,城市委员会为选民准备了一个张贴海报的 electoral 墙。 张贴规则如下: electoral…...

Python训练营打卡Day28

浙大疏锦行 DAY 28 类的定义和方法 知识点回顾: 1.类的定义 2.pass占位语句 3.类的初始化方法 4.类的普通方法 5.类的继承:属性的继承、方法的继承 作业 题目1:定义圆(Circle)类 要求: 1.包含属性&#x…...

MODBUS RTU通信协议详解与调试指南

一、MODBUS RTU简介 MODBUS RTU(Remote Terminal Unit)是一种基于串行通信(RS-485/RS-232)的工业标准协议,采用二进制数据格式,具有高效、可靠的特点,广泛应用于PLC、传感器、变频器等工业设备…...

CSP 2024 提高级第一轮(CSP-S 2024)单选题解析

单选题解析 第 1 题 在 Linux 系统中,如果你想显示当前工作目录的路径,应该使用哪个命令?(A) A. pwd B. cd C. ls D. echo 解析:Linux 系统中,pwd命令可以显示当前工作目录的路径。pwd&#x…...

六、绘制图片

文章目录 1.创建一个红色图片2.加载bmp图片3.加载png、jpg图片 前面的几个示例,我们已经展示过如果在Linux系统下使用xlib接口向窗口中绘制文本、线、矩形;并设置文本、线条的颜色。并利用xlib提供的接口结合事件处理机制完成了一个自绘按钮控件功能。有…...

Java 面向对象详解和JVM底层内存分析

先关注、点赞再看、人生灿烂!!!(谢谢) 神速熟悉面向对象 表格结构和类结构 我们在现实生活中,思考问题、发现问题、处理问题,往往都会用“表格”作为工具。实际上,“表格思维”就是…...

深度学习---知识蒸馏(Knowledge Distillation, KD)

一、知识蒸馏的本质与起源 定义: 知识蒸馏是一种模型压缩与迁移技术,通过将复杂高性能的教师模型(Teacher Model)所学的“知识”迁移到轻量级的学生模型(Student Model),使学生模型在参数量和计…...

基于CNN卷积神经网络的带频偏QPSK调制信号检测识别算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 2.算法运行软件版本 matlab2024b 3.部分核心程序 (完整版代码包含详细中文注释和操作步骤视频&#xff09…...

【DAY21】 常见的降维算法

内容来自浙大疏锦行python打卡训练营 浙大疏锦行 目录 PCA主成分分析 t-sne降维 线性判别分析 (Linear Discriminant Analysis, LDA) 作业: 什么时候用到降维 降维的主要应用场景 知识点回顾: PCA主成分分析t-sne降维LDA线性判别 通常情况下,…...

PostGIS实现栅格数据入库-raster2pgsql

raster2pgsql使用与最佳实践 一、工具概述 raster2pgsql是PostGIS提供的命令行工具,用于将GDAL支持的栅格格式(如GeoTIFF、JPEG、PNG等)导入PostgreSQL数据库,支持批量加载、分块切片、创建空间索引及金字塔概览,是栅格数据入库的核心工具。 二、核心功能与典型用法 1…...

校园社区小程序源码解析

基于ThinkPHP、FastAdmin和UniApp开发的校园社区小程序源码,旨在为校园内的学生和教职员工提供一个便捷的在线交流和服务平台。 该小程序前端采用UniApp进行开发,具有良好的跨平台兼容性,可以轻松发布到iOS和Android平台。同时,后…...

第6章:文件权限

一、文件权限概述 Linux为了保证系统中每个文件的安全,引入了文件权限机制。针对于系统中的每一个文件Linux都可以提供精确的权限控制。它可以做到不同的用户对同一个文件具有不同的操作权利。而通常这个权利包括以下3个: 读的权利(Read&…...

使用 Python 连接 Oracle 23ai 数据库完整指南

方法一:使用 oracledb 官方驱动(推荐) Oracle 官方维护的 oracledb 驱动(原 cx_Oracle)是最新推荐方案,支持 Thin/Thick 两种模式。 1. 环境准备 pip install oracledb2. 完整示例代码 import oracledb import getpass from typing import Unionclass Oracle23aiConn…...

C语言| 指针变量的定义

C语言| 指针的优点-CSDN博客 * 表示“指向”,为了说明指针变量和它所指向的变量之间的联系。 int * i;//表示指针变量i里面存放的地址,所指向的存储单元里的【数据】。 【指针变量的定义】 C语言规定所有变量,在使用前必须先定…...

HTML 中的 input 标签详解

HTML 中的 input 标签详解 一、基础概念 1. 定义与作用 HTML 中的 <input> 标签是表单元素的核心组件&#xff0c;用于创建各种用户输入字段。作为一个空标签&#xff08;没有闭合标签&#xff09;&#xff0c;它通过 type 属性来决定呈现何种输入控件&#xff0c;是实…...