UIView局部透明/打洞,點擊事件穿透

前言

當APP增加一個新功能時介陶,很可能產(chǎn)品會要求添加一個引導圖來引導用戶使用。簡單的引導圖無非添加個按鈕點擊之后引導圖消失衬潦,但是復雜的引導圖就不容易實現(xiàn)了斤蔓,比如我在做直播時遇到的需求:


直播引導圖

如圖所示,要滿足以下的需求:

  • 點擊區(qū)域1镀岛,能彈出本場排行榜弦牡,區(qū)域1的寬度隨豌豆數(shù)變化
  • 點擊區(qū)域2,能彈出對應(yīng)觀眾的個人資料

這個時候漂羊,好的處理方式就是可以在引導圖的區(qū)域1和2處打一個透明的洞驾锰,并在用戶點擊的時候響應(yīng)的是引導圖下面的事件。下面分兩步來處理這個問題走越。

打洞

以下的代碼可以保證BEMaskViewForGuide的實例局部透明

.h

#import <UIKit/UIKit.h>
#import "UIView+STHitTest.h"

@interface BEMaskViewForGuide : UIView

@property (nonatomic, strong) UIColor *maskColor;

- (void)addTransparentRect:(CGRect)rect;

- (void)addTransparentRoundedRect:(CGRect)rect
                     cornerRadius:(CGFloat)cornerRadius;

- (void)addTransparentRoundedRect:(CGRect)rect
                byRoundingCorners:(UIRectCorner)corners
                      cornerRadii:(CGSize)cornerRadii;

- (void)addTransparentOvalRect:(CGRect)rect;

- (void)reset;

@end

.m

#import "BEMaskViewForGuide.h"

@interface BEMaskViewForGuide ()

@property (nonatomic, weak) CAShapeLayer *fillLayer;

@property (nonatomic, strong) UIBezierPath *overlayPath;

@property (nonatomic, strong) NSMutableArray *transparentPaths;

@end

@implementation BEMaskViewForGuide

- (instancetype)initWithFrame:(CGRect)frame {
    self = [super initWithFrame:frame];
    if (self) {
        [self setUp];
    }
    
    return self;
}

- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    self = [super initWithCoder:aDecoder];
    if (self) {
        [self setUp];
    }
    
    return self;
}

- (void)layoutSubviews {
    [super layoutSubviews];
    
    [self refreshMask];
}

#pragma mark - Public Methods

- (void)reset {
    [self.transparentPaths removeAllObjects];
    
    [self refreshMask];
}

- (void)addTransparentPath:(UIBezierPath *)transparentPath {
    [self.overlayPath appendPath:transparentPath];
    
    [self.transparentPaths addObject:transparentPath];
    
    self.fillLayer.path = self.overlayPath.CGPath;
}

- (void)addTransparentRect:(CGRect)rect {
    UIBezierPath *transparentPath = [UIBezierPath bezierPathWithRect:rect];
    
    [self addTransparentPath:transparentPath];
}

- (void)addTransparentRoundedRect:(CGRect)rect cornerRadius:(CGFloat)cornerRadius {
    UIBezierPath *transparentPath = [UIBezierPath bezierPathWithRoundedRect:rect cornerRadius:cornerRadius];
    
    [self addTransparentPath:transparentPath];
}

- (void)addTransparentRoundedRect:(CGRect)rect
                byRoundingCorners:(UIRectCorner)corners
                      cornerRadii:(CGSize)cornerRadii {
    UIBezierPath *transparentPath = [UIBezierPath bezierPathWithRoundedRect:rect byRoundingCorners:corners cornerRadii:cornerRadii];
    
    [self addTransparentPath:transparentPath];
}

- (void)addTransparentOvalRect:(CGRect)rect {
    UIBezierPath *transparentPath = [UIBezierPath bezierPathWithOvalInRect:rect];
    
    [self addTransparentPath:transparentPath];
}

#pragma mark - Private Methods

- (void)setUp {
    self.backgroundColor = [UIColor clearColor];
    self.maskColor = [UIColor blackColor];
    
    self.fillLayer.path = self.overlayPath.CGPath;
    self.fillLayer.fillRule = kCAFillRuleEvenOdd;
    self.fillLayer.fillColor = self.maskColor.CGColor;
}

- (UIBezierPath *)generateOverlayPath {
    UIBezierPath *overlayPath = [UIBezierPath bezierPathWithRect:self.bounds];
    [overlayPath setUsesEvenOddFillRule:YES];
    
    return overlayPath;
}

- (void)refreshMask {
    
    UIBezierPath *path = [self generateOverlayPath];
    for (UIBezierPath *transparentPath in self.transparentPaths) {
        [path appendPath:transparentPath];
    }
    
    self.overlayPath = path;
    
    self.fillLayer.frame = self.bounds;
    self.fillLayer.path = self.overlayPath.CGPath;
    self.fillLayer.fillColor = self.maskColor.CGColor;
}

#pragma mark - Setter and Getter Methods

- (UIBezierPath *)overlayPath {
    if (!_overlayPath) {
        _overlayPath = [self generateOverlayPath];
    }
    
    return _overlayPath;
}

- (CAShapeLayer *)fillLayer {
    if (!_fillLayer) {
        CAShapeLayer *fillLayer = [CAShapeLayer layer];
        fillLayer.frame = self.bounds;
        [self.layer addSublayer:fillLayer];
        
        _fillLayer = fillLayer;
    }
    
    return _fillLayer;
}

- (NSMutableArray *)transparentPaths {
    if (!_transparentPaths) {
        _transparentPaths = [NSMutableArray array];
    }
    
    return _transparentPaths;
}

- (void)setMaskColor:(UIColor *)maskColor {
    _maskColor = maskColor;
    
    [self refreshMask];
}

@end

事件穿透

以下代碼來保證BEMaskViewForGuide的實例局部事件穿透

.h

#import <UIKit/UIKit.h>

/**
 * @abstract hitTestBlock
 *
 * @param 其余參數(shù) 參考UIView hitTest:withEvent:
 * @param returnSuper 是否返回Super的值椭豫。如果*returnSuper=YES,則代表會返回 super hitTest:withEvent:, 否則則按照block的返回值(即使是nil)
 *
 * @discussion 切記,千萬不要在這個block中調(diào)用self hitTest:withPoint,否則則會造成遞歸調(diào)用旨指。這個方法就是hitTest:withEvent的一個代替赏酥。
 */
typedef UIView * (^STHitTestViewBlock)(CGPoint point, UIEvent *event, BOOL *returnSuper);
typedef BOOL (^STPointInsideBlock)(CGPoint point, UIEvent *event, BOOL *returnSuper);

@interface UIView (STHitTest)

/// althought this is strong ,but i deal it with copy
@property(nonatomic, strong) STHitTestViewBlock hitTestBlock;

@property(nonatomic, strong) STPointInsideBlock pointInsideBlock;

@end

.m

#import "UIView+STHitTest.h"

@implementation UIView (STHitTest)

const static NSString *STHitTestViewBlockKey = @"STHitTestViewBlockKey";
const static NSString *STPointInsideBlockKey = @"STPointInsideBlockKey";

+ (void)load {
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(hitTest:withEvent:)),
                                   class_getInstanceMethod(self, @selector(st_hitTest:withEvent:)));
    method_exchangeImplementations(class_getInstanceMethod(self, @selector(pointInside:withEvent:)),
                                   class_getInstanceMethod(self, @selector(st_pointInside:withEvent:)));
}

- (UIView *)st_hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    NSMutableString *spaces = [NSMutableString stringWithCapacity:20];
    UIView *superView = self.superview;
    while (superView) {
        [spaces appendString:@"----"];
        superView = superView.superview;
    }
//    NSLog(@"%@%@:[hitTest:withEvent:]", spaces, NSStringFromClass(self.class));
    UIView *deliveredView = nil;
    // 如果有hitTestBlock的實現(xiàn),則調(diào)用block
    if (self.hitTestBlock) {
        BOOL returnSuper = NO;
        deliveredView = self.hitTestBlock(point, event, &returnSuper);
        if (returnSuper) {
            deliveredView = [self st_hitTest:point withEvent:event];
        }
    } else {
        deliveredView = [self st_hitTest:point withEvent:event];
    }
    //    NSLog(@"%@%@:[hitTest:withEvent:] Result:%@", spaces, NSStringFromClass(self.class), NSStringFromClass(deliveredView.class));
    return deliveredView;
}

- (BOOL)st_pointInside:(CGPoint)point withEvent:(UIEvent *)event {
    NSMutableString *spaces = [NSMutableString stringWithCapacity:20];
    UIView *superView = self.superview;
    while (superView) {
        [spaces appendString:@"----"];
        superView = superView.superview;
    }
//    NSLog(@"%@%@:[pointInside:withEvent:]", spaces, NSStringFromClass(self.class));
    BOOL pointInside = NO;
    if (self.pointInsideBlock) {
        BOOL returnSuper = NO;
        pointInside =  self.pointInsideBlock(point, event, &returnSuper);
        if (returnSuper) {
            pointInside = [self st_pointInside:point withEvent:event];
        }
    } else {
        pointInside = [self st_pointInside:point withEvent:event];
    }
    return pointInside;
}

- (void)setHitTestBlock:(STHitTestViewBlock)hitTestBlock {
    objc_setAssociatedObject(self, (__bridge const void *)(STHitTestViewBlockKey), hitTestBlock, OBJC_ASSOCIATION_COPY);
}

- (STHitTestViewBlock)hitTestBlock {
    return objc_getAssociatedObject(self, (__bridge const void *)(STHitTestViewBlockKey));
}

- (void)setPointInsideBlock:(STPointInsideBlock)pointInsideBlock {
    objc_setAssociatedObject(self, (__bridge const void *)(STPointInsideBlockKey), pointInsideBlock, OBJC_ASSOCIATION_COPY);
}

- (STPointInsideBlock)pointInsideBlock {
    return objc_getAssociatedObject(self, (__bridge const void *)(STPointInsideBlockKey));
}

@end

實例

如圖所示:


xib文件
  • 父view(1)保證透明
  • 蒙層view(2)谆构,也就是BEMaskViewForGuide的實例裸扶,可以設(shè)置顏色

使用方法

//豌豆,頭像打洞搬素,設(shè)置顏色
[self.guideView.maskView addTransparentRoundedRect:CGRectMake(4, 67, self.beanBG.width + 8, 28) cornerRadius:14];
    [self.guideView.maskView addTransparentRoundedRect:CGRectMake(SCREEN_WIDTH - 32 - self.headerListView.width - 3, 24, self.headerListView.width + 6, 44) cornerRadius:22];
    self.guideView.maskView.maskColor = [UIColor gjw_colorWithHex:0x1E2327 alpha:0.86];
    WEAKSELF;
    self.guideView.hitTestBlock =  ^UIView*(CGPoint point, UIEvent *event, BOOL *returnSuper){
        
        CGFloat x = point.x;
        CGFloat y = point.y;
        
        //豌豆事件傳遞
        if (x > 4 && x < weakSelf.beanBG.width + 8 + 4 && y > 67 && y < 67 + 28) {
            
            [weakSelf.guideView removeFromSuperview];
            weakSelf.guideView = nil;
            *returnSuper = NO;
            return nil;//方便呵晨,直接讓事件傳遞,無需返回具體的view
        }
        
        //頭像事件傳遞
        if (x > (SCREEN_WIDTH - 32 - weakSelf.headerListView.width - 3)
            && x < (SCREEN_WIDTH - 32 - 3 + 6)
            && y > 24
            && y < (24 + 44)) {
            
            [weakSelf.guideView removeFromSuperview];
            weakSelf.guideView = nil;
            *returnSuper = NO;
            return nil;
        }
        
        *returnSuper = YES;
        return weakSelf.guideView;
    };

上一篇:應(yīng)用內(nèi)發(fā)短信/郵件熬尺,messenger摸屠、line、whatsapp分享
下一篇:代碼規(guī)范(Objective-C)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末粱哼,一起剝皮案震驚了整個濱河市季二,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌皂吮,老刑警劉巖戒傻,帶你破解...
    沈念sama閱讀 222,183評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件税手,死亡現(xiàn)場離奇詭異蜂筹,居然都是意外死亡需纳,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,850評論 3 399
  • 文/潘曉璐 我一進店門艺挪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來不翩,“玉大人,你說我怎么就攤上這事麻裳】隍穑” “怎么了?”我有些...
    開封第一講書人閱讀 168,766評論 0 361
  • 文/不壞的土叔 我叫張陵津坑,是天一觀的道長妙蔗。 經(jīng)常有香客問我,道長疆瑰,這世上最難降的妖魔是什么眉反? 我笑而不...
    開封第一講書人閱讀 59,854評論 1 299
  • 正文 為了忘掉前任,我火速辦了婚禮穆役,結(jié)果婚禮上寸五,老公的妹妹穿的比我還像新娘。我一直安慰自己耿币,他們只是感情好梳杏,可當我...
    茶點故事閱讀 68,871評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著淹接,像睡著了一般十性。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上塑悼,一...
    開封第一講書人閱讀 52,457評論 1 311
  • 那天劲适,我揣著相機與錄音,去河邊找鬼拢肆。 笑死减响,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的郭怪。 我是一名探鬼主播支示,決...
    沈念sama閱讀 40,999評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼鄙才!你這毒婦竟也來了颂鸿?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,914評論 0 277
  • 序言:老撾萬榮一對情侶失蹤攒庵,失蹤者是張志新(化名)和其女友劉穎嘴纺,沒想到半個月后败晴,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,465評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡栽渴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,543評論 3 342
  • 正文 我和宋清朗相戀三年尖坤,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闲擦。...
    茶點故事閱讀 40,675評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡慢味,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出墅冷,到底是詐尸還是另有隱情纯路,我是刑警寧澤,帶...
    沈念sama閱讀 36,354評論 5 351
  • 正文 年R本政府宣布寞忿,位于F島的核電站驰唬,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏腔彰。R本人自食惡果不足惜叫编,卻給世界環(huán)境...
    茶點故事閱讀 42,029評論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望萍桌。 院中可真熱鬧宵溅,春花似錦、人聲如沸上炎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,514評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽藕施。三九已至寇损,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間裳食,已是汗流浹背矛市。 一陣腳步聲響...
    開封第一講書人閱讀 33,616評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留诲祸,地道東北人浊吏。 一個月前我還...
    沈念sama閱讀 49,091評論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像救氯,于是被迫代替她去往敵國和親找田。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,685評論 2 360

推薦閱讀更多精彩內(nèi)容