viewgroup自定义组件 --- 侧滑组件

继承ViewGroup制作的自定义组件常常是一些布局或者组合组件:
和View的自定义组件类似,需要重写onMeause和onLayout测量组件的宽高和布局,因为viewGruop里面包含了子组件,所有在测量和定位的时候都是以子组件为中心进行测量和定位;不需要重写onDraw方法,因为viewGroup是一个容器类,我们只需要调用它的子组件的draw方法即可

还要注意事件分发时,要重写onInteceptTouchEvent,判断事件来临时需不需要向子组件传递下去;

下面以一下侧滑组件为例:

我们创建的组件由两个子组件组合而成,该组件可以左右侧滑,而里面的子组件“菜单栏”可以上下滑动;
1. 创建一个类,继承viewGroup,重写onMeasure和onLayout和构造器
    Measure  
        菜单栏( 宽度等于自己的真实宽度   高度和父类的控件相同)
        主界面  (宽高都和父控件相同)
    layout:
        菜单栏  (右边为 0 ,0  左边为 -自己宽度长 0 )
        主界面   (和父类的相同)

2. 监控事件分发:
    当我们在屏幕上滑动时,如何区分是左右滑动还是数值滑动,传递给viewgroup还是它的子类scrollView;
    需要一个判断用户滑动的长度,大于此长度就认为是滑动;此长度 == ViewConfiguration.get(context).getScaleTouchSlop
    用上面的长度,在OnInteceptTouchEvent的Move动作中,如果x轴上变化距离大于此变量并且y小于此长度就是水平滑动,返回true,阻断事件分发;反之fasle

3. 左右的滑动效果
    滑动效果的主要在onTouchEvent方法里面完成:
    switch(down  move up)三个动作,down里面记录初始的x坐标,move里面用当前坐标减去初始坐标就是x轴的位移;
    在move中,
        view.getScrollX()获取开始接触屏幕时的坐标,此坐标+移动的x轴坐标差就是移动后的坐标,判断此坐标如果大于0直接scrollTo(0,0);如果小于-菜单栏的,则移动到左边的菜单栏
        view.scrollTo(int x,int y)滑动到具体的xy坐标
        view.scrollBy(int dx,int dy); 滑动了多少距离  当前坐标加dx和dy

    慢慢滑动的效果:
        在我们抬起手指,查看当前的x轴坐标,如果小于菜单栏/2,往左慢慢滑动菜单栏,大于菜单栏/2,慢慢滑动右边
        借助Scroller辅助类帮助滑动:
            startScroll(int startX, int startY, int dx, int dy, int duration)    在规定的时间内完成滑动,模拟滑动,但此方法不会使view滑动

            computeScrollOffset()   如果模拟没完成,返回true,完成则false
            //重写view的方法,drawchild  -- >> child.draw() -->> computeScroll
            computeScroll()
            在此方法中,我们通过判断computeScrollOffset看模拟是否完成,如果没完成就滑动组件scrollTo(Scroller.getcurrX,0)然后调用              invalidate重画draw即可实现再次调用此方法,直到模拟完成
public class SildView extends ViewGroup {

    private static final String TAG = "SilDView";
    private Scroller scroller;
    private int lastX,currentX,lastY;
    private int mTouchSlip;
    private boolean viewFlag = false;       //viewFlag为界面标志

    //AttributeSet attr是啥意思
    public SildView(Context context,AttributeSet attrs) {
        super(context,attrs);
        // TODO Auto-generated constructor stub
        scroller = new Scroller(context);
        mTouchSlip = ViewConfiguration.get(context).getScaledTouchSlop();       //获取滑动识别的最小距离
        Log.i(TAG, ""+mTouchSlip);
    }

    /*测量布局*/
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        // TODO Auto-generated method stub
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //菜单布局
        View menuView = this.getChildAt(0);
        menuView.measure(menuView.getLayoutParams().width, heightMeasureSpec);
        //内容布局
        View contextView = this.getChildAt(1);
        contextView.measure(widthMeasureSpec, heightMeasureSpec);
    }
    /*布局下去*/
    @Override
    protected void onLayout(boolean arg0, int l, int t, int r, int b) {
        // TODO Auto-generated method stub
        View menuView = this.getChildAt(0);
        menuView.layout(-menuView.getMeasuredWidth(), 0, 0, b);

        View contextView = this.getChildAt(1);
        contextView.layout(0, 0, r, b);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        // TODO Auto-generated method stub
        switch(event.getAction()){
            case MotionEvent.ACTION_DOWN:
                lastX = (int)event.getX();          //左后一次down动作的x
                break;

            case MotionEvent.ACTION_MOVE:
                currentX = (int)event.getX();           //滑动过程中
                int change = lastX - currentX;          //增量
                int scrollX = getScrollX() + change;        //getScrollX是获取屏幕左边的X坐标 ,加上增量就是滑动后的坐标  
                if(scrollX < -this.getChildAt(0).getWidth()){
                    //最左边了
                    scrollTo(-this.getChildAt(0).getWidth(),0);
                }else if(scrollX > 0){
                    //大于0直接滑到内容界面
                    scrollTo(0,0);
                }else{
                    scrollBy(change,0);
                }
                lastX = currentX;
                break;

            case MotionEvent.ACTION_UP:
                int midX = -this.getChildAt(0).getWidth()/2;
                int x = getScrollX();
                if(x < midX){
                    viewFlag = false;
                }else{
                    viewFlag = true;    
                }
                switchUI(viewFlag);
                break;
        }
        return true;
    }

    /**
     * 菜单界面和功能界面的切换
     * true为功能界面
     * false为菜单界面
     * 注:滑动是从当前界面滑动到目标位置,而不是直接跳转到目标position
     */
    public void switchUI(boolean flagUI){
        int dx = 0;
        int currentX = getScrollX();
        if(flagUI){
            dx = 0 - currentX;
        }else{
            dx = -this.getChildAt(0).getWidth() - currentX;
        }

        //模拟滑动过程

        scroller.startScroll(currentX, 0, dx, 0,Math.abs(dx)*5);

        invalidate();           //此函数会经过viewGroup下面的drawChild()-->child.draw()-->computeScroll()最终完成滑动并显示出来
    }

    @Override
    public void computeScroll() {
        // TODO Auto-generated method stub
        if(scroller.computeScrollOffset()){         //true表示正在模拟数据
            //更新左X坐标
            scrollTo(scroller.getCurrX(),0);

            invalidate();   
        }
    }


    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        // TODO Auto-generated method stub
        switch(ev.getAction()){
            case MotionEvent.ACTION_DOWN:
                lastX = (int)ev.getX();
                lastY = (int)ev.getY();
                Log.i(TAG, "ACTION_DOWN: "+ lastX);
                break;
            case MotionEvent.ACTION_MOVE:
                int currentX = (int)ev.getX();
                int currentY = (int)ev.getY();
                int diffX = lastX - currentX;
                int diffY = lastY - currentY;
                diffX = Math.abs(diffX);
                diffY = Math.abs(diffY);
                Log.i(TAG, "ACTION_MOVE: "+ diffX);
                /*****
                 * 必须加一个竖直判断,因为上下滑动的平率很容易让x之间的距离大于设定的距离,
                 * 所以,缓慢上下滑动还能够在scrollView组件里面滑动,速度稍微快一点就转到
                 * viewgroup的左右滑动当中去了
                 */
                if(diffX > mTouchSlip && diffY < mTouchSlip){   // 
                    return true;
                }
                break;
            case MotionEvent.ACTION_UP:
                Log.i(TAG, "ACTION_UP: ");
                break;
        }

        return super.onInterceptTouchEvent(ev);
    }

    //功能界面是否打开
    public boolean isShown(){
        return viewFlag;
    }

    //打开功能界面
    public void showView(){
        switchUI(true);
        viewFlag = true;
    }

    //掩藏功能界面
    public void hideView(){
        switchUI(false);
        viewFlag = false;
    }

}
posted @ 2015-10-12 20:50  帅气好男人_jack  阅读(4)  评论(0编辑  收藏  举报  来源