自定义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>