unity实现使不同尺寸物体总能较完整的显示在相机范围内

unity通常会有将一个模型展示在界面上用于旋转缩放观察的功能,通常是使用相机看向模型,然后使用RawImage显示相机看到的画面。

但很多时候展示的模型尺寸与中心点是差别较大的,就需要自适应的修改模型的位置使其能够完整的显示在屏幕中心位置,下面就是相关代码

首先将物体的子物体进行网格合并

cs 复制代码
/// <summary>
/// 计算模型全部展示位置
/// </summary>
/// <param name="root"></param>
/// <returns></returns>
private Vector3 CalculateAllShowPos(GameObject root)
{
    if (root.transform.childCount > 0)
    {
        //若物体是由一堆子物体拼接而成, 可以先将他们的Mesh合并
        MeshFilter[] meshFilters = root.transform.GetComponentsInChildren<MeshFilter>();
        CombineInstance[] combineInstances = new CombineInstance[meshFilters.Length];
        for (int k = 0; k < meshFilters.Length; k++)
        {
            combineInstances[k].mesh = meshFilters[k].sharedMesh;
            combineInstances[k].transform = meshFilters[k].transform.localToWorldMatrix;
        }
        Mesh newMesh = new Mesh();
        //若合并的Mesh面数太多,需要将indexFormat 改为UInt32,默认为UInt16
        newMesh.indexFormat = UnityEngine.Rendering.IndexFormat.UInt32;
        newMesh.CombineMeshes(combineInstances);
        root.AddComponent<MeshFilter>().sharedMesh = newMesh;

        float xx = root.GetComponent<MeshFilter>().sharedMesh.bounds.size.x;
        float yy = root.GetComponent<MeshFilter>().sharedMesh.bounds.size.y;
        float zz = root.GetComponent<MeshFilter>().sharedMesh.bounds.size.z;

        float sizeF = 1;

        if (xx >= yy)
        {
            sizeF = xx;
        }
        else
        {
            sizeF = yy;
        }

        if (sizeF >= zz)
        {

        }
        else
        {
            sizeF = zz;
        }
        //返回模型需要向后移动的距离
        return new Vector3(0, 0, 1.1f * sizeF);
    }

    return Vector3.zero;
}

然后计算物体的几何中心点

cs 复制代码
/// <summary>
/// 计算模型的几何中心点(局部坐标系下)
/// </summary>
/// <param name="tran"></param>
/// <returns></returns>
private Vector3 CalculateModelCenter(Transform tran)
{
    //计算每个子物体包围盒的中心,平均值得到物体的中心
    Vector3 center = Vector3.zero;
    Renderer[] renders = tran.GetComponentsInChildren<Renderer>();
    foreach (Renderer child in renders)
    {
        center += child.bounds.center;
    }
    center /= renders.Length;
    //返回相对位置
    return center - tran.position;
}

最后就是展示模型时调用上述方法即可

cs 复制代码
/// <summary>
/// 显示三维模型
/// </summary>
public void ShowModel(GameObject model)
{
    if (model == null)
    {
        return;
    }
    
    //ModelPoint为相机看向的点,是一个空物体,将展示的物体设置为其子物体
    model.transform.SetParent(ModelPoint, false);

    Vector3 backPos = CalculateAllShowPos(model);
    Vector3 pos = CalculateModelCenter(model.transform);

    model.transform.localPosition = -pos;
    model.transform.localRotation = Quaternion.identity;
    ModelPoint.position += backPos;
}

需要注意的是创建RenderTexture时需要设置宽高与展示模型的RawImage的宽高一致,否则展示画面会有拉伸感

cs 复制代码
 private void Start()
 {
     RenderTexture render = new RenderTexture((int)rect.rect.width, (int)rect.rect.height, 16, RenderTextureFormat.ARGB32);
    ModelCamera.targetTexture = render;
    ModelImage.texture = render;
 }

最后是简单的鼠标控制相机拖拽,旋转,缩放观察模型,若需要射线监测点击物体则需要注意射线发射的位置为鼠标位置减去RawImage左下角位置,具体代码如下

cs 复制代码
using BehaviorDesigner.Runtime.Tasks.Unity.UnityPhysics;
using DG.Tweening;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
using UnityEngine.EventSystems;
using UnityEngine.InputSystem.HID;

/// <summary>
/// 控制模型移动,旋转,缩放,点击模型
/// </summary>
public class ModelControl : MonoBehaviour,IPointerEnterHandler,IPointerExitHandler
{
    /// <summary>
    /// 模型相机
    /// </summary>
    public Camera ModelCamera;
    /// <summary>
    /// 模型父节点
    /// </summary>
    public Transform ModelPoint;

    public RectTransform ModelRect;

    /// <summary>
    /// 是否鼠标进入操作区域
    /// </summary>
    private bool isMouseEnter = false;

    RaycastHit hit;

    public void OnPointerEnter(PointerEventData eventData)
    {
        isMouseEnter = true;
    }

    public void OnPointerExit(PointerEventData eventData)
    {
        isMouseEnter = false;
    }

    private void Update()
    {
        if (!isMouseEnter)
            return;

        if(Input.GetMouseButton(0))
        {
            float x = Input.GetAxis("Mouse X");
            float y = Input.GetAxis("Mouse Y");

            //相机上下左右移动实现拖拽移动
            ModelCamera.transform.position -= ModelCamera.transform.right * x * 0.2f;
            ModelCamera.transform.position -= ModelCamera.transform.up * y * 0.2f;
        }

        float scrollWheel = Input.GetAxis("Mouse ScrollWheel");

        if(scrollWheel != 0)
        {
            //相机前后移动实现缩放
            ModelCamera.transform.position += ModelCamera.transform.forward * scrollWheel * 10f;
        }

        if(Input.GetMouseButton(2))
        {
            float x = Input.GetAxis("Mouse X");
            float y = Input.GetAxis("Mouse Y");

            float xnew = Mathf.Abs(x);
            float ynew = Mathf.Abs(y);

            if(xnew > ynew)
            {
                ModelCamera.transform.RotateAround(ModelPoint.position, Vector3.up, x * 5f);
            }
            else
                ModelCamera.transform.RotateAround(ModelPoint.position, ModelCamera.transform.right, -y * 5f);
        }
        
        //右键点击
        if(Input.GetMouseButtonDown(1))
        {
            //为保证射线在任何分辨率下无异常,需要进行位置的换算,其中1920与1080为CanvasScaler设定的值,若不为1920*1080需要对应修改
            float x = 1920f / Screen.width;
            float y = 1080f / Screen.height;

            //鼠标位置减去模型Image左下角位置
            Vector3 mousePos = Input.mousePosition;
            mousePos = new Vector3(mousePos.x * x, mousePos.y * y);

            Vector3 rectPos = ModelRect.position;
            rectPos = new Vector3(rectPos.x * x, rectPos.y * y);

            Vector3 leftDownPos = rectPos - new Vector3(ModelRect.rect.width * 0.5f, ModelRect.rect.height * 0.5f);

            Ray ray = ModelCamera.ScreenPointToRay(mousePos - leftDownPos);

            if (Physics.Raycast(ray, out hit, 300)) //如果碰撞检测到物体
            {
                //点击到物体,自行处理
            }
        }
    }
}
相关推荐
向宇it5 小时前
【unity小技巧】unity 什么是反射?反射的作用?反射的使用场景?反射的缺点?常用的反射操作?反射常见示例
开发语言·游戏·unity·c#·游戏引擎
Heaphaestus,RC6 小时前
【Unity3D】获取 GameObject 的完整层级结构
unity·c#
芋芋qwq6 小时前
Unity UI射线检测 道具拖拽
ui·unity·游戏引擎
tealcwu7 小时前
【Unity服务】关于Unity LevelPlay的基本情况
unity·游戏引擎
大眼睛姑娘10 小时前
Unity3d场景童话梦幻卡通Q版城镇建筑植物山石3D模型游戏美术素材
unity·游戏美术
鹿野素材屋14 小时前
Unity Dots下的动画合批工具:GPU ECS Animation Baker
unity·游戏引擎
那就举个栗子!14 小时前
多传感器融合slam过程解析【大白话版】
数码相机
xy1899017 小时前
相机触发模式
数码相机
小春熙子21 小时前
Unity图形学之着色器之间传递参数
unity·游戏引擎·技术美术·着色器
虾球xz1 天前
游戏引擎学习第15天
学习·游戏引擎