【WebView】WebView的基本使用

每个项目或简单或复杂的都会使用到WebView,那么这篇就对WebView的基本使用做一个简单的介绍。

WebView是什么,能做什么

官方类继承结构.png

WebView是android提供的展示网页的视图,基于这个类你可以在你的app内启动网页浏览器或者只是简单的展示在线的内容。
WebView是一个基于webkit引擎、展现web页面的控件。Android的WebView在4.4之前是基于WebKit内核实现的,而4.4开始基于Chrome内核实现。而Chrome所使用的WebKit内核与之前WebView采用的WebKit内核是不同的。

1. 首先,要使用WebView,自然创建一个WebView,创建的方式有两种:

  • 在xml中:
<WebView
    android:id="@+id/web_view"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"/>
  • 在java代码中:
WebView view = new WebView(this);

2. 创建WebView,最基本的当然是需要展示相关内容,根据内容来源可以采用不同的方法。

private String content = "<html>" +
            "<title>题目没想好</title>"+
            "<h1>这只是一个测试页面</h1>" +
            "<body>" +
            "<p>我开始选择1种生活方式,培养1种思维,考虑承担社会角色中的各种责任(现在的和未来的),开始养成1,2,3种习惯,而这其中还需要有很多细节,这所有的一切来源于追求。</p>" +
            "<p><font color=\"red\"> 任重道远,至死方休。</font></p>" +
            "<p><a href=\"http://www.baidu.com\">众里寻她千百度</a></p>"+
            "</body>" +
            "</html>";

/**
 * 加载一段html代码的字符串
 * data 内容字符串
 * mimeType  内容的媒体类型
 * encoding 编码类型
 */
wv.loadData(String data,String mimeType,String encoding);//方法原型
wv.loadData(content,"text/html","utf-8");//方法示例(错误),如果不是英文和数字会有乱码的问题,官方文档说除了URL和base64以外,其他的声明都是按照ascII来设置
wv.loadData(content,"text/html;charset=UTF-8",null);//方法示例,以这种方式会解决乱码问题

/**
 * 这个方法我同样用来加载一段html代码的字符串
 */
wv.loadDataWithBaseURL(String baseUrl,String data,String mimeType,String encoding,String historyUrl);//方法原型
wv.loadDataWithBaseURL(null,content,"text/html","utf-8",null);//方法示例,同样这种方式也没有问题

wv.loadUrl(String url);//方法原型
wv.loadUrl("file:///android_asset/***.html");//方法示例,加载项目assets下的html页面,html页面自己定义,其他部分不变
wv.loadUrl("http://www.baidu.com");//方法示例,加载网页

3. WebView将内容显示出来之后,你可能会发现效果跟自己想要的效果有些出入,那么怎么来对WebView进行一些定制化的设置呢?这就不得不提到WebSettings类。

WebSettings.png

WebSettings是用来管理WebView设置的抽象类。当WebView第一次创建时,相应包含一个WebSettings子类的对象,此WebSettings对象中有一套默认的设置,WebSettings对象的生命周期和其WebView的生命周期绑定,如果WebView对象被销毁,再去使用WebSettings的任何方法则会报错。

  • 看看WebSettings部分相关的具体使用方法
//获取WebSettings对象
WebSettings webSettings = webView.getSettings();

//设置自适应屏幕,两者合用
webSettings.setUseWideViewPort(true); //将内容调整到适合webview的大小 
webSettings.setLoadWithOverviewMode(true); // 缩放至屏幕的大小

//设置滚动条
mWebView.setHorizontalScrollBarEnabled(false);//水平滚动条不显示
mWebView.setVerticalScrollBarEnabled(false); //垂直滚动条不显示

//缩放操作
webSettings.setSupportZoom(true); //支持缩放,默认为true。可以不显示设置,但是如果设置为false,则下面方法如何设置都不再支持缩放
webSettings.setBuiltInZoomControls(true); //设置内置的缩放控件。若为false,则该WebView不可缩放
webSettings.setDisplayZoomControls(false); //隐藏原生的缩放控件,否则放大和缩小过程中会出现放大缩小图标,虽然放大和缩小动作结束后稍后就会消失,但是实在是有些丑

webSettings.setJavaScriptEnabled(true);  //如果访问的页面中要与Javascript交互,则webview必须设置支持Javascript
webSettings.setAllowFileAccess(true); //设置可以访问文件 ,比如当网页中存在文本文件时就可以打开查看

// 若加载的 html 里有JS 在执行动画等操作,会造成资源浪费(CPU、电量)
// 在 onStop 和 onResume 里分别把 setJavaScriptEnabled() 给设置成 false 和 true 即可

//其他细节操作
webSettings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK); //关闭webview中缓存 
webSettings.setJavaScriptCanOpenWindowsAutomatically(true); //支持通过JS打开新窗口 
webSettings.setLoadsImagesAutomatically(true); //支持自动加载图片
webSettings.setDefaultTextEncodingName("utf-8");//设置编码格式

4. 样式也调好了,如果仍然需要对WebView的内部一些链接做处理、WebView的加载状态甚至是具体的进度,这时候就需要对WebViewClient类和WebChromeClient类做一些简单的了解了。

这两个类直接继承自Object。如果WebView只是用来处理一些html的页面内容,只用WebViewClient就行了,如果需要更丰富的处理效果,比如Javascript 的对话框,网站图标,网站标题,进度条处理等,就要用到WebChromeClient。

  • WebViewClient代码示例,主要说明处理WebView中的Uri的方法。
//尽管有些方法已经废弃,但是代替它的方法在API级别较高时才支持,这样的话还是选择用之前的方法
wv.setWebViewClient(new WebViewClient(){ 
      @Override
      public boolean shouldOverrideUrlLoading(WebView view, String url) {// 重写此方法击网页里面的链接还是在当前的webview里跳转,不跳到手机浏览器
             if(isHttpLink(url)){//同样可以根据判断具体的url类型来做不同的处理,比如电话号码就可以跳转手机的拨号页面
                   view.loadUrl(url);
             }
             return true;
     }
        @Override
        public void onPageStarted(WebView view, String url, Bitmap favicon) {
            //设定加载开始的动作
        }

        @Override
        public void onPageFinished(WebView view, String url) {
            //设定结束时的动作,比如需要等待webview加载完毕才显示一些其他的布局或者加载框的消失
        }

        @Override
        public void onReceivedError(WebView view, int errorCode, String description, String failingUrl)  {
            //出现错误时,根据错误类型展示一个自定义的h5页面或者错误提示
        }
 });

//判断url的类型
private boolean isHttpLink(String url) {
        if (url.startsWith("http:") || url.startsWith("https:")) {
            return true;
        }
        return false;
    }
  • WebChromeClient代码示例
wv.setWebChromeClient(new WebChromeClient() {
            @Override
            public void onProgressChanged(WebView view, int newProgress) {
                super.onProgressChanged(view, newProgress);
                if (newProgress < 100) {
                    String progress = newProgress + "%";
                } else {

                }
            }

            @Override
            public void onReceivedTitle(WebView view, String title) {
                super.onReceivedTitle(view, title);
                tvTitle.setText(title);
            }

            /**
             *告诉客户端显示一个JavaScript警告对话框。
             * 如果客户端返回true,则WebView将假定客户端将处理该对话框。
             * 如果客户端返回false,它将继续执行。
             */
            @Override
            public boolean onJsAlert(WebView view, String url, String message, final JsResult result) {
                new AlertDialog.Builder(WebViewLoadActivity.this)
                        .setTitle("JsAlert")
                        .setMessage(message)
                        .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                result.confirm();
                            }
                        })
                        .setCancelable(false)
                        .show();
                return true;

            }

            /**
             * 告诉客户端向用户显示一个确认对话框。
             * 如果客户端返回true,WebView将假定客户端将处理确认对话框并调用相应的JsResult方法。
             * 如果客户端返回false,则默认值false将返回到JavaScript。 默认行为是返回false。
             */
            @Override
            public boolean onJsConfirm(WebView view, String url, String message, final JsResult result) {
                new AlertDialog.Builder(WebViewLoadActivity.this)
                        .setTitle("JsConfirm")
                        .setMessage(message)
                        .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                result.confirm();
                            }
                        })
                        .setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                result.cancel();
                            }
                        })
                        .setCancelable(false)
                        .show();
                return true;
            }

            /**
             * 告诉客户端向用户显示一个提示对话框。
             * 如果客户端返回true,则WebView将假定客户端将处理提示对话框并调用相应的JsPromptResult方法。
             * 如果客户端返回false,则默认值false将返回到javascript。 默认行为是返回false。
             */
            @Override
            public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, final JsPromptResult result) {
                final EditText et = new EditText(WebViewLoadActivity.this);
                et.setText(defaultValue);
                new AlertDialog.Builder(WebViewLoadActivity.this)
                        .setTitle(message)
                        .setView(et)
                        .setPositiveButton("OK", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                result.confirm(et.getText().toString());
                            }
                        })
                        .setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                result.cancel();
                            }
                        })
                        .setCancelable(false)
                        .show();

                return true;
            }
        });

5. 销毁WebView

//销毁Webview,释放资源
    @Override
    protected void onDestroy() {
        if (mWebview != null) {
            mWebview.loadDataWithBaseURL(null, "", "text/html", "utf-8", null);
            mWebview.clearHistory();
            ((ViewGroup) mWebview.getParent()).removeView(mWebview);
            mWebview.destroy();
            mWebview = null;
        }
        super.onDestroy();
    }

WebView的基本使用就先写到这里,这不是结束,而是开始!
对于WebView与js的互相调用,虽然demo也写了,但是觉得有不错的博文,思路也很清晰。传送地址下面已放好。

[参考博文]
http://www.jianshu.com/p/3c94ae673e2a
http://www.jianshu.com/p/345f4d8a5cfa

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

推荐阅读更多精彩内容