从头开始敲代码之《从BaseApplication/Activity开始(五)》(自定义控件,实现点击/滑动翻页)

转载请注明出处:王亟亟的大牛之路

开场白惯用鼓励诗句:

黑发不知勤学早,白首方悔读书迟。 —— 颜真卿《劝学诗》

这一系列的博文这是第五篇了,感谢大家的支持以及陪伴,往后我也会继续努力写出高质量的内容,谢谢

今天上的是一个自定义View,新鲜出炉,先上下效果(是一张张截图拼接的Gif动画都看不出来了,大家理解就行可以下Demo跑)

样例分析(最简单的描述了)

黑色线条是我们的手机

红色是我们自定义的”TitleBar”

蓝色是我们的自定义布局

紫色是自定义布局填充的内容

我们只需要配置我们蓝色内容的参数就可以对动画效果以及大小等进行设置。
PS:因为 蓝色内容吃掉了所有蓝色区域的OnTouch,所以紫色就不要做用户交互内容了,纯粹做展示吧TOT(小的该死)

看下项目结构:

就比上次的代码多了一些资源文件和4个类,一个就是我们的麦麦Activity,另外3个解释下

DraggableFlipView我们的自定义控件

DragGestureDetector我们的触碰效果处理类

FlipListener动作展示以及处理结果

OK,开始分析

DraggableFlipView

public class DraggableFlipView extends FrameLayout implements DragGestureDetector.DragGestureListener {
    //一系列的声明,不一一解释了,后面用到了会加以解释

    private static final float DRAG_THRESHOLD_PARAM = 50.0f;
    private static final int DEFAULT_VALUE = 0;
    private static final int DEFAULT_DRAGGABLE_VALUE = 50;
    private static final int DEFAULT_DRAG_DETECT_VALUE = 7;

    private DragGestureDetector mDragGestureDetector;
    private boolean isAnimation;
    private boolean isDragging;
    private int mAngle;
    private int mDraggableAngle;
    private int mDragDetectAngle;
    private boolean mIsReverse;
    private FlipListener mFlipListener;

    private RelativeLayout mFrontLayout;
    private RelativeLayout mBackLayout;

    //声明左右状态的枚举
    private enum RotateDirection {
        RIGHT(1), LEFT(-1);

        private int mValue;

        RotateDirection(int value) {
            this.mValue = value;
        }

        public int getValue() {
            return mValue;
        }
    }

    //构造函数
    public DraggableFlipView(Context context) {
        this(context, null);
    }

    public DraggableFlipView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DraggableFlipView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(context, attrs);
    }

    //初始化参数
    private void init(Context context, AttributeSet attrs) {
        //获取布局对象并加以填充,默认显示mBackLayout这个布局
        mFrontLayout = new RelativeLayout(context);
        RelativeLayout.LayoutParams params1 = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.MATCH_PARENT);
        mFrontLayout.setLayoutParams(params1);

        mBackLayout = new RelativeLayout(context);
        RelativeLayout.LayoutParams params2 = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.MATCH_PARENT);
        mBackLayout.setLayoutParams(params2);

        this.addView(mFrontLayout);
        this.addView(mBackLayout);
        mBackLayout.setVisibility(View.INVISIBLE);

        //初始化FlipListener,传入2个布局,第二个参数为不显示的布局
        mFlipListener = new FlipListener(mFrontLayout, mBackLayout, this);
        mDragGestureDetector = new DragGestureDetector(this);

        //获取 标签
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.DraggableFlipView);
        LayoutInflater.from(context).inflate(a.getResourceId(R.styleable.DraggableFlipView_frontView, DEFAULT_VALUE), mFrontLayout);
        LayoutInflater.from(context).inflate(a.getResourceId(R.styleable.DraggableFlipView_backView, DEFAULT_VALUE), mBackLayout);
        //填充标签数据
        mDraggableAngle = a.getInteger(R.styleable.DraggableFlipView_draggableAngle, DEFAULT_DRAGGABLE_VALUE);
        mDragDetectAngle = a.getInteger(R.styleable.DraggableFlipView_dragDetectAngle, DEFAULT_DRAG_DETECT_VALUE);
    }
    //onInterceptTouchEvent这个事件是从父控件开始往子控件传的,直到有拦截或者到没有这个事件的view,并且使用  mDragGestureDetector.setPointMap(ev);进行参数的传递
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        if (mDragGestureDetector == null) return false;
        int action = ev.getAction() & MotionEvent.ACTION_MASK;
        switch (action) {
            case MotionEvent.ACTION_UP:
                break;
            case MotionEvent.ACTION_MOVE:
                if (Math.abs(ev.getX() - mDragGestureDetector.getTouchPoint().getX())
                        > DRAG_THRESHOLD_PARAM
                        || Math.abs(ev.getY() - mDragGestureDetector.getTouchPoint().getY())
                        > DRAG_THRESHOLD_PARAM) {
                    mDragGestureDetector.setPointMap(ev);
                    return true;
                }
                break;
            case MotionEvent.ACTION_POINTER_DOWN:
                return true;
        }
        return false;
    }
    //onTouch这个事件是从子控件回传到父控件的,一层层向下传
    //mDragGestureDetector.onTouchEvent(event)来实现onTouchEvent的操作
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (mDragGestureDetector != null) {
            mDragGestureDetector.onTouchEvent(event);
        }
        return true;
    }

    //计算,并判断以哪种方式来实现切换动画
    @Override
    public void onDragGestureListener(DragGestureDetector dragGestureDetector, int action) {
        if (isAnimation) return;
        if (action == MotionEvent.ACTION_UP) {
            if (mAngle >= mDragDetectAngle) {
                startAutoRotateAnimation(RotateDirection.RIGHT);
            } else if (mAngle < -mDragDetectAngle) {
                startAutoRotateAnimation(RotateDirection.LEFT);
            }
            return;
        }

        mAngle = (dragGestureDetector.deltaX - dragGestureDetector.prevDeltaX) > 0 ? ++mAngle : --mAngle;
        if (Math.abs(mAngle) > mDragDetectAngle) isDragging = true;
        if(isDragging) this.setRotationY(mAngle);

        if (mAngle >= mDraggableAngle) {
            startAutoRotateAnimation(RotateDirection.RIGHT);
        } else if (mAngle < -mDraggableAngle) {
            startAutoRotateAnimation(RotateDirection.LEFT);
        }
    }

    private void startAutoRotateAnimation(RotateDirection rotateDirection) {
        isAnimation = true;
        if (mIsReverse) {
            mFlipListener.reverse();
        } else {
            mIsReverse = true;
        }

        mFlipListener.setRotateDirection(rotateDirection.getValue());
        //动画的平滑过渡 可以参照 http://blog.csdn.net/guolin_blog/article/details/43536355
        //讲的很详细
        ValueAnimator mFlipAnimator = ValueAnimator.ofFloat(0f, 1f);
        mFlipAnimator.addUpdateListener(mFlipListener);
        mFlipAnimator.start();
        mFlipAnimator.addListener(new Animator.AnimatorListener() {
            @Override
            public void onAnimationStart(Animator animation) {
            }

            @Override
            public void onAnimationEnd(Animator animation) {
                mAngle = 0;
                isAnimation = false;
                isDragging = false;
            }

            @Override
            public void onAnimationCancel(Animator animation) {
            }

            @Override
            public void onAnimationRepeat(Animator animation) {
            }
        });
    }
}

DragGestureDetector

public class DragGestureDetector {

    public float deltaX;
    public float deltaY;
    public float prevDeltaX;
    public float prevDeltaY;
    public int originalIndex;
    public float velocityX;
    public float velocityY;

    //储存用户操作路径
    private HashMap<Integer, TouchPoint> pointMap = new HashMap<>();

    private DragGestureListener dragGestureListener;

    //构造函数
    public DragGestureDetector(DragGestureListener dragGestureListener) {
        this.dragGestureListener = dragGestureListener;
        //初始化坐标
        pointMap.put(0, createPoint(0.f, 0.f));
    }

    //储存坐标点
    public void setPointMap(MotionEvent event) {
        float eventX = event.getX();
        float eventY = event.getY();
        TouchPoint downPoint = pointMap.get(0);
        if (downPoint != null) {
            downPoint.setXY(eventX, eventY);
            return;
        }
        downPoint = createPoint(eventX, eventY);
        pointMap.put(0, downPoint);
    }

    //获取坐标点
    public TouchPoint getTouchPoint() {
        return pointMap.get(originalIndex);
    }

    //用户触控坐标点相应的计算
    synchronized public boolean onTouchEvent(MotionEvent event) {

        float eventX = event.getX(originalIndex);
        float eventY = event.getY(originalIndex);

        int action = event.getAction() & MotionEvent.ACTION_MASK;
        switch (action) {
            case MotionEvent.ACTION_DOWN: {
                break;
            }
            case MotionEvent.ACTION_MOVE: {
                TouchPoint originalPoint = pointMap.get(originalIndex);
                if (originalPoint != null) {
                    deltaX = eventX - originalPoint.x;
                    deltaY = eventY - originalPoint.y;

                    if (dragGestureListener != null) {
                        dragGestureListener.onDragGestureListener(this, action);
                    }

                    velocityX = deltaX - prevDeltaX;
                    velocityY = deltaY - prevDeltaY;
                    prevDeltaX = deltaX;
                    prevDeltaY = deltaY;
                }
                break;
            }

            case MotionEvent.ACTION_UP: {
                TouchPoint originalPoint = pointMap.get(originalIndex);
                if (originalPoint != null && dragGestureListener != null) {
                    dragGestureListener.onDragGestureListener(this, action);
                }
                velocityX = velocityY = 0;
                prevDeltaX = prevDeltaY = 0;
                deltaX = deltaY = 0;
                break;
            }
            default:
        }
        return false;
    }

    private TouchPoint createPoint(float x, float y) {
        return new TouchPoint(x, y);
    }

    public interface DragGestureListener {
        void onDragGestureListener(DragGestureDetector dragGestureDetector, int action);
    }

    //坐标类
    public class TouchPoint {

        private float x;
        private float y;

        public TouchPoint(float x, float y) {
            this.x = x;
            this.y = y;
        }

        public TouchPoint setXY(float x, float y) {
            this.x = x;
            this.y = y;
            return this;
        }

        public float getX() {
            return this.x;
        }

        public float getY() {
            return this.y;
        }
    }
}

FlipListener

public class FlipListener implements ValueAnimator.AnimatorUpdateListener {

    private View mParentView;
    private View mFrontView;
    private View mBackView;
    private boolean mFlipped;
    private int mDirection;

    //构造函数
    public FlipListener(final View front, final View back, final View parent) {
        this.mParentView = parent;
        this.mFrontView = front;
        this.mBackView = back;
        this.mBackView.setVisibility(View.GONE);
    }

    @Override
    public void onAnimationUpdate(final ValueAnimator animation) {
        final float value = animation.getAnimatedFraction();
        final float scaleValue = 0.625f + (1.5f * (value - 0.5f) * (value - 0.5f));

        //根据传入的mDirection(1或者-1)进行计算并且逻辑判断
        if (value <= 0.5f) {
            this.mParentView.setRotationY(180 * value * mDirection);
            if (mFlipped) setStateFlipped(false);
        } else {
            this.mParentView.setRotationY(-180 * (1 - value) * mDirection);
            if (!mFlipped) setStateFlipped(true);
        }
        this.mParentView.setScaleX(scaleValue);
        this.mParentView.setScaleY(scaleValue);
    }

    //初始化自定义View时调用
    public void reverse() {
        View temp = mBackView;
        mBackView = mFrontView;
        mFrontView = temp;
    }

    public void setRotateDirection(int direction) {
        mDirection = direction;
    }

    //具体切换试图
    private void setStateFlipped(boolean flipped) {
        mFlipped = flipped;
        this.mFrontView.setVisibility(flipped ? View.GONE : View.VISIBLE);
        this.mBackView.setVisibility(flipped ? View.VISIBLE : View.GONE);
    }
}

OK!!!!实现就这些啦
源码:http://yunpan.cn/cHwL97TfNdApF 访问密码 d11f

时间: 2024-10-24 18:24:14

从头开始敲代码之《从BaseApplication/Activity开始(五)》(自定义控件,实现点击/滑动翻页)的相关文章

从头开始敲代码之《从BaseApplication/Activity开始(三)》

转载请注明出处:王亟亟的大牛之路 上一篇写了对蓝牙,对NFC的监听实现,今天我们又给我们的BaseActivity加了一个<include>的操作以及实现了一个界面,上一篇请点击这里:从头开始敲代码之<从BaseApplication/Activity开始(二)> 总觉得,做事得有自己对自己所从事内容的理解,也许这不会让我们多挣钱,但我觉得会让我们活得明白 OK,那我们说下这篇我们做哪些点: 1.给我们的BaseActivity增添一个通用"TitleBar"

安卓下如何在一个activity下左右滑动翻页

问题描述 安卓下如何在一个activity下左右滑动翻页 在网上查了一下左右滑动翻页的讲解,用的都是图片数组填充的内容,我想在一个activity下实现滑动翻页,实现数据的填充,哪位大神帮忙指点迷津. 解决方案 viewpager,里面放置多个view 解决方案二: http://blog.csdn.net/lmj623565791/article/details/41087219

android ViewPager实现滑动翻页效果实例代码

实现ViewPager的滑动翻页效果可以使用ViewPager的setPageTransformer方法,如下: import android.content.Context; import android.support.v4.view.ViewPager; import android.util.AttributeSet; import android.view.View; public class ReadViewPager extends ViewPager { public ReadV

从头开始敲代码之《从BaseApplication/Activity开始》

转载请注明出处王亟亟的大牛之路 其安易持,其未兆易谋:其脆易泮,其微易散.为之于未有,治之于未乱.合抱之木,生于毫末:九层之台,起于垒土:千里之行,始于足下.为者败之,执者失之.是以圣人无为故无败,无执故无失.民之从事,常于几成而败之.慎终如始,则无败事.是以圣人欲不欲,不贵难得之货,学不学,复众人之所过,以辅万物之自然而不敢为. 作为系列专题的第一篇,这一篇文章属于小难产,中间夹杂着一些工作上的事,一些蛋疼的事(学车之类的),说实在的,做了Coder之后发现业余时间还真不是太多....唉...

从头开始敲代码之《从BaseApplication/Activity开始(二)》

转载请注明出处:王亟亟的大牛之路 愿意花时间写东西不容易,人啊,都是有血有肉有思想的,借鉴是学习,纯Copy就不好了,谢谢 部分资料参考于网上. <赠梁任父同年>黄遵宪 寸寸河山寸寸金,侉离分裂力谁任? 杜鹃再拜忧天泪,精卫无穷填海心. 上一篇我们讲到了简易的封装 对我们提高效率的好处,这一篇继续写下去,如果第一篇没看过的希望能看下,方便理解.链接:亟亟在安卓的进阶实例 这一次我们利用最基本的Activity生命周期中的方法,来对用户蓝牙进行识别操作,并在过程中考虑用户层面的操作理解,顺便补充

从头开始敲代码之《从BaseApplication/Activity开始(四)》

转载请注明出处:王亟亟的大牛之路 早上无聊看以前下的一大堆资料,发现一个用JNI实现的模糊效果,效果都差不多,但是对JNI的不熟悉让我不太推荐这种办法(不了解的总不方便,调试,修改都是) 然后在Git上找到个不错的实现,还是分2种的,应对于各种需要. 这一篇文章会介绍什么 1.模糊视图处理 2.线程操作优化 1.Renderscript 2.FastBlur 效果图 布局: <?xml version="1.0" encoding="utf-8"?> &

jquery带翻页动画的电子杂志代码分享_jquery

这是一款基于jquery实现的带翻页动画的电子杂志,小编有时就在想我们读的纸质课本可不可以都改成电子书,这要是不是会更节省资源呐? 记得以前我们有介绍过不少书本翻页的动画,比如这款CSS3书本翻页动画,制作就非常逼真.今天要分享的这款jQuery书本翻页3D动画功能更加强大,可以支持任意数量页的翻阅,并且也有非常美观的视觉效果.书本的内容支持任意HTML元素,相当灵活. 运行效果图                         --------------------------------查

学做界面#-想学做界面的信息安全专业的会敲代码的色影丝小学渣

问题描述 想学做界面的信息安全专业的会敲代码的色影丝小学渣 自身具备的艺术素养对做出优质的界面有助推作用吗?我对别人做的界面的构图位置美观吧啦吧啦很敏感,脑中会形成一个自己感觉更舒服的界面版式,这对做出优质的界面有助推作用吗,还是小学生胡思乱想了,第一次提问,求指示,轻喷. 解决方案 有艺术细胞是好事.但是如果过于强调这个,而忽略了系统的学习.理性和理论,那么是没什么好处的. 一个靠直觉和自发得到的经验而进行界面设计的人,可能你能设计用户群体和你本人有着相同背景的简单的小软件. 但是一个前端交互

编程题-敲代码时真不想再用大小写交换了,如何让首字母置为大写

问题描述 敲代码时真不想再用大小写交换了,如何让首字母置为大写 一段可以将由大于等于一个单词组成的字符串中的每个单词的首字母置为大写状态的源代码?诸位的想法是什么可以在下面留言看看 解决方案 现有的编辑工具应该只会按 英文 的规则进行大小写自动转换,如果你想要你定制的每个英文单词的首字母大写,就可能自己做一个简单的编辑软件来实现. 另外,你可以换编码风格,按 linux 的风格是不需要大写字母的. 解决方案二: 在C++中可以用string类中的字符转换. 解决方案三: 你是写代码实现,还是希望