[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
/**
|
|
|
|
* 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 "RCTModuleData.h"
|
|
|
|
|
|
|
|
#import "RCTBridge.h"
|
2015-12-15 13:39:30 +00:00
|
|
|
#import "RCTBridge+Private.h"
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
#import "RCTModuleMethod.h"
|
|
|
|
#import "RCTLog.h"
|
2015-10-29 01:41:49 +00:00
|
|
|
#import "RCTUtils.h"
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
|
|
|
|
@implementation RCTModuleData
|
2015-08-07 13:42:34 +00:00
|
|
|
{
|
2016-02-09 11:29:18 +00:00
|
|
|
NSDictionary<NSString *, id> *_constantsToExport;
|
2015-08-07 23:07:15 +00:00
|
|
|
NSString *_queueName;
|
2015-11-25 11:09:00 +00:00
|
|
|
__weak RCTBridge *_bridge;
|
2016-01-05 17:04:08 +00:00
|
|
|
NSLock *_instanceLock;
|
|
|
|
BOOL _setupComplete;
|
2015-08-07 13:42:34 +00:00
|
|
|
}
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
|
2015-11-03 22:45:46 +00:00
|
|
|
@synthesize methods = _methods;
|
2015-11-25 11:09:00 +00:00
|
|
|
@synthesize instance = _instance;
|
|
|
|
@synthesize methodQueue = _methodQueue;
|
2015-11-03 22:45:46 +00:00
|
|
|
|
2015-11-25 11:09:00 +00:00
|
|
|
- (instancetype)initWithModuleClass:(Class)moduleClass
|
|
|
|
bridge:(RCTBridge *)bridge
|
|
|
|
{
|
|
|
|
if ((self = [super init])) {
|
|
|
|
_moduleClass = moduleClass;
|
|
|
|
_bridge = bridge;
|
2016-01-05 17:04:08 +00:00
|
|
|
|
|
|
|
_implementsBatchDidComplete = [_moduleClass instancesRespondToSelector:@selector(batchDidComplete)];
|
|
|
|
_implementsPartialBatchDidFlush = [_moduleClass instancesRespondToSelector:@selector(partialBatchDidFlush)];
|
|
|
|
|
|
|
|
_instanceLock = [NSLock new];
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)initWithModuleInstance:(id<RCTBridgeModule>)instance
|
2016-01-05 17:04:08 +00:00
|
|
|
bridge:(RCTBridge *)bridge
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
{
|
2016-01-05 17:04:08 +00:00
|
|
|
if ((self = [self initWithModuleClass:[instance class] bridge:bridge])) {
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
_instance = instance;
|
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2015-08-24 10:14:33 +00:00
|
|
|
RCT_NOT_IMPLEMENTED(- (instancetype)init);
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
|
2016-01-05 17:04:08 +00:00
|
|
|
#pragma mark - private setup methods
|
|
|
|
|
|
|
|
- (void)setBridgeForInstance
|
2015-11-25 11:09:00 +00:00
|
|
|
{
|
2016-01-05 17:04:08 +00:00
|
|
|
RCTAssert(_instance, @"setBridgeForInstance called before %@ initialized", self.name);
|
2016-01-07 18:15:45 +00:00
|
|
|
if ([_instance respondsToSelector:@selector(bridge)] && _instance.bridge != _bridge) {
|
2016-01-05 17:04:08 +00:00
|
|
|
@try {
|
|
|
|
[(id)_instance setValue:_bridge forKey:@"bridge"];
|
|
|
|
}
|
|
|
|
@catch (NSException *exception) {
|
|
|
|
RCTLogError(@"%@ has no setter or ivar for its bridge, which is not "
|
|
|
|
"permitted. You must either @synthesize the bridge property, "
|
|
|
|
"or provide your own setter method.", self.name);
|
|
|
|
}
|
|
|
|
}
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
|
|
|
|
2016-01-07 09:29:43 +00:00
|
|
|
- (void)finishSetupForInstance
|
|
|
|
{
|
|
|
|
if (!_setupComplete) {
|
|
|
|
_setupComplete = YES;
|
|
|
|
[self setUpMethodQueue];
|
|
|
|
[_bridge registerModuleForFrameUpdates:_instance withModuleData:self];
|
|
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTDidInitializeModuleNotification
|
|
|
|
object:_bridge
|
|
|
|
userInfo:@{@"module": _instance}];
|
2016-02-09 11:29:18 +00:00
|
|
|
|
|
|
|
if (RCTClassOverridesInstanceMethod(_moduleClass, @selector(constantsToExport))) {
|
|
|
|
RCTAssertMainThread();
|
|
|
|
_constantsToExport = [_instance constantsToExport];
|
|
|
|
}
|
2016-01-07 09:29:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-04 14:23:51 +00:00
|
|
|
- (void)setUpMethodQueue
|
2015-11-25 11:09:00 +00:00
|
|
|
{
|
2016-01-04 14:23:51 +00:00
|
|
|
if (!_methodQueue) {
|
2016-01-05 17:04:08 +00:00
|
|
|
RCTAssert(_instance, @"setUpMethodQueue called before %@ initialized", self.name);
|
|
|
|
BOOL implementsMethodQueue = [_instance respondsToSelector:@selector(methodQueue)];
|
2016-01-04 14:23:51 +00:00
|
|
|
if (implementsMethodQueue) {
|
2016-01-05 17:04:08 +00:00
|
|
|
_methodQueue = _instance.methodQueue;
|
|
|
|
}
|
|
|
|
if (!_methodQueue) {
|
|
|
|
|
|
|
|
// Create new queue (store queueName, as it isn't retained by dispatch_queue)
|
|
|
|
_queueName = [NSString stringWithFormat:@"com.facebook.React.%@Queue", self.name];
|
|
|
|
_methodQueue = dispatch_queue_create(_queueName.UTF8String, DISPATCH_QUEUE_SERIAL);
|
|
|
|
|
|
|
|
// assign it to the module
|
|
|
|
if (implementsMethodQueue) {
|
|
|
|
@try {
|
|
|
|
[(id)_instance setValue:_methodQueue forKey:@"methodQueue"];
|
|
|
|
}
|
|
|
|
@catch (NSException *exception) {
|
2016-01-07 09:29:43 +00:00
|
|
|
RCTLogError(@"%@ is returning nil for its methodQueue, which is not "
|
2016-01-05 17:04:08 +00:00
|
|
|
"permitted. You must either return a pre-initialized "
|
|
|
|
"queue, or @synthesize the methodQueue to let the bridge "
|
|
|
|
"create a queue for you.", self.name);
|
|
|
|
}
|
2016-01-04 14:23:51 +00:00
|
|
|
}
|
|
|
|
}
|
2016-01-05 17:04:08 +00:00
|
|
|
}
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
|
|
|
|
2016-01-05 17:04:08 +00:00
|
|
|
#pragma mark - public getters
|
2016-01-04 14:23:51 +00:00
|
|
|
|
|
|
|
- (BOOL)hasInstance
|
|
|
|
{
|
|
|
|
return _instance != nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (id<RCTBridgeModule>)instance
|
|
|
|
{
|
2016-01-05 17:04:08 +00:00
|
|
|
[_instanceLock lock];
|
|
|
|
if (!_setupComplete) {
|
|
|
|
if (!_instance) {
|
|
|
|
_instance = [_moduleClass new];
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
2016-01-05 17:04:08 +00:00
|
|
|
// Bridge must be set before methodQueue is set up, as methodQueue
|
|
|
|
// initialization requires it (View Managers get their queue by calling
|
|
|
|
// self.bridge.uiManager.methodQueue)
|
|
|
|
[self setBridgeForInstance];
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
2016-01-05 17:04:08 +00:00
|
|
|
[_instanceLock unlock];
|
2016-01-07 09:29:43 +00:00
|
|
|
|
|
|
|
[self finishSetupForInstance];
|
|
|
|
|
2016-01-05 17:04:08 +00:00
|
|
|
return _instance;
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *)name
|
|
|
|
{
|
|
|
|
return RCTBridgeModuleNameForClass(_moduleClass);
|
|
|
|
}
|
|
|
|
|
2015-11-03 22:45:46 +00:00
|
|
|
- (NSArray<id<RCTBridgeMethod>> *)methods
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
{
|
2015-08-07 13:42:34 +00:00
|
|
|
if (!_methods) {
|
2015-11-03 22:45:46 +00:00
|
|
|
NSMutableArray<id<RCTBridgeMethod>> *moduleMethods = [NSMutableArray new];
|
2015-09-18 22:01:21 +00:00
|
|
|
|
2016-01-05 17:04:08 +00:00
|
|
|
if ([_moduleClass instancesRespondToSelector:@selector(methodsToExport)]) {
|
|
|
|
[self instance];
|
2015-09-18 22:01:21 +00:00
|
|
|
[moduleMethods addObjectsFromArray:[_instance methodsToExport]];
|
|
|
|
}
|
|
|
|
|
2015-08-07 13:42:34 +00:00
|
|
|
unsigned int methodCount;
|
|
|
|
Method *methods = class_copyMethodList(object_getClass(_moduleClass), &methodCount);
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < methodCount; i++) {
|
|
|
|
Method method = methods[i];
|
|
|
|
SEL selector = method_getName(method);
|
|
|
|
if ([NSStringFromSelector(selector) hasPrefix:@"__rct_export__"]) {
|
|
|
|
IMP imp = method_getImplementation(method);
|
2015-11-03 22:45:46 +00:00
|
|
|
NSArray<NSString *> *entries =
|
|
|
|
((NSArray<NSString *> *(*)(id, SEL))imp)(_moduleClass, selector);
|
2015-09-18 22:01:21 +00:00
|
|
|
id<RCTBridgeMethod> moduleMethod =
|
2015-12-10 18:09:04 +00:00
|
|
|
[[RCTModuleMethod alloc] initWithMethodSignature:entries[1]
|
|
|
|
JSMethodName:entries[0]
|
|
|
|
moduleClass:_moduleClass];
|
2015-08-07 13:42:34 +00:00
|
|
|
|
|
|
|
[moduleMethods addObject:moduleMethod];
|
|
|
|
}
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
}
|
|
|
|
|
2015-08-07 13:42:34 +00:00
|
|
|
free(methods);
|
|
|
|
|
|
|
|
_methods = [moduleMethods copy];
|
|
|
|
}
|
|
|
|
return _methods;
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
}
|
|
|
|
|
2015-10-29 01:41:49 +00:00
|
|
|
- (NSArray *)config
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
{
|
2016-02-09 11:29:18 +00:00
|
|
|
__block NSDictionary<NSString *, id> *constants = _constantsToExport;
|
|
|
|
_constantsToExport = nil; // Not needed any more
|
2015-11-25 11:09:00 +00:00
|
|
|
|
|
|
|
if (constants.count == 0 && self.methods.count == 0) {
|
2015-10-29 01:41:49 +00:00
|
|
|
return (id)kCFNull; // Nothing to export
|
2015-10-22 12:51:21 +00:00
|
|
|
}
|
|
|
|
|
2015-11-03 22:45:46 +00:00
|
|
|
NSMutableArray<NSString *> *methods = self.methods.count ? [NSMutableArray new] : nil;
|
|
|
|
NSMutableArray<NSNumber *> *asyncMethods = nil;
|
2015-10-29 01:41:49 +00:00
|
|
|
for (id<RCTBridgeMethod> method in self.methods) {
|
2015-10-28 15:26:50 +00:00
|
|
|
if (method.functionType == RCTFunctionTypePromise) {
|
2015-10-29 01:41:49 +00:00
|
|
|
if (!asyncMethods) {
|
|
|
|
asyncMethods = [NSMutableArray new];
|
|
|
|
}
|
|
|
|
[asyncMethods addObject:@(methods.count)];
|
2015-10-23 17:14:26 +00:00
|
|
|
}
|
2015-10-29 22:35:15 +00:00
|
|
|
[methods addObject:method.JSMethodName];
|
2015-10-23 17:14:26 +00:00
|
|
|
}
|
2015-10-28 15:26:50 +00:00
|
|
|
|
2015-10-29 01:41:49 +00:00
|
|
|
NSMutableArray *config = [NSMutableArray new];
|
2015-11-25 11:09:00 +00:00
|
|
|
[config addObject:self.name];
|
|
|
|
if (constants.count) {
|
|
|
|
[config addObject:constants];
|
2015-10-29 01:41:49 +00:00
|
|
|
}
|
|
|
|
if (methods) {
|
|
|
|
[config addObject:methods];
|
|
|
|
if (asyncMethods) {
|
|
|
|
[config addObject:asyncMethods];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return config;
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
}
|
|
|
|
|
2015-11-25 11:09:00 +00:00
|
|
|
- (dispatch_queue_t)methodQueue
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
{
|
2016-01-05 17:04:08 +00:00
|
|
|
[self instance];
|
2015-11-25 11:09:00 +00:00
|
|
|
return _methodQueue;
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
}
|
|
|
|
|
2015-11-25 11:09:00 +00:00
|
|
|
- (void)invalidate
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
{
|
2015-11-25 11:09:00 +00:00
|
|
|
_methodQueue = nil;
|
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 23:34:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|