XABlurEffect-iOS7直接使用UIVisualEffectView实现毛玻璃效果

黑魔法——“低版本中使用高版本中出现的类”之技术实现原理详解

下载XABlurEffect代码

UIVisualEffectView是在iOS8上实现毛玻璃效果的系统库,在iOS7上无法使用。XABlurEffect可以实现在iOS7上直接使用UIVisualEffectView来实现毛玻璃效果。

使用方法:直接拷贝导入

直接拷贝导入 直接将XABlurEffect拷贝到工程项目中,无需改动代码。

XAVisualEffectView.h

#import <UIKit/UIKit.h>

typedef NS_ENUM(NSInteger, XABlurEffectStyle) {
    XABlurEffectStyleExtraLight,
    XABlurEffectStyleLight,
    XABlurEffectStyleDark,
    XABlurEffectStyleExtraDark,
    XABlurEffectStyleRegular,
    XABlurEffectStyleProminent
};

NS_ASSUME_NONNULL_BEGIN

@interface XAVisualEffect : NSObject <NSCopying, NSSecureCoding> @end

@interface XABlurEffect : XAVisualEffect
+ (XABlurEffect *)effectWithStyle:(XABlurEffectStyle)style;
@end

@interface XAVibrancyEffect : XAVisualEffect
+ (XAVibrancyEffect *)effectForBlurEffect:(XABlurEffect *)blurEffect;
@end

@interface XAVisualEffectView : UIView <NSSecureCoding>
@property (nonatomic, strong, readonly) UIView *contentView;
@property (nonatomic, copy, nullable) XAVisualEffect *effect;
- (instancetype)initWithEffect:(nullable XAVisualEffect *)effect;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder;
@end

NS_ASSUME_NONNULL_END

XAVisualEffectView.m

#import "XAVisualEffectView.h"
#import <objc/runtime.h>

@implementation XAVisualEffect

#pragma mark - NSSecureCoding
+ (BOOL)supportsSecureCoding {
    return YES;
}

- (void)encodeWithCoder:(NSCoder *)aCoder {
    
}

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

#pragma mark - NSCopying
- (id)copyWithZone:(NSZone *)zone {
    return [[self class] allocWithZone:zone];
}

@end

@implementation XABlurEffect

+ (XABlurEffect *)effectWithStyle:(XABlurEffectStyle)style {
    XABlurEffect *effect = [[XABlurEffect alloc] init];
    objc_setAssociatedObject(effect, _cmd, @(style), OBJC_ASSOCIATION_RETAIN);
    return effect;
}

@end

@implementation XAVibrancyEffect

+ (XAVibrancyEffect *)effectForBlurEffect:(XABlurEffect *)blurEffect {
    XAVibrancyEffect *effect = [[XAVibrancyEffect alloc] init];
    objc_setAssociatedObject(effect, _cmd, blurEffect, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    return effect;
}

@end

@implementation XAVisualEffectView
{
    UIToolbar *_toolBar;
}

- (instancetype)initWithEffect:(XAVisualEffect *)effect {
    if (self = [super init]) {
        [self setEffect:effect];
    }
    return self;
}

// VisualEffect
- (void)setEffect:(XAVisualEffect *)effect {
    if ([effect isMemberOfClass:[XABlurEffect class]]) {
        [self setBlurEffect:(XABlurEffect *)effect];
    } else if ([effect isMemberOfClass:[XAVibrancyEffect class]]) {
        [self setVibrancyEffect:(XAVibrancyEffect *)effect];
    }
}

// BlurEffect
- (void)setBlurEffect:(XABlurEffect *)effect {
    XABlurEffectStyle style = [objc_getAssociatedObject(effect, @selector(effectWithStyle:)) integerValue];
    switch (style) {
        case XABlurEffectStyleExtraLight: {
            [self effectWithBarStyle:UIBarStyleDefault];
        }
            break;
        case XABlurEffectStyleLight: {
            [self effectWithBarStyle:UIBarStyleDefault];
        }
            break;
        case XABlurEffectStyleDark: {
            [self effectWithBarStyle:UIBarStyleBlack];
        }
            break;
        case XABlurEffectStyleRegular: {
            [self effectWithBarStyle:UIBarStyleDefault];
        }
            break;
        case XABlurEffectStyleProminent: {
            [self effectWithBarStyle:UIBarStyleDefault];
        }
            break;
        default:
            break;
    }
}

// VibrancyEffect
- (void)setVibrancyEffect:(XAVibrancyEffect *)effect {
    XABlurEffect *blurEffect = objc_getAssociatedObject(effect, @selector(effectForBlurEffect:));
    [self setBlurEffect:blurEffect];
}

#pragma mark - setEffect
- (void)effectWithBarStyle:(UIBarStyle)barStyle {
    UIToolbar *toolBar = [[UIToolbar alloc] init];
    toolBar.barStyle = barStyle;
    [self addSubview:toolBar];
    
    _toolBar = toolBar;
    
    [self setLayoutConstraints];
}

#pragma mark - Layout
- (void)setLayoutConstraints {
    _toolBar.translatesAutoresizingMaskIntoConstraints = NO;
    
    NSLayoutConstraint *topConstraint = [NSLayoutConstraint constraintWithItem:_toolBar attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeTop multiplier:1.0 constant:0];
    NSLayoutConstraint *leftConstraint = [NSLayoutConstraint constraintWithItem:_toolBar attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeLeft multiplier:1.0 constant:0];
    NSLayoutConstraint *bottomConstraint = [NSLayoutConstraint constraintWithItem:_toolBar attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeBottom multiplier:1.0 constant:0];
    NSLayoutConstraint *rightConstraint = [NSLayoutConstraint constraintWithItem:_toolBar attribute:NSLayoutAttributeRight relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeRight multiplier:1.0 constant:0];
    [self addConstraints:@[topConstraint, leftConstraint, bottomConstraint, rightConstraint]];
}

#pragma mark - NSSecureCoding
+ (BOOL)supportsSecureCoding {
    return YES;
}

- (instancetype)initWithCoder:(NSCoder *)aDecoder {
    if (self = [super initWithCoder:aDecoder]) {
        _contentView = [aDecoder decodeObjectOfClass:[UIView class] forKey:@"contentView"];
        _effect = [aDecoder decodeObjectOfClass:[UIVisualEffect class] forKey:@"effect"];
    }
    return self;
}

- (void)encodeWithCoder:(NSCoder *)aCoder {
    [aCoder encodeObject:_contentView forKey:@"contentView"];
    [aCoder encodeObject:_effect forKey:@"effect"];
}

@end

#pragma mark - Runtime Injection
__asm(
      ".section        __DATA,__objc_classrefs,regular,no_dead_strip\n"
#if TARGET_RT_64_BIT
      ".align          3\n"
      "L_OBJC_CLASS_UIVisualEffect:\n"
      ".quad           _OBJC_CLASS_$_UIVisualEffect\n"
      
      ".align          3\n"
      "L_OBJC_CLASS_UIBlurEffect:\n"
      ".quad           _OBJC_CLASS_$_UIBlurEffect\n"
      
      ".align          3\n"
      "L_OBJC_CLASS_UIVibrancyEffect:\n"
      ".quad           _OBJC_CLASS_$_UIVibrancyEffect\n"
      
      ".align          3\n"
      "L_OBJC_CLASS_UIVisualEffectView:\n"
      ".quad           _OBJC_CLASS_$_UIVisualEffectView\n"
#else
      ".align          2\n"
      "_OBJC_CLASS_UIVisualEffect:\n"
      ".long           _OBJC_CLASS_$_UIVisualEffect\n"
      
      ".align          2\n"
      "_OBJC_CLASS_UIBlurEffect:\n"
      ".long           _OBJC_CLASS_$_UIBlurEffect\n"
      
      ".align          2\n"
      "_OBJC_CLASS_UIVibrancyEffect:\n"
      ".long           _OBJC_CLASS_$_UIVibrancyEffect\n"
      
      ".align          2\n"
      "_OBJC_CLASS_UIVisualEffectView:\n"
      ".long           _OBJC_CLASS_$_UIVisualEffectView\n"
#endif
      ".weak_reference _OBJC_CLASS_$_UIVisualEffect\n"
      ".weak_reference _OBJC_CLASS_$_UIBlurEffect\n"
      ".weak_reference _OBJC_CLASS_$_UIVibrancyEffect\n"
      ".weak_reference _OBJC_CLASS_$_UIVisualEffectView\n"
      );

// Constructors are called after all classes have been loaded.
__attribute__((constructor)) static void XAVisualEffectViewPatchEntry(void) {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        @autoreleasepool {
            
            // >= iOS8.
            if (objc_getClass("UIVisualEffectView")) {
                return;
            }
            
            Class *visualEffect = NULL;
            Class *blurEffect = NULL;
            Class *vibrancyEffect = NULL;
            Class *visualEffectView = NULL;
            
#if TARGET_CPU_ARM
            __asm("movw %0, :lower16:(_OBJC_CLASS_UIVisualEffect-(LPC0+4))\n"
                  "movt %0, :upper16:(_OBJC_CLASS_UIVisualEffect-(LPC0+4))\n"
                  "LPC0: add %0, pc" : "=r"(visualEffect));
            __asm("movw %0, :lower16:(_OBJC_CLASS_UIBlurEffect-(LPC1+4))\n"
                  "movt %0, :upper16:(_OBJC_CLASS_UIBlurEffect-(LPC1+4))\n"
                  "LPC1: add %0, pc" : "=r"(blurEffect));
            __asm("movw %0, :lower16:(_OBJC_CLASS_UIVibrancyEffect-(LPC2+4))\n"
                  "movt %0, :upper16:(_OBJC_CLASS_UIVibrancyEffect-(LPC2+4))\n"
                  "LPC2: add %0, pc" : "=r"(vibrancyEffect));
            __asm("movw %0, :lower16:(_OBJC_CLASS_UIVisualEffectView-(LPC3+4))\n"
                  "movt %0, :upper16:(_OBJC_CLASS_UIVisualEffectView-(LPC3+4))\n"
                  "LPC3: add %0, pc" : "=r"(visualEffectView));
#elif TARGET_CPU_ARM64
            __asm("adrp %0, L_OBJC_CLASS_UIVisualEffect@PAGE\n"
                  "add  %0, %0, L_OBJC_CLASS_UIVisualEffect@PAGEOFF" : "=r"(visualEffect));
            __asm("adrp %0, L_OBJC_CLASS_UIBlurEffect@PAGE\n"
                  "add  %0, %0, L_OBJC_CLASS_UIBlurEffect@PAGEOFF" : "=r"(blurEffect));
            __asm("adrp %0, L_OBJC_CLASS_UIVibrancyEffect@PAGE\n"
                  "add  %0, %0, L_OBJC_CLASS_UIVibrancyEffect@PAGEOFF" : "=r"(vibrancyEffect));
            __asm("adrp %0, L_OBJC_CLASS_UIVisualEffectView@PAGE\n"
                  "add  %0, %0, L_OBJC_CLASS_UIVisualEffectView@PAGEOFF" : "=r"(visualEffectView));
#elif TARGET_CPU_X86_64
            __asm("leaq L_OBJC_CLASS_UIVisualEffect(%%rip), %0" : "=r"(visualEffect));
            __asm("leaq L_OBJC_CLASS_UIBlurEffect(%%rip), %0" : "=r"(blurEffect));
            __asm("leaq L_OBJC_CLASS_UIVibrancyEffect(%%rip), %0" : "=r"(vibrancyEffect));
            __asm("leaq L_OBJC_CLASS_UIVisualEffectView(%%rip), %0" : "=r"(visualEffectView));
#elif TARGET_CPU_X86
            void *pc1 = NULL;
            __asm("calll L0\n"
                  "L0: popl %0\n"
                  "leal _OBJC_CLASS_UIVisualEffect-L0(%0), %1" : "=r"(pc1), "=r"(visualEffect));
            void *pc2 = NULL;
            __asm("calll L0\n"
                  "L0: popl %0\n"
                  "leal _OBJC_CLASS_UIBlurEffect-L0(%0), %1" : "=r"(pc2), "=r"(blurEffect));
            void *pc3 = NULL;
            __asm("calll L0\n"
                  "L0: popl %0\n"
                  "leal _OBJC_CLASS_UIVibrancyEffect-L0(%0), %1" : "=r"(pc3), "=r"(vibrancyEffect));
            void *pc4 = NULL;
            __asm("calll L0\n"
                  "L0: popl %0\n"
                  "leal _OBJC_CLASS_UIVisualEffectView-L0(%0), %1" : "=r"(pc4), "=r"(visualEffectView));
#else
#error Unsupported CPU
#endif
            
            if (visualEffect && !*visualEffect) {
                Class class = objc_allocateClassPair([XAVisualEffect class], "UIVisualEffect", 0);
                if (class) {
                    objc_registerClassPair(class);
                    *visualEffect = class;
                }
            }
            
            if (blurEffect && !*blurEffect) {
                Class class = objc_allocateClassPair([XABlurEffect class], "UIBlurEffect", 0);
                if (class) {
                    objc_registerClassPair(class);
                    *blurEffect = class;
                }
            }
            
            if (vibrancyEffect && !*vibrancyEffect) {
                Class class = objc_allocateClassPair([XAVibrancyEffect class], "UIVibrancyEffect", 0);
                if (class) {
                    objc_registerClassPair(class);
                    *vibrancyEffect = class;
                }
            }
            
            if (visualEffectView && !*visualEffectView) {
                Class class = objc_allocateClassPair([XAVisualEffectView class], "UIVisualEffectView", 0);
                if (class) {
                    objc_registerClassPair(class);
                    *visualEffectView = class;
                }
            }
        }
    });
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,016评论 4 62
  • 话说苹果在iOS7.0之后,很多系统界面都使用了毛玻璃效果,增加了界面的美观性,比如下图的通知中心界面; 效果图:...
    Mr_董阅读 1,289评论 0 0
  • 吝啬鬼也有大方的时候。 "买!" 2014年元旦第二天,我打电话给老公,说看中了一套房子,带个小院,非常喜欢,就是...
    专抓耗子阅读 1,151评论 1 3
  • 一次往事重提,让我回忆起了那深情亦多情的少女时代。那些笑过哭过傻过痴过的青春记忆,让已为人母的我又少女了一回。我想...
    麗芝姑娘阅读 286评论 0 1
  • 作者: 拥抱之力_故 自从很多年前与“宅”这个字相遇后,我对他始终抱有相见恨晚,一见如故的感觉。 ...
    拥抱之力_故阅读 280评论 2 2