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

【Unity】Unity Transform缩放控制教程:实现3D模型缩放交互,支持按钮/鼠标/手势操作


【Unity 】Transform缩放控制教程:实现3D模型缩放交互,支持按钮/鼠标/手势操作

在Unity开发中,Transform组件承担着场景中物体的空间信息控制,包括位置、旋转和缩放。而缩放(Scale)操作,作为三维交互中最常用的一种变换形式,不仅在模型调整、UI动画、场景编辑中被广泛使用,在许多实际项目中更是交互控制的核心。今天我们就来实现一个简单而实用的缩放控制脚本,并结合多个应用场景探讨它的使用方式与拓展思路。


一、为什么需要自定义缩放控制?

Unity虽然提供了对Transform的基本控制,但在实际开发中,我们常常需要自定义缩放行为,比如:

  • UI按钮控制3D模型的缩放
  • 鼠标滚轮或手势手势控制物体缩放
  • 限制缩放的最大/最小值,防止模型变得太大或太小
  • 在编辑器中交互式缩放对象,用于地图编辑器、安全区域编辑等功能

如果你正在开发一款需要用户交互的3D编辑工具、模拟器、教育培训产品或者VR/AR应用,那么“可控的缩放行为”将是你无法绕过的一个功能。


二、实现核心:ScaleController 脚本

我们从一个最基础的脚本出发,它提供两个公共方法,分别用于放大和缩小当前对象的 Transform。

using UnityEngine;/// <summary>
/// 控制对象缩放的通用脚本
/// </summary>
public class ScaleController : MonoBehaviour
{// 缩放因子,默认为1.1,即每次放大10%public float scaleFactor = 1.1f;// 最小缩放值public float minScale = 0.1f;// 最大缩放值public float maxScale = 10f;/// <summary>/// 放大当前对象/// </summary>public void ScaleUp(){Vector3 newScale = transform.localScale * scaleFactor;if (newScale.x <= maxScale && newScale.y <= maxScale && newScale.z <= maxScale){transform.localScale = newScale;}}/// <summary>/// 缩小当前对象/// </summary>public void ScaleDown(){Vector3 newScale = transform.localScale / scaleFactor;if (newScale.x >= minScale && newScale.y >= minScale && newScale.z >= minScale){transform.localScale = newScale;}}
}

这个脚本提供了:

  • 简洁的放大/缩小方法
  • 缩放限制,防止极端数值
  • Inspector 面板可调参数

三、应用场景详解

场景一:UI 控制 3D 模型大小

在产品展示类应用中,用户常常需要点击按钮对模型进行缩放。我们可以将 ScaleController 挂载到模型上,并通过 UI 按钮绑定 ScaleUp()ScaleDown() 方法:

public class ScaleUIHandler : MonoBehaviour
{public ScaleController controller;public void OnClickScaleUp(){controller.ScaleUp();}public void OnClickScaleDown(){controller.ScaleDown();}
}

这样就可以通过按钮交互控制模型大小,适用于家具展示、角色查看等场景。


场景二:鼠标滚轮控制缩放

在场景查看器或编辑工具中,常见交互是使用鼠标滚轮进行缩放。我们可以在 Update 方法中监听滚轮输入:

void Update()
{float scroll = Input.GetAxis("Mouse ScrollWheel");if (scroll > 0f){ScaleUp();}else if (scroll < 0f){ScaleDown();}
}

这个功能适合:

  • 3D场景查看器
  • 场景地图编辑器
  • 自由浏览VR空间的桌面版本

场景三:手势控制(适配VR/AR)

在 VR 或 AR 中,用户更希望使用手势进行缩放控制,比如双指开合(pinch gesture)。以 Unity 的 AR Foundation 或 XR Toolkit 为例,可以通过监听 PinchGesture 实现缩放,进而调用我们封装好的 ScaleController 方法。

void OnPinch(float pinchDelta)
{if (pinchDelta > 0)scaleController.ScaleUp();elsescaleController.ScaleDown();
}

这种交互在以下场景中非常常见:

  • AR 中的家具放置和缩放
  • VR 设计工具中的精细调整
  • 虚拟展馆中的展品查看

场景四:编辑器工具中的对象缩放

你可以将该组件结合 Unity 的自定义 Editor 工具,制作一个可交互调整对象缩放的 Scene 工具:

#if UNITY_EDITOR
using UnityEditor;
[CustomEditor(typeof(ScaleController))]
public class ScaleControllerEditor : Editor
{public override void OnInspectorGUI(){DrawDefaultInspector();ScaleController sc = (ScaleController)target;GUILayout.Space(10);if (GUILayout.Button("放大")){sc.ScaleUp();}if (GUILayout.Button("缩小")){sc.ScaleDown();}}
}
#endif

只需选中带有该脚本的 GameObject,即可在 Inspector 中直接控制缩放。


四、进阶拓展

1. 非等比缩放

目前的实现是等比缩放,但有些时候我们可能希望只在某一个轴上缩放,例如:

transform.localScale += new Vector3(0.1f, 0, 0); // 仅放大X轴

这种方式适合:

  • 面积、长度模拟的可视化工具
  • 在某些动画过程中拉伸物体,如技能特效

2. 增加缩放动画效果

为了增强用户体验,可以将缩放改为渐变效果,例如使用 LerpDOTween

void ScaleTo(Vector3 targetScale)
{StartCoroutine(ScaleSmoothly(targetScale));
}IEnumerator ScaleSmoothly(Vector3 targetScale)
{float t = 0;Vector3 start = transform.localScale;while (t < 1){transform.localScale = Vector3.Lerp(start, targetScale, t);t += Time.deltaTime * 5f;yield return null;}transform.localScale = targetScale;
}

这种方式可以带来更柔和、真实的体验。


五、实际开发建议

控制范围必须设置

缩放无限放大/缩小会导致模型渲染异常或穿模,建议始终设置最大/最小限制。

统一缩放方式

如果一个项目中多个模型使用缩放逻辑,建议统一封装为工具类或抽象接口,便于统一管理与测试。

缩放中心需注意

默认的缩放中心是模型的原点(Pivot),如果缩放出现偏移问题,可以调整模型导入时的中心点或在父物体上处理。

缩放带来的碰撞问题

缩放会改变物体的碰撞盒大小,尤其是 MeshCollider,注意需要在缩放后重新 RecalculateBounds() 或刷新碰撞体。


六、结语

本文从一个简单的Transform缩放脚本出发,介绍了其基础实现、参数控制以及多个实际应用场景。


专栏《VR 360°全景视频开发》,持续更新中,敬请关注!

【专栏预告】《VR 360°全景视频开发:从GoPro到Unity VR眼镜应用实战》

《VR 360°全景视频开发》将带你深入探索从GoPro拍摄到Unity眼镜端应用开发的全流程技术。专栏内容涵盖安卓原生VR播放器开发、Unity VR视频渲染与手势交互、360°全景视频制作与优化,以及高分辨率视频性能优化等实战技巧。敬请关注每周更新的技术分享!


相关文章:

【Unity】Unity Transform缩放控制教程:实现3D模型缩放交互,支持按钮/鼠标/手势操作

【Unity 】Transform缩放控制教程&#xff1a;实现3D模型缩放交互&#xff0c;支持按钮/鼠标/手势操作 在Unity开发中&#xff0c;Transform组件承担着场景中物体的空间信息控制&#xff0c;包括位置、旋转和缩放。而缩放&#xff08;Scale&#xff09;操作&#xff0c;作为三…...

【Linux篇】缓冲区的工作原理:如何影响你程序的输入输出速度

从内存到磁盘&#xff1a;缓冲区如何提升文件I/O效率 一. 缓冲区1.1 什么是缓冲区1.2 为什么要引入缓冲区1.3 缓冲区类型1.4 FILE1.4.1 基本概念1.4.2 FILE 结构体的作用1.4.3 FILE 的工作机制 二. 最后 在程序开发中&#xff0c;缓冲区是一个经常被提及却不容易深入理解的概念…...

kotlin,Android,jetpack compose,日期时间设置

AI生成&#xff0c;调试出来学习&#xff0c;这些小组件会用了&#xff0c;就可以组合一个大点的程序了。 package com.example.mydatetimeimport android.app.AlertDialog import android.os.Bundle import androidx.activity.ComponentActivity import androidx.activity.co…...

ASP.NET图书馆借阅系统(源码+lw+部署文档+讲解),源码可白嫖!

摘要 近些年来&#xff0c;随着科技的飞速发展&#xff0c;互联网的普及逐渐延伸到各行各业中&#xff0c;给人们生活带来了十分的便利&#xff0c;图书馆借阅系统利用计算机网络实现信息化管理&#xff0c;使图书信息、图书借阅、归还的管理发展和服务水平有显著提升。 本文拟…...

LeetCode算法题(Go语言实现)_35

题目 给你一棵根为 root 的二叉树&#xff0c;请你返回二叉树中好节点的数目。 「好节点」X 定义为&#xff1a;从根到该节点 X 所经过的节点中&#xff0c;没有任何节点的值大于 X 的值。 一、代码实现 func goodNodes(root *TreeNode) int {if root nil {return 0}return d…...

vi/vim常用快捷键

那么今天我们继续昨天没有介绍完的vi编辑器,来看看常用的一些快捷键,方便我们对文件的编辑. 1.拷贝当前行yy,拷贝当前行向下的5行5yy,并粘贴(输入p) 2.删除当前行dd,删除当前行向下的5行5d 3.在文件中查找某个单词[命令模式/关键字,回车查找,输入n就是查找下一个] ⭐️&…...

JVM核心机制:类加载×字节码引擎×垃圾回收机制

&#x1f680;前言 “为什么你的Spring应用启动慢&#xff1f;为什么GC总是突然卡顿&#xff1f;答案藏在JVM的核心机制里&#xff01; 本文将用全流程图解字节码案例&#xff0c;带你穿透三大核心机制&#xff1a; 类加载&#xff1a;双亲委派如何防止恶意代码入侵&#xff…...

opencv无法设置禁用RGB转换问题

树莓派连接摄像头,摄像头输出格式为YUYV(YUV422)。 通过执行 v4l2-ctl --list-formats --device/dev/video0 可以看的具体的摄像头的数据格式。 使用opencv获取视频流&#xff0c;通过cap.set(cv2.CAP_PROP_CONVERT_RGB, 0)设置禁用自动转换RGB格式&#xff0c;但是打印输出…...

k8s 1.30.6版本部署(使用canal插件)

#系统环境准备 参考 https://blog.csdn.net/dingzy1/article/details/147062698?spm1001.2014.3001.5501 #配置下载源 curl -fsSL https://mirrors.aliyun.com/kubernetes-new/core/stable/v1.30/deb/Release.key |gpg --dearmor -o /etc/apt/keyrings/kubernetes-apt-keyri…...

GZ036区块链卷一 EtherStore合约漏洞详解

题目 pragma solidity >0.8.3;contract EtherStore {mapping(address > uint) public balances;function deposit() public payable {balances[msg.sender] msg.value;emit Balance(balances[msg.sender]);}function withdraw() public {uint bal balances[msg.sender…...

MCP+Blender创建电力塔

MCP&#xff08;Model Context Protocol&#xff09;与Blender的结合是当前AI与3D建模领域的热门技术&#xff0c;它通过协议化的方式让Claude等AI模型直接控制Blender&#xff0c;实现自动化3D建模。 1. 功能与原理 • 核心能力&#xff1a;用户通过自然语言指令&#xff08;…...

什么是RACI矩阵,应用在什么场景?

一、什么是RACI RACI矩阵是一种用于明确项目或任务中角色与责任的管理工具&#xff0c;通过定义不同人员在任务中的参与程度来避免职责不清的问题。以下是其核心要点&#xff1a; ‌RACI的含义‌ ● ‌R&#xff08;Responsible&#xff09;执行者‌&#xff1a;直接完成任务…...

Selenium自动化:玩转浏览器,搞定动态页面爬取

嘿&#xff0c;各位爬虫爱好者和自动化达人们&#xff01;是不是经常遇到这种情况&#xff1a;信心满满地写好爬虫&#xff0c;requests一把梭&#xff0c;结果抓下来的HTML里&#xff0c;想要的数据空空如也&#xff1f;定睛一看&#xff0c;原来数据是靠JavaScript动态加载出…...

QAI AppBuilder 快速上手(8): 图像修复应用实例2

LaMa-Dilated模型旨在通过扩张卷积技术实现高效的图像擦除和修复。该模型采用先进的卷积神经网络架构&#xff0c;能够处理复杂的图像输入&#xff0c;并填补图像中的缺失部分&#xff0c;使修复后的图像更加自然和逼真。LaMa-Dilated不仅在图像编辑领域表现出色&#xff0c;还…...

`ConstantPositionProperty` 的使用与应用

ConstantPositionProperty 的使用与应用 1. 什么是 ConstantPositionProperty&#xff1f; ConstantPositionProperty 是 Cesium 中用于表示实体位置的属性类。它表示一个实体在三维空间中的位置是固定的&#xff0c;不会随时间变化。与动态位置属性&#xff08;如 SampledPo…...

【计网】作业4

一. 单选题&#xff08;共22题&#xff0c;64分&#xff09; 1. (单选题)主机甲采用停止-等待协议向主机乙发送数据&#xff0c;数据传输速率是4kb/s&#xff0c;单向传播时延为30ms&#xff0c;忽略确认帧的发送时延。当信道利用率等于80%时&#xff0c;数据帧的长度为&#…...

MPDrive:利用基于标记的提示学习提高自动驾驶的空间理解能力

25年4月来自南方科技大学、百度、英国 KCL和琶洲实验室&#xff08;广东 AI 和数字经济实验室&#xff09;的论文“MPDrive: Improving Spatial Understanding with Marker-Based Prompt Learning for Autonomous Driving”。 自动驾驶视觉问答&#xff08;AD-VQA&#xff09;…...

QTSql全解析:从连接到查询的数据库集成指南

概览 与数据库的有效集成是确保数据管理效率和应用性能的关键&#xff0c;Qt框架就提供了强大的QtSql模块&#xff0c;使得开发者能够轻松地进行数据库操作&#xff0c;包括连接、查询执行以及结果处理等 一、引入QtSql模块 首先&#xff0c;需要在项目中引入QtSql模块&…...

FreeRTOS临界区

在FreeRTOS中&#xff0c;临界区通过关闭可管理的中断来保护共享资源&#xff0c;具体关闭的中断层级由configMAX_SYSCALL_INTERRUPT_PRIORITY宏定义决定。以下是关键点解析&#xff1a; 中断优先级分类&#xff1a; 高优先级中断&#xff1a;数值低于configMAX_SYSCALL_INTERR…...

【学习笔记】HTTP和HTTPS的核心区别及工作原理

一、基础概念 HTTP&#xff08;超文本传输协议&#xff09;&#xff1a;明文传输数据&#xff0c;默认端口80&#xff0c;容易被窃听或篡改。 HTTPS&#xff08;HTTP SSL/TLS&#xff09;&#xff1a;通过加密传输数据&#xff0c;默认端口443&#xff0c;保障安全性。 二、…...

Dubbo的简单介绍

Dubbo的简单介绍 Dubbo 是一个高性能的 Java RPC 框架&#xff0c;最初由阿里巴巴开发&#xff0c;用于构建分布式服务。它主要用于提供服务间的通信&#xff0c;支持高效的远程调用和服务治理&#xff0c;常用于大规模分布式系统中。Dubbo 提供了以下几个核心功能&#xff1a…...

7.2 重复推送(每日、每周等)

1. 核心方法 使用 UNCalendarNotificationTrigger 的 dateMatching 参数配置日历组件&#xff08;DateComponents&#xff09;&#xff0c;结合 repeats: true 实现周期性触发。 2. 不同频率的重复推送配置 2.1 每日重复 每天固定时间触发&#xff08;如上午 10:00&#xff…...

【STL】list介绍(附与vector的比较)

文章目录 1.关于list2.使用2.1 list的构造2.2 list 迭代器的使用2.3 list 容量操作2.3.1 size()2.3.2 empty()2.3.3 resize() 2.4 list 元素访问2.4.1 front()2.4.2 back() 2.5 list 修改操作2.5.1 push_front()2.5.2 pop_front()2.5.3 push_back()2.5.4 pop_back()2.5.5 inser…...

Ansible:roles角色

文章目录 Roles角色Ansible Roles目录编排Roles各目录作用创建 roleplaybook调用角色调用角色方法1&#xff1a;调用角色方法2&#xff1a;调用角色方法3&#xff1a; roles 中 tags 使用实战案例 Roles角色 角色是ansible自1.2版本引入的新特性&#xff0c;用于层次性、结构化…...

找不到导入的项目“xxx\QtMsBuild\Qt.props”。请确认 Import 声明“$(QtMsBuild)\Qt.props”中计算结果为

系列文章目录 文章目录 系列文章目录前言一、问题原因 前言 新建的项目visual studio2022 使用Qt vs tools 找不到导入的项目“E:\osgEarth\DigitalSimulationPlatform\DigitalSimulationPlatform\QtMsBuild\Qt.props”。 请确认 Import 声明“$(QtMsBuild)\Qt.props”中计算结…...

Rust 是如何层层防错的

一、Rust 的多层防错机制 &#x1f9f1; 第一层&#xff1a;Rust语言自带的“编译时护盾” —— 错误连运行都跑不起来 错误类型Rust 怎么发现的&#xff1f;工具/机制举个例子✅ 语法缺陷写错了代码格式或语法Rust Analyzer&#xff08;智能补全&#xff09;少写了分号、括号…...

SQL Server 数据库邮件配置失败:SMTP 连接与权限问题

问题现象&#xff1a; 配置数据库邮件时&#xff0c;发送测试邮件失败&#xff0c;提示 “邮件无法发送到 SMTP 服务器&#xff0c;操作超时”&#xff08;错误 14661&#xff09;或 “服务器拒绝发件人地址”&#xff08;错误 15009&#xff09;。 快速诊断 检查数据库邮件配置…...

2025 年福建交安安全员考试:结合本省交通特点备考​

福建地处东南沿海&#xff0c;交通建设具有独特特点&#xff0c;这对交安安全员考试备考意义重大。在桥梁建设方面&#xff0c;由于面临复杂的海洋环境&#xff0c;桥梁的防腐、防台风等安全措施成为重点。考生在学习桥梁施工安全知识时&#xff0c;要特别关注福建本地跨海大桥…...

OpenBMC:BmcWeb 处理http请求5 检查权限

OpenBMC:BmcWeb 处理http请求4 处理路由对象-CSDN博客 在通过url获取了路由对象后,如果该请求是有session的,那么下一步需要检查权限 1.validatePrivilege调用时传入了一个lambda(1)做为回调 validatePrivilege(req, asyncResp, rule,[req, asyncResp, &rule, params =…...

996引擎-源码学习:Cocos2d-Lua 的 class(classname, ...)

996引擎-源码学习:Cocos2d-Lua 的 class(classname, ...) 一、核心方法调用顺序用户调用入口完整调用链二、__create 工厂方法的三种情形情形1:父类为函数(自定义工厂)情形2:父类为Cocos原生类情形3:父类为普通Lua表三、方法职责与内存管理对照表四、正确使用示例示例1…...