Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮

简介: Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮前言之前写过屏蔽系统导航栏功能的文章,具体可看Android6.0 源码修改之屏蔽导航栏虚拟按键(Home和RecentAPP)/动态显示和隐藏NavigationBar在某些特殊定制的版本中要求完全去掉导航栏,那么当用户点进一些系统自带的应用界面如设置、联系人等,就没法退出了,虽然可以在actionBar中添加back按钮,但总不能每一个app都去添加吧。

Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮
前言
之前写过屏蔽系统导航栏功能的文章,具体可看Android6.0 源码修改之屏蔽导航栏虚拟按键(Home和RecentAPP)/动态显示和隐藏NavigationBar

在某些特殊定制的版本中要求完全去掉导航栏,那么当用户点进一些系统自带的应用界面如设置、联系人等,就没法退出了,虽然可以在actionBar中添加back按钮,但总不能每一个app都去添加吧。所以灵机一动我们就给系统添加一个全屏可拖拽的浮窗按钮,点击的时候处理返回键的逻辑。它大概长这样(审美可能丑了点,你们可以自由发挥)

在这里插入图片描述
图1 最终效果图

思路分析
通过分析之前的NavigationBar代码,发现系统是通过WindowManager添加View的方式来实现,此处我们也可以模拟这种方法来添加
添加悬浮窗以后监听触摸事件,跟随手指移动重新修改view的layoutParam
松手后获取当前X坐标,小于屏幕width的一半则平移归位至屏幕左边
添加系统的返回按键功能
一、添加悬浮窗
private void showFloatingWindow() {

DisplayMetrics outMetrics = new DisplayMetrics();
mWindowManager.getDefaultDisplay().getMetrics(outMetrics);
screenWidth = outMetrics.widthPixels;
screenHeight = outMetrics.heightPixels;

layoutParams = new WindowManager.LayoutParams();
layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
layoutParams.format = PixelFormat.RGBA_8888;
layoutParams.gravity = Gravity.LEFT | Gravity.TOP;
layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
layoutParams.width = 100;
layoutParams.height = 100;
layoutParams.x = 200;
layoutParams.y = 200;

button = new ImageButton(mContext);
button.setBackground(mContext.getResources().getDrawable(R.drawable.fab_background));//系统通讯录里的蓝色圆形图标
button.setImageResource(R.drawable.ic_sysbar_back);//系统本身的back图标
mWindowManager.addView(button, layoutParams);
isShowFloatingView = true;

}
代码很简单,就是通过windowManager添加一个ImageButton,宽高都是100的,位置在屏幕左上角为原点的200,200。需要注意的是因为我们是在源码里添加,而且是M的版本,所以type为WindowManager.LayoutParams.TYPE_PHONE。如果是在普通的app里注意事项可参考这篇

二、添加触摸事件监听
button.setOnTouchListener(new FloatingOnTouchListener());

private class FloatingOnTouchListener implements View.OnTouchListener {

private int lastX;
private int lastY;

@Override
public boolean onTouch(View view, MotionEvent event) {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            isDrag = false;
            lastX = (int) event.getRawX();
            lastY = (int) event.getRawY();
            break;

        case MotionEvent.ACTION_MOVE:
            isDrag = true;
            int nowX = (int) event.getRawX();
            int nowY = (int) event.getRawY();
            int movedX = nowX - lastX;
            int movedY = nowY - lastY;
            lastX = nowX;
            lastY = nowY;
            layoutParams.x = layoutParams.x + movedX;
            layoutParams.y = layoutParams.y + movedY;
            //获取当前手指移动的x和y,通过updateViewLayout方法将改变后的x和y设置给button
            mWindowManager.updateViewLayout(view, layoutParams);
            break;

        case MotionEvent.ACTION_UP:
            if (isDrag) {
                log("lastX=" + lastX + "  screenWidth=" + screenWidth);
                //手指抬起时,判断是需要滑动到屏幕左边还是屏幕右边
                if (lastX >= screenWidth / 2) {
                    setAnimation(view, lastX, screenWidth);
                } else {
                    setAnimation(view, -lastX, 0);
                }
            }
            break;
    }
    //返回true则消费事件,返回false则传递事件,此处特殊处理是为了和点击事件区分
    return isDrag || view.onTouchEvent(event);
}

三、添加抬起滑动归位动画
private void setAnimation(final View view, int fromX, int toX) {

    final ValueAnimator animator = ValueAnimator.ofInt(fromX, toX);
    if (Math.abs(fromX) < screenWidth / 4 || fromX > screenWidth * 3 / 4)
        animator.setDuration(300);
    else
        animator.setDuration(600);

    animator.setInterpolator(new LinearInterpolator());
    animator.addListener(new Animator.AnimatorListener() {
        @Override
        public void onAnimationStart(Animator animation) {}
        @Override
        public void onAnimationEnd(Animator animation) {
            log("onAnimationEnd=");
            savePreValue(layoutParams.x, layoutParams.y);
        }
        @Override
        public void onAnimationCancel(Animator animation) {}
        @Override
        public void onAnimationRepeat(Animator animation) {}
    });
    animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
        @Override
        public void onAnimationUpdate(ValueAnimator animation) {
            int current = (int) animator.getAnimatedValue();
            log("current=" + current);

            layoutParams.x = Math.abs(current);
            mWindowManager.updateViewLayout(view, layoutParams);
        }
    });
    animator.start();
}

}
同样是通过改变button的x和y值来达到滑动效果,只不过我只需要x平移,y为0,需要斜着滑的你们可自由发挥,为了使滑动看上去平滑,给动画添加了一个线性插值器,设置滑动时间,监听返回插值进度,这样动态设置给button。为了保存button的最终位置,添加了一个动画完成监听,并将x和y写入到SharedPreferences中保存。

四、添加点击返回功能
通过打印日志分析,系统导航栏的返回按键,发现其原理是通过KeyButtonView的触摸事件发送一个KeyEvent事件给系统来实现返回功能

源码位置frameworksbasepackagesSystemUIsrccomandroidsystemuistatusbarpolicyKeyButtonView.java

public boolean onTouchEvent(MotionEvent ev) {

final int action = ev.getAction();
int x, y;
if (action == MotionEvent.ACTION_DOWN) {
    mGestureAborted = false;
}
if (mGestureAborted) {
    return false;
}

switch (action) {
    case MotionEvent.ACTION_DOWN:
        //按下的时间
        mDownTime = SystemClock.uptimeMillis();
        setPressed(true);
        if (mCode != 0) {//按下事件
            sendEvent(KeyEvent.ACTION_DOWN, 0, mDownTime);
        } else {
            // Provide the same haptic feedback that the system offers for virtual keys.
            performHapticFeedback(HapticFeedbackConstants.VIRTUAL_KEY);
        }
        removeCallbacks(mCheckLongPress);
        postDelayed(mCheckLongPress, ViewConfiguration.getLongPressTimeout());
        break;
    case MotionEvent.ACTION_MOVE:
        x = (int)ev.getX();
        y = (int)ev.getY();
        setPressed(x >= -mTouchSlop
                && x < getWidth() + mTouchSlop
                && y >= -mTouchSlop
                && y < getHeight() + mTouchSlop);
        break;
    case MotionEvent.ACTION_CANCEL:
        setPressed(false);
        if (mCode != 0) {
            sendEvent(KeyEvent.ACTION_UP, KeyEvent.FLAG_CANCELED);
        }
        removeCallbacks(mCheckLongPress);
        break;
    case MotionEvent.ACTION_UP:
        final boolean doIt = isPressed();
        setPressed(false);
        if (mCode != 0) {
            if (doIt) {//抬起事件
                sendEvent(KeyEvent.ACTION_UP, 0);
                sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
                playSoundEffect(SoundEffectConstants.CLICK);
            } else {
                sendEvent(KeyEvent.ACTION_UP, KeyEvent.FLAG_CANCELED);
            }
        } else {
            // no key code, just a regular ImageView
            if (doIt) {
                performClick();
            }
        }
        removeCallbacks(mCheckLongPress);
        break;
}

return true;

}

//以下为我们给button添加的点击事件
private void sendEvent(int action, int flags, long when) {

int mCode = 4;
Log.e(TAG, "mCode="+mCode + "  flags="+flags);
final int repeatCount = (flags & KeyEvent.FLAG_LONG_PRESS) != 0 ? 1 : 0;
final KeyEvent ev = new KeyEvent(when - 100, when, action, mCode, repeatCount,
        0, KeyCharacterMap.VIRTUAL_KEYBOARD, 0,
        flags | KeyEvent.FLAG_FROM_SYSTEM | KeyEvent.FLAG_VIRTUAL_HARD_KEY,
        InputDevice.SOURCE_KEYBOARD);
InputManager.getInstance().injectInputEvent(ev,
        InputManager.INJECT_INPUT_EVENT_MODE_ASYNC);

}

button.setOnClickListener(new View.OnClickListener() {

    @Override
    public void onClick(View v) {
        Log.e(TAG,"click dragButton ...");
        final long mDownTime = SystemClock.uptimeMillis();
        //onBackPressed();
        sendEvent(KeyEvent.ACTION_DOWN, 0, mDownTime);

        new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                sendEvent(KeyEvent.ACTION_UP, 0, SystemClock.uptimeMillis());
            }
        }, 300);
    }
});

需要注意的地方,系统返回键对应的code为4,所以mCode=4,KeyButtonView的触摸事件包含按下和抬起,所以我们只需模拟发送按下和抬起事件,可以看到抬起事件加了300ms的延时发送,这是关键不然系统不会处理。
原文地址https://www.cnblogs.com/cczheng-666/p/10741082.html

相关文章
|
4月前
|
Ubuntu 开发工具 Android开发
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
本文介绍了在基于Ubuntu 22.04的环境下配置Python 3.9、安装repo工具、下载和同步AOSP源码包以及处理repo同步错误的详细步骤。
257 0
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
|
1天前
|
JavaScript 前端开发 iOS开发
ios样式开关按钮jQuery插件
ios样式开关按钮jQuery插件
22 7
|
1月前
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户点击按钮时,按钮将从圆形变为椭圆形,颜色从蓝色渐变到绿色;释放按钮时,动画以相反方式恢复。通过UIView的动画方法和弹簧动画效果,实现平滑自然的过渡。
55 1
|
2月前
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
【10月更文挑战第18天】本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户按下按钮时,按钮将从圆形变为椭圆形并从蓝色渐变为绿色;释放按钮时,动画恢复原状。通过UIView的动画方法和弹簧动画效果,实现平滑自然的动画过渡。
55 5
|
4月前
|
开发工具 git 索引
repo sync 更新源码 android-12.0.0_r34, fatal: 不能重置索引文件至版本 ‘v2.27^0‘。
本文描述了在更新AOSP 12源码时遇到的repo同步错误,并提供了通过手动git pull更新repo工具来解决这一问题的方法。
150 1
|
4月前
|
开发工具 uml git
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
本文分享了下载AOSP源码的方法,包括如何使用repo工具和处理常见的repo sync错误,以及配置Python环境以确保顺利同步特定版本的AOSP代码。
538 0
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
|
4月前
|
Java Android开发 芯片
使用Android Studio导入Android源码:基于全志H713 AOSP,方便解决编译、编码问题
本文介绍了如何将基于全志H713芯片的AOSP Android源码导入Android Studio以解决编译和编码问题,通过操作步骤的详细说明,展示了在Android Studio中利用代码提示和补全功能快速定位并修复编译错误的方法。
161 0
使用Android Studio导入Android源码:基于全志H713 AOSP,方便解决编译、编码问题
|
4月前
|
开发工具 Android开发 git
全志H713 Android 11 :给AOSP源码,新增一个Product
本文介绍了在全志H713 Android 11平台上新增名为myboard的产品的步骤,包括创建新的device目录、编辑配置文件、新增内核配置、记录差异列表以及编译kernel和Android系统的详细过程。
167 0
|
4月前
|
Ubuntu 开发工具 Android开发
Repo下载、编译AOSP源码:基于Ubuntu 21.04,android-12.1.0_r27
文章记录了作者在Ubuntu 21.04服务器上配置环境、下载并编译基于Android 12.1.0_r27版本的AOSP源码的过程,包括解决编译过程中遇到的问题和错误处理方法。
223 0