原鏈接:http://blog.saymagic.cn/2015/11/08/webview-upload.html?utm_source=tuicool&utm_medium=referral
從零開始
我們在xml中寫入一個簡單的Webview組件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"><WebViewandroid:id="@+id/webview"android:layout_width="fill_parent"android:layout_height="fill_parent"android:layout_margin="5dp"></WebView></RelativeLayout>
然后在Java代碼中使用其加載一個能夠提供上傳服務的URL:
WebView webview = (WebView) findViewById(R.id.webview);webview.loadUrl(A_UPLOAD_URL);
之后,要加網絡權限:
<uses-permission android:name="android.permission.INTERNET"></uses-permission>
如果想讓Webview能夠訪問本地資源,SD卡的讀寫權限也是避免不了的:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/><uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
最后,我們運行,會發現根本不能訪問本地資源。Why?
讓我們來填補第一個坑:
支持上傳文件
Webview執行上傳操作的邏輯是這樣的:首先准備上傳時會回調WebChromeClient類下的openFileChooser方法,在這個方法中給我們機會發起Intent來打開支持提供文件的第三方應用,最后在onActivityResult回調中將第三方應用提供的內容通過一個叫做ValueCallback的參數返回給Webview(詳細點來說:ValueCallback是在openFileChooser 方法里由webview提供給我們的,里面包裹一個Uri,我們在onActivityResult 里將選中的Uri反饋給ValueCallback,這時候相當於Webview就知道我們選擇了什么文件),因此,我們需要為Webview設置一個提供openFileChooser方法的WebChromeClient,這個方法在不同版本的Android中參數是不同的,為此我們一般需要寫三個重載函數,大致像這個樣子:
private ValueCallback<Uri> mUploadMessage;//設置`WebChromeClient`:webview.setWebChromeClient(new WebChromeClient(){public void openFileChooser(ValueCallback<Uri> uploadMsg) {Log.d(TAG, "openFileChoose(ValueCallback<Uri> uploadMsg)");mUploadMessage = uploadMsg;Intent i = new Intent(Intent.ACTION_GET_CONTENT);i.addCategory(Intent.CATEGORY_OPENABLE);i.setType("*/*");MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), FILECHOOSER_RESULTCODE);}public void openFileChooser( ValueCallback uploadMsg, String acceptType ) {Log.d(TAG, "openFileChoose( ValueCallback uploadMsg, String acceptType )");mUploadMessage = uploadMsg;Intent i = new Intent(Intent.ACTION_GET_CONTENT);i.addCategory(Intent.CATEGORY_OPENABLE);i.setType("*/*");MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Browser"),FILECHOOSER_RESULTCODE);}public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture){Log.d(TAG, "openFileChoose(ValueCallback<Uri> uploadMsg, String acceptType, String capture)");mUploadMessage = uploadMsg;Intent i = new Intent(Intent.ACTION_GET_CONTENT);i.addCategory(Intent.CATEGORY_OPENABLE);i.setType("*/*");MainActivity.this.startActivityForResult( Intent.createChooser( i, "File Browser" ), MainActivity.FILECHOOSER_RESULTCODE );}});//onActivityResult回調@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if(requestCode==FILECHOOSER_RESULTCODE){if (null == mUploadMessage && null == mUploadCallbackAboveL) return;Uri result = data == null || resultCode != RESULT_OK ? null : data.getData();if (mUploadMessage != null) {mUploadMessage.onReceiveValue(result);mUploadMessage = null;}}}
還有重要的一點:如果這個上傳操作涉及到JS操作,別忘記對Webview開啟對JS的支持:
WebSettings settings = webview.getSettings();settings.setJavaScriptEnabled(true);
這樣,打個debug包測試看以下,不出意外我們的Webview應該可以支持上傳操作了。
別高興得太早,如果這個時候產品要將release包推向市場,當你把release包交給產品時,你會發現你的Webview又不能上傳了,什么情況?
請聽Webview上傳操作的第二個坑。
支持release版
debug版是好的,為什么release就不行了呢?准確的說,開啟了混淆的release包是不可以的,究其原因在於,openFileChooser 方法並不是WebChromeClient 的對外開放的方法,因此這個方法會被混淆,解決辦法也比較簡單,只需要在混淆文件里控制一下即可:
-keepclassmembers class * extends android.webkit.WebChromeClient{public void openFileChooser(...);}
好了,我們的Webview可以作為應用內的一個部分對外發布了,等等,有5.0以上用戶反映用不了?納尼????
別回心,來看看這第三個坑。
支持5.0
在5.0發布后,Android人家說了,這次我們回調的不是openFileChooser方法,而是onShowFileChooser方法,並且上文提到的ValueCallback參數里包裹着不再是Uri,而是Uri數組,因此我們必須為5.0+的機器做適配,大致思路如下:
webview.setWebChromeClient(new WebChromeClient(){public void openFileChooser(ValueCallback<Uri> uploadMsg) {...}public void openFileChooser( ValueCallback uploadMsg, String acceptType ) {...}public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture){...}// For Android 5.0+public boolean onShowFileChooser (WebView webView, ValueCallback<Uri[]> filePathCallback, WebChromeClient.FileChooserParams fileChooserParams) {mUploadCallbackAboveL = filePathCallback;Intent i = new Intent(Intent.ACTION_GET_CONTENT);i.addCategory(Intent.CATEGORY_OPENABLE);i.setType("*/*");MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Browser"),FILECHOOSER_RESULTCODE);return true;}});@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if(requestCode==FILECHOOSER_RESULTCODE){if (null == mUploadMessage && null == mUploadCallbackAboveL) return;Uri result = data == null || resultCode != RESULT_OK ? null : data.getData();if (mUploadCallbackAboveL != null) {onActivityResultAboveL(requestCode, resultCode, data);}else if (mUploadMessage != null) {mUploadMessage.onReceiveValue(result);mUploadMessage = null;}}}@TargetApi(Build.VERSION_CODES.LOLLIPOP)private void onActivityResultAboveL(int requestCode, int resultCode, Intent data) {if (requestCode != FILECHOOSER_RESULTCODE|| mUploadCallbackAboveL == null) {return;}Uri[] results = null;if (resultCode == Activity.RESULT_OK) {if (data == null) {} else {String dataString = data.getDataString();ClipData clipData = data.getClipData();if (clipData != null) {results = new Uri[clipData.getItemCount()];for (int i = 0; i < clipData.getItemCount(); i++) {ClipData.Item item = clipData.getItemAt(i);results[i] = item.getUri();}}if (dataString != null)results = new Uri[]{Uri.parse(dataString)};}}mUploadCallbackAboveL.onReceiveValue(results);mUploadCallbackAboveL = null;return;}
如上,我們的Webview應該就可以適應5.0+的機器了。
