Android Webview 和 ScrollView 冲突和 WebView 使用总结

rlbx7836 8年前
   <h3>前言:</h3>    <p>今天review项目中的代码想起来之前修改一个有关Webview和ScrollView冲突的bug:</p>    <ul>     <li>1.因为Webview和ScrollView都用滑动事件,导致webview很难被滑动,即使被滑动了一点也非常不顺畅</li>     <li>2.解决滑动冲突问题后发现,如果webview嵌套的html中含有轮播图等还是有问题。</li>    </ul>    <p style="text-align:center"><img src="https://simg.open-open.com/show/6581bd3d2c018bc93caa5fee48e4973e.gif"></p>    <h3>解决方案:</h3>    <p>其实这两个问题属于同一类的问题,都是Webview和ScrollView滑动时产生的冲突</p>    <p>解决方法很简单:</p>    <pre>  <code class="language-java">public class ScrollWebView extends WebView{      private float startx;      private float starty;      private float offsetx;      private float offsety;        public ScrollWebView(Context context) {          super(context);      }        public ScrollWebView(Context context, AttributeSet attrs) {          super(context, attrs);      }        public ScrollWebView(Context context, AttributeSet attrs, int defStyleAttr) {          super(context, attrs, defStyleAttr);      }        @Override      public boolean onTouchEvent(MotionEvent event) {          switch (event.getAction()) {              case MotionEvent.ACTION_DOWN:                  getParent().requestDisallowInterceptTouchEvent(true);                  startx = event.getX();                  starty = event.getY();                  Log.e("MotionEvent", "webview按下");                  break;              case MotionEvent.ACTION_MOVE:                  Log.e("MotionEvent", "webview滑动");                  offsetx = Math.abs(event.getX() - startx);                  offsety = Math.abs(event.getY() - starty);                  if (offsetx > offsety) {                      getParent().requestDisallowInterceptTouchEvent(true);                      Log.e("MotionEvent", "屏蔽了父控件");                  } else {                      getParent().requestDisallowInterceptTouchEvent(false);                      Log.e("MotionEvent", "事件传递给父控件");                  }                  break;              default:                  break;          }          return super.onTouchEvent(event);      }  }</code></pre>    <p>自定义一个WebView,重写onTouchEvent方法。判断当手指横向滑动的偏移量(offsetx)大于纵向滑动的偏移量(offsety)时屏蔽父控件的滑动。</p>    <p>解决方法很简单,只要我们在遇到问题的时候多多思考,弄清楚错误的原因,有针对性的研究,绝大数的问题都是可以解决的。</p>    <h2>webview使用总结</h2>    <p>本来写到这里就想结束了,但是发现写的东西太少了,估计会被骂,既然写的是webview,索性就把我对webview的使用总结整理一下。</p>    <h3>数据加载</h3>    <ul>     <li>加载本地资源 <pre>  <code class="language-java">webView.loadUrl("file:///android_asset/text.html");</code></pre> </li>     <li>加载网络资源 <pre>  <code class="language-java">webView.loadUrl("www.xxx.com/text.html");</code></pre> </li>     <li>添加请求头信息 <pre>  <code class="language-java">Map<String,String> map=new HashMap<String,String>();    map.put("User-Agent","Android");    webView.loadUrl("www.xxx.com/text.html",map);</code></pre> </li>     <li>直接加载html代码片段 <pre>  <code class="language-java">String html = "数据";    webView.loadDataWithBaseURL(null,html, "text/html", "utf-8",null);</code></pre> </li>    </ul>    <h3>支持JavaScript</h3>    <ul>     <li>设置支持JavaScript <pre>  <code class="language-java">WebSettings webSettings = webView.getSettings();    webSettings.setJavaScriptEnabled(true);//设置支持javascript    webView.addJavascriptInterface(new JavaScriptInterface(), "tyk");//添加一个对象, 让JS可以访问该对象的方法, 该对象中可以调用JS中的方法  </code></pre> </li>    </ul>    <ul>     <li> <p>JavaScriptInterface 接口定义</p> <pre>  <code class="language-java">private final class JavaScriptInterface {          //JavaScript调用此方法拨打电话          public void call(String phone) {              //startActivity(new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + phone)));              Toast.makeText(MainActivity.this, phone, Toast.LENGTH_LONG).show();          }            //Html调用此方法传递数据          public void showcontacts() {              String json = "[{\"name\":\"tyk\", \"amount\":\"9999999\", \"phone\":\"1831041486.\"}]";               // 调用JS中的方法              webView.loadUrl("javascript:show('" + json + "')");          }      }</code></pre> </li>    </ul>    <h3>WebViewClient</h3>    <ul>     <li> <p>主要辅助WebView处理各种通知、请求事件</p> <pre>  <code class="language-java">onLoadResource//加载资源时响应    onPageStart//在加载页面时响应    onPageFinish//在加载页面结束时响应    onReceiveError//在加载出错时响应    onReceivedHttpAuthRequest//获取返回信息授权请求</code></pre> </li>     <li> <p>要实现WebView中链接在WebView内部跳转</p> <pre>  <code class="language-java">webView.setWebViewClient(new WebViewClient() {        public boolean shouldOverrideUrlLoading(WebView view, String url) {            view.loadUrl(url);            return true;        }    });</code></pre> </li>    </ul>    <h3>WebChromeClient</h3>    <ul>     <li> <p>主要辅助WebView处理Javascript的对话框,网站图标,网站title,加载进度等</p> <pre>  <code class="language-java">        onCloseWindow//关闭WebView            onCreateWindow() //触发创建一个新的窗口            onJsAlert //触发弹出一个对话框            onJsPrompt //触发弹出一个提示            onJsConfirm//触发弹出确认提示            onProgressChanged //加载进度            onReceivedIcon //获取网页icon            onReceivedTitle//获取网页title</code></pre> </li>     <li> <p>加载进度获取title</p> <pre>  <code class="language-java">webView.setWebChromeClient(new WebChromeClient() {        @Override        public void onProgressChanged(WebView view, int newProgress) {            if (newProgress == 100) {                //网页加载完成            } else {                //网页加载中            }        }    });</code></pre> </li>    </ul>    <h3>WebView 缓存控制</h3>    <pre>  <code class="language-java">LOAD_CACHE_ONLY//不使用网络,只读取本地缓存数据      LOAD_DEFAULT//根据cache-control决定是否从网络上取数据。      LOAD_CACHE_NORMAL//API level 17中已经废弃, 从API level 11开始作用同LOAD_DEFAULT模式      LOAD_NO_CACHE//不使用缓存,只从网络获取数据.      LOAD_CACHE_ELSE_NETWORK//只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。</code></pre>    <p>一般都是根据网络来判断缓存使用情况</p>    <pre>  <code class="language-java">//加载缓存形式          if (CommonUtils.getNetWorkStatus(context)){//判断网络是否可用              // 根据cache-control决定是否从网络上取数据。              websettings.setCacheMode(WebSettings.LOAD_NO_CACHE);          }else{              // 只要本地有,无论是否过期,或者no-cache,都使用缓存中的数据。              websettings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);          }</code></pre>    <h3>页面返回</h3>    <ul>     <li> <p>我们有时需要实现回退到上一目录</p> <pre>  <code class="language-java">@Override    public boolean onKeyDown(int keyCode, KeyEvent event) {        if (keyCode == KeyEvent.KEYCODE_BACK) {            if (webView.canGoBack()) {                webView.goBack();//返回上一浏览页面                return true;            } else {                finish();//关闭Activity            }        }        return super.onKeyDown(keyCode, event);    }</code></pre> </li>    </ul>    <h3>其他设置</h3>    <pre>  <code class="language-java">WebSettings webSettings = webView.getSettings();      //支持缩放      webSettings.setSupportZoom(true);        //支持内容重新布局      webSettings.setLayoutAlgorithm(WebSettings.LayoutAlgorithm.SINGLE_COLUMN);        //多窗口      webSettings.supportMultipleWindows();       //当webview调用requestFocus时为webview设置节点      webSettings.setNeedInitialFocus(true);       //设置支持缩放      webSettings.setBuiltInZoomControls(true);       //支持通过JS打开新窗口      webSettings.setJavaScriptCanOpenWindowsAutomatically(true);       //支持自动加载图片      webSettings.setLoadsImagesAutomatically(true);        //提高渲染的优先级       websettings.setRenderPriority(WebSettings.RenderPriority.HIGH);       // 开启H5(APPCache)缓存功能       websettings.setAppCacheEnabled(true);      // 开启 DOM storage 功能       websettings.setDomStorageEnabled(true);      // 应用可以有数据库      websettings.setDatabaseEnabled(true);      // 可以读取文件缓存(manifest生效)      websettings.setAllowFileAccess(true);</code></pre>    <p> </p>    <p>来自:https://juejin.im/post/58f714ac570c35005651a347</p>    <p> </p>