2019-07-26 14:19:42 +08:00
|
|
|
//
|
|
|
|
// DoricJSEngine.m
|
|
|
|
// Doric
|
|
|
|
//
|
|
|
|
// Created by pengfei.zhou on 2019/7/26.
|
|
|
|
//
|
|
|
|
|
|
|
|
#import "DoricJSEngine.h"
|
|
|
|
#import "DoricJSExecutorProtocal.h"
|
|
|
|
#import "DoricJSCoreExecutor.h"
|
|
|
|
#import "DoricConstant.h"
|
|
|
|
#import "DoricUtil.h"
|
2019-07-29 20:51:53 +08:00
|
|
|
#import "DoricBridgeExtension.h"
|
2019-07-26 14:19:42 +08:00
|
|
|
|
|
|
|
@interface DoricJSEngine()
|
|
|
|
@property(nonatomic,strong) id<DoricJSExecutorProtocal> jsExecutor;
|
2019-07-27 19:16:02 +08:00
|
|
|
@property(nonatomic,strong) NSMutableDictionary *timers;
|
2019-07-29 20:51:53 +08:00
|
|
|
@property(nonatomic,strong) DoricBridgeExtension *bridgeExtension;
|
2019-07-26 14:19:42 +08:00
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation DoricJSEngine
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (instancetype)init {
|
2019-07-26 14:19:42 +08:00
|
|
|
if(self = [super init]){
|
|
|
|
_jsQueue = dispatch_queue_create("doric.jsengine", DISPATCH_QUEUE_SERIAL);
|
2019-07-29 20:51:53 +08:00
|
|
|
_bridgeExtension = [[DoricBridgeExtension alloc] init];
|
2019-07-26 14:19:42 +08:00
|
|
|
dispatch_async(_jsQueue, ^(){
|
2019-07-27 19:16:02 +08:00
|
|
|
self.timers = [[NSMutableDictionary alloc] init];
|
2019-07-26 14:19:42 +08:00
|
|
|
self.jsExecutor = [[DoricJSCoreExecutor alloc] init];
|
2019-07-27 14:36:41 +08:00
|
|
|
self.registry = [[DoricRegistry alloc] init];
|
2019-07-29 13:48:27 +08:00
|
|
|
[self initJSExecutor];
|
2019-07-26 14:19:42 +08:00
|
|
|
[self initDoricEnvironment];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)initJSExecutor {
|
2019-07-27 14:36:41 +08:00
|
|
|
__weak typeof(self) _self = self;
|
|
|
|
|
2019-07-27 19:16:02 +08:00
|
|
|
[self.jsExecutor injectGlobalJSObject:INJECT_LOG obj:^(NSString * type, NSString * message) {
|
2019-07-26 14:19:42 +08:00
|
|
|
DoricLog(@"JS:%@",message);
|
|
|
|
}];
|
2019-07-27 14:36:41 +08:00
|
|
|
|
2019-07-26 14:19:42 +08:00
|
|
|
[self.jsExecutor injectGlobalJSObject:INJECT_REQUIRE obj:^(NSString *name){
|
2019-07-27 14:36:41 +08:00
|
|
|
__strong typeof(_self) self = _self;
|
|
|
|
if(!self) return NO;
|
|
|
|
NSString *content = [self.registry acquireJSBundle:name];
|
|
|
|
if(!content){
|
|
|
|
DoricLog(@"require js bundle:%@ is empty", name);
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
@try{
|
|
|
|
[self.jsExecutor loadJSScript:[self packageModuleScript:name content:content]
|
|
|
|
source:[@"Module://" stringByAppendingString:name]];
|
|
|
|
}@catch(NSException *e){
|
|
|
|
DoricLog(@"require js bundle:%@ error,for %@", name, e.reason);
|
|
|
|
}
|
|
|
|
return YES;
|
2019-07-26 14:19:42 +08:00
|
|
|
}];
|
2019-07-27 19:16:02 +08:00
|
|
|
[self.jsExecutor injectGlobalJSObject:INJECT_TIMER_SET
|
|
|
|
obj:^(NSNumber *timerId,NSNumber *interval,NSNumber *isInterval) {
|
|
|
|
__strong typeof(_self) self = _self;
|
|
|
|
NSString *timerId_str = [timerId stringValue];
|
|
|
|
BOOL repeat = [isInterval boolValue];
|
2019-07-30 19:07:46 +08:00
|
|
|
NSTimer *timer = [NSTimer timerWithTimeInterval:[interval doubleValue]/1000 target:self selector:@selector(callbackTimer:) userInfo:@{@"timerId":timerId,@"repeat":isInterval} repeats:repeat];
|
2019-07-27 19:16:02 +08:00
|
|
|
[self.timers setValue:timer forKey:timerId_str];
|
|
|
|
dispatch_async(dispatch_get_main_queue(), ^(){
|
|
|
|
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
|
|
|
|
});
|
|
|
|
}];
|
2019-07-27 14:36:41 +08:00
|
|
|
|
2019-07-27 19:16:02 +08:00
|
|
|
[self.jsExecutor injectGlobalJSObject:INJECT_TIMER_CLEAR
|
|
|
|
obj:^(NSString *timerId) {
|
|
|
|
__strong typeof(_self) self = _self;
|
|
|
|
NSTimer *timer = [self.timers valueForKey:timerId];
|
|
|
|
if(timer){
|
|
|
|
[timer invalidate];
|
|
|
|
[self.timers removeObjectForKey:timerId];
|
|
|
|
}
|
|
|
|
}];
|
2019-07-29 20:51:53 +08:00
|
|
|
|
|
|
|
[self.jsExecutor injectGlobalJSObject:INJECT_BRIDGE obj:^(NSString *contextId, NSString *module, NSString *method, NSString *callbackId, id argument){
|
|
|
|
return [self.bridgeExtension callNativeWithContextId:contextId module:module method:method callbackId:callbackId argument:argument];
|
|
|
|
}];
|
2019-07-26 14:19:42 +08:00
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)initDoricEnvironment {
|
2019-07-27 10:33:38 +08:00
|
|
|
[self loadBuiltinJS:DORIC_BUNDLE_SANDBOX];
|
2019-08-14 14:13:08 +08:00
|
|
|
NSString *path = [DoricBundle() pathForResource:DORIC_BUNDLE_LIB ofType:@"js" inDirectory:@"bundle"];
|
2019-07-27 10:33:38 +08:00
|
|
|
NSString *jsContent = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];
|
|
|
|
[self.jsExecutor loadJSScript:[self packageModuleScript: DORIC_MODULE_LIB content:jsContent]
|
|
|
|
source: [@"Module://" stringByAppendingString:DORIC_MODULE_LIB]];
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)loadBuiltinJS:(NSString *)fileName {
|
2019-08-14 14:13:08 +08:00
|
|
|
NSString *path = [DoricBundle() pathForResource:DORIC_BUNDLE_SANDBOX ofType:@"js" inDirectory:@"bundle"];
|
2019-07-27 10:33:38 +08:00
|
|
|
NSString *jsContent = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];
|
|
|
|
[self.jsExecutor loadJSScript:jsContent source:[@"Assets://" stringByAppendingString:fileName]];
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (JSValue *)invokeDoricMethod:(NSString *)method,... {
|
2019-07-26 14:19:42 +08:00
|
|
|
va_list args;
|
|
|
|
va_start(args, method);
|
2019-07-27 14:36:41 +08:00
|
|
|
JSValue *ret = [self invokeDoricMethod:method arguments:args];
|
|
|
|
va_end(args);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (JSValue *)invokeDoricMethod:(NSString *)method arguments:(va_list)args {
|
2019-07-27 14:36:41 +08:00
|
|
|
NSMutableArray *array = [[NSMutableArray alloc] init];
|
2019-07-27 10:33:38 +08:00
|
|
|
id arg = va_arg(args, id);
|
2019-07-27 14:36:41 +08:00
|
|
|
while(arg != nil){
|
2019-07-26 14:19:42 +08:00
|
|
|
[array addObject:arg];
|
|
|
|
arg = va_arg(args, JSValue *);
|
|
|
|
}
|
2019-07-27 14:36:41 +08:00
|
|
|
return [self.jsExecutor invokeObject:GLOBAL_DORIC method:method args:array];
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (JSValue *)invokeDoricMethod:(NSString *)method argumentsArray:(NSArray *)args {
|
2019-07-27 14:36:41 +08:00
|
|
|
return [self.jsExecutor invokeObject:GLOBAL_DORIC method:method args:args];
|
2019-07-26 14:19:42 +08:00
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (NSString *)packageContextScript:(NSString *)contextId content:(NSString *)content {
|
2019-07-27 10:33:38 +08:00
|
|
|
NSString *ret = [NSString stringWithFormat:TEMPLATE_CONTEXT_CREATE, content, contextId, contextId, contextId];
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (NSString *)packageModuleScript:(NSString *)moduleName content:(NSString *)content {
|
2019-07-27 10:33:38 +08:00
|
|
|
NSString *ret = [NSString stringWithFormat:TEMPLATE_MODULE, moduleName, content];
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)prepareContext:(NSString *)contextId script:(NSString *)script source:(NSString *)source {
|
2019-07-27 10:33:38 +08:00
|
|
|
[self.jsExecutor loadJSScript:[self packageContextScript:contextId content:script]
|
|
|
|
source:[@"Context://" stringByAppendingString:source]];
|
|
|
|
}
|
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)destroyContext:(NSString *)contextId {
|
2019-07-27 10:33:38 +08:00
|
|
|
[self.jsExecutor loadJSScript:[NSString stringWithFormat:TEMPLATE_CONTEXT_DESTROY, contextId]
|
|
|
|
source:[@"_Context://" stringByAppendingString:contextId]];
|
|
|
|
}
|
2019-07-26 14:19:42 +08:00
|
|
|
|
2019-07-29 13:48:27 +08:00
|
|
|
- (void)callbackTimer:(NSTimer *)timer {
|
2019-07-28 01:26:19 +08:00
|
|
|
NSDictionary *userInfo = timer.userInfo;
|
|
|
|
NSNumber *timerId = [userInfo valueForKey:@"timerId"];
|
|
|
|
NSNumber *repeat = [userInfo valueForKey:@"repeat"];
|
|
|
|
__weak typeof(self) _self = self;
|
|
|
|
dispatch_async(self.jsQueue, ^(){
|
|
|
|
__strong typeof(_self) self = _self;
|
|
|
|
@try {
|
2019-07-30 19:07:46 +08:00
|
|
|
[self invokeDoricMethod:DORIC_TIMER_CALLBACK, timerId, nil];
|
2019-07-28 01:26:19 +08:00
|
|
|
} @catch (NSException *exception) {
|
|
|
|
DoricLog(@"Timer Callback error:%@", exception.reason);
|
|
|
|
}
|
|
|
|
if(![repeat boolValue]){
|
|
|
|
[self.timers removeObjectForKey:[timerId stringValue]];
|
|
|
|
}
|
|
|
|
});
|
2019-07-27 19:16:02 +08:00
|
|
|
}
|
2019-07-26 14:19:42 +08:00
|
|
|
@end
|