zoukankan      html  css  js  c++  java
  • invalidate和requestLayout

    Invalidate:
    To farce a view to draw,call invalidate().——摘自View类源码
    从上面这句话看出,invalidate方法会执行draw过程,重绘View树。
    当View的appearance发生改变,比如状态改变(enable,focus),背景改变,隐显改变等,这些都属于appearance范畴,都会引起invalidate操作。

    所以当我们改变了View的appearance,需要更新界面显示,就可以直接调用invalidate方法。

    View(非容器类)调用invalidate方法只会重绘自身,ViewGroup调用则会重绘整个View树。

    RequestLayout:
    To initiate a layout, call requestLayout(). This method is typically called by a view on itself when it believes that it can no longer fit within its current bounds.——摘自View源码

    从上面这句话看出,当View的边界,也可以理解为View的宽高,发生了变化,不再适合现在的区域,可以调用requestLayout方法重新对View布局。

    View执行requestLayout方法,会向上递归到顶级父View中,再执行这个顶级父View的requestLayout,所以其他View的onMeasure,onLayout也可能会被调用。

    总结:

    View绘制分三个步骤,顺序是:onMeasure,onLayout,onDraw。经代码亲测,log输出显示:调用invalidate方法只会执行onDraw方法;调用requestLayout方法只会执行onMeasure方法和onLayout方法,并不会执行onDraw方法。

    所以当我们进行View更新时,若仅View的显示内容发生改变且新显示内容不影响View的大小、位置,则只需调用invalidate方法;若View宽高、位置发生改变且显示内容不变,只需调用requestLayout方法;若两者均发生改变,则需调用两者,按照View的绘制流程,推荐先调用requestLayout方法再调用invalidate方法。

    requestLayout示例:实现可移动组件:https://blog.csdn.net/qq_39658819/article/details/78994308

    import android.content.Context;
    import android.util.AttributeSet;
    import android.view.MotionEvent;
    import android.widget.FrameLayout;
     
    /**
     * 步骤一:自定义可移动组件
     * @author NewBies
     * @date 2017/12/26
     */
    public class VertexView extends android.support.v7.widget.AppCompatTextView{
     
        private int startX;
        private int startY;
        private int endX;
        private int endY;
        private FrameLayout.LayoutParams layoutParams;
     
        public VertexView(Context context) {
            super(context);
        }
     
        public VertexView(Context context, AttributeSet attrs) {
            super(context, attrs);
        }
     
        public VertexView(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
        }
     
     
        /**
         * 步骤二:重写onTouchEvent事件
         * @param event
         * @return
         */
        @Override
        public boolean onTouchEvent(MotionEvent event){
            //步骤三:获取手机触摸点的横坐标和纵坐标
            endX = (int)event.getX();
            endY = (int)event.getY();
     
            //步骤四:获取布局参数实例,注意:xx.LayoutParams这里的xx应该是该组件的父布局类型
            //注意:这句话必须在该组件已经添加到父布局中才会起作用,所以这句话我没有写在构造函数中,而是写在这里
            layoutParams = (FrameLayout.LayoutParams) this.getLayoutParams();
     
            switch (event.getAction()){
                //监听按下去的事件,这个事件在每次拖动时,必定会执行,也只执行一次
                case MotionEvent.ACTION_DOWN:
                    //将按下去的点记录为起始点
                    startX = endX;
                    startY = endY;
                    break;
                //步骤五:监听移动事件,该事件会在拖动时执行N次
                case MotionEvent.ACTION_MOVE:
                    //计算移动的距离
                    int offsetX = endX - startX;
                    int offsetY = endY - startY;
     
                    //调用layout方法来重新放置它的位置
                    layoutParams.setMargins(getLeft() + offsetX, getTop() + offsetY, getRight() + offsetX, getBottom() + offsetY);
                    //刷新
                    requestLayout();
                    break;
                //监听抬起事件,该事件同按下去的时间一样,只执行一次
                case MotionEvent.ACTION_UP:
                    break;
                default:break;
            }
            //这里应该返回true,这里涉及到了android的事件拦截机制,大致意思是,我的事件是在哪里处理,就在那里的事件返回TRUE
            return  true;
        }
    }
  • 相关阅读:
    gitlab web端使用
    1、gitlab的理论知识
    git命令
    gitlab web客户端的使用
    jenkins
    jenkins pipeline
    nginx
    ELK(+Redis)-开源实时日志分析平台
    OpenStack构架知识梳理
    Linux 下的dd命令使用详解
  • 原文地址:https://www.cnblogs.com/genggeng/p/10014121.html
Copyright © 2011-2022 走看看