Android中WebView的開發(fā)

在Android手機(jī)中內(nèi)置了一款高性能webkit內(nèi)核瀏覽器棉浸,在SDK中封裝為一個(gè)叫做WebView組件抵卫。下面總結(jié)一下使用webview

一、webview的基本使用方法
1. 添加權(quán)限:AndroidManifest.xml中設(shè)置權(quán)限"android.permission.INTERNET",否則會出Web page not available錯(cuò)誤逻炊。
2. 在要Activity中生成一個(gè)WebView組件:WebView webView = new WebView(this);或者可以在activity的layout文件里添加webview控件
3. 設(shè)置WebView基本信息:

 mWebView = (WebView) findViewById(R.id.wb);
 mWebView.getSettings().setJavaScriptEnabled(true);//支持javascript
 mWebView.requestFocus();//觸摸焦點(diǎn)起作用    
 mWebView.setScrollBarStyle(WebView.SCROLLBARS_OUTSIDE_OVERLAY);//取消滾動條
 mWebView.getSettings().setJavaScriptCanOpenWindowsAutomatically(true);//設(shè)置允許js彈出alert對話框
 
 mWebView.addJavascriptInterface(new JsInteration(), "control");//js訪問android秤掌,定義接口
   //設(shè)置了Alert才會彈出,重新onJsAlert()方法return true可以自定義處理信息
 mWebView.setWebChromeClient(new WebChromeClient() {    
@Override    
  public boolean onJsAlert(WebView view, String url, String message, JsResult result)
  {        
  //return super.onJsAlert(view, url, message, result);        
  Toast.makeText(MainActivity.this, message, Toast.LENGTH_LONG).show();        
  return true;  
  }
  });

4. 設(shè)置WevView要顯示的網(wǎng)頁:互聯(lián)網(wǎng)用:webView.loadUrl("http://www.google.com"); 本地文件用:webView.loadUrl("file:///android_asset/XX.html"); 本地文件存放在:assets文件中
5. 如果希望點(diǎn)擊鏈接由自己處理再来,而不是新開Android的系統(tǒng)browser中響應(yīng)該鏈接蒙兰。給WebView添加一個(gè)事件監(jiān)聽對象(WebViewClient)并重寫其中的一些方法: shouldOverrideUrlLoading:對網(wǎng)頁中超鏈接按鈕的響應(yīng)。當(dāng)按下某個(gè)連接時(shí)WebViewClient會調(diào)用這個(gè)方法芒篷,并傳遞參數(shù)

 public boolean shouldOverrideUrlLoading(WebView view,String url){ view.loadUrl(url); return true; }

6. 處理https請求webView默認(rèn)是不處理https請求的搜变,頁面顯示空白,需要進(jìn)行如下設(shè)置:

  webView.setWebViewClient(new WebViewClient() { 
    @Override
      public void onReceivedSslError(WebView view, 
     SslErrorHandler handler, SslError error) { 
       handler.proceed(); 
     // handler.cancel(); 
   // handler.handleMessage(null); }
   });

onReceivedSslError為webView處理ssl證書設(shè)置

其中handler.proceed();表示等待證書響應(yīng)handler.cancel();表示掛起連接针炉,為默認(rèn)方式handler.handleMessage(null);
可做其他處理另外還有其他一些可重寫的方法
1挠他,接收到Http請求的事件onReceivedHttpAuthRequest(WebView view, HttpAuthHandler handler, String host, String realm)
2,載入頁面完成的事件public void onPageFinished(WebView view, String url){ }同樣道理篡帕,我們知道一個(gè)頁面載入完成殖侵,于是我們可以關(guān)閉loading條,切換程序動作镰烧。
3拢军,載入頁面開始的事件public void onPageStarted(WebView view, String url, Bitmap favicon) { }這個(gè)事件就是開始載入頁面調(diào)用的,通常我們可以在這設(shè)定一個(gè)loading的頁面怔鳖,告訴用戶程序在等待網(wǎng)絡(luò)響應(yīng)茉唉。 通過這幾個(gè)事件,我們可以很輕松的控制程序操作结执,一邊用著瀏覽器顯示內(nèi)容度陆,一邊監(jiān)控著用戶操作實(shí)現(xiàn)我們需要的各種顯示方式,同時(shí)可以防止用戶產(chǎn)生誤操作昌犹。

4. 如果用webview點(diǎn)鏈接看了很多頁以后坚芜,如果不做任何處理,點(diǎn)擊系統(tǒng)“Back”鍵斜姥,整個(gè)瀏覽器會調(diào)用finish()而結(jié)束自身,如果希望瀏覽的網(wǎng)頁回退而不是退出瀏覽器沧竟,需要在當(dāng)前Activity中處理并消費(fèi)掉該Back事件铸敏。 覆蓋Activity類的onKeyDown(int keyCoder,KeyEvent event)方法。

 @Overridepublic boolean onKeyDown(int keyCode, KeyEvent event) {  
 if (keyCode == KeyEvent.KEYCODE_BACK && webView.canGoBack()) {    
 webView.goBack();// 返回前一個(gè)頁面   
  return true;   
}    
return super.onKeyDown(keyCode, event);
}

二悟泵、webview與js的交互(相互調(diào)用參數(shù)杈笔、傳值)
前端網(wǎng)頁全部代碼,文章最后有示例項(xiàng)目完整源碼

<!DOCTYPE html><html><head>    <meta charset="utf-8">    
<title>jaydenxiao遇上了webview</title>    
<script>
function sayHello() {
alert("我是無參無返回toast") 
}
function alertMessage(message) { 
alert(message)  
}
function toastMessage(message) {
window.control.toastMessage(message)
}
function sumToJava(number1, number2){
window.control.onSumResult(number1 + number2)
}
function sumToJava2(number1, number2) {
return number1 + number2;
}    
</script>
</head><body>
 <button type="button" id="button" onclick="toastMessage('js調(diào)用了android方法')">js訪問android中方法
      
</button>
  </body>
  </html>

調(diào)用示例:
js調(diào)用Java
調(diào)用格式為window.jsInterfaceName.methodName(parameterValues)此例中我們使用的是control作為注入接口名稱。

function toastMessage(message) { 
window.control.toastMessage(message) 
}
function sumToJava(number1, number2){ 
window.control.onSumResult(number1 + number2)
 }

Java調(diào)用JS
webView調(diào)用js的基本格式為webView.loadUrl(“javascript:methodName(parameterValues)”)****1. android調(diào)用js無參無返回值函數(shù)

 final String call = "javascript:sayHello()";
 mWebView.post(new Runnable() {    
  @Override    
  public void run() { 
   mWebView.loadUrl(call);    
 }
 });

2. android調(diào)用js有參無返回值函數(shù)

final String call = "javascript:alertMessage(\"" + "我是android傳過來的內(nèi)容,hey man" + "\")";
 mWebView.post(new Runnable() {    
 @Override    
   public void run() { 
    mWebView.loadUrl(call);    
  }
}
);

3. android調(diào)用js有參有返回值函數(shù)(4.4之前)
Android在4.4之前并沒有提供直接調(diào)用js函數(shù)并獲取值的方法糕非,所以在此之前蒙具,常用的思路是 java調(diào)用js方法球榆,js方法執(zhí)行完畢,再次調(diào)用java代碼將值返回禁筏。
(1).android調(diào)用js代碼

   final String call = "javascript:sumToJava(1,2)";mWebView.post(new Runnable() { 
   @Override public void run() { 
   mWebView.loadUrl(call); }
   });

(2).js函數(shù)處理持钉,并將結(jié)果通過調(diào)用android方法返回
網(wǎng)頁端:

function sumToJava(number1, number2){ window.control.onSumResult(number1 + number2) }

(3).android在回調(diào)方法中獲取js函數(shù)返回值

@JavascriptInterfacepublic void onSumResult(int result) { 
  Toast.makeText(getApplicationContext(), "我是android調(diào)用js方法(4.4前),入?yún)⑹?和2篱昔,
       js返回結(jié)果是" +    result, Toast.LENGTH_LONG).show();
   }

4. android調(diào)用js有參有返回值函數(shù)(4.4以上):****
Android 4.4以上使用evaluateJavascript即可每强。這里展示一個(gè)簡單的交互示例 具有返回值的js方法
js代碼如下:

 function sumToJava2(number1, number2) {return number1 + number2;}

android代碼如下:

@TargetApi(Build.VERSION_CODES.KITKAT)
public void Android2JsHaveParmHaveResult2(View view) { 
mWebView.evaluateJavascript("sumToJava2(3,4)", new ValueCallback<String>() {
@Override 
public void onReceiveValue(String Str) {   
 Toast.makeText(getApplicationContext(), "我是android調(diào)用js方法(4.4后),入?yún)⑹?和4州刽,
  js返回結(jié)果是" +    Str, Toast.LENGTH_LONG).show(); 
   }    
 } );
}

知識點(diǎn)

1.圖片自適應(yīng)
當(dāng)我用webview直接顯示該網(wǎng)頁的時(shí)候空执,圖片并不能自適應(yīng),只能夠顯示一部分穗椅。那么如何做到自適應(yīng)手機(jī)屏幕呢辨绊?具體見代碼:

WebSettings webSettings = view.getSettings(); 
 // User settings           
webSettings.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);//適應(yīng)內(nèi)容大小
webSettings.setUseWideViewPort(true);
webSettings.setLoadWithOverviewMode(true);

注意:對于比較復(fù)雜的網(wǎng)頁,該方法不一定有效匹表,需要開發(fā)者自行測試邢羔。

2.獲取圖片地址

那如何下載網(wǎng)頁中的圖片呢?我們可以通過webview中的setOnLongClickListener獲取HitTestResult對象桑孩,通過判斷該對象的類型獲取點(diǎn)擊圖片的URL拜鹤,然后把圖片下載到本地。

webwiew.setOnLongClickListener(new View.OnLongClickListener() {      
 @Override     
 public boolean onLongClick(final View v) {             
   final WebView.HitTestResult result = ((WebView) v).getHitTestResult();             
   int type = result.getType();              
   if (type == WebView.HitTestResult.IMAGE_TYPE) {                    
   WebView.HitTestResult result = ((WebView) v).getHitTestResult();           
         String imgurl = result.getExtra();//圖片地址                   
       }             
    return false;        
    }     
   });

HitTestResult的類型有:

類型 名稱 作用
int ANCHOR_TYPE 已經(jīng)廢棄
int EDIT_TEXT_TYPE 打開一個(gè)可編輯的區(qū)域
int EMAIL_TYPE 郵件
int GEO_TYPE map地址
int IMAGE_ANCHOR_TYPE 已經(jīng)廢棄
int IMAGE_TYPE 圖片
int PHONE_TYPE 電話號碼
int SRC_ANCHOR_TPE html的a標(biāo)簽流椒,內(nèi)容是一個(gè)http地址
int SRC_IMAGE_ANCHOR_TYPE html的a標(biāo)簽敏簿,內(nèi)容由http及image
int UNKNOW_TYPE 未知內(nèi)容
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市宣虾,隨后出現(xiàn)的幾起案子惯裕,更是在濱河造成了極大的恐慌,老刑警劉巖绣硝,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蜻势,死亡現(xiàn)場離奇詭異,居然都是意外死亡鹉胖,警方通過查閱死者的電腦和手機(jī)握玛,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來甫菠,“玉大人挠铲,你說我怎么就攤上這事〖庞眨” “怎么了拂苹?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長痰洒。 經(jīng)常有香客問我瓢棒,道長浴韭,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任脯宿,我火速辦了婚禮念颈,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘嗅绰。我一直安慰自己舍肠,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布窘面。 她就那樣靜靜地躺著翠语,像睡著了一般。 火紅的嫁衣襯著肌膚如雪财边。 梳的紋絲不亂的頭發(fā)上肌括,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天,我揣著相機(jī)與錄音酣难,去河邊找鬼谍夭。 笑死,一個(gè)胖子當(dāng)著我的面吹牛憨募,可吹牛的內(nèi)容都是我干的紧索。 我是一名探鬼主播,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼菜谣,長吁一口氣:“原來是場噩夢啊……” “哼珠漂!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起尾膊,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤媳危,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后冈敛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體待笑,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年抓谴,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了暮蹂。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,040評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡齐邦,死狀恐怖椎侠,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情措拇,我是刑警寧澤,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布慎宾,位于F島的核電站丐吓,受9級特大地震影響浅悉,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜券犁,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一术健、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧粘衬,春花似錦荞估、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至褂删,卻和暖如春飞醉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背屯阀。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工缅帘, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人难衰。 一個(gè)月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓钦无,卻偏偏與公主長得像,于是被迫代替她去往敵國和親盖袭。 傳聞我的和親對象是個(gè)殘疾皇子失暂,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評論 2 355

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