IOS-Swift:可插拔的Mixin设计范式

上面的这个『可插拔的Mixin』是我自己取的名字。其主要代指的是这种情况:常常我们需要给一个类添加一个通用性的功能,例如给一个有网络访问的ViewController添加一个进度条或者是加一个Loading的标志等。这时候如果逐一在各个类里面实现,会在调试和维护的时候产生很多麻烦。而如果通过类继承的方法来做,则限制其使用范围。下面这种Mixin的方法是我之前在做Java的时候收到启发想出来的,也许现在也有其他人已经做过了吧。

我通过下面这个给ViewController添加loading标识的例子来说明这个设计思路,其核心是利用Protocol的方法的Default Implementation来给ViewController添加新的功能:

实现Mixin的插件:

protocol LoadingMixin: class {
    /**
     调用这个函数来显示Loading标识
     */
    func lm_start()
    
    /**
     调用这个函数来隐藏Loading标识
     */
    func lm_stop()
    
    var lm_container: UIView! { get }
    var lm_loadingView: UIView! { get set }
    var lm_timer: UIView! { get set }
    var lm_delayTask: dispatch_block_t? { get set }
}

private var containerHandle: UInt8 = 1
private var loadingViewHandle: UInt8 = 2
private var timerHandle: UInt8 = 3
private var taskHandle: UInt8 = 4

// MARK: - 在extension里面我们可以给protocol添加默认实现
extension LoadingMixin where Self: UIViewController {
    var lm_container: UIView! {
        get {
            if let container = objc_getAssociatedObject(self, &containerHandle) as? UIView {
                return container
            } else {
                let container = UIView()
                // TODO: 在这里创建LoadingView的样式
                return container
            }
        }
    }
    
    var lm_loadingView: UIImageView! {
        set {
            objc_setAssociatedObject(self, &loadingViewHandle, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
        }
        
        get {
                  // 确保 LoadingViwe被创建了
            self.lp_container
            return objc_getAssociatedObject(self, &loadingViewHandle) as? UIImageView
        }
    }
    
    var lm_timer: NSTimer! {
        set {
            objc_setAssociatedObject(self, &timerHandle, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
        }
        
        get {
            return objc_getAssociatedObject(self, &timerHandle) as? NSTimer
        }
    }
    
    var lm_delayTask: dispatch_block_t? {
        set {
            objc_setAssociatedObject(self, &taskHandle, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
        }
        
        get {
            return objc_getAssociatedObject(self, &taskHandle) as? NSTimer
        }
    }
    
    func lp_stop() {
        if lm_timer != nil {
            lm_timer.invalidate()
            lm_timer = nil
            lm_container.hidden = true
            lm_container.removeFromSuperview()
        } else if self.delayTask != nil {
            dispatch_block_cancel(lm_delayTask!)
        } else {
            assertionFailure()
        }
        delayTask = nil
    }
    
    func lp_start() {
        // kLoadingAppearDelay是一个延时常量,需要你自己在外部定义,或者直接替换成一个常数
        let delay = dispatch_time(DISPATCH_TIME_NOW, Int64(NSEC_PER_MSEC) * kLoadingAppearDelay)
        lm_timer = nil
        lm_delayTask = dispatch_block_create(DISPATCH_BLOCK_INHERIT_QOS_CLASS, { [weak self] in
            guard let sSelf = self else {
                return
            }
            let timer = NSTimer.schedule(repeatInterval: 0.05, handler: { (_) in
                         // 你可以重写这个部分来自定义Loading动画
                let curTrans = sSelf.lm_loadingView.transform
                let newTrans = CGAffineTransformRotate(curTrans, 0.4)
                sSelf.lm_loadingView.transform = newTrans
            })
            sSelf.lm_timer = timer
            })
        dispatch_after(delay, dispatch_get_main_queue(), lm_delayTask!)
        
    }
}

在上面的代码中,我们定义了一个名为LoadingMixin的协议,并且extension中实现其涉及的主要方法:

  • lm_start: 最终供viewController使用的接口,调用这个函数来显示Loading标识;
  • lm_stop: 也是供viewController使用的接口,调用这个函数来隐藏Loading标识;
  • lm_container: Loading标志的view hierarchy的顶层view;
  • lm_loadingView: Loading标志中需要动画旋转的图;
  • lm_timer: 驱动动画的timer;
  • lm_delayTask: 涉及这个的功能我们在下面解释。

绝大多数的情况下你不用重写上面的属性或者方法。

进一步的代码解释:

在上面的代码中,有两个问题还需要解释一下:

一是我在lm_start中采用了一种延时启动的方式,当lm_start被调用以后,并不会马上显示LoadingView,显示的代码被封装在一个block中,定在kLoadingAppearDelay毫秒之后执行。而如果在这之前lm_stop被调用的话,其调用会被取消。在当网络环境比较好,请求在kLoadingAppearDelay时间之内完成时,不需要显示Loading画面。

二是下面这部分

 let timer = NSTimer.schedule(repeatInterval: 0.05, handler: { (_) in
                let curTrans = sSelf.lm_loadingView.transform
                let newTrans = CGAffineTransformRotate(curTrans, 0.4)
                sSelf.lm_loadingView.transform = newTrans
            })

使用如下定义的extension:

extension NSTimer {
    /**
     Creates and schedules a one-time NSTimer instance.
     - Parameters:
     - delay: The delay before execution.
     - handler: A closure to execute after delay.
     
     - Returns: The newly-created `NSTimer` instance.
     */
    class func schedule(delay delay: NSTimeInterval, handler: NSTimer! -> Void) -> NSTimer {
        let fireDate = delay + CFAbsoluteTimeGetCurrent()
        let timer = CFRunLoopTimerCreateWithHandler(kCFAllocatorDefault, fireDate, 0, 0, 0, handler)
        CFRunLoopAddTimer(CFRunLoopGetCurrent(), timer, kCFRunLoopCommonModes)
        return timer
    }
    
    /**
     Creates and schedules a repeating NSTimer instance.
     
     - Parameters:
     - repeatInterval: The interval (in seconds) between each execution of
     handler. Note that individual calls may be delayed; subsequent calls
     to handler will be based on the time the timer was created.
     - handler: A closure to execute at each repeatInterval.
     
     - Returns: The newly-created NSTimer instance.
     */
    class func schedule(repeatInterval interval: NSTimeInterval, handler: NSTimer! -> Void) -> NSTimer {
        let fireDate = interval + CFAbsoluteTimeGetCurrent()
        let timer = CFRunLoopTimerCreateWithHandler(kCFAllocatorDefault, fireDate, interval, 0, 0, handler)
        CFRunLoopAddTimer(CFRunLoopGetCurrent(), timer, kCFRunLoopCommonModes)
        return timer
    }
}

使用上面的Mixin:

如下例子:

class MyViewController: UIViewController, LoadingMixin {
    func request() {
        lm_start()
        send_request(..., onFinish: {
            ...
            self.lm_stop()
        })
    }
}

上面只是一个例子,大家可以用类似的思路来实现自己的『可插拔的Mixin』

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,799评论 6 342
  • 夏至,阵阵压抑。 是灰暗的天空, 是零落的人流, 是暴雨前隐忍的雷鸣。 悠扬的季节已悄然滑过, 片片碎裂的温情被高...
    夏洛克鱿尔鱼丝阅读 320评论 0 1
  • 1、看了部电影,讲的是一万年后的世界,说不定那时的人们考古发现现代人都喜欢看段子,所以断定段子是古代的文学瑰宝,哈...
    1e81dcf20286阅读 254评论 0 1
  • (全文约6200字,阅读需16分钟) (本文不涉及重度剧透,不影响动漫观看效果) 现在市面上的动漫作品越来越多,每...
    阿正说动漫阅读 2,210评论 5 24