需要测试一下H5页面在安卓API 28版本下的兼容问题,所以使用IDEA编写一个带有webview的安卓程序,记录步骤和遇到问题如下:
使用的IDEA版本为2019.1
一、新建一个空白安卓项目
File-New Project(IDEA需要安装好安卓开发插件)
选择你需要的最低支持API版本(就是最低支持运行的安卓版本)
完成后生产以下项目目录
安卓依赖需要使用gradle管理,所以先下载并安装gradle
并配置系统环境变量(具体安装方法略)
成功后根据IDEA提示导入Gradle项目,注意需要配置为本地gradle目录,且Gradle JVM需要配置为本地Javahome
等待Gradle安装依赖
成功后build项目,可能会报错:License for package Android SDK Build-Tools 28.0.2 not accepted.
原因是现在使用某个版本的安卓sdk都需要同意他的License。
解决方式:
进到sdk/tools/bin目录下
在此目录打开cmd或powershell 执行
./sdkmanager --licenses
sdkmanager会打开多个License 让你一一同意,全部输入y就可以。
同意后再进行编译即可成功
编译版本配置可在build.gradle文件中查看修改:
二、使用真机运行
配置Run/Debug Configurations,选择要运行的模块
点击Run或者Debug,选择连上的真机(开启usb调试)
真机上启动成功后可看到Hello world,至此新建并启动空白项目成功。
三、为APP添加并配置一个webview
打开AndroidManifest.xml文件,配置app主题、主页面、并允许访问网络。
此处使用了无头部主题且Activity类型为AppCompat。
打开src/main/res/layout/activity_main.xml配置布局
使用FrameLayout布局,放一个webview和一个ProgressBar(进度条)
打开src/main/java/com.example.myapplication/MainActivity开发webview和进度条功能。
完整代码:
package com.example.myapplication;
import android.app.AlertDialog;
import android.graphics.Bitmap;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.webkit.*;
import android.widget.ProgressBar;
public class MainActivity extends AppCompatActivity {
private WebView webView;
private ProgressBar progressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
progressBar= (ProgressBar)findViewById(R.id.progressbar);//进度条
webView = (WebView) findViewById(R.id.webview);
webView.loadUrl("https://www.baidu.com");//加载url
//使用webview显示html代码
// webView.loadDataWithBaseURL(null,"<html><head><title> 欢迎您 </title></head>" +
// "<body><h2>使用webview显示 html代码</h2></body></html>", "text/html" , "utf-8", null);
webView.addJavascriptInterface(this,"android");//添加js监听 这样html就能调用客户端
webView.setWebChromeClient(webChromeClient);
webView.setWebViewClient(webViewClient);
WebSettings webSettings=webView.getSettings();
/**
* LOAD_CACHE_ONLY: 不使用网络,只读取本地缓存数据
* LOAD_DEFAULT: (默认)根据cache-control决定是否从网络上取数据。
* LOAD_NO_CACHE: 不使用缓存,只从网络获取数据.
* LOAD_CACHE_ELSE_NETWORK,只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。
*/
webSettings.setCacheMode(WebSettings.LOAD_DEFAULT);//不使用缓存,只从网络获取数据.
webView.getSettings().setTextZoom(100);
webView.getSettings().setJavaScriptCanOpenWindowsAutomatically(true);//设置js可以直接打开窗口,如window.open(),默认为false
webView.getSettings().setJavaScriptEnabled(true);//是否允许执行js,默认为false。设置true时,会提醒可能造成XSS漏洞
webView.getSettings().setSupportZoom(true);//是否可以缩放,默认true
webView.getSettings().setBuiltInZoomControls(true);//是否显示缩放按钮,默认false
webView.getSettings().setUseWideViewPort(true);//设置此属性,可任意比例缩放。大视图模式
webView.getSettings().setLoadWithOverviewMode(true);//和setUseWideViewPort(true)一起解决网页自适应问题
webView.getSettings().setAppCacheEnabled(true);//是否使用缓存
webView.getSettings().setDomStorageEnabled(true);//DOM Storage
}
//WebViewClient主要帮助WebView处理各种通知、请求事件
private WebViewClient webViewClient=new WebViewClient(){
@Override
public void onPageFinished(WebView view, String url) {//页面加载完成
progressBar.setVisibility(View.GONE);
}
@Override
public void onPageStarted(WebView view, String url, Bitmap favicon) {//页面开始加载
progressBar.setVisibility(View.VISIBLE);
}
@Override
public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
Log.i("ansen","拦截url:"+request.getUrl());
return super.shouldOverrideUrlLoading(view, request);
}
};
//WebChromeClient主要辅助WebView处理Javascript的对话框、网站图标、网站title、加载进度等
private WebChromeClient webChromeClient=new WebChromeClient(){
//不支持js的alert弹窗,需要自己监听然后通过dialog弹窗
@Override
public boolean onJsAlert(WebView webView, String url, String message, JsResult result) {
AlertDialog.Builder localBuilder = new AlertDialog.Builder(webView.getContext());
localBuilder.setMessage(message).setPositiveButton("确定",null);
localBuilder.setCancelable(false);
localBuilder.create().show();
//注意:
//必须要这一句代码:result.confirm()表示:
//处理结果为确定状态同时唤醒WebCore线程
//否则不能继续点击按钮
result.confirm();
return true;
}
//获取网页标题
@Override
public void onReceivedTitle(WebView view, String title) {
super.onReceivedTitle(view, title);
Log.i("ansen","网页标题:"+title);
}
//加载进度回调
@Override
public void onProgressChanged(WebView view, int newProgress) {
progressBar.setProgress(newProgress);
}
};
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
Log.i("ansen","是否有上一个页面:"+webView.canGoBack());
if (webView.canGoBack() && keyCode == KeyEvent.KEYCODE_BACK){//点击返回按钮的时候判断有没有上一页
webView.goBack(); // goBack()表示返回webView的上一页面
return true;
}
return super.onKeyDown(keyCode,event);
}
/**
* JS调用android的方法
* @param str
* @return
*/
@JavascriptInterface //仍然必不可少
public void getClient(String str){
Log.i("ansen","html调用客户端:"+str);
}
@Override
protected void onDestroy() {
super.onDestroy();
//释放资源
webView.destroy();
webView=null;
}
}
特别需要注意其中的webview配置,如是否允许js、是否允许js新打开窗口、系统缩放字体、缓存策略、DOM Storage等,安卓默认情况下都是false的,需要全部打开才可保证h5运行正常。
之后只要配置 webView.loadUrl 来选择需要打开的页面,编译在真机运行调试即可。
四、打包为apk文件
build菜单中选择:
这里选择普通apk
选择你的key文件和密码,如果没有可以选择create new新建到本地
选择生成包路径、版本和签名类型:
等待gradle打包,成功后即可看到apk文件。
在真机中允许未知来源应用,即可使用此apk安装使用了!