mirror of
https://github.com/status-im/react-native.git
synced 2025-01-10 09:35:48 +00:00
7b718b03eb
Summary: By default we run the the JS display link, even if there are no modules listening. Given that most listeners will be lazily constructed, let's make it paused by default. Since RCTTiming almost never unpauses due to some long-lived timers, implement a sleep timer that pauses the displaylink but uses an NSTimer to wake up in time. Reviewed By: mhorowitz Differential Revision: D3235044 fbshipit-source-id: 4a340fea552ada1bd8bc0d83b596a7df6f992387
278 lines
7.6 KiB
Objective-C
278 lines
7.6 KiB
Objective-C
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "RCTTiming.h"
|
|
|
|
#import "RCTAssert.h"
|
|
#import "RCTBridge.h"
|
|
#import "RCTBridge+Private.h"
|
|
#import "RCTLog.h"
|
|
#import "RCTUtils.h"
|
|
|
|
static const NSTimeInterval kMinimumSleepInterval = 1;
|
|
|
|
@interface _RCTTimer : NSObject
|
|
|
|
@property (nonatomic, strong, readonly) NSDate *target;
|
|
@property (nonatomic, assign, readonly) BOOL repeats;
|
|
@property (nonatomic, copy, readonly) NSNumber *callbackID;
|
|
@property (nonatomic, assign, readonly) NSTimeInterval interval;
|
|
|
|
@end
|
|
|
|
@implementation _RCTTimer
|
|
|
|
- (instancetype)initWithCallbackID:(NSNumber *)callbackID
|
|
interval:(NSTimeInterval)interval
|
|
targetTime:(NSTimeInterval)targetTime
|
|
repeats:(BOOL)repeats
|
|
{
|
|
if ((self = [super init])) {
|
|
_interval = interval;
|
|
_repeats = repeats;
|
|
_callbackID = callbackID;
|
|
_target = [NSDate dateWithTimeIntervalSinceNow:targetTime];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
/**
|
|
* Returns `YES` if we should invoke the JS callback.
|
|
*/
|
|
- (BOOL)updateFoundNeedsJSUpdate
|
|
{
|
|
if (_target && _target.timeIntervalSinceNow <= 0) {
|
|
// The JS Timers will do fine grained calculating of expired timeouts.
|
|
_target = _repeats ? [NSDate dateWithTimeIntervalSinceNow:_interval] : nil;
|
|
return YES;
|
|
}
|
|
return NO;
|
|
}
|
|
|
|
@end
|
|
|
|
@interface _RCTTimingProxy : NSObject
|
|
|
|
@end
|
|
|
|
// NSTimer retains its target, insert this class to break potential retain cycles
|
|
@implementation _RCTTimingProxy
|
|
{
|
|
__weak id _target;
|
|
}
|
|
|
|
+ (instancetype)proxyWithTarget:(id)target
|
|
{
|
|
_RCTTimingProxy *proxy = [self new];
|
|
if (proxy) {
|
|
proxy->_target = target;
|
|
}
|
|
return proxy;
|
|
}
|
|
|
|
- (void)timerDidFire
|
|
{
|
|
[_target timerDidFire];
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RCTTiming
|
|
{
|
|
NSMutableDictionary<NSNumber *, _RCTTimer *> *_timers;
|
|
NSTimer *_sleepTimer;
|
|
}
|
|
|
|
@synthesize bridge = _bridge;
|
|
@synthesize paused = _paused;
|
|
@synthesize pauseCallback = _pauseCallback;
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (void)setBridge:(RCTBridge *)bridge
|
|
{
|
|
RCTAssert(!_bridge, @"Should never be initialized twice!");
|
|
|
|
_paused = YES;
|
|
_timers = [NSMutableDictionary new];
|
|
|
|
for (NSString *name in @[UIApplicationWillResignActiveNotification,
|
|
UIApplicationDidEnterBackgroundNotification,
|
|
UIApplicationWillTerminateNotification]) {
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(stopTimers)
|
|
name:name
|
|
object:nil];
|
|
}
|
|
|
|
for (NSString *name in @[UIApplicationDidBecomeActiveNotification,
|
|
UIApplicationWillEnterForegroundNotification]) {
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(startTimers)
|
|
name:name
|
|
object:nil];
|
|
}
|
|
|
|
_bridge = bridge;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[_sleepTimer invalidate];
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
}
|
|
|
|
- (dispatch_queue_t)methodQueue
|
|
{
|
|
return RCTJSThread;
|
|
}
|
|
|
|
- (void)invalidate
|
|
{
|
|
[self stopTimers];
|
|
_bridge = nil;
|
|
}
|
|
|
|
- (void)stopTimers
|
|
{
|
|
if (!_paused) {
|
|
_paused = YES;
|
|
if (_pauseCallback) {
|
|
_pauseCallback();
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)startTimers
|
|
{
|
|
if (!_bridge || _timers.count == 0) {
|
|
return;
|
|
}
|
|
|
|
if (_paused) {
|
|
_paused = NO;
|
|
if (_pauseCallback) {
|
|
_pauseCallback();
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)didUpdateFrame:(__unused RCTFrameUpdate *)update
|
|
{
|
|
NSDate *nextScheduledTarget = [NSDate distantFuture];
|
|
NSMutableArray<NSNumber *> *timersToCall = [NSMutableArray new];
|
|
for (_RCTTimer *timer in _timers.allValues) {
|
|
NSDate *target = timer.target;
|
|
if ([timer updateFoundNeedsJSUpdate]) {
|
|
[timersToCall addObject:timer.callbackID];
|
|
}
|
|
if (!timer.target) {
|
|
[_timers removeObjectForKey:timer.callbackID];
|
|
} else {
|
|
nextScheduledTarget = [nextScheduledTarget earlierDate:timer.target];
|
|
}
|
|
}
|
|
|
|
// Call timers that need to be called
|
|
if (timersToCall.count > 0) {
|
|
[_bridge enqueueJSCall:@"JSTimersExecution.callTimers" args:@[timersToCall]];
|
|
|
|
// If we call at least one timer this frame, don't switch to a paused state yet, so if
|
|
// in response to this timer another timer is scheduled, we don't pause and unpause
|
|
// the displaylink frivolously.
|
|
return;
|
|
}
|
|
|
|
// No need to call the pauseCallback as RCTDisplayLink will ask us about our paused
|
|
// status immediately after completing this call
|
|
if (_timers.count == 0) {
|
|
_paused = YES;
|
|
}
|
|
// If the next timer is more than 1 second out, pause and schedule an NSTimer;
|
|
else if ([nextScheduledTarget timeIntervalSinceNow] > kMinimumSleepInterval) {
|
|
[self scheduleSleepTimer:nextScheduledTarget];
|
|
_paused = YES;
|
|
}
|
|
}
|
|
|
|
- (void)scheduleSleepTimer:(NSDate *)sleepTarget
|
|
{
|
|
if (!_sleepTimer || !_sleepTimer.valid) {
|
|
_sleepTimer = [[NSTimer alloc] initWithFireDate:sleepTarget
|
|
interval:0
|
|
target:[_RCTTimingProxy proxyWithTarget:self]
|
|
selector:@selector(timerDidFire)
|
|
userInfo:nil
|
|
repeats:NO];
|
|
[[NSRunLoop currentRunLoop] addTimer:_sleepTimer forMode:NSDefaultRunLoopMode];
|
|
} else {
|
|
_sleepTimer.fireDate = [_sleepTimer.fireDate earlierDate:sleepTarget];
|
|
}
|
|
}
|
|
|
|
- (void)timerDidFire
|
|
{
|
|
_sleepTimer = nil;
|
|
if (_paused) {
|
|
[self startTimers];
|
|
|
|
// Immediately dispatch frame, so we don't have to wait on the displaylink.
|
|
[self didUpdateFrame:nil];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* There's a small difference between the time when we call
|
|
* setTimeout/setInterval/requestAnimation frame and the time it actually makes
|
|
* it here. This is important and needs to be taken into account when
|
|
* calculating the timer's target time. We calculate this by passing in
|
|
* Date.now() from JS and then subtracting that from the current time here.
|
|
*/
|
|
RCT_EXPORT_METHOD(createTimer:(nonnull NSNumber *)callbackID
|
|
duration:(NSTimeInterval)jsDuration
|
|
jsSchedulingTime:(NSDate *)jsSchedulingTime
|
|
repeats:(BOOL)repeats)
|
|
{
|
|
if (jsDuration == 0 && repeats == NO) {
|
|
// For super fast, one-off timers, just enqueue them immediately rather than waiting a frame.
|
|
[_bridge _immediatelyCallTimer:callbackID];
|
|
return;
|
|
}
|
|
|
|
NSTimeInterval jsSchedulingOverhead = MAX(-jsSchedulingTime.timeIntervalSinceNow, 0);
|
|
|
|
NSTimeInterval targetTime = jsDuration - jsSchedulingOverhead;
|
|
if (jsDuration < 0.018) { // Make sure short intervals run each frame
|
|
jsDuration = 0;
|
|
}
|
|
|
|
_RCTTimer *timer = [[_RCTTimer alloc] initWithCallbackID:callbackID
|
|
interval:jsDuration
|
|
targetTime:targetTime
|
|
repeats:repeats];
|
|
_timers[callbackID] = timer;
|
|
if (_paused) {
|
|
if ([timer.target timeIntervalSinceNow] > kMinimumSleepInterval) {
|
|
[self scheduleSleepTimer:timer.target];
|
|
} else {
|
|
[self startTimers];
|
|
}
|
|
}
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(deleteTimer:(nonnull NSNumber *)timerID)
|
|
{
|
|
[_timers removeObjectForKey:timerID];
|
|
if (_timers.count == 0) {
|
|
[self stopTimers];
|
|
}
|
|
}
|
|
|
|
@end
|