可拖拽可点击,可吸附button(2)

布局

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="toasttest.luuuzi.clicktoastdemo.MainActivity">

    <Button
        android:id="@+id/btn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!" />
</android.support.constraint.ConstraintLayout>

---------------## MainActivity代码 ##-

* 这个效果类似于ios的AitiveTouch,但是不会自动吸附在屏幕边缘,
* 当我们拖到什么位置时就停在什么位置,也解决了同时给按钮加上OnTouch和OnClick事件后拖动时会触发click事件的冲突,
* 在程序中应用这样的一种效果还是很出彩的,实现的原理也比较简单,
* 首先你得有一个按钮,其次你得把这个按钮放进一个帧布局,当然帧布局要铺满,这样按钮才能被你拖着满屏幕跑
* 相比上一篇就是添加了了一个可吸附的功能,这个功能就是在手指抬起的时候让button继续显示在左右2侧,
* 只需要在MotionEvent.ACTION_UP:下添加相应的代码即可,之前的代码不变,因为ACTION_MOVE事件和ACTION_UP事件都要设置button显示位置,所以,将left,right,top,bottom设置为全局的

package toasttest.luuuzi.clicktoastdemo;

import android.content.Context;
import android.graphics.PixelFormat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;

/*
* */
public class MainActivity extends AppCompatActivity {
    private Context mContext;
    //记录btn初始位置
    int startX;
    int startY;
    //记录手指抬起后的坐标
    int moveX;
    int moveY;
   //计算移动后的位置
    int left;
    int right;
    int top;
    int bottom;

    //isclick这个布尔值在全局声明,是解决两种事件冲突的关键,
    // 因为只有当onTouch返回false的时候,click事件才会生效,
    // 所以我们利用isclick值来控制事件的响应,当按下的时候设其为false,倘若不移动,返回false,响应click的事件
    //,如果按钮坐标发生移动就设置为true,这样在移动完按钮以后就不会同时响应OnClick事件了。
    boolean isclick;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mContext = this;
        final Button btn = (Button) findViewById(R.id.btn);
        btn.setOnTouchListener(new View.OnTouchListener() {

            @Override
            public boolean onTouch(View view, MotionEvent motionEvent) {
                switch (motionEvent.getAction()) {
                    case MotionEvent.ACTION_DOWN://按下事件,获取出事位置
                        isclick = false;//当按下的时候设置isclick为false
                        //1.获取控件的初始位置
                        /*getX:手按下的位置和当前控件(手要移动的控件)x轴的位置
                        * getRawX:手按下的位置与x轴的距离
                        * */
                        //motionEvent.getX();
                        startX = (int) motionEvent.getRawX();
                        startY = (int) motionEvent.getRawY();
                        break;
                    case MotionEvent.ACTION_MOVE://移动事件
                        isclick = true;//当按下的时候设置isclick为false
                        //2.记录手指抬起后的位置
                        moveX = (int) motionEvent.getRawX();
                        moveY = (int) motionEvent.getRawY();
                        //3.计算偏移量
                        int disX = moveX - startX;
                        int disY = moveY - startY;
                        //让偏移量设置给button的位置上去
                        //4.移动后的控件所在屏幕的(左,上)角位置,
                        left = btn.getLeft() + disX;//当前控件左边缘与屏幕左边缘的间距
                         right = btn.getRight() + disX;//右侧坐标,都是距左上角的距离
                         top = btn.getTop() + disY;//上边缘间距
                         bottom = btn.getBottom() + disY;//底部坐标

                        //7.容错处理
                        if (left < 0) {
                            return true;
                        }
                        if (right > ((WindowManager) getSystemService(WINDOW_SERVICE)).getDefaultDisplay().getWidth()) {
                            return true;
                        }
                        //上边缘不能超过屏幕可显示区域
                        if (top < 0) {
                            return true;
                        }
                        //下边缘(屏幕的高度-22(状态栏高度))
                        //状态栏高度没有api去拿到,只能手动去指定一个高度
                        if (bottom > ((WindowManager) getSystemService(WINDOW_SERVICE)).getDefaultDisplay().getHeight() - 150) {
                            return true;
                        }
                        //5.告知移动的控件按计算出来的坐标去展示
                        btn.layout(left, top, right, bottom);
                        //6.重至btn的位置
                        startX = (int) motionEvent.getRawX();
                        startY = (int) motionEvent.getRawY();
                        break;
                    case MotionEvent.ACTION_UP://抬起事件
                        //吸附在左右两边逻辑处理逻辑处理
                        if (left<((WindowManager)getSystemService(WINDOW_SERVICE)).getDefaultDisplay().getWidth()/2-btn.getWidth()){
                            left=0;
                            right=left+btn.getWidth();
                        }else {
                            left=((WindowManager)getSystemService(WINDOW_SERVICE)).getDefaultDisplay().getWidth()-btn.getWidth();
                            right=left+btn.getWidth();
                        }

                        //5.告知移动的控件按计算出来的坐标去展示
                        btn.layout(left, top, right, bottom);
                        //6.重至btn的位置
                        startX = (int) motionEvent.getRawX();
                        startY = (int) motionEvent.getRawY();
                        break;
                }
                /*
                * 返回值
                * false:不相应事件,
                * true响应事件
                * */
                return isclick;
            }
        });
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Toast.makeText(mContext, "哈哈", Toast.LENGTH_SHORT).show();
            }
        });
    }

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值