LayaBox:工具类HitArea(绘制可视区域)

2020-10-13  本文已影响0人  一眼就认出你

代码:

package laya.utils {
    import laya.display.Graphics;
    import laya.maths.Point;
    import laya.maths.Rectangle;
    import laya.renders.Render;
    import laya.renders.RenderContext;
    
    /**
     * 鼠标点击区域,可以设置绘制一系列矢量图作为点击区域和非点击区域(目前只支持圆形,矩形,多边形)
     */
    public class HitArea {
        
        /**@private */
        private static var _cmds:Array = [];
        /**@private */
        private static var _rec:Rectangle = new Rectangle();
        /**@private */
        private static var _ptPoint:Point = new Point();
        /**@private */
        private var _hit:Graphics;
        /**@private */
        private var _unHit:Graphics;
        
        /**
         * 是否包含某个点
         * @param x x坐标
         * @param y y坐标
         * @return 是否点击到
         */
        public function isHit(x:Number, y:Number):Boolean {
            if (!isHitGraphic(x, y, hit)) return false;
            return !isHitGraphic(x, y, unHit);
        }
        
        /**
         * 检测对象是否包含指定的点。
         * @param   x   点的 X 轴坐标值(水平位置)。
         * @param   y   点的 Y 轴坐标值(垂直位置)。
         * @return  如果包含指定的点,则值为 true;否则为 false。
         */
        public function contains(x:Number, y:Number):Boolean {
            return isHit(x, y);
        }
        
        /**
         * @private
         * 是否击中Graphic
         */
        public static function isHitGraphic(x:Number, y:Number, graphic:Graphics):Boolean {
            if (!graphic) return false;
            var cmds:Array;
            cmds = graphic.cmds;
            if (!cmds && graphic._one) {
                cmds = _cmds;
                cmds.length = 1;
                cmds[0] = graphic._one;
                
            }
            if (!cmds) return false;
            var i:int, len:int;
            len = cmds.length;
            var cmd:Array;
            for (i = 0; i < len; i++) {
                cmd = cmds[i];
                if (!cmd) continue;
                var context:RenderContext = Render._context;
                switch (cmd.callee) {
                
                case context._translate: 
                case 6://translate
                    x -= cmd[0];
                    y -= cmd[1];
                default:
                    
                }
                if (isHitCmd(x, y, cmd)) return true;
            }
            return false;
        }
        
        /**
         * @private
         * 是否击中绘图指令
         */
        public static function isHitCmd(x:Number, y:Number, cmd:Array):Boolean {
            if (!cmd) return false;
            var context:RenderContext = Render._context;
            var rst:Boolean = false;
            switch (cmd["callee"]) {
            
            case context._drawRect: 
            case 13://case context._drawRect:
                _rec.setTo(cmd[0], cmd[1], cmd[2], cmd[3]);
                rst = _rec.contains(x, y);
                break;
            case context._drawCircle: 
            case context._fillCircle: 
            case 14://case context._drawCircle
                var d:Number;
                x -= cmd[0];
                y -= cmd[1];
                d = x * x + y * y;
                rst = d < cmd[2] * cmd[2];
                break;
            
            case context._drawPoly: 
            case 18://drawpoly
                x -= cmd[0];
                y -= cmd[1];
                rst = ptInPolygon(x, y, cmd[2]);
                break;
            default: 
                break;
            }
            return rst;
        }
        
        /**
         * @private
         * 坐标是否在多边形内
         */
        public static function ptInPolygon(x:Number, y:Number, areaPoints:Array):Boolean {
            var p:Point;
            p = _ptPoint;
            p.setTo(x, y);
            // 交点个数
            var nCross:int = 0;
            var p1x:Number, p1y:Number, p2x:Number, p2y:Number;
            var len:int;
            len = areaPoints.length;
            for (var i:int = 0; i < len; i += 2) {
                p1x = areaPoints[i];
                p1y = areaPoints[i + 1];
                p2x = areaPoints[(i + 2) % len];
                p2y = areaPoints[(i + 3) % len];
                //var p1:Point = areaPoints[i];
                //var p2:Point = areaPoints[(i + 1) % areaPoints.length]; // 最后一个点与第一个点连线
                if (p1y == p2y)
                    continue;
                if (p.y < Math.min(p1y, p2y))
                    continue;
                if (p.y >= Math.max(p1y, p2y))
                    continue;
                // 求交点的x坐标
                var tx:Number = (p.y - p1y) * (p2x - p1x) / (p2y - p1y) + p1x;
                // 只统计p1p2与p向右射线的交点
                if (tx > p.x) {
                    nCross++;
                }
            }
            // 交点为偶数,点在多边形之外
            return (nCross % 2 == 1);
        }
        
        /**
         * 可点击区域,可以设置绘制一系列矢量图作为点击区域(目前只支持圆形,矩形,多边形)
         */
        public function get hit():Graphics {
            if (!_hit) _hit = new Graphics();
            return _hit;
        }
        
        public function set hit(value:Graphics):void {
            _hit = value;
        }
        
        /**
         * 不可点击区域,可以设置绘制一系列矢量图作为非点击区域(目前只支持圆形,矩形,多边形)
         */
        public function get unHit():Graphics {
            if (!_unHit) _unHit = new Graphics();
            return _unHit;
        }
        
        public function set unHit(value:Graphics):void {
            _unHit = value;
        }
    }
}

上一篇 下一篇

猜你喜欢

热点阅读