需要测试一下H5页面在安卓API 28版本下的兼容问题,所以使用IDEA编写一个带有webview的安卓程序,记录步骤和遇到问题如下:

使用的IDEA版本为2019.1

一、新建一个空白安卓项目

File-New Project(IDEA需要安装好安卓开发插件)

android idea 创建工程 idea怎么写安卓程序_android


选择你需要的最低支持API版本(就是最低支持运行的安卓版本)

android idea 创建工程 idea怎么写安卓程序_android_02


完成后生产以下项目目录

android idea 创建工程 idea怎么写安卓程序_ide_03


安卓依赖需要使用gradle管理,所以先下载并安装gradle

并配置系统环境变量(具体安装方法略)

成功后根据IDEA提示导入Gradle项目,注意需要配置为本地gradle目录,且Gradle JVM需要配置为本地Javahome

android idea 创建工程 idea怎么写安卓程序_webview_04


等待Gradle安装依赖

android idea 创建工程 idea怎么写安卓程序_webview_05


成功后build项目,可能会报错:License for package Android SDK Build-Tools 28.0.2 not accepted.

原因是现在使用某个版本的安卓sdk都需要同意他的License。解决方式:

进到sdk/tools/bin目录下

android idea 创建工程 idea怎么写安卓程序_ide_06


在此目录打开cmd或powershell 执行

./sdkmanager --licenses

sdkmanager会打开多个License 让你一一同意,全部输入y就可以。

同意后再进行编译即可成功

android idea 创建工程 idea怎么写安卓程序_webview_07


编译版本配置可在build.gradle文件中查看修改:

android idea 创建工程 idea怎么写安卓程序_webview_08

二、使用真机运行

配置Run/Debug Configurations,选择要运行的模块

android idea 创建工程 idea怎么写安卓程序_ide_09


点击Run或者Debug,选择连上的真机(开启usb调试)

android idea 创建工程 idea怎么写安卓程序_android idea 创建工程_10


真机上启动成功后可看到Hello world,至此新建并启动空白项目成功。

三、为APP添加并配置一个webview

打开AndroidManifest.xml文件,配置app主题、主页面、并允许访问网络。

此处使用了无头部主题且Activity类型为AppCompat。

android idea 创建工程 idea怎么写安卓程序_android_11

打开src/main/res/layout/activity_main.xml配置布局

使用FrameLayout布局,放一个webview和一个ProgressBar(进度条)

android idea 创建工程 idea怎么写安卓程序_android idea 创建工程_12


打开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菜单中选择:

android idea 创建工程 idea怎么写安卓程序_android idea 创建工程_13


这里选择普通apk

android idea 创建工程 idea怎么写安卓程序_android_14


选择你的key文件和密码,如果没有可以选择create new新建到本地

android idea 创建工程 idea怎么写安卓程序_Android_15


选择生成包路径、版本和签名类型:

android idea 创建工程 idea怎么写安卓程序_Android_16


等待gradle打包,成功后即可看到apk文件。

android idea 创建工程 idea怎么写安卓程序_ide_17

在真机中允许未知来源应用,即可使用此apk安装使用了!