iOS 深度好文

FGPopupView

2023-03-13  本文已影响0人  melonsong

1、基本使用

FGPopupScheduler *Scheduler = FGPopupSchedulerGetForPSS(FGPopupSchedulerStrategyFIFO);
AnimationShowPopupView *pop1 =  [[AnimationShowPopupView alloc] initWithDescrption:@"自定义动画 pop2" scheduler:Scheduler];
ConditionsPopView *pop2 =  [[ConditionsPopView alloc] initWithDescrption:@"条件弹窗 pop3 Discard" scheduler:Scheduler];

[Scheduler add:pop];
[Scheduler add:pop2];

2、AnimationShowPopupView、ConditionsPopView都是需要实现FGPopupView protocol的NSObject(一般是一个UIView的类)

FGPopupView protocol中的方法都是optional可选的:

@protocol FGPopupView <NSObject>

@optional
/*
 FGPopupSchedulerStrategyQueue会根据 -showPopupView: 做显示逻辑,如果含有动画请实现-showPopupViewWithAnimation:方法
 */
- (void)showPopupView;

/*
 FGPopupSchedulerStrategyQueue会根据 -dismissPopupView: 做隐藏逻辑,如果含有动画请实现-showPopupViewWithAnimation:方法
 */
- (void)dismissPopupView;

/*
 FGPopupSchedulerStrategyQueue会根据 -showPopupViewWithAnimation: 来做显示逻辑。如果block不传可能会出现意料外的问题
 */
- (void)showPopupViewWithAnimation:(FGPopupViewAnimationBlock)block;

/*
 FGPopupSchedulerStrategyQueue会根据 -dismissPopupView: 做隐藏逻辑,如果含有动画请实现-dismissPopupViewWithAnimation:方法,如果block不传可能会出现意料外的问题
 */
- (void)dismissPopupViewWithAnimation:(FGPopupViewAnimationBlock)block;

/**
 FGPopupSchedulerStrategyQueue会根据-canRegisterFirstPopupView判断,当队列顺序轮到它的时候是否能够成为响应的第一个优先级PopupView。默认为YES
 */
- (BOOL)canRegisterFirstPopupViewResponder;

/** 0.4.0 新增*/

/**
 FGPopupSchedulerStrategyQueue 会根据 - popupViewUntriggeredBehavior:来决定触发时弹窗的显示行为,默认为 FGPopupViewUntriggeredBehaviorAwait
 */
- (FGPopupViewUntriggeredBehavior)popupViewUntriggeredBehavior;


/**
 FGPopupViewSwitchBehavior 会根据 - popupViewSwitchBehavior:来决定已经显示的弹窗,是否会被后续更高优先级的弹窗锁影响,默认为 FGPopupViewSwitchBehaviorAwait  ⚠️⚠️ 只在FGPopupSchedulerStrategyPriority生效
 */
- (FGPopupViewSwitchBehavior)popupViewSwitchBehavior;

@end

3、调度策略

typedef NS_ENUM(NSUInteger, FGPopupSchedulerStrategy) {
    FGPopupSchedulerStrategyFIFO = 1 << 0,           //先进先出
    FGPopupSchedulerStrategyLIFO = 1 << 1,           //后进先出
    FGPopupSchedulerStrategyPriority = 1 << 2        //优先级调度
};

可以根据需求选择合适的策略,另外实际上还可以结合 FGPopupSchedulerStrategyPriority | FGPopupSchedulerStrategyFIFO 一起使用,来处理当选择优先级策略时,如何决定同一优先级弹窗的排序。

4、触发策略

用户可以根据它来决定,当弹窗触发显示逻辑时是否要继续等待

typedef NS_ENUM(NSUInteger, FGPopupViewUntriggeredBehavior) {
    FGPopupViewUntriggeredBehaviorDiscard,          //未满足条件时会被直接丢弃
    FGPopupViewUntriggeredBehaviorAwait,          //未满足条件时会继续等待
};

5、添加策略

在使用FGPopupSchedulerStrategyPriority时碰到优先级相同时,应该如何添加,这里有两个选择,一个先进先出,一个先进后出

typedef NS_ENUM(NSUInteger, FGPopupPriorityAddStrategy) {
    FGPopupPriorityAddStrategyFIFO, //FIFO
    FGPopupPriorityAddStrategyLIFO, //LIFO
};

6、切换策略

目前支持3种切换行为, 用户可以根据它来决定,当该弹窗已经显示时,是否会被后续高优线级的弹窗影响。仅在优先级调度策略时生效:FGPopupSchedulerStrategyPriority

typedef NS_ENUM(NSUInteger, FGPopupViewSwitchBehavior) {
    FGPopupViewSwitchBehaviorDiscard,  //当该弹窗已经显示,如果后面来了弹窗优先级更高的弹窗时,显示更高优先级弹窗并且当前弹窗会被抛弃
    FGPopupViewSwitchBehaviorLatent,   //当该弹窗已经显示,如果后面来了弹窗优先级更高的弹窗时,显示更高优先级弹窗并且当前弹窗重新进入队列, PS:优先级相同时同 FGPopupViewSwitchBehaviorDiscard
    FGPopupViewSwitchBehaviorAwait,    //当该弹窗已经显示时,不会被后续高优线级的弹窗影响
};

7、核心逻辑

//类FGPopupScheduler

//注册observer 通过Runloop监听主线程空闲的时刻
+ (void)initialize{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        CFRunLoopObserverRef observer = CFRunLoopObserverCreate(CFAllocatorGetDefault(), kCFRunLoopBeforeWaiting | kCFRunLoopExit, true, 0xFFFFFF, FGRunLoopObserverCallBack, nil);
        CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);
        CFRelease(observer);
    });
}
/*
以下三个方法调用了registerFirstPopupViewResponder
*/
- (void)setSuspended:(BOOL)suspended{
    dispatch_async_main_safe(^(){
        self->_suspended = suspended;
        if (!suspended) [self registerFirstPopupViewResponder];
    });
}
//添加弹窗对象的时候
- (void)add:(id<FGPopupView>)view  Priority:(FGPopupStrategyPriority)Priority{
    dispatch_async_main_safe(^(){
        [self->_list addPopupView:view Priority:Priority];
        [self registerFirstPopupViewResponder];
    });
}
//通过Runloop监听主线程空闲的时刻
static void FGRunLoopObserverCallBack(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info){
    for (FGPopupScheduler *scheduler in FGPopupSchedulers()) {
        if (![scheduler isEmpty]) {
            [scheduler registerFirstPopupViewResponder];
        }
    }
}

//类FGPopupScheduler
/**
 向调度器主动发送一个执行显示弹窗的命令, 支持线程安全
 */
- (void)registerFirstPopupViewResponder{
    if (!self.suspended && self.canRegisterFirstPopupViewResponder) {
        dispatch_async_main_safe(^(){
            [self->_list execute];
        });
    }
}
//类FGPopupScheduler
/**
 返回当前调度器是否拥有已经显示的弹窗, 如果canRegisterFirstPopupViewResponder为true,-registerFirstPopupViewResponder将执行无效
 */
@property (nonatomic, assign, readonly) BOOL canRegisterFirstPopupViewResponder;



//类FGPopupList : NSObject <FGPopupSchedulerStrategyQueue> 
//FGPopupSchedulerStrategyQueue可在第8点查看

/*这里有个疑问,这里传进来的priority并没有存储也没有使用不会影响功能?
答案是不会,因为FGPopupList是基类,其子类FGPopupPriorityList对这个方法进行了重写,
这也对应了其仅在优先级调度策略时生效:FGPopupSchedulerStrategyPriority
*/
- (void)addPopupView:(id<FGPopupView>)view Priority:(FGPopupStrategyPriority)Priority{
    [self monitorRemoveEventWith:view];
}
//类FGPopupList : NSObject <FGPopupSchedulerStrategyQueue>
- (void)execute{
    PopupElement *elemt = [self _hitTestFirstPopupResponder];
    id<FGPopupView> view = elemt.data;
    if (!view) {
        return;
    }
    self.FirstFirstResponderElement = elemt;
    
    if ([view respondsToSelector:@selector(showPopupViewWithAnimation:)]) {
        [view showPopupViewWithAnimation:^{}];
    }
    else if([view respondsToSelector:@selector(showPopupView)]){
        [view showPopupView];
    }else{
        NSAssert(NO, @"You must have to implementation -showPopupViewWithAnimation: or -showPopupView");
    }
}
//类FGPopupList : NSObject <FGPopupSchedulerStrategyQueue>
/*
 进行第一响应者测试并返回对应的节点
 
 @returns 作为第一响应者的节点
 */
- (PopupElement *)_hitTestFirstPopupResponder{
    PopupElement *element;
    for(auto itor=_list.begin(); itor!=_list.end();) {
        PopupElement *temp = *itor;
        id<FGPopupView> data = temp.data;
        __block BOOL canRegisterFirstPopupViewResponder = YES;
        if ([data respondsToSelector:@selector(canRegisterFirstPopupViewResponder)]) {
            canRegisterFirstPopupViewResponder = [data canRegisterFirstPopupViewResponder];
        }
        
        if (canRegisterFirstPopupViewResponder) {
            element = temp;
            break;
        }
        /// 这里只能由为显示的popup所触发
        else if([data respondsToSelector:@selector(popupViewUntriggeredBehavior)] && [data popupViewUntriggeredBehavior] == FGPopupViewUntriggeredBehaviorDiscard){
            itor = _list.erase(itor++);
        }
        else{
            itor++;
        }
    }
    return element;
}

8、FGPopupSchedulerStrategyQueue

@protocol FGPopupSchedulerStrategyQueue <NSObject>

/**
 向当前队列中添加弹窗对象,根据不同的FGPopupSchedulerStrategy,每个subList自己都需要重构的-addPopupView:方法
 
 @param view 弹窗对象
 @param Priority 优先级
 */
- (void)addPopupView:(id<FGPopupView>)view  Priority:(FGPopupStrategyPriority)Priority;

/**
 从当前队列删除指定的弹窗对象,根据不同的FGPopupSchedulerStrategy,每个subList自己都需要重构的-removePopupView:方法
 */
- (void)removePopupView:(id<FGPopupView>)view;

/**
 从当前队列中进行-hitTest,返回对象作为当前的FirstFirstResponder,并执行显示操作
 */
- (void)execute;

/**
 清除当前队列弹窗,
 */
- (void)clear;

/**
 返回当前队列是否存在弹窗
 */
- (BOOL)isEmpty;

/**
 返回是否能注册新的显示弹窗,如果当前已经有显示的弹窗返回NO
 */
- (BOOL)canRegisterFirstFirstPopupViewResponder;
@end
上一篇 下一篇

猜你喜欢

热点阅读