UIButton使用RACCommand后setEnable方法失效的问题

问题描述

首先,我们知道RACCommand有如下初始化方法,可以传入enabledSignal,用于控制按钮的enable状态。

- (id)initWithEnabled:(RACSignal *)enabledSignal signalBlock:(RACSignal * (^)(id input))signalBlock;

但有时我们想自己控制按钮的enable状态。这个时候就可能出现标题所说的问题。标题为了表达意思,并不准确。真正的问题是RACCommand内部调用setEnable方法,可能会覆盖你手动设置的UIButton状态。无论你在初始化RACommand时是否传入enabledSignal,RACCommand内部都会控制UIButton的enable/disable状态。我们在外部也调用setEnable方法设置enable状态,可能会冲突。


举个栗子

RACCommand *sendCodeCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
    [sendCodeButton setEnabled:NO];
    //TODO: Send code, count down and then setEnabled:YES
    return [RACSignal empty];
}];
sendCodeButton.rac_command = sendCodeCommand;

上面这段代码,我们的预期是点击发送验证码按钮之后disable掉按钮,进行倒计时,倒计时结束重新enable按钮。事实上,我们点击按钮之后,按钮并没有被disable掉。


简单解决方案

放弃手动控制按钮的enable状态,在初始化RACCommand时候传入enabledSignal,让RACCommand统一负责。不过使用Signal控制enable状态需要额外的转换操作,要费些事。


RACCommand内部是如何控制UIButton enable状态的

如果我们知道RACCommand内部是如何控制enable状态的,我们就可以避免冲突。
我们在UIButton+RACCommandSupport.m可以找到下面这条语句:

disposable = [command.enabled setKeyPath:@keypath(self.enabled) onObject:self];

这条语句将UIButton的enable属性与RACCommand中的enabled信号绑定起来了,那么我们就来看看RACCommand中的enabled信号的如何创建的,如何变化的。
我们在RACCommand.m中找到下面这段代码:

    if (enabledSignal == nil) {
        enabledSignal = [RACSignal return:@YES];
    } else {
        enabledSignal = [[[enabledSignal
            startWith:@YES]
            takeUntil:self.rac_willDeallocSignal]
            replayLast];
    }
    _immediateEnabled = [[RACSignal
        combineLatest:@[ enabledSignal, moreExecutionsAllowed ]]
        and];

    _enabled = [[[[[self.immediateEnabled
        take:1]
        concat:[[self.immediateEnabled skip:1] deliverOn:RACScheduler.mainThreadScheduler]]
        distinctUntilChanged]
        replayLast]
        setNameWithFormat:@"%@ -enabled", self];

可以看到enabled信号受我们传入的enabled信号(不传,默认为[RACSignal return:@YES])和内部的moreExecutionsAllowed信号影响。它们进行与操作。再看看moreExecutionsAllowed信号的定义。

    RACSignal *immediateExecuting = [RACObserve(self, activeExecutionSignals) map:^(NSArray *activeSignals) {
        return @(activeSignals.count > 0);
    }];

    RACSignal *moreExecutionsAllowed = [RACSignal
        if:RACObserve(self, allowsConcurrentExecution)
        then:[RACSignal return:@YES]
        else:[immediateExecuting not]];

moreExecutionsAllowed信号受allowsConcurrentExecution属性和immediateExecuting信号影响。如果允许并发,moreExecutionsAllowed发出的是@YES;如果不允许并发,目前有未完成任务,发出@NO,否则发出@YES。这里的任务完成是指signalBlock创建出来的Signal已经complete了,signalBlock是在- (id)initWithEnabled:(RACSignal *)enabledSignal signalBlock:(RACSignal * (^)(id input))signalBlock方法中传入的。

回到我们上面举的栗子,在signalBlock中我们disable了sendCodeButton,但是返回了[RACSignal empty][RACSignal empty]会立即complete,activeExecutionSignals会remove掉这个signal,然后activeSignals.count变为0,immediateExecuting发出@NO,moreExecutionsAllowed发出@YES,最终enabled信号发出@YES,sendCodeButton被enable。我们刚才设置的状态被冲掉了。

猜你喜欢

转载自blog.csdn.net/fly1183989782/article/details/52315029