Android WebView 與JS的數據交互

轉載自:http://blog.csdn.net/cappuccinolau/article/details/8262821

關於WebView

我們知道目前android市場上的一些應用採用的開發方式大致分爲三種:Native AppWeb AppHybrid App。本文主要是Hybrid App中實現的主要技術native組件與js的數據交互的理解以及實現。

 

Android API中提供了WebView組件來實現對html的渲染。所謂的HybridApp開發方式即是彙集了HTML5CSS3jS的相關開發技術,以及數據交換格式json/XML。這顯然是Web開發工程師的技能。正是因爲如此,衆多中小企業選擇了這種開發方式來減少對android開發工程師的過度依賴,至於這三種開發方式的比較與優劣不在本文考慮之列。

 

有了WebView這個組件,Android應用開發技術也就轉嫁到htmljava數據交互上來。說白了就是jsWebView的數據交互,這就是本文所要討論的。

WebViewjs的數據交互

1.        WebView中載入靜態頁面

 

WebView添加到應用中。和原生控件一樣,在layout引入WebView控件。代碼片段如下:

[html] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:id="@+id/linearLayout"  
  4.     android:layout_width="match_parent"  
  5.     android:layout_height="match_parent"  
  6.     android:background="#000"  
  7.     android:orientation="horizontal" >  
  8. <WebView  
  9.     android:id="@+id/webview"  
  10.     android:layout_width="match_parent"  
  11.     android:layout_height="match_parent"   
  12.     />  
  13. </LinearLayout>  

載入頁面:

 

[java] view plaincopy
  1. webView = (WebView) findViewById(R.id.webview);  
  2. webView.loadUrl("file:///file:///android_asset/page.html");  

page.html存儲在工程文件的assets根目錄下。

2.        引入jquery mobile

引入js框架讓我們編寫的html頁面更接近於原生控件的顯示效果。目前主流的移動應用js框架有:jquery mobilesencha touchjquery mobilesencha touch的選型不在本文討論範圍)。本文選擇使用jquery mobile

 

首先,在webview添加對js的支持:

[java] view plaincopy
  1. WebSettings setting = webView.getSettings();  
  2. setting.setJavaScriptEnabled(true);//支持js  

增加對中文的支持:

[java] view plaincopy
  1. WebSettings setting = webView.getSettings();  
  2. setting.setDefaultTextEncodingName("GBK");//設置字符編碼  

設置頁面滾動條風格:

[java] view plaincopy
  1. webView.setScrollBarStyle(0);//滾動條風格,爲0指滾動條不佔用空間,直接覆蓋在網頁上  

jquery mobile提供的標準頁面模板TemplateForJQuery.html

[html] view plaincopy
  1. <!DOCTYPE html>   
  2. <html>   
  3.     <head>   
  4.     <title>Page Title</title>   
  5.       
  6.     <meta name="viewport" content="width=device-width, initial-scale=1">   
  7.   
  8.     <link rel="stylesheet" href="css/jquery.mobile-1.1.1.min.css" />  
  9.     <script src="js/jquery.js"></script>  
  10.     <script src="js/jquery.mobile-1.1.1.min.js"></script>  
  11. </head>   
  12. <body>   
  13.   
  14. <div data-role="page">  
  15.   
  16.     <div data-role="header">  
  17.         <h1>Page Title</h1>  
  18.     </div><!-- /header -->  
  19.   
  20.     <div data-role="content">   
  21.         <p>Page content goes here.</p>        
  22.     </div><!-- /content -->  
  23.   
  24.     <div data-role="footer">  
  25.         <h4>Page Footer</h4>  
  26.     </div><!-- /footer -->  
  27. </div><!-- /page -->  
  28.   
  29. </body>  
  30. </html>  

頁面依賴的js庫、css等均放在assets目錄下,目錄組織結構如下:

運行應用後的截圖:

下面是button 的截圖,與原生控件沒什麼明顯區別,有種以假亂真的感覺:

3.        良好的用戶體驗

運行我們的應用發現,在擁有大量js的頁面被載入時,一直處於等待中,這是很糟糕的用戶體驗。可以加入進度條解決。注意到webview提供的兩個方法:setWebViewClientsetWebChromeClient。其中setWebChromeClient方法正是可以處理progress的加載,此外,還可以處理js對話框,在webview中顯示icon圖標等。對於處理progress的代碼片段如下:

[java] view plaincopy
  1. webView.setWebChromeClient(new WebChromeClient() {  
  2.     public void onProgressChanged(WebView view, int progress) {// 載入進度改變而觸發  
  3.             if (progress == 100) {  
  4.                     handler.sendEmptyMessage(1);// 如果全部載入,隱藏進度對話框  
  5.             }  
  6.                 super.onProgressChanged(view, progress);  
  7.         }  
  8. });  

其中通過handler 消息機制來處理UI線程的更新:

 

[java] view plaincopy
  1. handler = new Handler() {  
  2.     public void handleMessage(Message msg) {// 定義一個Handler,用於處理下載線程與UI間通訊  
  3.         if (!Thread.currentThread().isInterrupted()){  
  4.             switch (msg.what) {  
  5.             case 0:  
  6.                 pd.show();// 顯示進度對話框  
  7.                 break;  
  8.             case 1:  
  9.                 pd.hide();// 隱藏進度對話框,不可使用dismiss()、cancel(),否則再次調用show()時,顯示的對話框小圓圈不會動。  
  10.                 break;  
  11.             }  
  12.         }  
  13.         super.handleMessage(msg);  
  14.     }  
  15. };  

對於setWebViewClient方法,一般用來處理html的加載(需要重載onPageStarted(WebView view, String url, Bitmap favicon))、關閉(需要重載onPageFinishedWebViewview, String url)方法)。

 

setWebViewClientsetWebChromeClient的作用:前者主要用於處理webView的控制問題,如加載、關閉、錯誤處理等;後者主要處理js對話框、圖標、頁面標題等。

4.        獲取java中的數據

單獨構建一個接口,作爲處理jsjava的數據交互的橋樑,本文封裝的代碼AndroidToastForJs.java如下:

[java] view plaincopy
  1. public class AndroidToastForJs {  
  2.       
  3.     private Context mContext;  
  4.   
  5. public AndroidToastForJs(Context context){  
  6.         this.mContext = context;  
  7.     }  
  8.       
  9. //webview中調用toast原生組件  
  10. public void showToast(String toast) {  
  11.         Toast.makeText(mContext, toast, Toast.LENGTH_SHORT).show();  
  12.     }  
  13.       
  14. //webview中求和  
  15. public int sum(int a,int b){  
  16.         return a+b;  
  17.     }  
  18.       
  19.  //以json實現webview與js之間的數據交互  
  20. public String jsontohtml(){  
  21.         JSONObject map;  
  22.         JSONArray array = new JSONArray();  
  23.         try {  
  24.             map = new JSONObject();  
  25.             map.put("name","aaron");  
  26.             map.put("age"25);  
  27.             map.put("address""中國上海");  
  28.             array.put(map);  
  29.               
  30.             map = new JSONObject();  
  31.             map.put("name","jacky");  
  32.             map.put("age"22);  
  33.             map.put("address""中國北京");  
  34.             array.put(map);  
  35.               
  36.             map = new JSONObject();  
  37.             map.put("name","vans");  
  38.             map.put("age"26);  
  39.             map.put("address""中國深圳");  
  40.             map.put("phone","13888888888");  
  41.             array.put(map);  
  42.         } catch (JSONException e) {  
  43.             e.printStackTrace();  
  44.         }  
  45.         return array.toString();  
  46.     }  
  47. }  


 

Webview提供的傳入js的方法:

[java] view plaincopy
  1. webView.addJavascriptInterface(new AndroidToastForJs(mContext), "JavaScriptInterface");  

Html頁面jsonData.html設計的部分代碼如下:

[html] view plaincopy
  1.     <script type="text/javascript">  
  2.     var result = JavaScriptInterface.jsontohtml();  
  3.     var obj = eval("("+result+")");//解析json字符串  
  4.     function showAndroidToast(toast)   
  5.     {          
  6.         JavaScriptInterface.showToast(toast);   
  7.     }  
  8.     function getjsonData(){  
  9.         var result = JavaScriptInterface.jsontohtml();  
  10.         var obj = eval("("+result+")");//解析json字符串  
  11.         for(i=0;i<obj.length;i++){  
  12.             var user=obj[i];  
  13.             document.write("<p>姓名:"+user.name+"</p>");  
  14.             document.write("<p>年齡:"+user.age+"</p>");  
  15.             document.write("<p>地址:"+user.address+"</p>");  
  16.             if(user.phone!=null){  
  17.                 document.write("<p>手機號碼:"+user.address+"</p>");  
  18.             }  
  19.         }  
  20.     }     
  21.     function list(){  
  22.         document.write("<div data-role='header'><p>another</p></div>");  
  23.     }  
  24.     </script>  
  25. </head>   
  26. <body>   
  27. <div data-role="page" >  
  28.     <div data-role="header" data-theme="c">  
  29.         <h1>Android via Interface</h1>  
  30.     </div><!-- /header -->  
  31.     <div data-role="content">   
  32.         <button value="say hello" onclick="showAndroidToast('Hello,Android!')" data-theme="e"></button>  
  33.         <button value="get json data" onclick="getjsonData()" data-theme="e"></button>    
  34.     </div><!-- /content -->  
  35. <div data-role="collapsible" data-theme="c" data-content-theme="f">  
  36.    <h3>I'm <script>document.write(obj[0].name);</script>,click to see my info</h3>  
  37.    <p><script>document.write("<p>姓名:"+obj[0].name+"</p>");</script></p>  
  38.    <p><script>document.write("<p>年齡:"+obj[0].age+"</p>");</script></p>  
  39.    <p><script>document.write("<p>地址:"+obj[0].address+"</p>");</script></p>  
  40. </div>  
  41.     <div data-role="footer" data-theme="c">  
  42.         <h4>Page Footer</h4>  
  43.     </div><!-- /footer -->  
  44. </div><!-- /page -->  
  45. </body>  

點擊say hello按鈕運行的截圖如下:

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章