mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-08-07 16:11:13 +00:00
228 lines
5.1 KiB
Objective-C
228 lines
5.1 KiB
Objective-C
#import "SSignal+Meta.h"
|
|
|
|
#import "SDisposableSet.h"
|
|
#import "SMetaDisposable.h"
|
|
#import "SSignal+Mapping.h"
|
|
#import "SAtomic.h"
|
|
|
|
#import <libkern/OSAtomic.h>
|
|
|
|
@interface SSignalQueueState : NSObject <SDisposable>
|
|
{
|
|
OSSpinLock _lock;
|
|
bool _executingSignal;
|
|
bool _terminated;
|
|
|
|
id<SDisposable> _disposable;
|
|
SMetaDisposable *_currentDisposable;
|
|
SSubscriber *_subscriber;
|
|
|
|
NSMutableArray *_queuedSignals;
|
|
bool _queueMode;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation SSignalQueueState
|
|
|
|
- (instancetype)initWithSubscriber:(SSubscriber *)subscriber queueMode:(bool)queueMode
|
|
{
|
|
self = [super init];
|
|
if (self != nil)
|
|
{
|
|
_subscriber = subscriber;
|
|
_currentDisposable = [[SMetaDisposable alloc] init];
|
|
_queuedSignals = queueMode ? [[NSMutableArray alloc] init] : nil;
|
|
_queueMode = queueMode;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)beginWithDisposable:(id<SDisposable>)disposable
|
|
{
|
|
_disposable = disposable;
|
|
}
|
|
|
|
- (void)enqueueSignal:(SSignal *)signal
|
|
{
|
|
bool startSignal = false;
|
|
OSSpinLockLock(&_lock);
|
|
if (_queueMode && _executingSignal)
|
|
{
|
|
[_queuedSignals addObject:signal];
|
|
}
|
|
else
|
|
{
|
|
_executingSignal = true;
|
|
startSignal = true;
|
|
}
|
|
OSSpinLockUnlock(&_lock);
|
|
|
|
if (startSignal)
|
|
{
|
|
id<SDisposable> disposable = [signal startWithNext:^(id next)
|
|
{
|
|
[_subscriber putNext:next];
|
|
} error:^(id error)
|
|
{
|
|
[_subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[self headCompleted];
|
|
}];
|
|
|
|
[_currentDisposable setDisposable:disposable];
|
|
}
|
|
}
|
|
|
|
- (void)headCompleted
|
|
{
|
|
SSignal *nextSignal = nil;
|
|
|
|
bool terminated = false;
|
|
OSSpinLockLock(&_lock);
|
|
_executingSignal = false;
|
|
|
|
if (_queueMode)
|
|
{
|
|
if (_queuedSignals.count != 0)
|
|
{
|
|
nextSignal = _queuedSignals[0];
|
|
[_queuedSignals removeObjectAtIndex:0];
|
|
_executingSignal = true;
|
|
}
|
|
else
|
|
terminated = _terminated;
|
|
}
|
|
else
|
|
terminated = _terminated;
|
|
OSSpinLockUnlock(&_lock);
|
|
|
|
if (terminated)
|
|
[_subscriber putCompletion];
|
|
else if (nextSignal != nil)
|
|
{
|
|
id<SDisposable> disposable = [nextSignal startWithNext:^(id next)
|
|
{
|
|
[_subscriber putNext:next];
|
|
} error:^(id error)
|
|
{
|
|
[_subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[self headCompleted];
|
|
}];
|
|
|
|
[_currentDisposable setDisposable:disposable];
|
|
}
|
|
}
|
|
|
|
- (void)beginCompletion
|
|
{
|
|
bool executingSignal = false;
|
|
OSSpinLockLock(&_lock);
|
|
executingSignal = _executingSignal;
|
|
_terminated = true;
|
|
OSSpinLockUnlock(&_lock);
|
|
|
|
if (!executingSignal)
|
|
[_subscriber putCompletion];
|
|
}
|
|
|
|
- (void)dispose
|
|
{
|
|
[_currentDisposable dispose];
|
|
[_disposable dispose];
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation SSignal (Meta)
|
|
|
|
- (SSignal *)switchToLatest
|
|
{
|
|
return [[SSignal alloc] initWithGenerator:^id<SDisposable> (SSubscriber *subscriber)
|
|
{
|
|
SSignalQueueState *state = [[SSignalQueueState alloc] initWithSubscriber:subscriber queueMode:false];
|
|
|
|
[state beginWithDisposable:[self startWithNext:^(id next)
|
|
{
|
|
[state enqueueSignal:next];
|
|
} error:^(id error)
|
|
{
|
|
[subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[state beginCompletion];
|
|
}]];
|
|
|
|
return state;
|
|
}];
|
|
}
|
|
|
|
- (SSignal *)mapToSignal:(SSignal *(^)(id))f
|
|
{
|
|
return [[self map:f] switchToLatest];
|
|
}
|
|
|
|
- (SSignal *)mapToQueue:(SSignal *(^)(id))f
|
|
{
|
|
return [[self map:f] queue];
|
|
}
|
|
|
|
- (SSignal *)then:(SSignal *)signal
|
|
{
|
|
SDisposableSet *compositeDisposable = [[SDisposableSet alloc] init];
|
|
|
|
SMetaDisposable *currentDisposable = [[SMetaDisposable alloc] init];
|
|
[compositeDisposable add:currentDisposable];
|
|
|
|
return [[SSignal alloc] initWithGenerator:^(SSubscriber *subscriber)
|
|
{
|
|
[currentDisposable setDisposable:[self startWithNext:^(id next)
|
|
{
|
|
[subscriber putNext:next];
|
|
} error:^(id error)
|
|
{
|
|
[subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[compositeDisposable add:[signal startWithNext:^(id next)
|
|
{
|
|
[subscriber putNext:next];
|
|
} error:^(id error)
|
|
{
|
|
[subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[subscriber putCompletion];
|
|
}]];
|
|
}]];
|
|
|
|
return compositeDisposable;
|
|
}];
|
|
}
|
|
|
|
- (SSignal *)queue
|
|
{
|
|
return [[SSignal alloc] initWithGenerator:^id<SDisposable> (SSubscriber *subscriber)
|
|
{
|
|
SSignalQueueState *state = [[SSignalQueueState alloc] initWithSubscriber:subscriber queueMode:true];
|
|
|
|
[state beginWithDisposable:[self startWithNext:^(id next)
|
|
{
|
|
[state enqueueSignal:next];
|
|
} error:^(id error)
|
|
{
|
|
[subscriber putError:error];
|
|
} completed:^
|
|
{
|
|
[state beginCompletion];
|
|
}]];
|
|
|
|
return state;
|
|
}];
|
|
}
|
|
|
|
@end
|