Unity UGUI Image使用图集透明度点击过滤BUG

将Sprite打入图集时会默认去掉边缘的alpha=0的部分

Unity的Image.IsRaycastLocationValid在判断alphaHitTestMinimumThreshold的时候会使用图集中的点判断alpha而不是原Sprite中的点

所以会导致无法正确获取到alpha,要修复这个BUG可以重写Image类

主要是添加了这段代码 其他都是原装的

csharp 复制代码
 //计算local在图集中的位置
 Rect spriteRect = overrideSprite.textureRect;
 local.x = local.x - overrideSprite.textureRectOffset.x + spriteRect.x;
 local.y = local.y - overrideSprite.textureRectOffset.y + spriteRect.y;
 //判断是否在当前图的范围内
 if (!overrideSprite.textureRect.Contains(local))
 {
     return false;
 }

其实只重写了IsRaycastLocationValid方法,但是要访问private方法 只能拷贝出来了

csharp 复制代码
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class OverrideImage : Image
{
    
    private void _PreserveSpriteAspectRatio(ref Rect rect, Vector2 spriteSize)
    {
        var spriteRatio = spriteSize.x / spriteSize.y;
        var rectRatio = rect.width / rect.height;

        if (spriteRatio > rectRatio)
        {
            var oldHeight = rect.height;
            rect.height = rect.width * (1.0f / spriteRatio);
            rect.y += (oldHeight - rect.height) * rectTransform.pivot.y;
        }
        else
        {
            var oldWidth = rect.width;
            rect.width = rect.height * spriteRatio;
            rect.x += (oldWidth - rect.width) * rectTransform.pivot.x;
        }
    }
    private Vector4 _GetAdjustedBorders(Vector4 border, Rect adjustedRect)
    {
        Rect originalRect = rectTransform.rect;

        for (int axis = 0; axis <= 1; axis++)
        {
            float borderScaleRatio;

            // The adjusted rect (adjusted for pixel correctness)
            // may be slightly larger than the original rect.
            // Adjust the border to match the adjustedRect to avoid
            // small gaps between borders (case 833201).
            if (originalRect.size[axis] != 0)
            {
                borderScaleRatio = adjustedRect.size[axis] / originalRect.size[axis];
                border[axis] *= borderScaleRatio;
                border[axis + 2] *= borderScaleRatio;
            }

            // If the rect is smaller than the combined borders, then there's not room for the borders at their normal size.
            // In order to avoid artefacts with overlapping borders, we scale the borders down to fit.
            float combinedBorders = border[axis] + border[axis + 2];
            if (adjustedRect.size[axis] < combinedBorders && combinedBorders != 0)
            {
                borderScaleRatio = adjustedRect.size[axis] / combinedBorders;
                border[axis] *= borderScaleRatio;
                border[axis + 2] *= borderScaleRatio;
            }
        }
        return border;
    }
    
    private Vector2 _MapCoordinate(Vector2 local, Rect rect)
    {
        Rect spriteRect = overrideSprite.rect;
        if (type == Type.Simple || type == Type.Filled)
            return new Vector2(spriteRect.position.x + local.x * spriteRect.width / rect.width, spriteRect.position.y + local.y * spriteRect.height / rect.height);

        Vector4 border = overrideSprite.border;
        Vector4 adjustedBorder = _GetAdjustedBorders(border / pixelsPerUnit, rect);

        for (int i = 0; i < 2; i++)
        {
            if (local[i] <= adjustedBorder[i])
                continue;

            if (rect.size[i] - local[i] <= adjustedBorder[i + 2])
            {
                local[i] -= (rect.size[i] - spriteRect.size[i]);
                continue;
            }

            if (type == Type.Sliced)
            {
                float lerp = Mathf.InverseLerp(adjustedBorder[i], rect.size[i] - adjustedBorder[i + 2], local[i]);
                local[i] = Mathf.Lerp(border[i], spriteRect.size[i] - border[i + 2], lerp);
            }
            else
            {
                local[i] -= adjustedBorder[i];
                local[i] = Mathf.Repeat(local[i], spriteRect.size[i] - border[i] - border[i + 2]);
                local[i] += border[i];
            }
        }

        return local + spriteRect.position;
    }
    
    public override bool IsRaycastLocationValid(Vector2 screenPoint, Camera eventCamera)
    {
        if (alphaHitTestMinimumThreshold <= 0)
            return true;

        if (alphaHitTestMinimumThreshold > 1)
            return false;

        if (overrideSprite == null)
            return true;

        Vector2 local;
        if (!RectTransformUtility.ScreenPointToLocalPointInRectangle(rectTransform, screenPoint, eventCamera, out local))
            return false;

        Rect rect = GetPixelAdjustedRect();

        if (preserveAspect)
            _PreserveSpriteAspectRatio(ref rect, new Vector2(overrideSprite.texture.width, overrideSprite.texture.height));

        // Convert to have lower left corner as reference point.
        local.x += rectTransform.pivot.x * rect.width;
        local.y += rectTransform.pivot.y * rect.height;

        local = _MapCoordinate(local, rect);
        
        //计算local在图集中的位置
        Rect spriteRect = overrideSprite.textureRect;
        local.x = local.x - overrideSprite.textureRectOffset.x + spriteRect.x;
        local.y = local.y - overrideSprite.textureRectOffset.y + spriteRect.y;
        //判断是否在当前图的范围内
        if (!overrideSprite.textureRect.Contains(local))
        {
            return false;
        }
        
        // Convert local coordinates to texture space.
        float x = local.x / overrideSprite.texture.width;
        float y = local.y / overrideSprite.texture.height;
    
        try
        {
            return overrideSprite.texture.GetPixelBilinear(x, y).a >= alphaHitTestMinimumThreshold;
        }
        catch (UnityException e)
        {
            Debug.LogError("Using alphaHitTestMinimumThreshold greater than 0 on Image whose sprite texture cannot be read. " + e.Message + " Also make sure to disable sprite packing for this sprite.", this);
            return true;
        }
    }
}
相关推荐
lingxiao1688816 小时前
WebApi详解+Unity注入--下篇:Unity注入
unity·c#·wpf
世洋Blog16 小时前
面经-CPU、内存、GPU的性能优化
unity·性能优化
lingxiao1688818 小时前
WebApi详解+Unity注入--中篇:.net core的WebAPI
unity·c#·.netcore
weixin_4239950021 小时前
unity 处理图片:截图,下载,保存
java·unity·游戏引擎
hui函数1 天前
Python系列Bug修复PyCharm控制台pip install报错:如何解决 pip install 网络报错 企业网关拦截 User-Agent 问题
python·pycharm·bug
hui函数1 天前
如何解决 pip install 代理报错 SOCKS5 握手失败 ReadTimeoutError 问题
bug·pip
呆呆敲代码的小Y1 天前
【Unity实战篇】| 游戏轮播图效果,多种实现思路及完整教程
游戏·unity·游戏引擎·实战·游戏开发·轮播图·u3d
f***24111 天前
Bug悬案:程序员破案实录
bug
小南家的青蛙1 天前
O3DE社区发布2510.1版本
游戏引擎·图形引擎
示申○言舌1 天前
Unity高性能参数差异化URP Shader圆角圆环UI进度条
ui·unity·游戏引擎·圆环进度条·参数差异化·材质参数独立·圆角圆环