Unity - 搬砖日志 - 修复 Billboard 闪烁的 BUG

Billboard 简称:BB。可以参考:


BB 的网格使用的是一个 Quad 作为 Mesh 的。

而 该 Mesh.bounds (AABB) 是很小的,如果走实时(RT)的 FrustumCulling 的话,那么肯定会因为 BB 的 Mesh.bounds 太小而闪烁

当初我以为可以在 Unity 离线时处理,代码如下:

    [MenuItem("实用工具/场景工具/将 BB 的 AABB 重置")]
    public static void _Handle()
    {
        int _BBSizePos = Shader.PropertyToID("_BBSizePos");
        var path_prefix = "Assets/xxx/";

        var BBMaxSize = new Vector3();

        for (int i = 0; i < 40; i++)
        {
            var suffix = i == 0 ? "" : i.ToString();
            var path = $"{path_prefix}xxx{suffix}.mat";
            var mat = AssetDatabase.LoadAssetAtPath<Material>(path);
            if (mat == null) continue;
            var size = mat.GetVector(_BBSizePos);
            BBMaxSize.x = Mathf.Max(size.x, BBMaxSize.x);
            BBMaxSize.y = Mathf.Max(size.y, BBMaxSize.y + size.z);
            BBMaxSize.z = Mathf.Max(size.x, BBMaxSize.z); // z 方向的深度使用 x 等宽的数值
        }

        _UpdateAABB(BBMaxSize);
    }

    private static void _UpdateAABB(Vector3 size)
    {
        int _BBSizePos_hash = Shader.PropertyToID("_BBSizePos");

        var curScene = EditorSceneManager.GetActiveScene();
        var goes = curScene.GetRootGameObjects();

        foreach (var go in goes)
        {
            var meshRenderers = go.GetComponentsInChildren<MeshRenderer>(true);

            // Debug.Log($"ResizeCurSceneAllBBMeshBoundsUtil.Run(), meshRenderers.Length = {meshRenderers.Length}");

            var count = meshRenderers.Length;
            for (int i = 0; i < count; i++)
            {
                var renderer = meshRenderers[i];
                if (renderer == null)
                {
                    continue;
                }
                Material sharedMat = renderer.sharedMaterial;
                if (sharedMat == null)
                {
                    continue;
                }

                // Debug.Log($"sharedMat.shader.name:{sharedMat.shader.name}");
                if (sharedMat.shader.name != "XXX")
                {
                    continue;
                }
                var bbSizePosition = sharedMat.GetVector(_BBSizePos_hash);
                var center = new Vector3(0, bbSizePosition.y * 0.5f, 0);
                var bounds = new Bounds(center, size);
                renderer.gameObject.GetComponent<MeshFilter>().sharedMesh.bounds = bounds;
                Debug.Log($"NewBounds : {bounds}");
            }
        }
    }

很快就确定自己想多了,于是换另一种处理方式:在运行时或是所有 BB 当中,所有的 BB的 AABB 的并集来作为每个 BB 的 AABB:

    private void OnBattleLoadingOver()
    {
        int _BBSizePos_hash = Shader.PropertyToID("_BBSizePos");

        Dictionary<Material, Mesh> meshDict_Key_SharedMat = new Dictionary<Material, Mesh>();

        var curScene = SceneManager.GetActiveScene();
        var goes = curScene.GetRootGameObjects();

        foreach (var go in goes)
        {
            var meshRenderers = go.GetComponentsInChildren<MeshRenderer>(true);

            // Debug.Log($"ResizeCurSceneAllBBMeshBoundsUtil.Run(), meshRenderers.Length = {meshRenderers.Length}");

            var count = meshRenderers.Length;
            for (int i = 0; i < count; i++)
            {
                var renderer = meshRenderers[i];
                if (renderer == null)
                {
                    continue;
                }
                Material sharedMat = renderer.sharedMaterial;
                if (sharedMat == null)
                {
                    continue;
                }

                if (meshDict_Key_SharedMat.TryGetValue(sharedMat, out Mesh sharedMesh))
                {
                    renderer.gameObject.GetComponent<MeshFilter>().sharedMesh = sharedMesh;
                    continue;
                }

                //sharedMesh = renderer.gameObject.GetComponent<MeshFilter>().sharedMesh; // new 一个,所以不使用 sharedMesh,改用 mesh的 getter 来new一个
                sharedMesh = renderer.gameObject.GetComponent<MeshFilter>().mesh; // 改用 mesh的 getter 来new一个
                renderer.gameObject.GetComponent<MeshFilter>().sharedMesh = sharedMesh;
                meshDict_Key_SharedMat[sharedMat] = sharedMesh;

                // Debug.Log($"sharedMat.shader.name:{sharedMat.shader.name}");
                if (sharedMat.shader.name == "XXX")
                {
                    var bbSizePosition = sharedMat.GetVector(_BBSizePos_hash);
                    var center = new Vector3(0, bbSizePosition.y * 0.5f, 0);
                    var size = new Vector3(bbSizePosition.x, bbSizePosition.y, bbSizePosition.x);
                    var bounds = new Bounds();
                    bounds.center = center;
                    bounds.size = size;
                    sharedMesh.bounds = bounds;
                    Debug.Log($"NewBounds : {bounds}");
                }
            }
        }
    }

OK,FrustumCulling 的闪烁得以解决

但是随着后续的迭代,我们添加了:Unity 内置的 Occlusion Culling(不支持 Compute Shader 的 HiZ 剔除,我们就使用这种基础 CPU 剔除的方式),所以又出现了 BB 的闪烁

原因:BB LOD3级对象的 StaticFlag 设置了 OccludeeFlag 导致的

解决:

    [MenuItem("实用工具/场景工具/将 BB 的 OccludeeFlag 去除")]
    public static void _DisabledBBOCStaicFlag()
    {
        var curScene = EditorSceneManager.GetActiveScene();
        var goes = curScene.GetRootGameObjects();

        var hashset = new HashSet<string>();

        foreach (var go in goes)
        {
            var meshRenderers = go.GetComponentsInChildren<MeshRenderer>(true);

            // Debug.Log($"ResizeCurSceneAllBBMeshBoundsUtil.Run(), meshRenderers.Length = {meshRenderers.Length}");

            var count = meshRenderers.Length;
            for (int i = 0; i < count; i++)
            {
                var renderer = meshRenderers[i];
                if (renderer == null)
                {
                    continue;
                }
                Material sharedMat = renderer.sharedMaterial;
                if (sharedMat == null)
                {
                    continue;
                }

                // Debug.Log($"sharedMat.shader.name:{sharedMat.shader.name}");
                if (sharedMat.shader.name != "xxx")
                {
                    continue;
                }

                var prefabInst = PrefabUtility.GetOutermostPrefabInstanceRoot(renderer.gameObject);
                if (prefabInst == null)
                {
                    continue;
                }
                var prefabInstPath = PrefabUtility.GetPrefabAssetPathOfNearestInstanceRoot(prefabInst);
                Debug.Log($"{prefabInst}, path : {prefabInstPath}");

                if (hashset.Contains(prefabInstPath))
                {
                    Debug.Log($"{prefabInstPath}, was handled, skip at this time.");
                    continue;
                }

                var srcStaticFlag = GameObjectUtility.GetStaticEditorFlags(renderer.gameObject);

                if (!srcStaticFlag.HasFlag(StaticEditorFlags.OccluderStatic)&&
                    !srcStaticFlag.HasFlag(StaticEditorFlags.OccludeeStatic))
                {
                    Debug.Log($"{prefabInst}, have no occludee or occluder static flag, skip at this time.");
                    continue;
                }

                // 去除两个:Occlude Staic Flag
                srcStaticFlag &= ~StaticEditorFlags.OccludeeStatic;
                srcStaticFlag &= ~StaticEditorFlags.OccluderStatic;

                GameObjectUtility.SetStaticEditorFlags(renderer.gameObject, srcStaticFlag);

                PrefabUtility.ApplyPrefabInstance(prefabInst, InteractionMode.AutomatedAction);

                hashset.Add(prefabInstPath);
            }
        }
    }

上面脚本批量处理完后,发现还是有部分闪烁,发现是因为部分美术同学没有遵守场景编辑原则,场景中的 prefab 要进入 prefab 编辑界面编辑,不要在 Scene 下编辑,否则一定要记得 apply prefab modifies

解决:恢复这些 modifies

    [MenuItem("实用工具/场景工具/将 BB 都 Revert Changes")]
    public static void _RevertBBOCStaicFlag()
    {
        var curScene = EditorSceneManager.GetActiveScene();
        var goes = curScene.GetRootGameObjects();

        foreach (var go in goes)
        {
            var meshRenderers = go.GetComponentsInChildren<MeshRenderer>(true);

            // Debug.Log($"ResizeCurSceneAllBBMeshBoundsUtil.Run(), meshRenderers.Length = {meshRenderers.Length}");

            var count = meshRenderers.Length;
            for (int i = 0; i < count; i++)
            {
                var renderer = meshRenderers[i];
                if (renderer == null)
                {
                    continue;
                }
                Material sharedMat = renderer.sharedMaterial;
                if (sharedMat == null)
                {
                    continue;
                }

                // Debug.Log($"sharedMat.shader.name:{sharedMat.shader.name}");
                if (sharedMat.shader.name != "xxx")
                {
                    continue;
                }

                var prefabInst = PrefabUtility.GetOutermostPrefabInstanceRoot(renderer.gameObject);
                if (prefabInst == null)
                {
                    continue;
                }
                var prefabInstPath = PrefabUtility.GetPrefabAssetPathOfNearestInstanceRoot(prefabInst);
                Debug.Log($"{prefabInst}, path : {prefabInstPath}");

                PrefabUtility.RevertPrefabInstance(prefabInst, InteractionMode.AutomatedAction);
            }
        }
    }
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spine-Unity-3.8-2021是一个Spine动画的Unity插件,可以让开发人员在游戏开发过程中轻松实现Spine动画。这个插件包含了最新的Spine Runtime库,可以让开发人员在使用Spine动画时获得更好的性能和可靠性。此版本还提供了对C# 9的支持,并修复了在以前的Spine Unity版本中存在的一些问题和异常。此外,此版本也提供了一些改进和优化,能够让开发人员更快速、更容易地实现动画效果并提高游戏的运行效率。总的来说,Spine-Unity-3.8-2021是一个值得开发人员关注和使用的插件,它可以帮助游戏开发人员更好地实现动画效果,提升游戏的质量和用户体验。 ### 回答2: spine-unity-3.8-2021是一款在Unity中使用的Spine动画软件包。Spine是一种2D骨骼动画软件,它可以帮助制作2D游戏中的角色动画。与传统的逐帧动画相比,Spine骨骼动画具有更高的效率和更好的表现效果。 spine-unity-3.8-2021是Spine运行在Unity中的版本。它提供了一个简便的方式来将Spine制作的动画集成到Unity游戏中。spine-unity-3.8-2021拥有许多实用的功能,例如动画的播放、循环、暂停和停止等。 此外,spine-unity-3.8-2021还支持动画的混合、遮罩、缩放等高级特性。这些功能可以大大提升2D游戏的动画表现效果。 总之,spine-unity-3.8-2021是一款强大的Spine骨骼动画软件包,它可以帮助Unity开发者更方便、更高效地制作2D游戏中的动画效果。 ### 回答3: Spine-Unity-3.8-2021是Spine动画引擎的一个版本,其主要特点是可以与Unity引擎无缝集成,提供了高效、灵活、可定制的动画解决方案。此版本相比之前版本,主要增加了一些新功能和改进,如支持GPU动画混合、2D环境的自适应、高效的顶点色边框渲染、Spine Atlas纹理集加载、支持Mecanim(动画过渡和状态机和蒙太奇)和 IK姿势、环境光遮蔽、大量优化和 bug 修复等。除此之外,它还易于使用和实现,并具有快速迭代的能力,可以让开发者轻松创建精美的动画效果,提升游戏的用户体验。该版本是Spine引擎的主要升级版本之一,同时也体现出Spine开发团队对于产品需求和用户反馈的重视和努力,不仅提高了动画制作领域的生产力和创造力,也为游戏行业推陈出新提供了有力支持。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值