原生应用使用cordova并与h5应用分离
个人原创地址:https://www.jianshu.com/p/1ad536e76640
1.需求与使用场景
打开一个新页面,要求能够加载本地zip格式的h5应用,该应用使用了某些原生能力;能够加载远程应用,该应用也使用了部分原生能力;能够在多个h5应用时同样适用;h5应用能够移植到其它场景,如web、第三方移动应用;h5应用无需复杂适配移动端,如android、iOS等;
2.目的
让h5应用只专注于开发h5,涉及到原生功能,则交给原生应用去实现,通过cordova js功能进行h5与原生功能的交互。
3.前提
需要3个应用,如下所示
- h5应用:前端开发者只专注于前端开发和打包应用,应支持ionic、vue、react等应用
- cordova应用:原生开发者创建,将cordova插件安装到cordova应用中,并生成对应的platforms即android、iOS
- 原生应用:原生开发者创建,iOS应用通过pod引入cordova插件,并将config.xml 通过group 方式添加,www则以文件夹folder的形式引入,即config.xml相对路径引入,www绝对路径引入;
对以上3个应用的解读:
- h5应用,正常的前端应用,若使用js则无需声明,若使用ts则需要声明 `declare let cordova: any;`全局变量
- cordova应用,目的是将config.xml文件和platforms/android/www、platforms/ios/www文件夹copy出来备用。将其归为原生开发者创建范畴,是因为cordova.js通过exec来调用原生插件,将插件的管理交给原生应用。
- 原生应用,管理插件,提供对应的原生能力。若使用cordova.exec方式交互(推荐),则将config.xml和www/cordova.js引入;若使用js service方式交互,则h5应用需要导入npm包,cordova应用添加插件,并将platforms/./www文件夹copy出来,放到原生应用下.
4. 原生部分
h5应用部分省略不表,原生部分以iOS为例
4.1 如何创建cordova项目
在mac上,安装node,再在terminal中npm install -g cordova@lastest;
- 创建应用`cordova create myApp org.apache.cordova.myApp myApp`
- 项目目录下`cd myApp`* 安装插件`cordova plugin add cordova-plugin-camera`
- 生成iOS应用 `cordova platform add ios`
- cordova应用下的config.xml和platforms/ios/www是我们需要用到的
4.2 由于UIWebView有缺陷以及不再维护,并且官方推荐使用WKWebView,所以以WKWebView为例,进行config.xml的修改以及cordova-plugin-wkwebview-engine的简单使用的讲解
4.2.1 config.xml的修改
1. 允许远程网页加载 ATS
<content src="index.html" />是本地加载的入口,也可设置远程地址
<allow-navigation href="https://*/*" />
<allow-navigation href="http://*/*" />
<allow-navigation href="data:*" />
以上三行等价于
<allow-navigation href="*" /> 允许所有的网址跳转。
2. 白名单设置
<access origin="*" /> 设置白名单,允许访问所有域 <allow-intent href="http://*/*" />
<allow-intent href="https://*/*" />
<allow-intent href="*" /> 等价于以上两行
3. 偏好设置
<preference name="StatusBarStyle" value="lightcontent" /> //设置状态栏颜色
4. feature功能引入,以便cordova.js识别
<feature name="Camera">
<param name="ios-package" value="CDVCamera" />
</feature>
添加后,cordova.exec才可以正确找到对应的原生文件,调用方式如`cordova.exec(success, failed, 'Camera', 'getPicker',[参数])`
5. 插件添加
<plugin name="cordova-plugin-wkwebview-engine" spec="^1.2.0" />
<plugin name="cordova-plugin-camera" spec="^4.1.0" />
<plugin name="cordova-plugin-device" spec="^2.0.3" />
指定添加的插件和版本号
6. 在info.plist文件添加私有权限
某些插件和功能需要开启相关的权限才能够使用,因此必须手动开启权限
4.2.2 wkwebviewengine的简单使用
先决条件:
- iOS原生应用pod 引入cordova以及插件
- 新建继承自CDVViewController的ViewController,如HtmlViewController
1. 简单的本地加载
cordova加载本地自己的h5应用,需要在HtmlViewController初始化init的地方修改startPage,一定要在viewDidLoad之前;
原因在于CDVViewController源码会在viewDidLoad的地方调用了与网页相关的三个方法:
`- loadSetting`、`- createGapView`、`- appUrl`,并设置了默认appUrl;
适用场景:
在应用内下载了zip格式的h5应用后,将其保存并解压,再将工程目录下的www文件夹copy到应用程序内,并将h5应用替换(不是合并)对应文件,如替换www/index.html,即将www/cordova_plugins.js、www/cordova-js-src、www/cordova.js、www/plugins copy到h5应用下。
2. 加载远程应用
uiwebview:只需要设置下 self.startPage为远程地址即可;
需要注意的地方:
- self.startPage的赋值,必须在[super viewDidLoad]之前,否则self.startPage 会被默认赋值为index.html。
- 需要在config.xml中修改一下配置,否则加载远程H5时,会自动打开浏览器加载。
<allow-navigation href="https://*/*" />
<allow-navigation href="http://*/*" />
- 远程H5中也要引用cordova.js文件。
- 在 info.plist 中添加 App Transport Security Setting的设置。
wkwebviewengine貌似则不能这样,我试了几次均不行,所以采用了以下方法:
- pod 引入cordova-plugin-wkwebview-engine
- 修改HtmlViewController.h
#import <Cordova/CDVViewController.h>
#import <Cordova/CDVCommandDelegateImpl.h>
@interface HtmlViewController : CDVViewController @end @interface CustomWKCommandDelegate: CDVCommandDelegateImpl // 核心指令 @end @interface CustomCmdQueue : CDVCommandQueue @end
HtmlViewController.m修改添加内容
#import <WebKit/WebKit.h>
@interface HtmlViewController ()<WKNavigationDelegate> @end @implementation CustomWKWebViewController - (id)init {
self = [super init];
if (self) {
// 重写CDVCommandDelegate,加载wkwebview的代理
_commandDelegate = [[CustomWKCommandDelegate alloc] initWithViewController:self];
_commandQueue = [[CustomCmdQueue alloc] initWithViewController:self]; }
return self;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view. // 由于支持滑动返回,导航栏可以隐藏了
[self.navigationController setNavigationBarHidden:YES animated:NO]; // 使用驱动器加载
[self.webViewEngine loadRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"https://www.baidu.com"] cachePolicy:NSURLRequestReloadIgnoringCacheData timeoutInterval:]]; } // 处理跨域等情况建议使用nginx
- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
// 不能跨域的进行单独处理
NSURL *url = navigationAction.request.URL;
if ([url.scheme isEqualToString:@"域名"]) { //在取消跨域请求之前,自定义处理,如使用safari浏览器打开网页,若带参数,则将其进行拼接
NSDictionary *param = nil;
if (@available(iOS 10.0, *)) {
[[UIApplication sharedApplication] openURL:url options:param completionHandler:^(BOOL success) { }];
} else {
// Fallback on earlier versions
[[UIApplication sharedApplication] openURL:url];
} decisionHandler(WKNavigationActionPolicyCancel);// web view取消本次
} else {
decisionHandler(WKNavigationActionPolicyAllow);
} } @end @implementation CustomWKCommandDelegate //- (id)initWithViewController:(CDVViewController *)viewController {
// self = [super initWithViewController:viewController];
// if (self) {
// self.viewController = viewController;
// }
// return self;
//} - (id)getCommandInstance:(NSString *)pluginName {
return [super getCommandInstance:pluginName];
} // 重写资源路径的方法,进行拦截
- (NSString *)pathForResource:(NSString *)resourcepath {
// if (<#condition#>) {
// <#statements#>
// }
NSLog(@"path ---> %@", resourcepath);
return [super pathForResource:resourcepath];
} @end @implementation CustomCmdQueue - (BOOL)execute:(CDVInvokedUrlCommand *)command {
return [super execute:command];
} @end
这样写的原因,在于CDVViewController会自动加载CDVWKWebViewEngine并自动实现WKUIDelegate,在wkwebview的代理函数中,实现自己的业务逻辑.
适用场景:本地h5应用、html字符串、远程应用(如果较少的www文件,可采用wkwebview预加载,提升加载速度)
5. 关于自定义插件部分
原有插件不能满足需求,需要自定义,大致流程为:
- 新建集成自CDVPLugin的原生类
- 修改原生类.h和.m,实现相应函数,
- 修改config.xml,将自定义插件以feature添加进去
- js通过cordova.exec的方式调用
建议将插件以pod的方式引入,方便管理,这就涉及到Podfile语法了
6. 持续优化:
webview预加载、js\css\image离线缓存、避免无用的js引入;
原生应用使用cordova并与h5应用分离的更多相关文章
- H5动静分离
1. 动静分离的实现思路(类似于iOS.安卓的思路,后台提供数据接口,前端用ajax异步请求json数据,再把json数据渲染到页面) 动静分离是将网站静态资源(HTML,JavaScript,CSS ...
- Android H5混合开发(3):原生Android项目里嵌入Cordova
前言 如果安卓项目已经存在了,那么如何使用Cordova做混合开发? 方案1(适用于插件会持续增加或变化的项目): 新建Cordova项目并添加Android平台,把我们的安卓项目导入Android平 ...
- H5程序员如何利用cordova开发跨平台应用
什么是Cordova? Cordova以前也叫PhoneGap,它提供了一组设备相关的API,通过这组API,移动应用能够以JavaScript访问原生的设备功能,如摄像头.麦克风等.Cordova还 ...
- Android H5混合开发(4):构建Cordova Jar包
前言 上一节,介绍了原生项目如何嵌入Cordova,我们对Cordova的依赖使用的是CordovaLib Module,这也是安卓项目常用的方式. 但是,也有项目希望以Jar包的方式依赖Cordov ...
- 原生Android项目里嵌入Cordova
Android H5混合开发():原生Android项目里嵌入Cordova 如果安卓项目已经存在了,那么如何使用Cordova做混合开发? 方案1(适用于插件会持续增加或变化的项目): 新建Cord ...
- 【quickhybrid】H5和原生的职责划分
前言 在JSBridge实现后,前端网页与原生的交互已经通了,接下来就要开始规划API,明确需要提供哪一些功能来供前端调用. 但是在这之前,还有一点重要工作需要做: 明确H5与Native的职责划分, ...
- 如何在原生工程中引入Cordova工程-for iOS 【转】
http://blog.csdn.net/e20914053/article/details/50170487 如今混合开发方兴未艾,有的项目可能一开始是原生开发的,后期需要加入混合开发,如将Cord ...
- Android 原生开发、H5、React-Native使用利弊和场景技术分享
http://m.blog.csdn.net/article/details?id=51778086 发表于2016/6/28 18:52:46 1176人阅读 最近工作中接触到React ...
- H5和原生APP之间的区别
最近项目中因各种客观因素,移动端都是默认用的纯H5 APP,感受最深的就是各种坑啊,好大的坑啊.产品上线后,带着各种坑后的总结原因方发现很多人都说纯H5 APP一次编写就能支持android和IOS两 ...
随机推荐
- 文件识别浅谈(含office文件区分)
前言 本文主要根据后台接口识别Office文件类型这一话题做一些分享,主要方向还是放在不能获取到文件名情况下的Office文件识别. 可获取到文件名 如果后端接口可以获取到完成的文件名称,则整个过程会 ...
- win10安装docker
配置首先需要Hyper-v和容器,这样就可以运行Linux的镜像了 如果是win10home版或者是其他版本就需要安装visulbox了, 然后去官网https://www.docker.com/pr ...
- OVS实现VXLAN隔离
一.实验环境 1.准备3个CentOS7 mini版本的虚拟机,每个主机3个网卡.如图: 图中OVS-1.OVS-2.OVS-3分别为三台CentOS7 mini版虚拟机,分别配备3个虚拟网卡.如图中 ...
- angular安装
安装时间:20190703安装环境:win10 1 安装Nodejs 1.1 下载地址:https://nodejs.org/en/ 1.2 下载完成之后双击安装,安装完成之后无需配置环境变量(安装的 ...
- Linux系统下解锁Oracle的Scott用户
1).在Oracle用户下面输入命令:lsnrctl status查看监听是否开启,如果未开启则需要开启监听,输入命令:lsnrctl start; 2).如果没有设置监听的话需要先建立一个监听,然后 ...
- java源码解析之String类(四)
/* * 返回指定字符第一次出现的字符串内的索引 */ public int indexOf(int ch) { return indexOf(ch, 0); } /* * 返回指定字符第一次出现的字 ...
- decimal.ToString()问题
decimal dt = 1.00M; decimal dt1 = 1M; bool d = dt == dt1; ...
- Python编程菜鸟成长记--A1--03--Python 环境安装(待完成)
1.重点知识 Windows 上如何安装 Python 3 Linux 上如何安装 Python 3 Mac 上如何安装 Python 3 Windows 上如何安装 Pycharm Mac 上如何安 ...
- HDU 5616:Jam's balance(背包DP)
http://acm.hdu.edu.cn/showproblem.php?pid=5616 题意:有n个物品,每个重量为w[i],有一个天平,你可以把物品放在天平的左边或者右边,接下来m个询问,问是 ...
- Ng-Matero:基于 Angular Material 搭建的中后台管理框架
前言 目前市面上关于 Angular Material 的后台框架比较少,大多都是收费主题,而且都不太好用. 很多人都说 Material 是一个面向 C 端的框架,其实在使用其它框架做管理系统的时候 ...