WebView(H5)調(diào)用Android端代碼

WebView調(diào)用Android代碼

做好戰(zhàn)斗準備,其實也不復(fù)雜

情景再現(xiàn):
運營的H5頁面上有個按鈕,叫“立即參加”悄雅,當用戶點擊按鈕的時候芹助,App上要彈一個吐司出來。

一囤踩、通過WebView的addJavascriptInterface ()方式進行映射

1.準備工作
定義我們將來要響應(yīng)js代碼的類

/**
 * 與h5交互的共同類
 */
public class ForJs {
    //添加注解,不添加注解方法不能夠被js調(diào)用
  //注意方法名稱彤守,提供給js的時候也一定要匹配
    @JavascriptInterface
    public void fromAndroid(){
        //我們具體要執(zhí)行的邏輯历极,本例只是彈一個吐司
        ToastUtil.showToast("這個是android里的方法");
    }
 // 同上,只是一個有參米死,一個無參而已
    @JavascriptInterface
    public void fromAndroid(String msg){
        //我們具體要執(zhí)行的邏輯锌历,本例只是彈一個吐司
        ToastUtil.showToast("msg");
    }
}

至此,準備工作就已經(jīng)結(jié)束峦筒。
2.使用

  • android端代碼
        //如果只需要讓js調(diào)用android代碼究西,這幾行就夠了
        WebSettings settings = mWebView.getSettings();
        settings.setJavaScriptEnabled(true);
        mWebView.addJavascriptInterface(new ForJs(), "testMall");

參數(shù)說明:
1、我們定義的與H5交互的java類
2物喷、java類在H5中映射的對象名稱
H5中的js在調(diào)用我們的方法的時候卤材,必須知道的是:
映射對象的名稱遮斥,即示例中的testMall
映射對象提供的方法名稱,即示例中的ForJs類中的fromAndroid()
H5中的關(guān)鍵代碼
注意這部分代碼并不需要我們Android端人員編寫商膊,但是卻需要我們Android端人員的配合

//js中的點擊方法調(diào)用伏伐,調(diào)用的是Android端的無參方法
<div onclick="test()" ></div>
<script>
//js中方法的定義
        function test(){
        testMall.fromAndroid()//注意 testMall 對象,以及fromAndroid()方法名稱
        }
</script>

重復(fù)一下注意事項:

Androdi端提供給js調(diào)用的方法名稱要準確無誤的告訴前端人員
Js中映射的Java對象名稱,要準確無誤的告訴Android端人員

一定要看

這種使用方式看起來很簡單晕拆,但是卻有一個致命漏洞:
http://www.reibang.com/p/3a345d27cd42

二藐翎、通過WebViewClinet的shouldOverrideUrlLoading()方法攔截url

  • 攔截該Url
  • 解析Url協(xié)議()//新的API可以直接獲取到uri
  • 按照約定調(diào)用Android端的方法
    1.Android端代碼
 mWebView.loadUrl("file:///android_asset/test.html");
        WebSettings settings = mWebView.getSettings();
        settings.setJavaScriptEnabled(true);
        mWebView.setWebViewClient(new WebViewClient(){
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
                //請求方式
                String method = request.getMethod();
                //請求頭
                Map<String, String> requestHeaders = request.getRequestHeaders();
                //uri,給里面有各種我們需要的東西,只需要get即可
                Uri uri = request.getUrl();
                String host = uri.getHost();
                String authority = uri.getAuthority();
                // TODO: 2018/12/15 判斷实幕,如果符合約定吝镣,即執(zhí)行Android提供好的方法,最后記得return true
              //模擬微信封殺抖音鏈接
                if (authority.contains("douyin")){
                    ToastUtil.showToast("不要看抖音,微信不夠豐富嗎");
                    return true;
                }
                return super.shouldOverrideUrlLoading(view, request);
            }

            @Override
            public boolean shouldOverrideUrlLoading(WebView view, String url) {
                //如果使用這個方法昆庇,那么需要自己手動獲取uri末贾,其實與上面并無本質(zhì)區(qū)別
                Uri uri = Uri.parse(url);
                return super.shouldOverrideUrlLoading(view, url);
            }
        });

至此,Android端代碼完成
2.H5端代碼
注意這部分代碼并不需要我們Android端人員編寫整吆,但是卻需要我們Android端人員與H5端人員協(xié)商好協(xié)議約定

//js中的點擊方法調(diào)用
<div onclick="test()" ></div>

    <script>
          //js中方法的定義
        function test(){
          document.location="https://www.douyin.com/"
        }
    </script>

與第一種方法比起來拱撵,這種寫法雙方寫起來都比較隨意。除了Url的協(xié)議約定外表蝙,別的沒有需要注意事項拴测。尤其H5端,其實只是重定向了一個網(wǎng)頁而已府蛇。
換句話說集索,在本示例中,如果我們直接在webView中加載
https://www.douyin.com/汇跨,也會觸發(fā)我們的Android中的方法务荆,即攔截邏輯。

更重要的是穷遂,第二種方式?jīng)]有第一種方式那種致命漏洞函匕。即利用shouldOverrideUrlLoading方法不存在漏洞

缺點:難以獲得返回值,如果想要獲得返回值蚪黑,那么就利用Android端調(diào)用Js的方法webView的loadUrl(restul)浦箱,將Android端的返回值result當做參數(shù)傳遞給js。

Android端示例代碼:

 WebSettings settings = mWebView.getSettings();
        settings.setJavaScriptEnabled(true);
        mWebView.setWebViewClient(new WebViewClient() {
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
                Uri uri = request.getUrl();
                String authority = uri.getAuthority();
                //模擬代碼
                if (authority.contains("douyin")) {
                    String result = "這是Android端執(zhí)行完成之后的結(jié)果";
//獲取android端的結(jié)果后調(diào)用js中的方法祠锣,把result當參數(shù)傳遞給js
mWebView.loadUrl("javascript:callJsFromAndroid("+result+")");
                    return true;
                }
                return super.shouldOverrideUrlLoading(view, request);
            }
        });

        //注意酷窥,本示例之所以加此行代碼,是為了能彈出js框伴网,具體要看實際需求
        mWebView.setWebChromeClient(new WebChromeClient());
        mWebView.loadUrl("file:///android_asset/test.html");

H5端示例代碼


<script>
        //result是android端的處理結(jié)果
        function callJsFromAndroid(result){
          alert(result)
        }
    </script>

三蓬推、通過WebChromeClient的alert()、prompt()澡腾、confirm()方法回調(diào)js對應(yīng)的對話框

我在http://www.reibang.com/p/22265f5c5d78中詳細介紹了Android端響應(yīng)這三種js彈框的方式沸伏,如果不熟悉的請先了解一下糕珊。一般來將,我們使用的最多的是prompt()方法毅糟,因為這個有任意類型的返回值,請看示例代碼

Android端代碼

  WebSettings settings = mWebView.getSettings();
        settings.setJavaScriptEnabled(true);
        //本行代碼添加只是為了讓網(wǎng)頁在app內(nèi)打開
        mWebView.setWebViewClient(new WebViewClient());
        
        mWebView.setWebChromeClient(new WebChromeClient(){
            @Override
            public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) {
                //message即是js彈框中的內(nèi)容
                //解析message红选,按照約定處理邏輯
                if (TextUtils.equals(message,"douyin")) {
                    // android端的處理邏輯
                    ToastUtil.showToast("你們怎么都那么好看又有錢");
                    return true;
                }
                return super.onJsPrompt(view, url, message, defaultValue, result);
            }
        });
        mWebView.loadUrl("file:///android_asset/test.html");

H5端示例代碼

//js中的點擊方法調(diào)用
<div onclick="test()" ></div>
<script>
        function test(){
          prompt("douyin")
        }

</script>

三種使用方式的對比

1.比較簡單,但存在致命漏洞
2.需要雙方約定協(xié)議姆另,但不存在漏洞
3.與方式2很相似喇肋,只是攔截對象不同,也不存在漏洞

寫在最后
這些交互方式各有特點迹辐,而且并不難以理解蝶防,個人認為都應(yīng)該掌握。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末明吩,一起剝皮案震驚了整個濱河市间学,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌印荔,老刑警劉巖低葫,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異仍律,居然都是意外死亡嘿悬,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進店門染苛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來鹊漠,“玉大人主到,你說我怎么就攤上這事茶行。” “怎么了登钥?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵畔师,是天一觀的道長。 經(jīng)常有香客問我牧牢,道長看锉,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任塔鳍,我火速辦了婚禮伯铣,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘轮纫。我一直安慰自己腔寡,他們只是感情好,可當我...
    茶點故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布掌唾。 她就那樣靜靜地躺著放前,像睡著了一般忿磅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上凭语,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天葱她,我揣著相機與錄音,去河邊找鬼似扔。 笑死吨些,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的虫几。 我是一名探鬼主播锤灿,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼辆脸!你這毒婦竟也來了但校?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤啡氢,失蹤者是張志新(化名)和其女友劉穎状囱,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體倘是,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡亭枷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了搀崭。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片叨粘。...
    茶點故事閱讀 38,724評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖瘤睹,靈堂內(nèi)的尸體忽然破棺而出升敲,到底是詐尸還是另有隱情,我是刑警寧澤轰传,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布驴党,位于F島的核電站,受9級特大地震影響获茬,放射性物質(zhì)發(fā)生泄漏港庄。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一恕曲、第九天 我趴在偏房一處隱蔽的房頂上張望鹏氧。 院中可真熱鬧,春花似錦佩谣、人聲如沸把还。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽笨篷。三九已至瞳秽,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間率翅,已是汗流浹背练俐。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留冕臭,地道東北人腺晾。 一個月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像辜贵,于是被迫代替她去往敵國和親悯蝉。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,627評論 2 350

推薦閱讀更多精彩內(nèi)容