This repository has been archived on 2024-07-22. You can view files and clone it, but cannot push or open issues or pull requests.
Doric/doric-iOS/Pod/Classes/Extension/DoricBridgeExtension.m

121 lines
5.1 KiB
Mathematica
Raw Normal View History

2019-10-21 09:59:22 +08:00
/*
* Copyright [2019] [Doric.Pub]
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2019-07-29 20:51:53 +08:00
//
// DoricBridgeExtension.m
// Doric
//
// Created by pengfei.zhou on 2019/7/29.
//
#import "DoricBridgeExtension.h"
#import "DoricRegistry.h"
#import "DoricContextManager.h"
#import "DoricNativePlugin.h"
2019-07-30 11:25:05 +08:00
#import "DoricUtil.h"
#import <JavaScriptCore/JavaScriptCore.h>
#import <objc/runtime.h>
2019-07-29 20:51:53 +08:00
@implementation DoricBridgeExtension
- (id)callNativeWithContextId:(NSString *)contextId module:(NSString *)module method:(NSString *)method callbackId:(NSString *)callbackId argument:(id)argument {
DoricContext *context = [[DoricContextManager instance] getContext:contextId];
DoricRegistry *registry = context.driver.registry;
Class pluginClass = [registry acquireNativePlugin:module];
2019-10-21 09:59:22 +08:00
DoricNativePlugin *nativePlugin = context.pluginInstanceMap[module];
2019-10-12 14:48:19 +08:00
if (nativePlugin == nil) {
2019-10-21 09:59:22 +08:00
nativePlugin = [(DoricNativePlugin *) [pluginClass alloc] initWithContext:context];
context.pluginInstanceMap[module] = nativePlugin;
2019-07-29 20:51:53 +08:00
}
return [self findClass:pluginClass target:nativePlugin context:context method:method callbackId:callbackId argument:argument];
}
- (id)createParamWithMethodName:(NSString *)method context:(DoricContext *)context callbackId:(NSString *)callbackId argument:(id)argument {
if ([method isEqualToString:@"withPromise"]) {
return [[DoricPromise alloc] initWithContext:context callbackId:callbackId];
}
return argument;
}
- (id)findClass:(Class)clz target:(id)target context:(DoricContext *)context method:(NSString *)name callbackId:(NSString *)callbackId argument:(id)argument {
2019-07-30 11:25:05 +08:00
unsigned int count;
id ret = nil;
Method *methods = class_copyMethodList(clz, &count);
BOOL isFound = NO;
2019-10-12 14:48:19 +08:00
for (int i = 0; i < count; i++) {
2019-07-30 11:25:05 +08:00
NSString *methodName = [NSString stringWithCString:sel_getName(method_getName(methods[i])) encoding:NSUTF8StringEncoding];
NSArray *array = [methodName componentsSeparatedByString:@":"];
2019-10-12 14:48:19 +08:00
if (array && [array count] > 0) {
if ([array[0] isEqualToString:name]) {
isFound = YES;
2019-07-30 11:25:05 +08:00
SEL selector = NSSelectorFromString(methodName);
NSMethodSignature *methodSignature = [target methodSignatureForSelector:selector];
2019-07-30 11:25:05 +08:00
if (methodSignature) {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSignature];
invocation.selector = selector;
invocation.target = target;
2019-10-12 14:48:19 +08:00
__weak __typeof__(self) _self = self;
2019-11-18 16:33:23 +08:00
dispatch_block_t block = ^() {
2019-10-12 14:48:19 +08:00
__strong __typeof__(_self) self = _self;
2019-07-30 11:25:05 +08:00
@try {
2019-11-18 11:45:48 +08:00
for (NSUInteger idx = 2; idx < methodSignature.numberOfArguments; idx++) {
if (idx - 2 > [array count]) {
2019-07-30 11:25:05 +08:00
break;
}
2019-11-18 11:45:48 +08:00
id args = [self createParamWithMethodName:array[idx - 2] context:context callbackId:callbackId argument:argument];
[invocation setArgument:&args atIndex:idx];
2019-07-30 11:25:05 +08:00
}
[invocation invoke];
} @catch (NSException *exception) {
DoricLog(@"CallNative Error:%@", exception.reason);
}
};
2019-10-12 14:48:19 +08:00
2019-07-30 11:25:05 +08:00
const char *retType = methodSignature.methodReturnType;
if (!strcmp(retType, @encode(void))) {
ret = nil;
2019-10-12 14:48:19 +08:00
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), block);
2019-07-30 11:25:05 +08:00
} else if (!strcmp(retType, @encode(id))) {
void *retValue;
block();
[invocation getReturnValue:&retValue];
2019-10-12 14:48:19 +08:00
id returnValue = (__bridge id) retValue;
2019-07-30 11:25:05 +08:00
ret = [JSValue valueWithObject:[returnValue copy] inContext:[JSContext currentContext]];
} else {
DoricLog(@"CallNative Error:%@", @"Must return object type");
ret = nil;
}
}
break;
}
}
}
if (methods) {
free(methods);
}
if (!isFound) {
Class superclass = class_getSuperclass(clz);
if (superclass && superclass != [NSObject class]) {
return [self findClass:superclass target:target context:context method:name callbackId:callbackId argument:argument];
}
}
2019-07-30 11:25:05 +08:00
return ret;
}
2019-07-29 20:51:53 +08:00
@end