iOS集成Weex页面的总结

由于项目的需求和未来业务发展的需要,在app版本迭代中,尝试将某个模块的功能用weex前端代码来实现,然后集成到原生app中,来减少原生业务的代码量.具体的集成过程,其实官方也已经提供了相应的文档,主要还是移动端和前端要约定好对应的方法和传参方式.保证之间的交互和数据传递是畅通的.下面我简单地描述一下整个的集成和对接过程.

一、集成WeexSDK到项目中,直接使用Cocoapods导入WeexSDK即可,这一步基本上没什么问题;

二、初始化Weex环境:     在AppDelegate导入<WeexSDK.h>,然后初始化:

    [WXAppConfiguration setAppGroup:@"WXApp"];

    [WXAppConfiguration setAppName:@"Merchants"];

    [WXAppConfiguration setAppVersion:[ToolManager getVersionStr]];

    [WXSDKEngine initSDKEnvironment];

    [WXSDKEngine registerModule:@"CommonModule" withClass:[CommonModule class]];

    [WXSDKEngine registerHandler:[WXimgLoader new] withProtocol:@protocol(WXImgLoaderProtocol)];

    [WXLog setLogLevel: WXLogLevelError];

注释:AppGroup可以自己自定义填写;AppName填自己的项目名称;AppVersion填写自己的项目版本号;initSDKEnvironment这个方法就是在初始化WeexSDK的运行环境,registModule方法和registerHandler方法是重点要讲的两个点.

三、创建Weex页面容器控制器:

1.继承自己的根视图控制器,我这里自己项目中写了一个BaseViewController,继承自UIViewController,里面自定义了控制器的一些方法,项目中其他所有的控制器都继承自BaseViewController.所以这里继承BaseViewController,新建了一个QuickOrderController容器,其中初始化容器的代码如下,网上有很多关于这一块的代码,基本上都是大同小异的.

#pragma mark-隐藏导航栏

-(void)viewWillAppear:(BOOL)animated{

    [super viewWillAppear:animated];

    self.navigationController.navigationBarHidden = YES;

}

-(void)viewDidAppear:(BOOL)animated{

    [super viewDidAppear:animated];

    [self updateInstanceState:WeexInstanceAppear];

}

#pragma mark-显示导航栏

-(void)viewWillDisappear:(BOOL)animated{

    [super viewWillDisappear:animated];

    self.navigationController.navigationBarHidden = NO;

    [self updateInstanceState:WeexInstanceDisappear];

}

- (void)viewDidLoad {

     [super viewDidLoad];

    self.view.backgroundColor = [UIColor whiteColor];

    [self render];

}

#pragma mark-初始化weex容器

-(void)render{

    _instance = [[WXSDKInstance alloc] init];

    _instance.viewController = self;

    _instance.frame = CGRectMake(0, kStatusBarHeight, self.view.frame.size.width, ScreenHeight-kStatusBarHeight);

    /**默认请求参数*/

    [self.params setValue:@"ios" forKey:@"platform"];

    [self.params setValue:[ToolManager getUUID] forKey:@"deviceID"];

    [self.params setValue:[ToolManager getVersionStr] forKey:@"version"];

    [self.params setValue:[UserAccountModel loadAccount].siteuserid forKey:@"opId"];

    [self.params setValue:[UserAccountModel loadAccount].password forKey:@"password"];

    [self.params setValue:[ZYUserStand objectForKey:@"shopId"] forKey:@"shopId"];

    [self.params setValue:[ZYUserStand objectForKey:@"bossPurchaseOrderType"] forKey:@"bossPurchaseOrderType"];

    [_instance renderWithURL:self.url options:@{@"params":self.params} data:nil];

    __weaktypeof(self) weakSelf =self;

    _instance.onCreate= ^(UIView*view) {

        [weakSelf.weexViewremoveFromSuperview];

        weakSelf.weexView= view;

        [weakSelf.viewaddSubview:weakSelf.weexView];

        UIAccessibilityPostNotification(UIAccessibilityScreenChangedNotification, weakSelf.weexView);

    };

    /**失败*/

    _instance.onFailed= ^(NSError*error) {

 };

    /**完成*/

    _instance.renderFinish = ^ (UIView *view) {

 };

/**更新状态*/

    _instance.updateFinish = ^(UIView *view) {


    };

}

#pragma mark-更新视图的state

-(void)updateInstanceState:(WXState)state {

    if(_instance&&_instance.state!= state) {

        _instance.state= state;

        if(state ==WeexInstanceAppear) {

            [[WXSDKManager bridgeMgr] fireEvent:_instance.instanceId ref:WX_SDK_ROOT_REF type:@"viewappear" params:nil domChanges:nil];

        }else if(state ==WeexInstanceDisappear) {

            [[WXSDKManager bridgeMgr] fireEvent:_instance.instanceId ref:WX_SDK_ROOT_REF type:@"viewdisappear" params:nil domChanges:nil];

        }

    }

}

#pragma mark-销毁

-(void)dealloc{

   [_instance destroyInstance];

}

注释:viewAppear里隐藏导航栏是因为weex页面自己写好了导航栏的内容,所以我这里将导航栏隐藏,将weexView(也就是weexInstance的frame设置为从状态栏下开始,避免页面顶到状态栏);params是一个字典,里面包含了请求时所需的默认参数,在这里,通过option传值,以键值对的形式将这些参数传递给weex,以供weex页面中的服务请求使用,当然这里是我和服务端沟通好的传值方式,如果有更好的方法,也可以按照自己的方法来.其他的代码可以完全拷贝使用.

四、自定义Module(registerModule方法)

1.继承NSObject创建一个自定义Module类,Module类名称可以自定义,最后在AppDelegate中注册Module时名称要保持一致即可;

2.暴露Module方法以供weex调用,这里我附上自己的部分代码来说明:

@implementation CommonModule

@synthesize weexInstance;

#pragma mark-暴露给weex调用的方法

/**吐司*/

WX_EXPORT_METHOD(@selector(showToast:))

/**js页面跳转方法*/

WX_EXPORT_METHOD(@selector(openPage:params:))

/**扫码*/

WX_EXPORT_METHOD(@selector(startScan:))

/**打印*/

WX_EXPORT_METHOD(@selector(print:content:))

/**拨号*/

WX_EXPORT_METHOD(@selector(call:))

这里我定义了几个供weex调用的方法,通过WX_EXPORT_METHOD将方法名暴露给js,当然,这里可以参考的方法是页面跳转的方法,其他的方法按自己的业务需求进行自定义即可,下面附上页面跳转的方法实现:

#pragma mark-跳转页面,带参,参数可传空

-(void)openPage:(NSString*)url params:(NSDictionary*)params{

    NSString*newURL = url;

    /**判断url是通过服务器下载还是本地js文件*/

    if([urlhasPrefix:@"//"]) {

        newURL = [NSStringstringWithFormat:@"http:%@", url];

    }elseif(![urlhasPrefix:@"http"]) {

        newURL = [NSURL URLWithString:url relativeToURL:weexInstance.scriptURL].absoluteString;

    }

    QuickOrderController *controller = [[QuickOrderController alloc] init];

    controller.url= [NSURLURLWithString:newURL];

    controller.params = [NSMutableDictionary dictionaryWithDictionary:params];

    [[weexInstance.viewController navigationController] pushViewController:controller animated:YES];

}

注释:这里的QuickOrderController是第三步创建的容器控制器,我们会发现这里在跳转时传递了两个参数,一个url和一个params,这个url就是第三步中 [_instance renderWithURL:self.url options:@{@"params":self.params} data:nil]方法中的url,params就是self.params,除了默认的参数,可以在这里添加其他的参数.视自己的需求而定.既然到这里,就再补充说一下原生和weex页面间的相互传值吧.

通过module里自定义的带参方法,我们可以将原生的一些参数传递给weex页面,同样,weex页面可以通过调用这些带参方法,将需要的参数传递给原生页面.参照我的代码具体说明如下:

1.weex传递到原生页面:在我的代码里有一个封装的打印方法,携带了type和content两个参数,weex页面在点击打印按钮时,会将当前订单的类型和需要打印的信息分别传递给这两个参数,而我们在module里实现这个方法时,就可以拿到weex页面传递过来的信息,去打印对应的订单小票.

2.原生传递到weex:我现在还有一个这样的需求,weex页面中有一个核销的按钮,点击后,会调用原生的扫码页面,扫码完成,需要将扫描到的结果传递给weex页面,然后weex页面进行对应的核销接口调用.这个扫码的方法就是上面代码里自定义Module中的扫码方法,这里使用了WXModuleCallback的回调机制.具体的实现代码如下:

#pragma mark-扫描,回调方法传递扫描结果

-(void)startScan:(WXModuleCallback)callBack{

    QuickScanCodeController *scanVC = [[QuickScanCodeController alloc] init];

    scanVC.title=@"扫描验货码";

    scanVC.callBack= callBack;

    [[weexInstance.viewController navigationController] pushViewController:scanVC animated:YES];

}

注释:这个QuickScanCodeController就是我这边原生的扫码页面,将callBack回调传递过去,同样,在扫码页面的扫码成功方法里,给这个callBack赋值,这里赋值的方式和具体传参的字段,需要和weex端统一好.我这里是以键值对的形式,将这个扫码结果传递.代码如下:

#pragma mark - 扫码完成后代理QRViewDelegate

- (void)qrView:(QRView*)view ScanResult:(NSString*)result{

    [view stopScan];/**扫描成功后先停止扫描*/

    NSLog(@"code=%@", result);

     /**扫码验货成功后的回调函数*/

     self.callBack(@{@"scanResult":result});

    [self.navigationController popViewControllerAnimated:YES];

}

到这里,基本集成就差不多了,有些人会发现,在集成好的weex页面已经可以正常跳转显示了,但是为什么weex页面上的图片全部都加载不出来.这是因为想要加载出图片还得实现一个图片加载器,回到AppDelegate里,可以发现,不仅有我注册的CommonModule,还有一个WXimgLoader,这个WXImgLoader就是用来加载图片的类.具体的代码实现如下,使用时可以直接拷贝到项目中,想深究的朋友可以自行去探讨.记住,一定要在AppDelegate中进行注册.

#import <Foundation/Foundation.h>

#import <WeexSDK/WeexSDK.h>

@interface WXimgLoader : NSObject<WXImgLoaderProtocol, WXImageOperationProtocol>

注意:.h文件,一定要遵循这个WXImgLoaderProtocol协议


#import "WXimgLoader.h"

#import <SDWebImage/UIImageView+WebCache.h>

#import <SDWebImageDownloader.h>

#define MIN_IMAGE_WIDTH 36

#define MIN_IMAGE_HEIGHT 36

#if OS_OBJECT_USE_OBJC

#undef  WXDispatchQueueRelease

#undef  WXDispatchQueueSetterSementics

#define WXDispatchQueueRelease(q)

#define WXDispatchQueueSetterSementics strong

#else

#undef  WXDispatchQueueRelease

#undef  WXDispatchQueueSetterSementics

#define WXDispatchQueueRelease(q) (dispatch_release(q))

#define WXDispatchQueueSetterSementics assign

#endif

@interface WXimgLoader ()

/**队列*/

@property (WXDispatchQueueSetterSementics, nonatomic) dispatch_queue_t ioQueue;

@end

@implementation WXimgLoader

#pragma mark WXImgLoaderProtocol

- (id)downloadImageWithURL:(NSString*)urlimageFrame:(CGRect)imageFrameuserInfo:(NSDictionary*)optionscompleted:(void(^)(UIImage*,NSError*,BOOL))completedBlock{

    if([urlhasPrefix:@"//"]) {

        url = [@"http:" stringByAppendingString:url];

    }

    // 加载本地图片

    if([urlhasPrefix:@"file://"]) {

        NSString *newUrl = [url stringByReplacingOccurrencesOfString:@"/images/" withString:@"/"];

        UIImage *image = [UIImage imageNamed:[newUrl substringFromIndex:7]];

        completedBlock(image,nil,YES);

        return (id<WXImageOperationProtocol>) self;


    }else{

        return (id<WXImageOperationProtocol>)  [[SDWebImageDownloader sharedDownloader]downloadImageWithURL:[NSURL URLWithString:url] options:0 progress:^(NSInteger receivedSize, NSInteger expectedSize, NSURL * _Nullable targetURL) {


          }completed:^(UIImage*_Nullableimage,NSData*_Nullabledata,NSError*_Nullableerror,BOOLfinished) {

              if(completedBlock){

                        completedBlock(image,error,finished);

              }

          }];

    }

}

#pragma mark-取消加载图片方法

- (void)cancel{

    [[SDWebImageManager sharedManager]cancelAll];

}

@end

注释:SDWebImage版本不宜过高,否则会报错,提示找不到cancel的方法.


整个集成过程中遇到的问题:

1.在weex容器页面加载的过程中,首次加载weex页面时,weex页面的网络请求始终无法响应,这里按照weex官方集成文档上所说的,viewappear方法对应ios端viewWillAppear方法,我将updateInstanceState的方法写在viewWillAppear方法里,weex首次加载时端始终无法捕捉到该事件,导致第一次进入页面数据未加载的情况.后来考虑是不是ios的页面尚未加载完成导致weex容器页面尚未加载,weex页面无法捕捉到该事件.于是尝试将updateInstanceState的方法写在了viewDIdLoad的方法里,问题得到了解决.但是具体原因是否是猜想的那样,还有待验证,如果有知道的或者有更好方法的小伙伴,可以提出来让大家参考一下.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352