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

开发游戏,特别是mmo手游的时候经常需要开发的一个需求是,点击某个装备,在它附近的位置生成一个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;


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值