自定义view事件分发机制的细枝末节

xlxh1295 8年前
   <p>当view确定自身已经不再适合现有的区域时,该view本身调用这个方法要求parent view(父类的视图)重新调用他的onMeasure onLayout来重新设置自己位置。特别是当view的layoutparameter发生改变,并且它的值还没能应用到view上时,这时候适合调用这个方法。</p>    <p>postInvalidate 与 invalidate</p>    <p>界面刷新 onDraw方法会执行,区别就是Invalidate不能直接在线程中调用,因为他是违背了单线程模型:Android UI操作并不是线程安全的,并且这些操作必须在UI线程中调用。 鉴于此,如果要使用invalidate的刷新,那我们就得配合handler的使用,使异步非ui线程转到ui线程中调用,如果要在非ui线程中直接使用就调用postInvalidate方法即可,这样就省去使用handler的烦恼。</p>    <p>view提供的方法</p>    <p>getTop:获取到的,是view自身的顶边到其父布局顶边的距离 getLeft:获取到的,是view自身的左边到其父布局左边的距离 getRight:获取到的,是view自身的右边到其父布局左边的距离 getBottom:获取到的,是view自身的底边到其父布局顶边的距离</p>    <p>MotionEvent提供的方法</p>    <p>getX():获取点击事件相对控件左边的x轴坐标,即点击事件距离控件左边的距离 getY():获取点击事件相对控件顶边的y轴坐标,即点击事件距离控件顶边的距离 getRawX():获取点击事件相对整个屏幕左边的x轴坐标,即点击事件距离整个屏幕左边的距离 getRawY():获取点击事件相对整个屏幕顶边的y轴坐标,即点击事件距离整个屏幕顶边的距离</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/9863bde2e328889ad2536099f56de25c.png"></p>    <p> </p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/c1e3bb73d5511f06edb74953c492253b.png"></p>    <p> </p>    <p>当以下代码时:</p>    <p>view中点击事件代码:</p>    <pre>  <code class="language-java">public boolean onTouchEvent(MotionEvent event) {          int touchEvent = event.getAction();          switch (touchEvent) {              case MotionEvent.ACTION_DOWN:                  System.out.println("CustomView getXdown="+event.getX()+" getYdown="+event.getY());                  System.out.println("CustomView getRawXdown="+event.getRawX()+" getRawYdown="+event.getRawY());                  System.out.println("CustomView eventdown="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_UP:                  System.out.println("CustomView getXup="+event.getX()+" getYup="+event.getY());                  System.out.println("CustomView getRawXup="+event.getRawX()+" getRawYup="+event.getRawY());                  System.out.println("CustomView eventup="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_MOVE:                  break;              default:                  break;          }          return true;      }</code></pre>    <p>activity中点击事件代码</p>    <pre>  <code class="language-java">@Override      public boolean onTouchEvent(MotionEvent event) {          int touchEvent = event.getAction();          switch (touchEvent) {              case MotionEvent.ACTION_DOWN:                  System.out.println("MainActivity getXdown="+event.getX()+" getYdown="+event.getY());                  System.out.println("MainActivity getRawXdown="+event.getRawX()+" getRawYdown="+event.getRawY());                  System.out.println("MainActivity eventdown="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_UP:                  System.out.println("MainActivity getXup="+event.getX()+" getYup="+event.getY());                  System.out.println("MainActivity getRawXup="+event.getRawX()+" getRawYup="+event.getRawY());                  System.out.println("MainActivity eventup="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_MOVE:                  break;              default:                  break;          }          return super.onTouchEvent(event);      }</code></pre>    <p>得到输出结果:</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/03823206275890dfd33a53b2b31e1e72.png"></p>    <p>因为view中返回true,消费掉了点击事件,所以activity中不打印。</p>    <p>而当把view中点击事件代码修改成以下情况:</p>    <p>view代码:</p>    <pre>  <code class="language-java">public boolean onTouchEvent(MotionEvent event) {          int touchEvent = event.getAction();          switch (touchEvent) {              case MotionEvent.ACTION_DOWN:                  System.out.println("CustomView getXdown="+event.getX()+" getYdown="+event.getY());                  System.out.println("CustomView getRawXdown="+event.getRawX()+" getRawYdown="+event.getRawY());                  System.out.println("CustomView eventdown="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_UP:                  System.out.println("CustomView getXup="+event.getX()+" getYup="+event.getY());                  System.out.println("CustomView getRawXup="+event.getRawX()+" getRawYup="+event.getRawY());                  System.out.println("CustomView eventup="+super.onTouchEvent(event));                  break;              case MotionEvent.ACTION_MOVE:                  break;              default:                  break;          }          return super.onTouchEvent(event);      }</code></pre>    <p>则返回如下</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/3f5221269facf097f3af89d7cac388e0.png"></p>    <p>因为view中返回false,所以点击没有被消费掉,只有down不执行up,因此执行至activity,而activity也是返回false,本来activity也不会执行up,但是没有更上级,所以这边只有在activity中吧up事件给执行了。(这句话如有错误请尽量提出)</p>    <p> </p>    <p>来自:http://www.jianshu.com/p/517368680bc1</p>    <p> </p>