react-native/React/CxxModule/RCTCxxModule.mm
Pieter De Baets d42ccca2e1 Allow native modules to customize the main-thread init behaviour
Summary:
We've simplified a lot of the conditions for eager of the module init so now we can introduce a final switch to allow modules to opt-out (and in the future opt-in if they still require the behaviour).

We now require you to be explicit about the intended behaviour and implement the `+ (BOOL)requiresMainQueueSetup` method on your module. When you return YES from this method, it tells the bridge the module needs to be created on the main thread (and to avoid deadlocks, we do so eagerly during bridge startup). When you return NO, the native module will be initialised when it's first accessed from JS.

The current behaviour is maintained but a warning is emitted until the new API is adopted.

Reviewed By: fkgozali

Differential Revision: D5527788

fbshipit-source-id: 56d38f81e58cf950547b9780e89bfac4667eeaaa
2017-08-07 07:04:36 -07:00

84 lines
1.9 KiB
Plaintext

/**
* 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 "RCTCxxModule.h"
#import <React/RCTBridge.h>
#import <React/RCTFollyConvert.h>
#import <React/RCTLog.h>
#import <cxxreact/CxxModule.h>
#import "RCTCxxMethod.h"
using namespace facebook::react;
@implementation RCTCxxModule
{
std::unique_ptr<facebook::xplat::module::CxxModule> _module;
}
+ (NSString *)moduleName
{
return @"";
}
+ (BOOL)requiresMainQueueSetup
{
return NO;
}
- (void)lazyInit
{
if (!_module) {
_module = [self createModule];
if (_module) {
RCTAssert([RCTBridgeModuleNameForClass([self class]) isEqualToString:@(_module->getName().c_str())],
@"CxxModule class name %@ does not match runtime name %s",
RCTBridgeModuleNameForClass([self class]), _module->getName().c_str());
}
}
}
- (std::unique_ptr<facebook::xplat::module::CxxModule>)createModule
{
RCTAssert(NO, @"Subclass %@ must override createModule", [self class]);
return nullptr;
}
- (NSArray<id<RCTBridgeMethod>> *)methodsToExport;
{
[self lazyInit];
if (!_module) {
return nil;
}
NSMutableArray *moduleMethods = [NSMutableArray new];
for (const auto &method : _module->getMethods()) {
[moduleMethods addObject:[[RCTCxxMethod alloc] initWithCxxMethod:method]];
}
return moduleMethods;
}
- (NSDictionary<NSString *, id> *)constantsToExport;
{
[self lazyInit];
if (!_module) {
return nil;
}
NSMutableDictionary *moduleConstants = [NSMutableDictionary new];
for (const auto &c : _module->getConstants()) {
moduleConstants[@(c.first.c_str())] = convertFollyDynamicToId(c.second);
}
return moduleConstants;
}
@end