本章主要讲的是使用 WebView 在应用内浏览网页
GitHub 地址:
完成第28章,未完成挑战
完成第28章挑战
1. WebView
如果不使用 WebView,我们可以使用隐式 intent,也就是用独立的浏览器打开一个网页。但是我们通常只想在 activity 中显示网页内容而不是打开浏览器:或许是想显示自己生成的 HTML,或许是想以某种方式限制用户使用浏览器。对于大多数需要帮助文档的应用,普遍做法是以网页的形式提供帮助文档,这样会方便后期的更新与维护。打开浏览器查看帮助文档,既不专业,又妨碍应用行为的定制,同时也无法将网页整合进自己的用户界面。
1.1 WebView 的使用
和其他的 View 一样,在 fragment 的布局文件中声明之后,在 fragment 中获取实例,然后进行一番设置:
1 2 3 4 5 6 7 8 9
| <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <WebView android:id="@+id/fragment_photo_page_web_view" android:layout_width="match_parent" android:layout_height="match_parent"/> </RelativeLayout>
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
| public class PhotoPageFragment extends VisibleFragment { private static final String ARG_URI = "photo_page_url";
private Uri mUri; private WebView mWebView;
……………… @SuppressLint("SetJavaScriptEnabled") @Nullable @Override public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { View v = inflater.inflate(R.layout.fragment_photo_page, container, false);
mWebView = (WebView) v.findViewById(R.id.fragment_photo_page_web_view); mWebView.getSettings().setJavaScriptEnabled(true); mWebView.setWebViewClient(new WebViewClient() { @Override public boolean shouldOverrideUrlLoading(WebView view, String url) { return false; } }); mWebView.loadUrl(mUri.toString());
return v; } }
|
可以看到,WebClient 是一个关键,如果不重写该方法,WebView 可能不会加载这个 url
1.2 使用 WebChromeClient 优化 WebView 显示
WebViewClient 主要帮助 WebView 处理各种通知、请求事件,而 WebChromeClient 主要辅助 WebView 处理 Javascript 的对话框、网站图标、网站 title、加载进度等。
我们在视图中加入 ProgressBar 放在 WebView 上方并默认隐藏,由于 WebChromeClient 返回进度值在0~100之间,所以我们要设置进度条的最大值为100。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| mProgressBar.setMax(100);
mWebView.setWebChromeClient(new WebChromeClient() { @Override public void onProgressChanged(WebView view, int newProgress) { if (newProgress == 100) { mProgressBar.setVisibility(View.GONE); } else { mProgressBar.setVisibility(View.VISIBLE); mProgressBar.setProgress(newProgress); } }
@Override public void onReceivedTitle(WebView view, String title) { AppCompatActivity activity = (AppCompatActivity) getActivity(); 设置网站标题 activity.getSupportActionBar().setSubtitle(title); } });
|
1.3 处理 WebView 的设备旋转问题
尝试旋转设备屏幕,可以发现 WebView 必须重新加载网页。这是因为 WebView 包含太多的数据,无法在 onSaveInstanceState(…)方法内全部保存。所以每次设备旋转,它都必须从头开始加载网页数据。
因为 WebView 是视图层级结构的一部分,所以旋转后它肯定会销毁并重建,不能用 retainFragment 来保留。对于一些类似的类(如 VideoView),Android 文档推荐让 activity 自己处理设备配置变更。也就是说,无需销毁重建 activity,就能直接调整自己的视图以适应新的屏幕尺寸。这样,WebView 也就不必重新加载全部数据了。
1 2
| <activity android:name=".PhotoPageActivity" android:configChanges="keyboardHidden|orientation|screenSize"/>
|
android:configChanges 属性表明,如果因键盘开或关、屏幕方向改变、屏幕大小改变(也包括 Android 3.2之后的屏幕方向变化)而发生设备配置更改,那么 activity 应自己处理配置更改。
2. 挑战练习
2.1 使用后退键浏览历史页面
我们覆盖 Activity.onBackPressed 方法,就可以对返回键的行为进行控制,但是要获取 WebView 的实例,所以在 fragment 中加入了一个 getter。
1 2 3 4 5 6 7 8 9 10 11
| @Override public void onBackPressed() { PhotoPageFragment fragment = (PhotoPageFragment) getSupportFragmentManager() .findFragmentById(R.id.fragment_container); if (fragment.getWebView().canGoBack()) { fragment.getWebView().goBack(); } else { super.onBackPressed(); } }
|
2.2 非 HTTP 链接支持
只需要修改重写的 shouldOverrideUrlLoading 方法即可。
1 2 3 4 5 6 7 8 9 10 11 12
| mWebView.setWebViewClient(new WebViewClient() { @Override public boolean shouldOverrideUrlLoading(WebView view, String url) { if (url.startsWith("https://") || url.startsWith("http://")) { return false; } else { Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url)); startActivity(intent); return true; } } });
|
GitHub Page: kniost.github.io
简书:http://www.jianshu.com/u/723da691aa42