前言

前文中就有提到,Hybrid模式的核心就是在原生,而本文就以此项目的Android部分为例介绍Android部分的实现。

提示,由于各种各样的原因,本项目中的Android容器确保核心交互以及部分重要API实现,关于底层容器优化等机制后续再考虑完善。

大致内容如下:

  • JSBridge核心交互部分

  • uipagenavigator等部分常用API的实现

  • 组件(自定义)API拓展的实现

  • 容器h5支撑的部分完善(如支持fileinput文件选择,地理定位等-默认不生效的)

  • API的权限校验仅预留了一个入口,模拟最简单的实现

  • 其它如离线资源加载更新,底层优化等机制暂时不提供

项目的结构

基于AndroidStudio的项目,为了便于管理,稍微分成了几个模块,

而且由于主要精力已经偏移到了JS前端,已经不想再花大力气重构Android代码了,

因此仅仅是将代码从业务中抽取出来,留下了一些稍微精简的代码(也不是特别精简)。

所以如果发现代码风格,规范等不太合适,请先将就着。

整体目录结构如下:

  1. quickhybrid-android
  2. |- app // application,应用主程序
  3. | |- api/PayApi // 拓展了一个组件API
  4. | |- MainActivity // 入口页面
  5. |- core // library,核心工具类模块,放一些通用工具类
  6. | |- baseapp
  7. | |- net
  8. | |- ui
  9. | |- util
  10. |- jsbridge // library,JSBridge模块,混合开发的核心实现
  11. | |- api
  12. | |- bean
  13. | |- bridge
  14. | |- control
  15. | |- view

代码架构

简单的三次架构:底层核心工具类->JSBridge桥接实现->app应用实现

  1. core
  2. |- application // 应用流程控制,Activity管理,崩溃日志等
  3. |- baseapp // 一些基础Activity,Fragment的定义
  4. |- net // 网络请求相关
  5. |- ui // 一些UI效果的定义与实现
  6. |- util // 通用工具类
  7. jsbridge
  8. |- api // 定义API,开放原生功能给H5
  9. |- bean // 放一些实体类
  10. |- bridge // 桥接的定义以及核心实现
  11. |- control // 控制类,包括回调控制,页面加载控制,文件选择控制等
  12. |- view // 定义混合开发需要的webview和fragment实现
  13. app
  14. |- api // 拓展项目需要的自定义组件API
  15. |- AppApplication.java // 应用的控制
  16. |- MainActivity.java // 入口界面的控制

权限配置

原生应用中,不可逃避的就是打包后的权限问题,没有权限,很多功能都使用不了,

简单起见,这里将应用中用的权限都列了出来(基于多种考虑,并没有遵循最小原则)

  1. <!-- ===============================权限配置声明=============================== -->
  2. <uses-permission android:name="android.permission.READ_PHONE_STATE" />
  3. <uses-permission android:name="android.permission.CAMERA" />
  4. <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
  5. <uses-permission android:name="android.permission.CALL_PHONE" />
  6. <uses-permission android:name="android.permission.READ_CALL_LOG" />
  7. <uses-permission android:name="android.permission.SEND_SMS" />
  8. <uses-permission android:name="android.permission.VIBRATE" />
  9. <uses-permission android:name="android.permission.READ_LOGS" />
  10. <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
  11. <uses-permission android:name="android.permission.WAKE_LOCK" />
  12. <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
  13. <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
  14. <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
  15. <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
  16. <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
  17. <uses-permission android:name="android.permission.INTERNET" />
  18. <uses-permission android:name="android.permission.WRITE_SETTINGS" />
  19. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
  20. <uses-permission android:name="android.permission.RECORD_AUDIO" />
  21. <uses-permission android:name="android.permission.READ_OWNER_DATA" />
  22. <uses-permission android:name="android.permission.READ_CONTACTS" />
  23. <uses-permission android:name="android.permission.WRITE_CONTACTS" />
  24. <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
  25. <uses-permission android:name="android.permission.DISABLE_KEYGUARD" />
  26. <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
  27. <uses-permission android:name="com.android.launcher.permission.READ_SETTINGS" />
  28. <uses-permission android:name="android.permission.GET_ACCOUNTS" />
  29. <uses-permission android:name="android.permission.READ_PROFILE" />

注意,6.0之上需要动态权限,请确保已经给应用开了对应的权限

Gradle配置

AndroidStudio中项目要正确运行起来,需要有一个正确的Gradle配置。

这里也就几个关键性的配置作说明,其余的可以参考源码

gradle-wrapper.properties

  1. distributionUrl=https\://services.gradle.org/distributions/gradle-4.2.1-all.zip

如果遇到gradle编译不动,可以像上述一样,把这个文件的gradle版本修改为本地用的版本

(否则的话,没有科学上网就很有可能卡住)

setting.gradle

  1. include ':app', ':jsbridge', ':core'

里面很简单,就是一行代码,将三个用到的模块都引用进来

build.gradle(core)

仅挑选了部分进行说明

  1. apply plugin: 'com.android.library'
  2. android {
  3. compileSdkVersion 25
  4. defaultConfig {
  5. minSdkVersion 16
  6. targetSdkVersion 22
  7. versionCode 1
  8. versionName "1.0"
  9. ...
  10. }
  11. ...
  12. }
  13. dependencies {
  14. compile fileTree(dir: 'libs', include: ['*.jar'])
  15. compile 'com.android.support:appcompat-v7:25.3.1'
  16. compile 'com.android.support:support-v4:25.3.1'
  17. compile 'com.android.support:design:25.3.1'
  18. compile 'com.android.support:recyclerview-v7:25.3.1'
  19. compile 'com.android.support.constraint:constraint-layout:1.0.2'
  20. compile 'com.jakewharton:butterknife:8.6.0'
  21. compile 'com.google.code.gson:gson:2.8.0'
  22. compile 'com.journeyapps:zxing-android-embedded:3.5.0'
  23. compile 'com.liulishuo.filedownloader:library:1.5.5'
  24. compile 'com.nostra13.universalimageloader:universal-image-loader:1.9.5'
  25. compile 'me.iwf.photopicker:PhotoPicker:0.9.10@aar'
  26. compile 'com.github.bumptech.glide:glide:4.1.1'
  27. ...
  28. }

上述的关键信息有几点:

  • apply plugin: 'com.android.library'代表是模块而不是主应用

  • minSdkVersion 16代表最低兼容4.1的版本

  • targetSdkVersion 25是编译版本,targetSdkVersion 22提供向前兼容的作用,22时不需要动态权限,

    主要作用是某些API在不同版本中使用不一样,或者根本就在低版本中没有。

  • versionNameversionCode进行版本控制

  • dependencies中是依赖信息,首先compile fileTree添加了libs下的所有离线依赖(里面有离线依赖包),

    然后compile一些必须的依赖(譬如用到了gson,自动注解,文件下载等等)

为什么这里没用implementation添加依赖,而是用compile?因为implementation不具有传递性,这样引用core的jsbridge就用不到了,

而我们需要确保jsbridge中也用到,所以就用了compile。

build.gradle(jsbridge)

一部分类似的代码就没有贴出来了

  1. apply plugin: 'com.android.library'
  2. ...
  3. dependencies {
  4. implementation project(':core')
  5. ...
  6. }

这里和core不同之处在于,内部依赖于core模块,使用了implementation project

这样在jsbridge内部就能使用core的源码了。

需要注意的是,implementation不具有传递性(core只会暴露给jsbridge,不会传递下去)

build.gradle(app)

一部分类似的代码就没有贴出来了

  1. apply plugin: 'com.android.application'
  2. android {
  3. defaultConfig {
  4. applicationId "com.quick.quickhybrid"
  5. versionCode 1
  6. versionName "1.0"
  7. }
  8. ...
  9. }
  10. dependencies {
  11. implementation project(':core')
  12. implementation project(':jsbridge')
  13. implementation fileTree(dir: 'libs', include: ['*.jar'])
  14. // butterknife8.0+版本支持控件注解必须在可运行的model加上
  15. annotationProcessor 'com.jakewharton:butterknife-compiler:8.6.0'
  16. ...
  17. }

与之前相比,有几点关键信息

  • apply plugin: 'com.android.application'代表是主应用而不是模块

  • applicationId定义了应用id

  • 同样有自己的版本控制,但是注意,这里是容器版本号,前面的如jsbridge中是quick的版本号,有区别的

  • implementation依赖了前面两个模块,同时,后面引入了应用中可能需要的依赖

  • annotationProcessor 'com.jakewharton:butterknife-compiler:8.6.0',这行代码是为了使得butterknife自动注解生效的配置

targetSdkVersion说明

配置中使用的版本是22,因为在这个版本以上会有动态权限问题,比较麻烦,需要更改部分逻辑。因此就暂时未修改了。

譬如操作私有文件的权限问题等等

一些关键代码

代码方面,也无法一一全部说明,这里仅列举一些比较重要的步骤实现,其余可参考源码

UA约定

前面的JS项目中就已经有提到UA约定,就是在加载对于webview时,统一在webview中加上如下UA标识

  1. WebSettings settings = getSettings();
  2. String ua = settings.getUserAgentString();
  3. // 设置浏览器UA,JS端通过UA判断是否属于Quick环境
  4. settings.setUserAgentString(ua + " QuickHybridJs/" + BuildConfig.VERSION_NAME);

一些关键的webview设置

  1. // 设置支持JS
  2. settings.setJavaScriptEnabled(true);
  3. // 设置是否支持meta标签来控制缩放
  4. settings.setUseWideViewPort(true);
  5. // 缩放至屏幕的大小
  6. settings.setLoadWithOverviewMode(true);
  7. // 设置内置的缩放控件(若SupportZoom为false,该设置项无效)
  8. settings.setBuiltInZoomControls(true);
  9. // 设置缓存模式
  10. // LOAD_DEFAULT 根据HTTP协议header中设置的cache-control属性来执行加载策略
  11. // LOAD_CACHE_ELSE_NETWORK 只要本地有无论是否过期都从本地获取
  12. settings.setCacheMode(WebSettings.LOAD_DEFAULT);
  13. settings.setDomStorageEnabled(true);
  14. // 设置AppCache 需要H5页面配置manifest文件(官方已不推介使用)
  15. String appCachePath = getContext().getCacheDir().getAbsolutePath();
  16. settings.setAppCachePath(appCachePath);
  17. settings.setAppCacheEnabled(true);
  18. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
  19. // 强制开启android webview debug模式使用Chrome inspect(https://developers.google.com/web/tools/chrome-devtools/remote-debugging/)
  20. WebView.setWebContentsDebuggingEnabled(true);
  21. }
  22. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
  23. CookieManager.getInstance().setAcceptThirdPartyCookies(this, true);
  24. }

上述的一系列配置下去才能让H5页面的大部分功能正常开启,如localstorage,cookie,viewport,javascript等

支持H5地理定位

在继承WebChromeClient的QuickWebChromeClient

  1. @Override
  2. public void onGeolocationPermissionsShowPrompt(String origin, GeolocationPermissions.Callback callback) {
  3. callback.invoke(origin, true, false);
  4. super.onGeolocationPermissionsShowPrompt(origin, callback);
  5. }

需要重新才支持地理定位,否则纯h5定位无法获取地理位置(或者被迫使用了网络定位)

支持文件选择

同样在继承WebChromeClient的QuickWebChromeClient

  1. /**
  2. * Android 4.1+适用
  3. *
  4. * @param uploadMsg
  5. * @param acceptType
  6. * @param capture
  7. */
  8. public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture) {
  9. loadPage.getFileChooser().showFileChooser(uploadMsg, acceptType, capture);
  10. }
  11. /**
  12. * Android 5.0+适用
  13. *
  14. * @param webView
  15. * @param filePathCallback
  16. * @param fileChooserParams
  17. * @return
  18. */
  19. @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
  20. @Override
  21. public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {
  22. loadPage.getFileChooser().showFileChooser(webView, filePathCallback, fileChooserParams);
  23. return true;
  24. }

上述的操作是主动监听文件的选择,然后自动调用原生中的处理方案,譬如弹出一个通用的选择框,进行选择等。

如果不实现,无法正常通过FileInput选择文件,而实际上,FileInput又是一个很常用的功能。

监听JSBridge的触发

同样在继承WebChromeClient的QuickWebChromeClient

  1. @Override
  2. public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) {
  3. result.confirm(JSBridge.callJava(loadPage.getFragment(), message,loadPage.hasConfig()));
  4. return true;
  5. }

为了方便,直接使用onJsPrompt来作为交互通道,前文中也相应提到过

其它

在直接提供API前,还有很多需要做的基础工作,譬如浏览历史记录管理,监听附件下载,页面加载报错处理等等,这里不再赘述,可以直接参考源码

最后,关于一些JSBridge实现,API实现,由于本系列的其它文中或多或少都已经提到,这里就不再赘述了,可以直接参考源码

另外,后续如果继续有容器优化等操作,也会单独整理,加入本系列。

前端页面示例

为了方便,直接集成到了app/assets/中,入口页面默认会加载它,也可以直接看源码

返回根目录

源码

github上这个框架的实现

quickhybrid/quickhybrid

【quickhybrid】Android端的项目实现的更多相关文章

  1. 【转】【Android】开源项目汇总-备用

    第一部分 个性化控件(View) 主要介绍那些不错个性化的View,包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Pro ...

  2. Android之开源项目工具库篇

    本文转自:http://www.trinea.cn/android/android-open-source-projects-dev-lib/ 本文中你可以找到那些精美App中各种有特性的View,如 ...

  3. Android常用开源项目

    Android开源项目第一篇——个性化控件(View)篇   包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Progre ...

  4. Android开源经典项目

    目前包括: Android开源项目第一篇--个性化控件(View)篇   包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView. ...

  5. 【Android】开源项目汇总-备用

    from://http://www.eoeandroid.com/home.php?mod=space&uid=765778&do=blog&id=47674 Android开 ...

  6. 【Android】开源项目汇总

    Android开源项目第一篇——个性化控件(View)篇  包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Progres ...

  7. Vue项目用于Ios和Android端开发

    起因 前公司商城App项目使用的是H5开发,有微信公众号.Ios和Android三个版本,H5版本是自己写的一套框架,已经用了有些年头了,承载不下不断涌现出的新需求.而Ios和Android端通过we ...

  8. [Android]Android端ORM框架——RapidORM(v2.0)

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5626716.html [Android]Android端ORM ...

  9. Java服务器对外提供接口以及Android端向服务器请求数据

    转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/5056780.html 讲解下java服务器是如何对移动终端提供接口的,以什么数据格式提供出去,移动端又是怎么 ...

随机推荐

  1. POJ 2250 (LCS,经典输出LCS序列 dfs)

    题目链接: http://poj.org/problem?id=2250 Compromise Time Limit: 1000MS   Memory Limit: 65536K Total Subm ...

  2. Centos7 yum安装Mysql5.7

    1.下载mysql安装源 curl -LO http://dev.mysql.com/get/mysql57-community-release-el7-11.noarch.rpm 2.安装yum源 ...

  3. 转换CLOB字段类型为VARCHAR2, lob类型不支持的sql语句

    转自:https://blog.csdn.net/e_wsq/article/details/7561209 步骤: 1.建立一个临时varchar2字段用来保存数据 2.将clob的内容截取后更新到 ...

  4. mysql/mariadb学习记录——查询

    连接查询:同时设计两个及以上的表的查询 连接条件或连接谓词:用来连接两个表的条件一般格式: [<表名1>]<列名1> <比较运算符> [<表名2>]&l ...

  5. n进制转十进制

    #include<cstdio> #include<iostream> using namespace std; ; int main(){ ,len=; char ch[ma ...

  6. 基于 Keras 用 LSTM 网络做时间序列预测

    目录 基于 Keras 用 LSTM 网络做时间序列预测 问题描述 长短记忆网络 LSTM 网络回归 LSTM 网络回归结合窗口法 基于时间步的 LSTM 网络回归 在批量训练之间保持 LSTM 的记 ...

  7. 20155307《网络对抗》PC平台逆向破解(二)

    20155307<网络对抗>PC平台逆向破解(二) shellcode注入 什么是shellcode? shellcode是一段代码,溢出后,执行这段代码能开启系统shell. 前期准备- ...

  8. Oracle OEM启动方法

    首先要启动 listener: lsnrctl start 对于dbconsole:emctl start dbconsole

  9. Linux下开发python django程序(Session读写)

    1.登陆设置session信息 def loginsession(req): if req.method == 'POST': loginform = LoginForm(req.POST) if l ...

  10. 9 README,全套代码

    BBS+ BLOG系统(仿博客园) 一.概要 欢迎您使用该BBS+BLOG系统,希望在您使用的过程中体验到便捷和愉快的使用感受,并对我们的软件提出您发现的问题和建议,谢谢. 联系邮箱:liangshu ...