FIFO和LIFO自動(dòng)管理modal控制器

在一個(gè)App中芋浮,彈窗一直是一個(gè)使用頻率較高的提示類控件兄世。蘋果對(duì)用戶體驗(yàn)方面的重視程度有多高甜孤,在彈窗的處理上就能體現(xiàn)出這一點(diǎn)來(lái)。不知你是否留意過(guò)新安裝的App上的彈窗顯示順序粥喜?通常是這樣的凸主,如果先出現(xiàn)的是通知權(quán)限彈窗然后是定位權(quán)限彈窗,前一個(gè)彈窗會(huì)暫時(shí)隱藏额湘,等用戶關(guān)掉后一個(gè)彈窗后才再次彈出來(lái)卿吐。顯然蘋果認(rèn)為后面的彈窗更重要,所以應(yīng)當(dāng)優(yōu)先被用戶處理锋华。蘋果對(duì)彈窗的彈出順序進(jìn)行了LIFO(last in, first out)后進(jìn)先出的管理嗡官。

本文后面會(huì)講解如何使用Runtime多線程實(shí)現(xiàn)LIFO和FIFO (first in, last out)先進(jìn)先出。先看一下最終效果毯焕,這是一個(gè)自定義轉(zhuǎn)場(chǎng)動(dòng)畫(TransitionAnimation)的UIViewController衍腥。

最終效果

UIAlertView支持自動(dòng)后進(jìn)先出管理。但是纳猫,它從iOS8開(kāi)始就已經(jīng)被廢棄了婆咸,再次使用UIAlertView,代碼中會(huì)出現(xiàn)黃色警告提示芜辕。以下是UIAlertView被廢棄的說(shuō)明尚骄,蘋果讓我們使用UIAlertController去替代前者。

UIAlert View is deprecated in iOS 8. (Note that UIAlert View Delegate is also deprecated.) To create and manage alerts in iOS 8 and later, instead use UIAlert Controller with a preferred Style of UIAlert Controller Style Alert.

UIAlertController是繼承于UIViewController并自定義了轉(zhuǎn)場(chǎng)動(dòng)畫的控制器侵续,和其它modal控制器一樣調(diào)用presentViewController:animated:completion:方法彈出倔丈。但是,每個(gè)控制器只能擁有一個(gè)presentedController询兴,也就是每次只能present一個(gè)別的控制器乃沙,強(qiáng)行present新的會(huì)出現(xiàn)以下提示:

Warning: Attempt to present <UIAlertController: 0x7fdb635045d0>  on <ViewController: 0x7fdb660090f0> which is already presenting <UIAlertController: 0x7fdb635084a0>

很多情況下,異步請(qǐng)求結(jié)束后诗舰,我們需要根據(jù)服務(wù)器的返回信息進(jìn)行彈窗提示警儒。卻無(wú)法保證當(dāng)時(shí)的所在控制器是否已經(jīng)present了別的UIAlertController,新的彈窗是彈不出來(lái)的眶根。所以使用UIAlertController會(huì)給我們帶來(lái)很大的困擾蜀铲。

還有一種是把自定義的UIView蓋到UIWindow的方式進(jìn)行彈窗。首先這種方式不支持彈出順序管理属百,其次同時(shí)彈出多個(gè)就是多次執(zhí)行addSubview方法记劝,很多半透明背景遮罩和view疊加在一起,顯示效果不言而喻族扰。更重要的是厌丑,很多系統(tǒng)控件也使用了window作為父控件定欧,比如鍵盤的window是UITextEffectsWindow,頻繁使用window可能會(huì)出現(xiàn)很多意想不到的問(wèn)題怒竿。

總結(jié)一下上述3種彈窗方式:

彈窗方式 存在的問(wèn)題
UIAlertView iOS8開(kāi)始已經(jīng)被廢棄
UIAlertController 每次只能彈一個(gè)
-[UIWindow addSubview:] 同時(shí)彈出多個(gè)的顯示效果較差

現(xiàn)在開(kāi)始講解如何用FIFOLIFO管理modal控制器

先講相對(duì)簡(jiǎn)單的FIFO

FIFO流程圖

效果如下

ps: gif圖中砍鸠,彈窗2的點(diǎn)擊事件中彈出彈窗4,觀察在FIFO和LIFO中的區(qū)別

自定義UIViewController
UIAlertController

首先耕驰,新建一個(gè)UIViewController的分類爷辱,設(shè)計(jì)成分類的目的是做到百分百解耦,供控制器調(diào)用朦肘,并支持對(duì)所有繼承于UIViewController的控制器進(jìn)行FIFO和LIFO的modal管理饭弓。

分類的頭文件只有一個(gè)方法,用來(lái)替代系統(tǒng)的presentViewController:animated:completion:方法媒抠。該方法接收一個(gè)UIViewController參數(shù)弟断,一個(gè)present完成的回調(diào)和dismiss完成的回調(diào)。

// 枚舉趴生,要用哪種方式管理modal控制器
typedef NS_OPTIONS (NSUInteger, JCPresentType) {
    JCPresentTypeLIFO = 0, // last in, first out
    JCPresentTypeFIFO      // first in, last out
};
// 新的present方法
- (void)jc_presentViewController:(UIViewController *)controller presentType:(JCPresentType)presentType presentCompletion:(void (^)(void))presentCompletion dismissCompletion:(void (^)(void))dismissCompletion;

.m文件中夫嗓,方法的實(shí)現(xiàn)是這樣的

// 判斷JCPresentType枚舉類型,跳轉(zhuǎn)到具體方法
- (void)jc_presentViewController:(UIViewController *)controller presentType:(JCPresentType)presentType presentCompletion:(void (^)(void))presentCompletion dismissCompletion:(void (^)(void))dismissCompletion {
    if (presentType == JCPresentTypeLIFO) {
        [self lifoPresentViewController:controller presentCompletion:presentCompletion dismissCompletion:dismissCompletion];
    } else {
        [self fifoPresentViewController:controller presentCompletion:presentCompletion dismissCompletion:dismissCompletion];
    }
}
// 核心方法
- (void)fifoPresentViewController:(UIViewController *)controller presentCompletion:(void (^)(void))presentCompletion dismissCompletion:(void (^)(void))dismissCompletion {
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        [controller setDeallocCompletion:^{
            if (dismissCompletion) {
                dismissCompletion();
            }
            // got to next operation
            dispatch_semaphore_signal(semaphore);
        }];
        dispatch_async(dispatch_get_main_queue(), ^{
            [self presentViewController:controller animated:YES completion:presentCompletion];
        });
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    }];

    // put in queue
    if ([self getOperationQueue].operations.lastObject) {
        [operation addDependency:[self getOperationQueue].operations.lastObject];
    }
    [[self getOperationQueue] addOperation:operation];
}
// NSOperationQueue單例冲秽,用于添加operation
- (NSOperationQueue *)getOperationQueue {
    static NSOperationQueue *operationQueue = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        operationQueue = [NSOperationQueue new];
    });
    return operationQueue;
}
// 使用關(guān)聯(lián)對(duì)象存取deallocCompletion這個(gè)block
- (void)setDeallocCompletion:(void (^)(void))completion {
    objc_setAssociatedObject(self, @selector(getDeallocCompletion), completion, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (void (^)(void))getDeallocCompletion {
    return objc_getAssociatedObject(self, _cmd);
}
// hook控制器的viewDidDisappear方法舍咖,在這個(gè)時(shí)候調(diào)deallocCompletion這個(gè)block
+ (void)load {
    SEL oldSel = @selector(viewDidDisappear:);
    SEL newSel = @selector(jc_viewDidDisappear:);
    Method oldMethod = class_getInstanceMethod([self class], oldSel);
    Method newMethod = class_getInstanceMethod([self class], newSel);
    
    BOOL didAddMethod = class_addMethod(self, oldSel, method_getImplementation(newMethod), method_getTypeEncoding(newMethod));
    if (didAddMethod) {
        class_replaceMethod(self, newSel, method_getImplementation(oldMethod), method_getTypeEncoding(oldMethod));
    } else {
        method_exchangeImplementations(oldMethod, newMethod);
    }
}

- (void)jc_viewDidDisappear:(BOOL)animated {
    [self jc_viewDidDisappear:animated];
    
    if ([self getDeallocCompletion] && ![self isTemporarilyDismissed]) {
        [self getDeallocCompletion]();
    }
}

其中dispatch_semaphore_t是多線程中的信號(hào)量,dispatch_semaphore_signal函數(shù)使信號(hào)量加一锉桑,dispatch_semaphore_wait使信號(hào)量減一排霉,并且在信號(hào)量小于0的時(shí)候暫停當(dāng)前線程。

presentViewController是一個(gè)耗時(shí)操作民轴,我把這個(gè)操作放在NSBlockOperation中攻柠,用dispatch_semaphore_t暫停線程直到present完成。并設(shè)置每個(gè)NSBlockOperation的前后依賴后裸,最后加到NSOperationQueue中瑰钮,組成一個(gè)串行的先進(jìn)先出隊(duì)列。

下面開(kāi)始講LIFO

LIFO流程圖

效果如下

自定義UIViewController
UIAlertController
// 核心方法
- (void)lifoPresentViewController:(UIViewController *)controller presentCompletion:(void (^)(void))presentCompletion dismissCompletion:(void (^)(void))dismissCompletion {
    
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    
    // put in stack
    NSMutableArray *stackControllers = [self getStackControllers];
    if (![stackControllers containsObject:controller]) {
        [stackControllers addObject:controller];
    }
    
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        __weak typeof(controller) weakController = controller;
        [controller setPresentCompletion:presentCompletion];
        [controller setDismissCompletion:dismissCompletion];
        [controller setDeallocCompletion:^{
            if (dismissCompletion) {
                dismissCompletion();
            }
            
            // fetch new next controller if exists, because button action after dismiss completion
            [weakController setDismissing:YES];
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(CGFLOAT_MIN * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                [weakController setDismissing:NO];
                // if the dismiss controller is the last one
                if (stackControllers.lastObject == controller) {
                    [stackControllers removeObject:weakController];
                    
                    // is there any previous controllers
                    if (stackControllers.count > 0) {
                        UIViewController *preController = [stackControllers lastObject];
                        [self lifoPresentViewController:preController presentCompletion:[preController getPresentCompletion] dismissCompletion:[preController getDismissCompletion]];
                    }
                } else {
                    NSUInteger index = [stackControllers indexOfObject:weakController];
                    [stackControllers removeObject:weakController];
                    
                    // is there any next controllers
                    NSArray *nextControllers = [stackControllers objectsAtIndexes:[NSIndexSet indexSetWithIndexesInRange:NSMakeRange(index, stackControllers.count - index)]];
                    for (UIViewController *nextController in nextControllers) {
                        [self lifoPresentViewController:nextController presentCompletion:[nextController getPresentCompletion] dismissCompletion:[nextController getDismissCompletion]];
                    }
                }
            });
        }];
        
        // if the previous controller is dismissing, wait it's completion
        if (stackControllers.count > 1) {
            for (UIViewController *preController in stackControllers) {
                if ([preController isDismissing]) {
                    return ;
                }
            }
        }
        
        // present a new controller before dismissing the presented controller if exists
        dispatch_async(dispatch_get_main_queue(), ^{
            if (self.presentedViewController) {
                [self.presentedViewController temporarilyDismissViewControllerAnimated:YES completion:^{
                    [self presentViewController:controller animated:YES completion:^{
                        dispatch_semaphore_signal(semaphore);
                    }];
                }];
            } else {
                [self presentViewController:controller animated:YES completion:^{
                    dispatch_semaphore_signal(semaphore);
                    if (presentCompletion) {
                        presentCompletion();
                    }
                }];
            }
        });
        
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    }];
    
    // put in queue
    if ([self getOperationQueue].operations.lastObject) {
        [operation addDependency:[self getOperationQueue].operations.lastObject];
    }
    [[self getOperationQueue] addOperation:operation];
}
// 使用關(guān)聯(lián)對(duì)象存取dismissCompletion
- (void)setDismissCompletion:(void (^)(void))completion {
    objc_setAssociatedObject(self, @selector(getDismissCompletion), completion, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (void (^)(void))getDismissCompletion {
    return objc_getAssociatedObject(self, _cmd);
}
// 使用關(guān)聯(lián)對(duì)象存取presentCompletion
- (void)setPresentCompletion:(void (^)(void))completion {
    objc_setAssociatedObject(self, @selector(getPresentCompletion), completion, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (void (^)(void))getPresentCompletion {
    return objc_getAssociatedObject(self, _cmd);
}

// 使用關(guān)聯(lián)對(duì)象存取temporarilyDismissed微驶,用于判斷是臨時(shí)隱藏還是用戶關(guān)閉控制器
- (void)setTemporarilyDismissed:(BOOL)temporarilyDismissed {
    objc_setAssociatedObject(self, @selector(isTemporarilyDismissed), @(temporarilyDismissed), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

- (BOOL)isTemporarilyDismissed {
    NSNumber *num = objc_getAssociatedObject(self, _cmd);
    return [num boolValue];
}

// 使用關(guān)聯(lián)對(duì)象存取dismissing
- (void)setDismissing:(BOOL)dismissing {
    objc_setAssociatedObject(self, @selector(isDismissing), @(dismissing), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

- (BOOL)isDismissing {
    NSNumber *num = objc_getAssociatedObject(self, _cmd);
    return [num boolValue];
}

// 數(shù)組棧浪谴,用于緩存所有傳進(jìn)來(lái)的控制器
- (NSMutableArray *)getStackControllers {
    static NSMutableArray *stackControllers = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        stackControllers = [NSMutableArray array];
    });
    return stackControllers;
}

// 臨時(shí)dismiss方法
- (void)temporarilyDismissViewControllerAnimated: (BOOL)flag completion: (void (^ __nullable)(void))completion {
    [self setTemporarilyDismissed:YES];
    [self dismissViewControllerAnimated:flag completion:^{
        [self setTemporarilyDismissed:NO];
        if (completion) {
            completion();
        }
    }];
}

和FIFO方法的區(qū)別是,LIFO方法中因苹,每次進(jìn)來(lái)會(huì)先判斷前面是否已經(jīng)有控制器了苟耻,如果有就先臨時(shí)dismiss。并且扶檐,控制器在被用戶關(guān)閉的時(shí)候凶杖,優(yōu)先判斷棧后面有沒(méi)有還沒(méi)有彈出來(lái)的控制器,然后才判斷棧前面有沒(méi)有控制器款筑。

這樣智蝠,一個(gè)具有FIFO和LIFO驅(qū)動(dòng)的present分類方法就完成了腾么,敢緊試試吧。


下載地址:UIViewController+JCPresentQueue.h

參考資料

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末杈湾,一起剝皮案震驚了整個(gè)濱河市哮翘,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌毛秘,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,204評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件阻课,死亡現(xiàn)場(chǎng)離奇詭異叫挟,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)限煞,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門抹恳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人署驻,你說(shuō)我怎么就攤上這事奋献。” “怎么了旺上?”我有些...
    開(kāi)封第一講書人閱讀 164,548評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵瓶蚂,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我宣吱,道長(zhǎng)窃这,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 58,657評(píng)論 1 293
  • 正文 為了忘掉前任征候,我火速辦了婚禮杭攻,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘疤坝。我一直安慰自己兆解,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,689評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布跑揉。 她就那樣靜靜地躺著锅睛,像睡著了一般。 火紅的嫁衣襯著肌膚如雪历谍。 梳的紋絲不亂的頭發(fā)上衣撬,一...
    開(kāi)封第一講書人閱讀 51,554評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音扮饶,去河邊找鬼具练。 笑死,一個(gè)胖子當(dāng)著我的面吹牛甜无,可吹牛的內(nèi)容都是我干的扛点。 我是一名探鬼主播哥遮,決...
    沈念sama閱讀 40,302評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼陵究!你這毒婦竟也來(lái)了眠饮?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,216評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤铜邮,失蹤者是張志新(化名)和其女友劉穎仪召,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體松蒜,經(jīng)...
    沈念sama閱讀 45,661評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡扔茅,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,851評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了秸苗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片召娜。...
    茶點(diǎn)故事閱讀 39,977評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖惊楼,靈堂內(nèi)的尸體忽然破棺而出玖瘸,到底是詐尸還是另有隱情,我是刑警寧澤檀咙,帶...
    沈念sama閱讀 35,697評(píng)論 5 347
  • 正文 年R本政府宣布雅倒,位于F島的核電站,受9級(jí)特大地震影響弧可,放射性物質(zhì)發(fā)生泄漏屯断。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,306評(píng)論 3 330
  • 文/蒙蒙 一侣诺、第九天 我趴在偏房一處隱蔽的房頂上張望殖演。 院中可真熱鬧,春花似錦年鸳、人聲如沸趴久。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,898評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)彼棍。三九已至,卻和暖如春膳算,著一層夾襖步出監(jiān)牢的瞬間座硕,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,019評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工涕蜂, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留华匾,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,138評(píng)論 3 370
  • 正文 我出身青樓机隙,卻偏偏與公主長(zhǎng)得像蜘拉,于是被迫代替她去往敵國(guó)和親萨西。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,927評(píng)論 2 355

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