记录一下控制相机围绕物体旋转与移动的脚本,相机操作思路分为两块,一部分为旋转,一部分为移动,旋转是根据当前center中心点的坐标,根据距离设置与默认的旋转进行位置移动,移动是根据相机的左右和前后进行计算,当获取到移动的值时把中心点进行移动,而不是移动相机,相机是根据中心点的坐标进行计算自己的坐标
这个中心点是一个空物体放在模型下方,不要把模型放进去
默认的角度设置需要提前说一下,例如当前我相机在场景的旋转为(39,135,0),
如果我想在运行的时候也是这个角度 就需要吧相机x轴填入到脚本的y轴上,需要取反为负数
相机的y轴放到脚本的x上,但是脚本的y需要自身的旋转减去180 设置就为(-45,-39,0)
这样做的理由就是把360的角度映射为正负1800
剩下的就是是否开启自旋转与是否可以低于地面
在这里插入图片描述在这里插入图片描述
在这里插入图片描述

using QFramework;
using System.Collections;
using System.Collections.Generic;
using TMPro;
using UnityEngine;public class Camera_模型相机 : MonoBehaviour
{// public Transform target; // 目标物体public GameObject center;//围绕点public float rotateSpeed = 5.0f; // 旋转速度public float scrollSpeed = 5.0f; // 滚轮速度public float minDistance = 20f; // 最小距离public float maxDistance = 150; // 最大距离public float initialDistance = 20.0f; // 初始距离public Vector2 initialAngles = new Vector2(0f, 30f); // 初始角度(水平,垂直)public float moveSpeed;//移动速度float move_X,move_Y;private float currentDistance;private float horizontalAngle;private float verticalAngle;public bool enableAutoRotate = false; // 是否启用自动旋转public float autoRotateSpeed = 10f; // 自动旋转速度bool bl_旋转控制;Vector3 initialPos;float time_自选择控制=5;public bool allowBelowGround = true; //是否允许低于地面public float groundHeight = 0f; // 地面高度(可自定义)void Start(){currentDistance = initialDistance;horizontalAngle = initialAngles.x;verticalAngle = initialAngles.y;initialPos = center.transform.position;UpdateCameraPosition();}void Update(){HandleMouseInput();HandleScrollInput();if (enableAutoRotate){if (!bl_旋转控制){time_自选择控制 += Time.deltaTime;if (time_自选择控制 >= 5) { bl_旋转控制 = true; time_自选择控制 = 0; }}AutoRotateCenter();}UpdateCameraPosition();}void AutoRotateCenter(){if (bl_旋转控制) horizontalAngle -= autoRotateSpeed * Time.deltaTime;}void HandleMouseInput(){if (Input.GetMouseButton(1)){bl_旋转控制 = false;time_自选择控制 = 0;// 获取鼠标移动量horizontalAngle += Input.GetAxis("Mouse X") * rotateSpeed;verticalAngle += Input.GetAxis("Mouse Y") * rotateSpeed;verticalAngle = Mathf.Clamp(verticalAngle, -89f, 89f); // 限制垂直角度}   // 新增右键平移逻辑// 鼠标左键移动if (Input.GetMouseButton(0)){bl_旋转控制 = false;time_自选择控制 = 0;move_X = moveSpeed * Input.GetAxis("Mouse X");move_Y= moveSpeed* Input.GetAxis("Mouse Y");Vector3 right = transform.right * move_X;Vector3 forward = transform.forward * move_Y;forward.y = 0; // 保持水平移动// print((right + forward));center.transform.Translate(-(right + forward), Space.World);}}void HandleScrollInput(){float scroll = Input.GetAxis("Mouse ScrollWheel");currentDistance -= scroll * scrollSpeed;currentDistance = Mathf.Clamp(currentDistance, minDistance, maxDistance);}public void ResetCamera(){// 重置到初始状态currentDistance = initialDistance;horizontalAngle = initialAngles.x;verticalAngle = initialAngles.y;center.Position(initialPos); bl_旋转控制 = false;}void UpdateCameraPosition(){// 计算旋转Quaternion yaw = Quaternion.AngleAxis(horizontalAngle, Vector3.up);Quaternion pitch = Quaternion.AngleAxis(verticalAngle, Vector3.right);Quaternion combinedRotation = yaw * pitch;// 计算相机位置Vector3 offset = combinedRotation * Vector3.forward * currentDistance;Vector3 targetPos = center.transform.position + offset;// 新增地面高度限制if (!allowBelowGround && targetPos.y < groundHeight){targetPos.y = groundHeight+0.2f; 重新计算距离保持原有水平距离//float horizontalDist = Mathf.Sqrt(//    Mathf.Pow(targetPos.x - center.transform.position.x, 2) +//    Mathf.Pow(targetPos.z - center.transform.position.z, 2)//);//currentDistance = horizontalDist / Mathf.Cos(Mathf.Asin(//    (groundHeight - center.transform.position.y) / currentDistance//));}transform.position = targetPos;// 确保相机看向目标transform.LookAt(center.transform.position);}
}

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

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

相关文章

python打卡day38@浙大疏锦行

知识点回顾&#xff1a; Dataset类的__getitem__和__len__方法&#xff08;本质是python的特殊方法&#xff09;Dataloader类minist手写数据集的了解 作业&#xff1a;了解下cifar数据集&#xff0c;尝试获取其中一张图片 一、首先加载CIFAR数据集 import torch import torchvi…

用户配置文件(Profile)

2.4.5 用户配置文件&#xff08;Profile&#xff09; 用户配置文件由以下组件构成&#xff1a; 一个运营商安全域&#xff08;MNO-SD&#xff09; 辅助安全域&#xff08;SSD&#xff09;和CASD Applets 应用程序&#xff08;如NFC应用&#xff09; 网络接入应用&#xff…

如何给自研MCP加上安全验证

前言 刚过去两个月,市面的 MCP 服务,如雨后春笋一般不断涌现出来,包括;百度、高德、网盘、支付宝。这些 MCP 服务,可以让我们基于 Spring AI 框架构建的 Agent 具备非常丰富的使用功能。同时这也说明,程序员👨🏻‍💻,应该具备开发 MCP 服务的能力,Spring AI 让 J…

Unity网络开发实践项目

摘要&#xff1a;该网络通信系统基于Unity实现&#xff0c;包含以下几个核心模块&#xff1a; 协议配置&#xff1a;通过XML定义枚举&#xff08;如玩家/英雄类型&#xff09;、数据结构&#xff08;如PlayerData&#xff09;及消息协议&#xff08;如PlayerMsg&#xff09;&a…

OpenCV CUDA模块图像过滤------创建一个 Sobel 滤波器函数createSobelFilter()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 该函数用于创建一个 Sobel 滤波器&#xff0c;用于在 GPU 上进行边缘检测。它基于图像的梯度计算&#xff1a; dx 表示对 x 方向求导的阶数&…

【JavaSE】枚举和注解学习笔记

枚举和注解 -枚举 规定多选一数据类型的解决方案-枚举 枚举对应英文(enumeration,简写 enum) 2)枚举是一组常量的集合。 3)可以这里理解:枚举属于一种特殊的类&#xff0c;里面只包含一组有限的特定的对象。 枚举的两种实现方式 自定义实现枚举 使用enum关键字实现枚举 自…

Spark SQL进阶:解锁大数据处理的新姿势

目录 一、Spark SQL&#xff0c;为何进阶&#xff1f; 二、进阶特性深剖析 2.1 窗口函数&#xff1a;数据洞察的新视角 2.2 高级聚合&#xff1a;挖掘数据深度价值 2.3 自定义函数&#xff08;UDF 和 UDTF&#xff09;&#xff1a;拓展功能边界 三、性能优化实战 3.1 数…

如何利用 Conda 安装 Pytorch 教程 ?

如何利用 Conda 安装 Pytorch 教程 &#xff1f; 总共分为六步走&#xff1a; &#xff08;1&#xff09;第一步&#xff1a;验证conda 环境是否安装好&#xff1f; 1) conda -V2) conda --version&#xff08;2&#xff09;第二步&#xff1a;查看现有环境 conda env list…

什么是HTTP

HTTP&#xff08;HyperText Transfer Protocol&#xff09;是万维网数据通信的基础协议&#xff0c;作为应用层协议具有以下关键特性&#xff1a; 客户端-服务器模型&#xff1a;基于请求/响应模式 无状态协议&#xff1a;默认不保留通信状态 可扩展性&#xff1a;通过首部字…

2025-05-27 学习记录--Python-模块

合抱之木&#xff0c;生于毫末&#xff1b;九层之台&#xff0c;起于累土&#xff1b;千里之行&#xff0c;始于足下。&#x1f4aa;&#x1f3fb; 一、模块 ⭐️ &#xff08;一&#xff09;模块的导入与使用 &#x1f36d; 模块的导入&#xff1a;&#x1f41d; 模块 就好比…

leetcode 131. Palindrome Partitioning

目录 一、题目描述 二、方法1、回溯法每次暴力判断回文子串 三、方法2、动态规划回溯法 一、题目描述 分割回文子串 131. Palindrome Partitioning 二、方法1、回溯法每次暴力判断回文子串 class Solution {vector<vector<string>> res;vector<string>…

重构开发范式!飞算JavaAI革新Spring Cloud分布式系统开发

分布式系统凭借高可用性、可扩展性等核心优势&#xff0c;成为大型软件项目的标配架构。Spring Cloud作为Java生态最主流的分布式开发框架&#xff0c;虽被广泛应用于微服务架构搭建&#xff0c;但其传统开发模式却面临效率瓶颈——从服务注册中心配置到网关路由规则编写&#…

python 生成复杂表格,自动分页等功能

py&#xff54;&#xff48;&#xff4f;&#xff4e; 生成复杂表格&#xff0c;自动分页等功能 解决将Python中的树形目录数据转换为Word表格&#xff0c;并生成带有合并单元格的检测报告的问题。首先&#xff0c;要解决“tree目录数据”和“Word表格互换”&#xff0c;指将树…

根据Cortex-M3(包括STM32F1)权威指南讲解MCU内存架构与如何查看编译器生成的地址具体位置

首先我们先查看官方对于Cortex-M3预定义的存储器映射 1.存储器映射 1.1 Cortex-M3架构的存储器结构 内部私有外设总线&#xff1a;即AHB总线&#xff0c;包括NVIC中断&#xff0c;ITM硬件调试&#xff0c;FPB, DWT。 外部私有外设总线&#xff1a;即APB总线&#xff0c;用于…

Linux中硬件信息查询利器——lshw命令详解!

lshw&#xff08;List Hardware&#xff09;是 Linux 系统下的一款命令行工具&#xff0c;用于全面检测并显示详细的硬件信息。它能够报告 CPU、内存、主板、存储设备、显卡、网络设备等几乎所有硬件组件的详细信息&#xff0c;适用于系统管理、故障排查和硬件兼容性检查等场景…

用llama3微调了一个WiFiGPT 用于室内定位

一段话总结 本文提出WiFiGPT,一种基于Decoder-Only Transformer(如LLaMA 3)的室内定位系统,通过将WiFi遥测数据(如CSI、FTM、RSSI)转换为文本序列进行端到端训练,无需手工特征工程即可实现高精度定位。实验表明,WiFiGPT在LOS环境中实现亚米级精度(MAE低至0.90米),在…

大模型系列22-MCP

大模型系列22-MCP 玩转 MCP 协议&#xff1a;用 Cline DeepSeek 接入天气服务什么是 MCP&#xff1f;环境准备&#xff1a;VScode Cline DeepSeek**配置 DeepSeek 模型&#xff1a;****配置 MCP 工具****uvx是什么&#xff1f;****安装 uv&#xff08;会自动有 uvx 命令&…

Go语言Map的底层原理

概念 map 又称字典&#xff0c;是一种常用的数据结构&#xff0c;核心特征包含下述三点&#xff1a; &#xff08;1&#xff09;存储基于 key-value 对映射的模式&#xff1b; &#xff08;2&#xff09;基于 key 维度实现存储数据的去重&#xff1b; &#xff08;3&#x…

循环神经网络(RNN):原理、架构与实战

循环神经网络&#xff08;Recurrent Neural Network, RNN&#xff09;是一类专门处理序列数据的神经网络&#xff0c;如时间序列、自然语言、音频等。与前馈神经网络不同&#xff0c;RNN 引入了循环结构&#xff0c;能够捕捉序列中的时序信息&#xff0c;使模型在不同时间步之间…

java 项目登录请求业务解耦模块全面

登录是统一的闸机&#xff1b; 密码存在数据库中&#xff0c;用的是密文&#xff0c;后端加密&#xff0c;和数据库中做对比 1、UserController public class UserController{Autowiredprivate IuserService userservicepublic JsonResult login(Validated RequestBody UserLo…