本文系作者原创,转载请注明出处

入门级的笔者想了一上午才搞懂那个欧拉角的Camera旋转..=.=

在调试场景的时候,每次都本能的按下W想前进,但是这是不可能的(呵呵)

于是便心血来潮想顺便添加个KeyMove事件给摄像机,来实现 伪”漫游“场景 吧。

笔者之前看到过一个叫First Person Controller的Prefab,直接就实现了第一人称的场景漫游(即是不能到处乱飞,只能在一个固定高度进行场景浏览,模拟人行走的意思)

盗张图嘿嘿

但是介于刚刚入门不知道哪儿去找,所以只能自己嗨几个脚本出来自己测....勿喷...

进入正题吧

众所周知 transfrom.Translate平移 里面提供了一个函数

function Translate (x : float, y : float, z : float, relativeTo : Space = Space.Self) : void

Description
描述 Moves the transform by x along the x axis, y along the y axis, and z along the z axis. 移动变换
x沿着x轴,y沿着y轴,z沿着z轴 If relativeTo is left out or set to Space.Self the movement is applied relative to the transform's local axes.
(the x, y and z axes shown when selecting the object inside the Scene View.)
If relativeTo is Space.World the movement is applied relative to the world coordinate system. 如果relativeTo留空或者设置为Space.Self,移动被应用相对于变换的自身轴(当在场景视图选择物体时,x、y和z轴显示)
如果相对于Space.World 移动被应用相对于世界坐标系统

函数的第四个形参relativeTo可以不加,默认为Space.Self

--------------------------------------------

那么什么是Space.World和Space.Self?

官方文档已经说了,Space.World使实体相对于世界坐标轴进行平移变换,Space.Self使实体相对于自己的坐标轴进行平移变换

举个栗子:

当不填写relativeTo(使用Space.Self)的时候,用以下移动一个物体

            if (Input.GetKey(KeyCode.W))
            {
                transform.Translate(, ,  * Time.deltaTime * moveSpeedForDebug);
            }
            if (Input.GetKey(KeyCode.S))
            {
                transform.Translate(, , - * Time.deltaTime * moveSpeedForDebug);
            }
            if (Input.GetKey(KeyCode.A))
            {
                transform.Translate(- * Time.deltaTime * moveSpeedForDebug, , );
            }
            if (Input.GetKey(KeyCode.D))
            {
                transform.Translate( * Time.deltaTime * moveSpeedForDebug, , );
            }

会以物体当前的Z轴正方向作为”前面“,即向着Z轴的实时朝向进行平移变换(前进后退左移右移等)。如果你朝向正前面按下W,就会往前面平移;向天上按下W就会向天上平移

即你可以到达三维空间中任何一个点

但是使用Space.World的时候不一样了

我们假设世界轴Z轴朝向北方

你继续以上的动作你会发现,不管你的视角面向哪儿,你按下W向着世界轴Z轴进行平移变换,你始终是往北边移动的

这个感觉就像你坐在汽车上,你可以踩油门但是不能摸方向盘,于是你踩着油门一直往前开。当前面遇到一个十字路口你想左转,于是你本能地看向左边

但是你只能踩油门,于是虽然你看向左边了,但是汽车还是开向了前方(如果还不懂的话就去自己试一试吧moew)

--------------------------------------------

于是我想到,可以新建一个camera,用GetComponent获取,然后直接用不写relativeTo的Translate函数对相机进行平移变换实现这种二维平面的移动

但是问题来了,就像上面说的,Space.Self平移变换是使用物体当前的Z轴正方向作为”前面“的,也就是说要实现在一定高度的场景预览,摄像机的X轴不能旋转

但是这不现实啊,哪儿有人去看风景的时候一直盯着前面不看天上不看地下的=.=这样的话Space.Self平移变换就不能实现二维平面的移动

用Space.World呢?也不行!因为实体的平移变换不会根据你的视角发生变化,也就是说实体一直往一个方向平移,不会”看哪儿走哪儿“,所以不能实现有指向的二维平面的移动

但是也只有Space.World能够控制实体不在空间的Y轴上乱动了,这个怎么办呢?

如果相机的移动朝向一直有左右没有上下(只有Yaw旋转没有Pitch旋转),但是相机的查看却能有左右和上下(包含Yaw与Pitch),那就可以直接用relativeTo Space.Self参数

进行一个有指向的二维平面移动了!

于是乎,利用父级子级的方法来实现!

也就是说可以将Camera绑在一个Cube上面。

Cube添加只有Yaw旋转的查看以及relativeTo Space.Self平移变换,那么Cube只能在xOz平面内进行平移,而因为相机是Cube的子级,平移和旋转照样适用

也就是说我们需要做的,只是在给相机添加一个只有Pitch旋转的查看就行了!

这样的话,相机能够实现”到处看“的查看,也能实现只在xOz平面内进行的平移

于是首先建个简单的场景吧

然后建立一个Cube实体,并在Cube的子级建立一个Camera,我把这个起导向作用的Cube实体命名为了GuideCube

向导方块2333

接下来做的是把Cube移到地面上来,再把相机放进Cube中间,关闭Cube的渲染

建立一个脚本(我的是MouseLook)在Update()函数里面添加如下片段(变量的定义我就不说了,详细点这里,另一篇文章里面给出了MouseLook的源码)

把脚本添加到Cube上,实现只有Yaw旋转的查看

if (axes == RotationAxes.MouseX)
        {
            //获取鼠标的X移动增量
            //希望摄像机移动Yaw即Y轴
            deltaMouseRotationX += Input.GetAxis("Mouse X") * sensitivityX * Damping;
            //比较X旋转角度与最大、最小旋转角度限制
            deltaMouseRotationX = Mathf.Clamp(deltaMouseRotationX, minimumX, maximumX);
            transform.localEulerAngles = , deltaMouseRotationX, );
            Debug.Log("左右看!");
        }

再将刚刚的代码写入另一个脚本(我的是KeyMove)添加上去

(并没有写完,但是具有基本功能,比如private enum movementLimit与mainCamera = GetComponentInChildren<Camera>();都还用不上)

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

[AddComponentMenu("Camera-Control/Key Move")]
public class KeyMove : MonoBehaviour {

    private Camera mainCamera;

    private enum movementLimit
    {
        noLimit =,
        limitYaxisMovement =
    }
    private movementLimit isMovementLimited = movementLimit.limitYaxisMovement;

    public float moveSpeedForDebug = 0.5f;

    // Use this for initialization
    void Start ()
    {
        mainCamera = GetComponentInChildren<Camera>();
        if (mainCamera == null)
            Debug.LogWarning("Warning:\nError Finging camera!!");
    }

    // Update is called once per frame
    void Update ()
    {

        if (isMovementLimited == movementLimit.limitYaxisMovement)
        {
            if (Input.GetKey(KeyCode.W))
            {
                transform.Translate(, ,  * Time.deltaTime * moveSpeedForDebug);
            }
            if (Input.GetKey(KeyCode.S))
            {
                transform.Translate(, , - * Time.deltaTime * moveSpeedForDebug);
            }
            if (Input.GetKey(KeyCode.A))
            {
                transform.Translate(- * Time.deltaTime * moveSpeedForDebug, , );
            }
            if (Input.GetKey(KeyCode.D))
            {
                transform.Translate( * Time.deltaTime * moveSpeedForDebug, , );
            }
        }
    }
}

之后给相机添加如下,源码还是在MouseLook里面,实现只有Pitch旋转的查看

        if (axes == RotationAxes.MouseY)
        {
            //获取鼠标的Y移动增量
            //希望摄像机移动Pitch即X轴
            deltaMouseRotationY += Input.GetAxis("Mouse Y") * sensitivityY * Damping;
            //比较Y旋转角度与最大、最小旋转角度限制
            deltaMouseRotationY = Mathf.Clamp(deltaMouseRotationY, minimumY, maximumY);
            transform.localEulerAngles = , );
            Debug.Log("上下看!");
        }

这样的话,鼠标的X轴偏移会变成Cube的Yaw旋转,鼠标的Y轴偏移变成Camera的Pitch旋转,又Camera跟着父级GuideCube进行Yaw旋转

那么在我们眼中看起来就只是Camera在作”四处查看“的动作啦,而GuideCube也能实现只在xOz平面上的漫游

。。是不是有点麻烦。。

注,我写的MouseLook源码里面能设置如下

也就是规定添加了此Component的实体能够XY旋转还是只能X还是只能Y旋转,方便了这个文章的实现

[Unity3D]巧妙利用父级子级实现Camera场景平面漫游的更多相关文章

  1. delphi 选中的展开0级 子级不展开

    TreeView1.Selected.Expand(False); //选中的展开0级 子级不展开 TreeView1.Selected.Expand(True); //全部展开 来自为知笔记(Wiz ...

  2. JS获取节点的兄弟,父级,子级元素的方法(js获取子级获取到换行与空格元素-FF)

    先说一下JS的获取方法,其要比JQUERY的方法麻烦很多,后面以JQUERY的方法作对比. JS的方法会比JQUERY麻烦很多,主要则是因为FF浏览器,FF浏览器会把你的换行也当最DOM元素 < ...

  3. mysql oracle sql获取树 父级 子级 及自己

    select * from ( select t.*,d.TABLE_NAME,d.QUERY_SQL,d.data_control_col,d.id table_id,d.where_sql fro ...

  4. 递归方式---通过子级id,获取子级和父级Name

    #region 递归--返回 父级|子级 名称 #region --返回 父级|子级 名称 public string RetrurnTypeNames(string TypeId) { String ...

  5. vue结合Ant Design实现后台系统的权限分配(支持无限子级嵌套)

    最近公司的业务需要,要做一个后台管理系统的管理系统类似于这样子 功能需求如下: 左边是权限菜单,右边对应的是具体权限. 1.父级权限菜单选中,父级权限菜单的权限包括其中所有子级权限菜单的权限也要选中, ...

  6. WPF利用VisualTreeHelper遍历寻找对象的子级对象或者父级对象

    原文:WPF利用VisualTreeHelper遍历寻找对象的子级对象或者父级对象 简介 本文将完整叙述我利用VisualTreeHelper实现题述功能的全部过程,想直接看函数实现的朋友可以跳到函数 ...

  7. ThinkPHP 关联模型中查询某条记录的父级(非查询子级)

    数据表 id      cat_name      cat_pid 76     手机.数码     0 84     手机配件        76 86     蓝牙耳机        84 从属关 ...

  8. 子级Repeater获取 父级Repeater

    第一种方法,子级Repeater中绑定父级的某个字段: <%# DataBinder.Eval((Container.NamingContainer.NamingContainer as Rep ...

  9. <转载>如何解决子级用float浮动父级div高度不能自适应的问题

    转载:http://www.kwstu.com/ArticleView/divcss_2013101582430202 解决子级对象使用css float浮动 而父级div不能自适应高度,不能被父级内 ...

随机推荐

  1. Windows下PowerShell监控Keepalived

    一.背景 某数据库服务器为CentOS,想要监控Keepalived的VIP是否有问题,通过邮件进行报警,但这台机器不能上外网,现在只能在Windows下通过PowerShell来完成发邮件预警. 二 ...

  2. CSS系列目录

    1.  在HTML中引入CSS的方法 2.  CSS选择器 2.1 CSS3新增选择器 3.  CSS的继承与层叠特性 4.  CSS中盒子模型 5.  CSS中盒子之间的关系 6.  CSS中盒子的 ...

  3. Ninesky源代码从Codeplex迁移到开源中国

    原来Ninesky代码一直发在Codeplex.com上,最近两三个星期了代码一直迁入不上去,网站访问也经常出错. 所以把代码放到开源中国去了,项目地址https://git.oschina.net/ ...

  4. ASP.NET MVC5+EF6+EasyUI 后台管理系统(60)-系统总结

    系列目录 前言: 起初写这个框架的时候,可以说在当时来说并不是很流行的设计模式,那是在2012年,面向对象的编程大家都很熟悉, 但是“注入.控制反转(DI,IOC,依赖注入).AOP切面编程”新兴名词 ...

  5. SQL Tuning 基础概述04 - Oracle 表的类型及介绍

    Tables A table describes an entity such as employees. You define a table with a table name, such as ...

  6. ENGLISH抠脚童鞋的福利--GitHub汉化插件

    今天在某前端群看到一个插件,激动万分啊!我就把插件使用实现的步骤分享一下! 打开chrome浏览器输入地址:chrome://extensions/ : 跳转到其他页面,点击左上角--扩展程序: 将T ...

  7. DDD及相关概念

    领域:指一个具体的应用范围,比如电商.订票管理.会议管理等,实现某一领域的功能,与其对应的商业领域一致.譬如Contoso会议管理系统从两个方面来阐述(1)系统概览:销售会议座位.创建新会议[领域的活 ...

  8. [Spring]04_最小化Spring XML配置

    4.1 自动装配 Bean Spring 装配 bean 时,有时非常明确,就是需要将某个 bean 的引用装配给指定属性. 例如,若应用上下文中只有一个 javax.sql.DataSource 类 ...

  9. 五步掌握OOM框架AutoMapper基本使用

    本文版权归博客园和作者吴双本人共同所有,转载和爬虫请注明原文地址 www.cnblogs.com/tdws  写在前面 OOM顾名思义,Object-Object-Mapping实体间相互转换,Aut ...

  10. HTML5简介

    HTML5简介 HTML5是HTML的最新修订标准.2014年10月29日,万维网联盟(W3C)宣布,经过8年的努力,HTML5标准规范制定完成. HTML5的设计目的是在移动设备上使用多媒体. HT ...