Android开发之仿滑动解锁

由于项目需求,现总结一下:自定义滑动开锁,先看效果图。

看了几眼美女啦!!!接下来我们看怎么实现的吧。

准备

1、在 build.gradle 文件中添加

compile 'com.nineoldandroids:library:2.4.0'
compile 'com.github.florent37:viewanimator:1.0.0@aar'

2、在清单文件中添加震动权限:

<uses-permission android:name="android.permission.VIBRATE" />

3、attrs.xml


<resources>

    
    <declare-styleable name="SlideToUnlockView">
        <attr name="slideImageViewWidth" format="dimension"/>
        <attr name="slideImageViewResId" format="reference"/>
        <attr name="slideImageViewResIdAfter" format="reference"/>
        <attr name="viewBackgroundResId" format="reference"/>
        <attr name="textHint" format="string"/>
        <attr name="textSize" format="integer"/>
        <attr name="textColorResId" format="color"/>
        <attr name="slideThreshold" format="float"/>
    declare-styleable>

resources>

自定义 View

package com.gyq.slideunclock.view;

import android.content.Context;
import android.content.res.TypedArray;
import android.util.AttributeSet;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.AccelerateInterpolator;
import android.widget.ImageView;
import android.widget.RelativeLayout;

import com.github.florent37.viewanimator.AnimationListener;
import com.github.florent37.viewanimator.ViewAnimator;
import com.gyq.slideunclock.R;
import com.gyq.slideunclock.utils.DensityUtil;
import com.nineoldandroids.view.ViewHelper;

/**
 * Created by gyq on 2017/8/7 11:16
 */
public class SlidingView extends RelativeLayout {
    private static final String TAG="CustomSlideToUnlockView";
    private static final long DEAFULT_DURATIN_LONG = 200;//左弹回,动画时长
    private static final long DEAFULT_DURATIN_SHORT = 100;//右弹,动画时长
    private static final boolean LOG = true;//打印开关
    private static int  DISTANCE_LIMIT = 600;//滑动阈值
    private static float  THRESHOLD = 0.5F;//滑动阈值比例:默认是0.5,即滑动超过父容器宽度的一半再松手就会触发
    protected Context mContext;
    private ImageView iv_slide;//滑块
    private RelativeLayout rl_slide;//滑动view
    private RelativeLayout rl_root;//父容器
    private boolean mIsUnLocked;//已经滑到最右边,将不再响应touch事件
    private CallBack mCallBack;//回调


    private int slideImageViewWidth;//滑块宽度
    private int  slideImageViewResId;//滑块资源
    private int  slideImageViewResIdAfter;//滑动到右边时,滑块资源id
    private int  viewBackgroundResId;//root 背景
    private String textHint;//文本
    private int textSize;//单位是sp,只拿数值
    private int textColorResId;//颜色,@color


    public SlidingView(Context mContext) {
        super(mContext);
        this.mContext = mContext;
        initView();
    }

    public SlidingView(Context mContext, AttributeSet attrs) {
        super(mContext, attrs);
        this.mContext = mContext;
        TypedArray mTypedArray = mContext.obtainStyledAttributes(attrs,
                R.styleable.SlideToUnlockView);
        init(mTypedArray);
        initView();
    }

    public SlidingView(Context mContext, AttributeSet attrs, int defStyleAttr) {
        super(mContext, attrs, defStyleAttr);
        this.mContext = mContext;

        TypedArray mTypedArray = mContext.obtainStyledAttributes(attrs,
                R.styleable.SlideToUnlockView);
        init(mTypedArray);

        initView();

    }

    private void init(TypedArray mTypedArray) {

        slideImageViewWidth= (int) mTypedArray.getDimension(R.styleable.SlideToUnlockView_slideImageViewWidth, DensityUtil.dp2px(getContext(), 50));
        slideImageViewResId= mTypedArray.getResourceId(R.styleable.SlideToUnlockView_slideImageViewResId, -1);
        slideImageViewResIdAfter= mTypedArray.getResourceId(R.styleable.SlideToUnlockView_slideImageViewResIdAfter, -1);
        viewBackgroundResId= mTypedArray.getResourceId(R.styleable.SlideToUnlockView_viewBackgroundResId, -1);
        textHint=mTypedArray.getString(R.styleable.SlideToUnlockView_textHint);
        textSize=mTypedArray.getInteger(R.styleable.SlideToUnlockView_textSize, 7);
        textColorResId= mTypedArray.getColor(R.styleable.SlideToUnlockView_textColorResId, getResources().getColor(android.R.color.white));
        THRESHOLD=mTypedArray.getFloat(R.styleable.SlideToUnlockView_slideThreshold, 0.5f);

        mTypedArray.recycle();
    }


    private int mActionDownX, mLastX, mSlidedDistance;


    /**
     * 初始化界面布局
     */
    protected void initView() {

        LayoutInflater.from(mContext).inflate(R.layout.layout_view_slide_to_unlock,
                this, true);

        rl_root = (RelativeLayout) findViewById(R.id.rl_root);
        rl_slide = (RelativeLayout) findViewById(R.id.rl_slide);
        iv_slide = (ImageView) findViewById(R.id.iv_slide);
        //tv_hint = (TextView) findViewById(R.id.tv_hint);

        LayoutParams params= (LayoutParams) iv_slide .getLayoutParams();
        //获取当前控件的布局对象
        params.width= slideImageViewWidth;//设置当前控件布局的高度
        iv_slide.setLayoutParams(params);//将设置好的布局参数应用到控件中

        setImageDefault();
        if(viewBackgroundResId>0){
            rl_slide.setBackgroundResource(viewBackgroundResId);//rootView设置背景
        }

        //MarginLayoutParams tvParams = (MarginLayoutParams) tv_hint.getLayoutParams();
       // tvParams.setMargins(0, 0, slideImageViewWidth, 0);//textview的marginRight设置为和滑块的宽度一致

        //添加滑动监听
        rl_slide.setOnTouchListener(new OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {

                DISTANCE_LIMIT= (int) (SlidingView.this.getWidth()*THRESHOLD);//默认阈值是控件宽度的一半

                switch (event.getAction()) {

                    case MotionEvent.ACTION_DOWN://按下时记录纵坐标

                        if(mIsUnLocked){//滑块已经在最右边则不处理touch
                            return false;
                        }

                        mLastX = (int) event.getRawX();//最后一个action时x值
                        mActionDownX = (int) event.getRawX();//按下的瞬间x
                        break;

                    case MotionEvent.ACTION_MOVE://上滑才处理,如果用户一开始就下滑,则过掉不处理


                        int dX = (int) event.getRawX() - mLastX;

                        mSlidedDistance = (int) event.getRawX() - mActionDownX;

                        final MarginLayoutParams params = (MarginLayoutParams) v.getLayoutParams();
                        int left = params.leftMargin;
                        int top = params.topMargin;
                        int right = params.rightMargin;
                        int bottom = params.bottomMargin;


                        int leftNew = left + dX;
                        int rightNew =right - dX;

                        if (mSlidedDistance > 0) {//直接通过margin实现滑动
                            params.setMargins(leftNew, top, rightNew, bottom);
                            v.setLayoutParams(params);

                            //回调
                            if(mCallBack!=null){
                                mCallBack.onSlide(mSlidedDistance);
                            }
                            mLastX = (int) event.getRawX();
                        } else {
                            return true;
                        }

                        break;


                    case MotionEvent.ACTION_UP:
                        if (Math.abs(mSlidedDistance) > DISTANCE_LIMIT) {
                            scrollToRight(v);//右边
                        } else {
                            scrollToLeft(v);//左边
                        }
                        break;


                    default:

                        break;

                }


                return true;
            }
        });



    }


    private void logI(String tag,String content){
        if(LOG){
            Log.i(tag,content);
        }
    }



    /**
     * 滑动未到阈值时松开手指,弹回到最左边
     **/
    private void scrollToLeft(final View v) {


        final MarginLayoutParams params1 = (MarginLayoutParams) v.getLayoutParams();


        ViewAnimator
                .animate( rl_slide)
                .translationX(ViewHelper.getTranslationX(v), -params1.leftMargin)
                .interpolator(new AccelerateInterpolator())
                .duration(DEAFULT_DURATIN_LONG)
                .onStop(new AnimationListener.Stop() {
                    @Override
                    public void onStop() {
                        MarginLayoutParams para = (MarginLayoutParams) v.getLayoutParams();
                        logI(TAG, "scrollToLeft动画结束para.leftMargin:" + para.leftMargin);
                        logI(TAG, "scrollToLeft动画结束para.rightMargin:" + para.rightMargin);
                        logI(TAG, "scrollToLeft动画结束,ViewHelper.getTranslationX(v):" + ViewHelper.getTranslationX(v));
                        mSlidedDistance = 0;
                        //tv_hint.setAlpha(1.0f);
                        mIsUnLocked=false;
                        if(mCallBack!=null){
                            mCallBack.onSlide(mSlidedDistance);
                        }
                        setImageDefault();
                    }
                })
                .start();
    }


    /**
     * @des:滑动到右边,并触发回调
     **/
    private void scrollToRight(final View v) {


        final MarginLayoutParams params1 = (MarginLayoutParams) v.getLayoutParams();

        //移动到最右端  移动的距离是 父容器宽度-leftMargin
        ViewAnimator
                .animate( rl_slide)
                //.translationX(ViewHelper.getTranslationX(v), ViewHelper.getTranslationX(v)+100)
                .translationX(ViewHelper.getTranslationX(v), ( rl_slide.getWidth() - params1.leftMargin-slideImageViewWidth))
                //.translationX(params1.leftMargin, ( rl_slide.getWidth() - params1.leftMargin-100))
                .interpolator(new AccelerateInterpolator())
                .duration(DEAFULT_DURATIN_SHORT)
                .onStop(new AnimationListener.Stop() {
                    @Override
                    public void onStop() {
                        MarginLayoutParams para = (MarginLayoutParams) v.getLayoutParams();
                        mSlidedDistance = 0;
                        //tv_hint.setAlpha(0.0f);
                        mIsUnLocked=true;

                        if(slideImageViewResIdAfter>0){
                            iv_slide.setImageResource(slideImageViewResIdAfter);//滑块imagview设置资源
                        }

                        //回调
                        if(mCallBack!=null){
                            mCallBack.onUnlocked();
                        }
                    }
                })
                .start();


    }


    public void resetView(){
        mIsUnLocked=false;
        setImageDefault();
        scrollToLeft(rl_slide);
    }

    private void setImageDefault() {

        if(slideImageViewResId>0){
            iv_slide.setImageResource(slideImageViewResId);//滑块imagview设置资源
        }
    }

    public interface CallBack{
        void onSlide(int distance);//右滑距离回调
        void onUnlocked();//滑动到了右边,事件回调
    }


    public CallBack getmCallBack() {
        return mCallBack;
    }

    public void setmCallBack(CallBack mCallBack) {
        this.mCallBack = mCallBack;
    }
}

MainActivity.java

package com.gyq.slideunclock;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.os.Vibrator;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.view.Window;
import android.widget.ImageView;
import android.widget.TextView;

import com.gyq.slideunclock.view.SlidingView;

import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {
    private SlidingView mSlide;

    private ImageView mGirl;

    private TextView mTime,mDate;

    private Vibrator vibrator;

    private ScreenReceiver receiver;

    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            refreshUI();
        }
    };
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.activity_main);

        // 获取系统振动器服务
        vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);

        mSlide = (SlidingView)findViewById(R.id.slide_to_unlock);
        mGirl = (ImageView)findViewById(R.id.iv_girl);

        mTime = (TextView) findViewById(R.id.tv_time);
        mDate = (TextView) findViewById(R.id.tv_date);

        SlidingView.CallBack callBack = new SlidingView.CallBack() {

            @Override
            public void onSlide(int distance) {
                //mText.setText("slide distance:"+distance);
            }

            @Override
            public void onUnlocked() {
                // 启动震动器 100ms
                vibrator.vibrate(100);
                mGirl.setVisibility(View.VISIBLE);

                mSlide.resetView();
                mSlide.setVisibility(View.GONE);
                mTime.setVisibility(View.GONE);
                mDate.setVisibility(View.GONE);

            }
        };

        mSlide.setmCallBack(callBack);


    }

    @Override
    protected void onStart() {
        super.onStart();
        initData();
        // 注册屏幕锁屏的广播
        registScreenOffReceiver();
    }


    private void initData() {
        SimpleDateFormat format = new SimpleDateFormat("E    yyyy/MM/dd");
        String str = format.format(new Date(System.currentTimeMillis()));
        if (str.contains("周"))
            str = "星期" + str.substring(1);
        mDate.setText(str);

        new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    try {
                        Thread.sleep(1000);
                        mHandler.sendMessage(mHandler.obtainMessage());
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();

    }

    /**
     * 通过handler及时更新时间显示效果
     */
    public void refreshUI() {
        SimpleDateFormat formatter = new SimpleDateFormat("HH:mm");
        Date date = new Date(System.currentTimeMillis());
        String curTime = formatter.format(date);
        mTime.setText(curTime);           //当前时间格式
    }

    /**
     * 注册一个屏幕锁屏的广播
     */
    private void registScreenOffReceiver() {
        // TODO Auto-generated method stub
        receiver = new ScreenReceiver();
        // 创建一个意图过滤器
        IntentFilter filter = new IntentFilter();
        // 添加屏幕锁屏的广播
        filter.addAction("android.intent.action.SCREEN_OFF");
        // 在代码里边来注册广播
        this.registerReceiver(receiver, filter);

    }

    class ScreenReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();

            // 关屏的操作
            if ("android.intent.action.SCREEN_OFF".equals(action)) {
                // 当手机关屏时,我们同时也锁屏
                mSlide.setVisibility(View.VISIBLE);
                mTime.setVisibility(View.VISIBLE);
                mDate.setVisibility(View.VISIBLE);
                // 设置图片消失
                mGirl.setVisibility(View.GONE);
            }
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 注销注册的广播
        unregisterReceiver(receiver);
        receiver = null;
    }
}

工具类:DensityUtil.java

package com.gyq.slideunclock.utils;

import android.content.Context;
import android.util.DisplayMetrics;
import android.util.TypedValue;
import android.view.WindowManager;

/**
 * Created by gyq on 2017/8/7 11:05
 */
public class DensityUtil {
    @Deprecated
    public static int dip2px(Context paramContext, float paramFloat) {
        return (int) (0.5F + paramFloat
                * paramContext.getResources().getDisplayMetrics().density);
    }

    @Deprecated
    public static int px2dip(Context context, float paramFloat) {
        return (int) (0.5F + paramFloat
                / context.getResources().getDisplayMetrics().density);
    }

    @Deprecated
    public static int sp2px(Context context, float spValue) {
        final float fontScale = context.getResources().getDisplayMetrics().scaledDensity;
        return (int) (spValue * fontScale + 0.5f);
    }

    @Deprecated
    public static int px2sp(Context context, float value) {
        final float fontScale = context.getResources().getDisplayMetrics().scaledDensity;
        return (int) (value / fontScale + 0.5f);
    }

    public static int dip2px(float density, float value) {
        return (int) (0.5F + value * density);
    }

    public static int px2dip(float density, float value) {
        return (int) (0.5F + value / density);
    }

    public static int sp2px(float scaledDensity, float value) {
        return (int) (0.5F + value * scaledDensity);
    }

    public static int px2sp(float fontScale /** scaledDensity */
            , float value) {
        return (int) (value / fontScale + 0.5f);


    }

    public static final int getStatusHeighByDensity(Context context) {
        int h = 38;
        int density = context.getResources().getDisplayMetrics().densityDpi;

        switch (density) {
            case 120:
                h = 19;
                break;
            case 160:
                h = 25;
                break;
            case 240:
                h = 38;
                break;
            case 320:
                h = 50;
                break;
            case 400:
                h = 63;
                break;
            case 480:
                h = 75;
                break;
            default:
                break;
        }

        return h;
    }

    private static int displayWidth, displayHeight;

    private static void initDisplay(Context context) {
        DisplayMetrics dm = new DisplayMetrics();
        ((WindowManager) context.getApplicationContext().getSystemService(
                Context.WINDOW_SERVICE)).getDefaultDisplay().getMetrics(dm);

        displayWidth = dm.widthPixels;

        displayHeight = dm.heightPixels;
    }


    public static final int getDisplayWidth(Context context) {
        if (displayWidth == 0) {
            initDisplay(context);
        }
        return displayWidth;
    }


    public static final int getDisplayHeight(Context context) {
        if (displayHeight == 0) {
            initDisplay(context);
        }
        return displayHeight;
    }

    public static int dp2px(Context context, int dp) {
        return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp,context.getResources().getDisplayMetrics());
    }

    public static int dp2px(Context context, float dp) {
        return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp,context.getResources().getDisplayMetrics());
    }

    public static int px2dp(Context context, int px) {
        return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_PX, px,context.getResources().getDisplayMetrics());
    }

    public static int px2sp(Context context, int px) {
        return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, px,context.getResources().getDisplayMetrics());
    }

    public static int getScreenWidth(Context context)
    {
        WindowManager wm = (WindowManager) context
                .getSystemService(Context.WINDOW_SERVICE );
        DisplayMetrics outMetrics = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics( outMetrics);
        return outMetrics .widthPixels ;
    }
}

记录下来方便今后开发使用。

你可能感兴趣的:(自定义View)