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

【小松教你手游开发】【系统模块开发】根据上一个GameObject坐标生成的tips界面...

开发游戏,特别是mmo手游的时候经常需要开发的一个需求是,点击某个装备,在它附近的位置生成一个tips界面,介绍装备功能和各种信息。
【小松教你手游开发】【系统模块开发】根据上一个GameObject坐标生成的tips界面
像上面红色框框里的这个。

这个主要的问题是 根据点击的GameObject对应生成这个详情界面时,详情界面位置需要合理摆放(不能显示不到,不能遮挡等)

基本的思路是,

首先找到GameObject的position,

把手机屏幕大概分成四个象限,知道这个GameObject大概在这个屏幕的哪个象限(左上,左下,右上,右下)

根据象限来判断详情界面应该在GameObject的哪一边生成

(比如如果GameObject在右下,则详情界面应该在GameObject的左上方生成)

根据上面的position加上GameObject的宽的一半,加上自己界面的宽的一半,再加几个像素进行偏移,

这样就可以摆放好详情界面的位置。

不过这里有个比较重要的问题是,通常情况下,GameObject和自己的详情界面一般不会在同一个camera下。

所以这里首先解决的是要这两边的坐标撸通。

这里简单说一下,

在unity里基本上有4种坐标

世界坐标,相对坐标,屏幕坐标,相对屏幕0,0点坐标

由于我们要在Camera之前切换坐标,为了简化,我们不要去算世界坐标和相对坐标,通过屏幕坐标进行转化

意思就是,

找到GameObject的世界坐标,通过接口算出它自己Camera的屏幕坐标,把屏幕坐标传给详情界面,

详情界面通过把GameObject的屏幕坐标通过自己Camera的转化接口转化成世界坐标,摆放位置。

这样就能计算出在不同camera的情况下,屏幕上同一个点坐标的转化。

下面贴上代码

public void ResetPosition(GameObject relativeObject,Camera cam)  
{  

    //获取触发这个dialog的gameObject相对于屏幕的position  
    Vector3 relativePosition = cam.WorldToScreenPoint(relativeObject.transform.position);//通过世界坐标去获取  
    Vector3 relativePositionPercent = cam.WorldToViewportPoint(relativeObject.transform.position);//获取GameObject相对于屏幕0,0点的位置(x,y范围在0,1之间)  
    Vector3 newPosition = CDialogManager.Instance.GetCamera(eUIIndex.Index_OrthographicThree).ScreenToWorldPoint(relativePosition);//转回世界坐标摆放  
    m_mainGameObject.transform.position = newPosition;//先摆放position,之后再把总偏移值加上  

    Bounds realativeObjectBounds = NGUIMath.CalculateRelativeWidgetBounds(relativeObject.transform);//算出GameObject边框大小  
    float realativeObjectWidth = realativeObjectBounds.extents.x;  
    float realativeObjectHeight = realativeObjectBounds.extents.y;  

    Bounds objectBounds = NGUIMath.CalculateRelativeWidgetBounds(m_mainGameObject.transform);//算出自己边框大小  
    float objectWidth = objectBounds.extents.x;  
    float objectHeight = objectBounds.extents.y;  

    float spacingX = 5;//5像素偏移  
    float spacingY = 5;  

    float offsetX = 0;//总偏移  
    float offsetY = 0;  

    if (relativePositionPercent.x >= 0.5 && relativePositionPercent.y >= 0.7)//为了效果更好,y轴上分为上中下三个象限  
    {  
        offsetX =  -realativeObjectWidth - objectWidth - spacingX;  
        offsetY =  -realativeObjectHeight - objectHeight - spacingY;  

    }  
    else if(relativePositionPercent.x >= 0.5 && relativePositionPercent.y <= 0.3)  
    {  
        offsetX = -realativeObjectWidth - objectWidth - spacingX;  
        offsetY = realativeObjectHeight +objectHeight + spacingY;  
    }  
    else if (relativePositionPercent.x >= 0.5 && relativePositionPercent.y > 0.3 && relativePositionPercent.y < 0.7)  
    {  
        offsetX = -realativeObjectWidth - objectWidth - spacingX;  
        offsetY = 0;  
    }  
    else if (relativePositionPercent.x <= 0.5 && relativePositionPercent.y >= 0.7)  
    {  
        offsetX = realativeObjectWidth + objectWidth + spacingX;  
        offsetY = -realativeObjectHeight - objectHeight - spacingY;  

    }  
    else if (relativePositionPercent.x <= 0.5 && relativePositionPercent.y <= 0.3)  
    {  
        offsetX = realativeObjectWidth + objectWidth + spacingX;  
        offsetY = realativeObjectHeight + objectHeight + spacingY;  
    }  
    else if (relativePositionPercent.x <= 0.5 && relativePositionPercent.y > 0.3 && relativePositionPercent.y < 0.7)  
    {  
        offsetX = realativeObjectWidth + objectWidth + spacingX;  
        offsetY = 0;  
    }  
span style="white-space:pre">    </span>//最后通过相对坐标进行偏移  
    m_mainGameObject.transform.localPosition = new Vector3(m_mainGameObject.transform.localPosition.x + offsetX, m_mainGameObject.transform.localPosition.y + offsetY, 0);  
}  

这里有个问题要注意!!

可能在计算widget宽度的时候GameObject不在原点上。所以优化一下应该是这样:

Bounds realativeObjectBounds = NGUIMath.CalculateRelativeWidgetBounds(relativeObject.transform);//算出GameObject边框大小  
    float realativeObjectWidth = realativeObjectBounds.extents.x - realativeObjectBounds.center.x;  
    float realativeObjectHeight = realativeObjectBounds.extents.y - realativeObjectBounds.center.y;  

    Bounds objectBounds = NGUIMath.CalculateRelativeWidgetBounds(m_mainGameObject.transform);//算出自己边框大小  
    float objectWidth = objectBounds.extents.x - objectBounds.center.x;  
    float objectHeight = objectBounds.extents.y - objectBounds.center.y;

转载于:https://blog.51cto.com/13638120/2084880

相关文章:

  • 观察者模式在One Order回调函数中的应用
  • grep sed awk 练习题
  • #define与typedef区别
  • Linux下命令设置别名--alias(同实用于mac)
  • Eclipse/MyEclipse导入导出注释模板
  • 正则介绍以及grep
  • AI的故事:半人马的诞生之路
  • 共享单车引发秩序问题增多,政府正在研究相关管理办法
  • web后台过程
  • 0314-布局遇到的问题(山东理工大)
  • java多线程处理导入数据拆分List集合 同步处理插入数据
  • 1011. A+B和C (15)
  • Pandora.js 视频介绍
  • display和position的值与用途
  • 吊销***用户
  • Django 博客开发教程 8 - 博客文章详情页
  • ES6系统学习----从Apollo Client看解构赋值
  • Gradle 5.0 正式版发布
  • Invalidate和postInvalidate的区别
  • javascript数组去重/查找/插入/删除
  • Java多线程(4):使用线程池执行定时任务
  • java架构面试锦集:开源框架+并发+数据结构+大企必备面试题
  • JS函数式编程 数组部分风格 ES6版
  • Netty 框架总结「ChannelHandler 及 EventLoop」
  • python3 使用 asyncio 代替线程
  • Vue官网教程学习过程中值得记录的一些事情
  • win10下安装mysql5.7
  • 每天10道Java面试题,跟我走,offer有!
  • 微信小程序--------语音识别(前端自己也能玩)
  • 数据可视化之下发图实践
  • ​MySQL主从复制一致性检测
  • # 透过事物看本质的能力怎么培养?
  • (C语言)编写程序将一个4×4的数组进行顺时针旋转90度后输出。
  • (ZT)薛涌:谈贫说富
  • (编译到47%失败)to be deleted
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (附源码)spring boot建达集团公司平台 毕业设计 141538
  • (附源码)ssm高校实验室 毕业设计 800008
  • (转)memcache、redis缓存
  • .NET Core WebAPI中使用swagger版本控制,添加注释
  • .net core 源码_ASP.NET Core之Identity源码学习
  • .NET Framework 的 bug?try-catch-when 中如果 when 语句抛出异常,程序将彻底崩溃
  • .NET/C# 如何获取当前进程的 CPU 和内存占用?如何获取全局 CPU 和内存占用?
  • .Net下使用 Geb.Video.FFMPEG 操作视频文件
  • :“Failed to access IIS metabase”解决方法
  • @RequestParam详解
  • [20171113]修改表结构删除列相关问题4.txt
  • [28期] lamp兄弟连28期学员手册,请大家务必看一下
  • [AAuto]给百宝箱增加娱乐功能
  • [APUE]进程关系(下)
  • [ArcPy百科]第三节: Geometry信息中的空间参考解析
  • [C/C++]数据结构 深入挖掘环形链表问题
  • [C++提高编程](三):STL初识
  • [CSAWQual 2019]Web_Unagi ---不会编程的崽
  • [delphi]保证程序只运行一个实例