├── .gitignore ├── .idea ├── compiler.xml ├── copyright │ └── profiles_settings.xml ├── encodings.xml ├── gradle.xml ├── misc.xml ├── modules.xml └── runConfigurations.xml ├── README.md ├── app ├── .gitignore ├── build.gradle ├── proguard-rules.pro └── src │ ├── androidTest │ └── java │ │ └── com │ │ └── ice │ │ └── floatingball │ │ └── ExampleInstrumentedTest.java │ ├── main │ ├── AndroidManifest.xml │ ├── java │ │ └── com │ │ │ └── ice │ │ │ └── floatingball │ │ │ ├── FloatingView.java │ │ │ ├── MainActivity.java │ │ │ ├── MyAccessibilityService.java │ │ │ └── ViewManager.java │ └── res │ │ ├── layout │ │ └── activity_main.xml │ │ ├── mipmap-hdpi │ │ └── ic_launcher.png │ │ ├── mipmap-mdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxhdpi │ │ └── ic_launcher.png │ │ ├── mipmap-xxxhdpi │ │ └── ic_launcher.png │ │ ├── values-w820dp │ │ └── dimens.xml │ │ ├── values │ │ ├── colors.xml │ │ ├── dimens.xml │ │ ├── strings.xml │ │ └── styles.xml │ │ └── xml │ │ └── accessibilityservice.xml │ └── test │ └── java │ └── com │ └── ice │ └── floatingball │ └── ExampleUnitTest.java ├── build.gradle ├── gradle.properties ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat └── settings.gradle /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | .externalNativeBuild 10 | -------------------------------------------------------------------------------- /.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /.idea/copyright/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | -------------------------------------------------------------------------------- /.idea/encodings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 17 | 18 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 19 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 46 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Android悬浮球及全局返回功能的实现 2 | ## 先来一发效果图: 3 | 前面是返回效果,最后一下是实现home键的效果 4 | ![效果图](http://upload-images.jianshu.io/upload_images/4774781-5da3a469f8adb4dc.gif?imageMogr2/auto-orient/strip) 5 | 6 | ## 前言 7 | 很久之前,就想做一个悬浮球了,毕竟是程序猿嘛,有想要的功能的时候总是想自己尝试一下,于是兴致勃勃的找了好久,都没有找到全局返回功能该如何实现!最后也无疾而终,就在前两天,又想到了这个功能,今天硬是花了好久,从一个同类软件获得了一点灵感,有一个关键的地方被我察觉到了,顺着这个思路找了很多资料,便实现了全局返回功能。 8 | ## 思路 9 | 废话不多说了,说说主要的思路吧,关键的一个类就是:`AccessibilityService`,[官方文档地址](https://developer.android.google.cn/reference/android/accessibilityservice/AccessibilityService.html "官方文档地址"),这个类与手机里面的一个功能密切相关:辅助功能-服务。官方文档来看,这个功能是为了方便有障碍的人士更好的使用手机。我们这里就不展开介绍里面的API了,为了实现我们的全局返回功能,我们只需要使用一个函数即可:`boolean performGlobalAction (int action)`,官方解释如下: 10 | > Performs a global action. Such an action can be performed at any moment regardless of the current application or user location in that application. For example going back, going home, opening recents, etc. 11 | 12 | 翻译过来就是: 13 | > 执行全局动作。无论该应用程序中的当前应用程序或用户位置如何,都可以随时执行此类操作。例如执行HOME键,BACK键,任务键等 14 | 15 | 其中可以传入的参数有四个: 16 | ` 17 | GLOBAL_ACTION_BACK 18 | GLOBAL_ACTION_HOME 19 | GLOBAL_ACTION_NOTIFICATIONS 20 | GLOBAL_ACTION_RECENTS 21 | ` 22 | 从字面就可以理解,我们返回功能需要的就是GLOBAL_ACTION_BACK。所以我们只需要开启服务,调用函数就可以实现全局返回功能了。 23 | ##编写代码 24 | ###最重要的服务类 25 | 我们要新建一个类去继承自上面那个类: 26 | ``` 27 | public class MyAccessibilityService extends AccessibilityService { 28 | public static final int BACK = 1; 29 | public static final int HOME = 2; 30 | private static final String TAG = "ICE"; 31 | 32 | @Override 33 | public void onCreate() { 34 | super.onCreate(); 35 | //使用EventBus代替广播 36 | EventBus.getDefault().register(this); 37 | } 38 | 39 | @Override 40 | public void onAccessibilityEvent(AccessibilityEvent event) { } 41 | 42 | @Override 43 | public void onInterrupt() {} 44 | 45 | @Subscribe 46 | public void onReceive(Integer action){ 47 | switch (action){ 48 | case BACK: 49 | performGlobalAction(AccessibilityService.GLOBAL_ACTION_BACK); 50 | break; 51 | case HOME: 52 | performGlobalAction(AccessibilityService.GLOBAL_ACTION_HOME); 53 | break; 54 | } 55 | } 56 | 57 | } 58 | ``` 59 | 上面的`onReceive`方法是我们使用EventBus的订阅函数,当其他地方发送消息之后,我们这里就可以收到,然后判断是要执行后退还是回到桌面。 60 | 然后我们在AndroiManifest里面要注册我们的服务,但是这个注册的比较特殊: 61 | 首先加入权限声明: 62 | `` 63 | 然后注册服务: 64 | ``` 65 | 68 | 69 | 70 | 71 | 74 | 75 | ``` 76 | 其中resource中的内容我们要在xml包中声明,首先新建一个xml包,如下: 77 | 78 | ![包结构](http://upload-images.jianshu.io/upload_images/4774781-651671110ba6105e.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 79 | 80 | 然后新建一个accessibilityservice.xml文件,内容如下: 81 | ``` 82 | 83 | 86 | ``` 87 | 里面还可以设置许多属性,在这里就不介绍了,有兴趣的可以在官方文档里面查看。 88 | 到时候description的显示效果如下: 89 | 90 | ![description显示效果](http://upload-images.jianshu.io/upload_images/4774781-c128d092dcc95594.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 91 | 92 | 好了,到现在就已经完成了AccessibilityService服务的创建与注册了,接下来在Activity中启动服务就可以了: `startService(new 93 | Intent(this,MyAccessibilityService.class));` 94 | 使用EventBus传递事件即可实现返回:`EventBus.getDefault().post(MyAccessibilityService.BACK);` 95 | 但是要打开服务才行,简单办法是直接调用Intent跳到设置界面: 96 | `startActivity(new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS));` 97 | 或者手动进入设置->辅助功能->服务->找到自己的app,然后开启服务即可。(不同的系统可能略有差异,小米就是在无障碍里面),界面如下: 98 | 99 | ![服务界面](http://upload-images.jianshu.io/upload_images/4774781-ff7d110a4857ca10.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240) 100 | 101 | ### 悬浮球的简单实现 102 | 1.自定义一个View,画一个悬浮球: 103 | ``` 104 | public class FloatingView extends View { 105 | 106 | public int height = 150; 107 | public int width = 150; 108 | private Paint paint; 109 | 110 | 111 | public FloatingView(Context context){ 112 | super(context); 113 | paint = new Paint(); 114 | } 115 | 116 | @Override 117 | protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 118 | super.onMeasure(widthMeasureSpec, heightMeasureSpec); 119 | setMeasuredDimension(height,width); 120 | } 121 | 122 | @Override 123 | protected void onDraw(Canvas canvas) { 124 | super.onDraw(canvas); 125 | //画大圆 126 | paint.setStyle(Paint.Style.FILL); 127 | paint.setAntiAlias(true); 128 | paint.setColor(getResources().getColor(R.color.state_one)); 129 | canvas.drawCircle(width/2,width/2,width/2,paint); 130 | //画小圆圈 131 | paint.setStyle(Paint.Style.STROKE); 132 | paint.setColor(Color.WHITE); 133 | canvas.drawCircle(width/2,width/2, (float) (width*1.0/4),paint); 134 | 135 | } 136 | ``` 137 | 代码很简单,是画了一个大圆,然后一个小点的圆圈。 138 | 接下来,把这个view展示在桌面: 139 | ``` 140 | 141 | public class ViewManager { 142 | FloatingView floatBall; 143 | WindowManager windowManager; 144 | public static ViewManager manager; 145 | Context context; 146 | private WindowManager.LayoutParams floatBallParams; 147 | 148 | private ViewManager(Context context) { 149 | this.context = context; 150 | } 151 | 152 | public static ViewManager getInstance(Context context) { 153 | if (manager == null) { 154 | manager = new ViewManager(context); 155 | } 156 | return manager; 157 | } 158 | public void showFloatBall() { 159 | floatBall = new FloatingView(context); 160 | windowManager = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE); 161 | if (floatBallParams == null) { 162 | floatBallParams = new WindowManager.LayoutParams(); 163 | floatBallParams.width = floatBall.width; 164 | floatBallParams.height = floatBall.height; 165 | floatBallParams.gravity = Gravity.TOP | Gravity.LEFT; 166 | floatBallParams.type = WindowManager.LayoutParams.TYPE_TOAST; 167 | floatBallParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL; 168 | floatBallParams.format = PixelFormat.RGBA_8888; 169 | } 170 | 171 | windowManager.addView(floatBall, floatBallParams); 172 | 173 | floatBall.setOnClickListener(new View.OnClickListener() { 174 | @Override 175 | public void onClick(View v) { 176 | EventBus.getDefault().post(MyAccessibilityService.BACK); 177 | Toast.makeText(context, "点击了悬浮球 执行后退操作", Toast.LENGTH_SHORT).show(); 178 | } 179 | }); 180 | 181 | floatBall.setOnLongClickListener(new View.OnLongClickListener() { 182 | @Override 183 | public boolean onLongClick(View v) { 184 | EventBus.getDefault().post(MyAccessibilityService.HOME); 185 | Toast.makeText(context, "长按了悬浮球 执行返回桌面", Toast.LENGTH_SHORT).show(); 186 | return false; 187 | } 188 | }); 189 | 190 | } 191 | 192 | public int getScreenWidth() { 193 | return windowManager.getDefaultDisplay().getWidth(); 194 | } 195 | 196 | } 197 | ``` 198 | 上面代码把view加入到window中,并给view设置了点击事件,以及长按事件,向AccessibilityService传递消息,执行相应的事件。 199 | 要显示悬浮窗,要声明权限: 200 | `` 201 | 然后手动开启权限!不然无法显示悬浮窗。 202 | 最后我们在Activity中开启我们自定义的悬浮窗即可: 203 | `ViewManager.getInstance(MainActivity.this).showFloatBall();` 204 | ## 结束语 205 | 现在看来,实现一个全局返回功能真的非常简单,但是当初就真的找了非常久,怎么找,怎么试都没法实现这个功能,于是尝试着去学学别的悬浮窗的代码,但是没办法,加壳了,反编译后没法看。但是我注意到了一个细节,它要我打开服务才能使用悬浮窗的功能,所以就从这里下手,慢慢找到了实现全局返回的方法。 206 | -------------------------------------------------------------------------------- /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 25 5 | buildToolsVersion "25.0.2" 6 | defaultConfig { 7 | applicationId "com.ice.floatingball" 8 | minSdkVersion 19 9 | targetSdkVersion 25 10 | versionCode 1 11 | versionName "1.0" 12 | testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" 13 | } 14 | buildTypes { 15 | release { 16 | minifyEnabled false 17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 18 | } 19 | } 20 | } 21 | 22 | dependencies { 23 | compile fileTree(dir: 'libs', include: ['*.jar']) 24 | androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', { 25 | exclude group: 'com.android.support', module: 'support-annotations' 26 | }) 27 | compile 'com.android.support:appcompat-v7:25.2.0' 28 | testCompile 'junit:junit:4.12' 29 | compile 'org.greenrobot:eventbus:3.0.0' 30 | 31 | } 32 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in D:\AndroidStudioSDK\sdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/ice/floatingball/ExampleInstrumentedTest.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import android.content.Context; 4 | import android.support.test.InstrumentationRegistry; 5 | import android.support.test.runner.AndroidJUnit4; 6 | 7 | import org.junit.Test; 8 | import org.junit.runner.RunWith; 9 | 10 | import static org.junit.Assert.*; 11 | 12 | /** 13 | * Instrumentation test, which will execute on an Android device. 14 | * 15 | * @see Testing documentation 16 | */ 17 | @RunWith(AndroidJUnit4.class) 18 | public class ExampleInstrumentedTest { 19 | @Test 20 | public void useAppContext() throws Exception { 21 | // Context of the app under test. 22 | Context appContext = InstrumentationRegistry.getTargetContext(); 23 | 24 | assertEquals("com.ice.floatingball", appContext.getPackageName()); 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 4 | 5 | 6 | 7 | 8 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 25 | 26 | 27 | 28 | 31 | 32 | 33 | 34 | 35 | 36 | -------------------------------------------------------------------------------- /app/src/main/java/com/ice/floatingball/FloatingView.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import android.content.Context; 4 | import android.graphics.Canvas; 5 | import android.graphics.Color; 6 | import android.graphics.Paint; 7 | import android.view.View; 8 | 9 | /** 10 | * Created by asd on 4/20/2017. 11 | */ 12 | 13 | public class FloatingView extends View { 14 | 15 | public int height = 150; 16 | public int width = 150; 17 | private Paint paint; 18 | 19 | 20 | public FloatingView(Context context){ 21 | super(context); 22 | paint = new Paint(); 23 | } 24 | 25 | @Override 26 | protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 27 | super.onMeasure(widthMeasureSpec, heightMeasureSpec); 28 | setMeasuredDimension(height,width); 29 | } 30 | 31 | @Override 32 | protected void onDraw(Canvas canvas) { 33 | super.onDraw(canvas); 34 | //画大圆 35 | paint.setStyle(Paint.Style.FILL); 36 | paint.setAntiAlias(true); 37 | paint.setColor(getResources().getColor(R.color.state_one)); 38 | canvas.drawCircle(width/2,width/2,width/2,paint); 39 | //画小圆圈 40 | paint.setStyle(Paint.Style.STROKE); 41 | paint.setColor(Color.WHITE); 42 | canvas.drawCircle(width/2,width/2, (float) (width*1.0/4),paint); 43 | 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /app/src/main/java/com/ice/floatingball/MainActivity.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import android.content.Intent; 4 | import android.os.Bundle; 5 | import android.provider.Settings; 6 | import android.support.v7.app.AppCompatActivity; 7 | import android.view.View; 8 | import android.widget.Button; 9 | 10 | public class MainActivity extends AppCompatActivity { 11 | 12 | Button btn_openSetting,btn_openFloatingBall; 13 | 14 | @Override 15 | protected void onCreate(Bundle savedInstanceState) { 16 | super.onCreate(savedInstanceState); 17 | setContentView(R.layout.activity_main); 18 | 19 | btn_openSetting = (Button) findViewById(R.id.btn_openSetting); 20 | btn_openFloatingBall = (Button) findViewById(R.id.btn_openFloatingBall); 21 | 22 | btn_openSetting.setOnClickListener(new View.OnClickListener() { 23 | @Override 24 | public void onClick(View v) { 25 | //打开设置 打开服务才能实现返回功能 26 | startActivity(new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS)); 27 | } 28 | }); 29 | 30 | btn_openFloatingBall.setOnClickListener(new View.OnClickListener() { 31 | @Override 32 | public void onClick(View v) { 33 | ViewManager.getInstance(MainActivity.this).showFloatBall(); 34 | } 35 | }); 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /app/src/main/java/com/ice/floatingball/MyAccessibilityService.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import android.accessibilityservice.AccessibilityService; 4 | import android.content.Intent; 5 | import android.util.Log; 6 | import android.view.accessibility.AccessibilityEvent; 7 | 8 | import org.greenrobot.eventbus.EventBus; 9 | import org.greenrobot.eventbus.Subscribe; 10 | 11 | /** 12 | * Created by asd on 4/20/2017. 13 | */ 14 | 15 | 16 | public class MyAccessibilityService extends AccessibilityService { 17 | 18 | public static final int BACK = 1; 19 | public static final int HOME = 2; 20 | private static final String TAG = "ICE"; 21 | 22 | @Override 23 | public void onCreate() { 24 | super.onCreate(); 25 | EventBus.getDefault().register(this); 26 | } 27 | 28 | @Override 29 | public int onStartCommand(Intent intent, int flags, int startId) { 30 | Log.d(TAG, "onStartCommand: 点击了"); 31 | return super.onStartCommand(intent, flags, startId); 32 | } 33 | 34 | @Override 35 | public void onAccessibilityEvent(AccessibilityEvent event) { 36 | 37 | } 38 | 39 | @Override 40 | public void onInterrupt() { 41 | 42 | } 43 | 44 | @Subscribe 45 | public void onReceive(Integer action){ 46 | switch (action){ 47 | case BACK: 48 | performGlobalAction(AccessibilityService.GLOBAL_ACTION_BACK); 49 | break; 50 | case HOME: 51 | performGlobalAction(AccessibilityService.GLOBAL_ACTION_HOME); 52 | break; 53 | } 54 | } 55 | 56 | 57 | } -------------------------------------------------------------------------------- /app/src/main/java/com/ice/floatingball/ViewManager.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import android.content.Context; 4 | import android.graphics.PixelFormat; 5 | import android.view.Gravity; 6 | import android.view.MotionEvent; 7 | import android.view.View; 8 | import android.view.WindowManager; 9 | import android.widget.Toast; 10 | 11 | import org.greenrobot.eventbus.EventBus; 12 | 13 | /** 14 | * Created by asd on 1/1/2017. 15 | */ 16 | 17 | public class ViewManager { 18 | FloatingView floatBall; 19 | WindowManager windowManager; 20 | public static ViewManager manager; 21 | Context context; 22 | private WindowManager.LayoutParams floatBallParams; 23 | 24 | private ViewManager(Context context) { 25 | this.context = context; 26 | } 27 | 28 | public static ViewManager getInstance(Context context) { 29 | if (manager == null) { 30 | manager = new ViewManager(context); 31 | } 32 | return manager; 33 | } 34 | public void showFloatBall() { 35 | floatBall = new FloatingView(context); 36 | windowManager = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE); 37 | if (floatBallParams == null) { 38 | floatBallParams = new WindowManager.LayoutParams(); 39 | floatBallParams.width = floatBall.width; 40 | floatBallParams.height = floatBall.height; 41 | floatBallParams.gravity = Gravity.TOP | Gravity.LEFT; 42 | floatBallParams.type = WindowManager.LayoutParams.TYPE_TOAST; 43 | floatBallParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL; 44 | floatBallParams.format = PixelFormat.RGBA_8888; 45 | } 46 | 47 | windowManager.addView(floatBall, floatBallParams); 48 | 49 | floatBall.setOnClickListener(new View.OnClickListener() { 50 | @Override 51 | public void onClick(View v) { 52 | EventBus.getDefault().post(MyAccessibilityService.BACK); 53 | Toast.makeText(context, "点击了悬浮球 执行后退操作", Toast.LENGTH_SHORT).show(); 54 | } 55 | }); 56 | 57 | floatBall.setOnLongClickListener(new View.OnLongClickListener() { 58 | @Override 59 | public boolean onLongClick(View v) { 60 | EventBus.getDefault().post(MyAccessibilityService.HOME); 61 | Toast.makeText(context, "长按了悬浮球 执行返回桌面", Toast.LENGTH_SHORT).show(); 62 | return false; 63 | } 64 | }); 65 | 66 | floatBall.setOnTouchListener(new View.OnTouchListener() { 67 | float startX; 68 | float startY; 69 | float tempX; 70 | float tempY; 71 | 72 | @Override 73 | public boolean onTouch(View v, MotionEvent event) { 74 | switch (event.getAction()) { 75 | case MotionEvent.ACTION_DOWN: 76 | startX = event.getRawX(); 77 | startY = event.getRawY(); 78 | 79 | tempX = event.getRawX(); 80 | tempY = event.getRawY(); 81 | break; 82 | case MotionEvent.ACTION_MOVE: 83 | float dx = event.getRawX() - startX; 84 | float dy = event.getRawY() - startY; 85 | //计算偏移量,刷新视图 86 | floatBallParams.x += dx; 87 | floatBallParams.y += dy; 88 | windowManager.updateViewLayout(floatBall, floatBallParams); 89 | startX = event.getRawX(); 90 | startY = event.getRawY(); 91 | break; 92 | case MotionEvent.ACTION_UP: 93 | //判断松手时View的横坐标是靠近屏幕哪一侧,将View移动到依靠屏幕 94 | float endX = event.getRawX(); 95 | float endY = event.getRawY(); 96 | if (endX < getScreenWidth() / 2) { 97 | endX = 0; 98 | } else { 99 | endX = getScreenWidth() - floatBall.width; 100 | } 101 | floatBallParams.x = (int) endX; 102 | windowManager.updateViewLayout(floatBall, floatBallParams); 103 | //如果初始落点与松手落点的坐标差值超过6个像素,则拦截该点击事件 104 | //否则继续传递,将事件交给OnClickListener函数处理 105 | if (Math.abs(endX - tempX) > 6 && Math.abs(endY - tempY) > 6) { 106 | return true; 107 | } 108 | break; 109 | } 110 | return false; 111 | } 112 | 113 | }); 114 | } 115 | 116 | public int getScreenWidth() { 117 | return windowManager.getDefaultDisplay().getWidth(); 118 | } 119 | 120 | } 121 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 13 | 14 | 20 | 21 | 27 | 28 | 29 | 30 | 31 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #3F51B5 4 | #303F9F 5 | #FF4081 6 | 7 | #90fa1098 8 | 9 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | FloatingBall 3 | 开启悬浮球功能 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /app/src/main/res/xml/accessibilityservice.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | -------------------------------------------------------------------------------- /app/src/test/java/com/ice/floatingball/ExampleUnitTest.java: -------------------------------------------------------------------------------- 1 | package com.ice.floatingball; 2 | 3 | import org.junit.Test; 4 | 5 | import static org.junit.Assert.*; 6 | 7 | /** 8 | * Example local unit test, which will execute on the development machine (host). 9 | * 10 | * @see Testing documentation 11 | */ 12 | public class ExampleUnitTest { 13 | @Test 14 | public void addition_isCorrect() throws Exception { 15 | assertEquals(4, 2 + 2); 16 | } 17 | } -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 | 3 | buildscript { 4 | repositories { 5 | jcenter() 6 | } 7 | dependencies { 8 | classpath 'com.android.tools.build:gradle:2.2.2' 9 | 10 | // NOTE: Do not place your application dependencies here; they belong 11 | // in the individual module build.gradle files 12 | } 13 | } 14 | 15 | allprojects { 16 | repositories { 17 | jcenter() 18 | } 19 | } 20 | 21 | task clean(type: Delete) { 22 | delete rootProject.buildDir 23 | } 24 | -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | org.gradle.jvmargs=-Xmx1536m 13 | 14 | # When configured, Gradle will run in incubating parallel mode. 15 | # This option should only be used with decoupled projects. More details, visit 16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 17 | # org.gradle.parallel=true 18 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/CHNicelee/FloatingBall/42815c722aead4aba30af8b8379e913f540db382/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Mon Dec 28 10:00:20 PST 2015 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.14.1-all.zip 7 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | ############################################################################## 4 | ## 5 | ## Gradle start up script for UN*X 6 | ## 7 | ############################################################################## 8 | 9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 10 | DEFAULT_JVM_OPTS="" 11 | 12 | APP_NAME="Gradle" 13 | APP_BASE_NAME=`basename "$0"` 14 | 15 | # Use the maximum available, or set MAX_FD != -1 to use that value. 16 | MAX_FD="maximum" 17 | 18 | warn ( ) { 19 | echo "$*" 20 | } 21 | 22 | die ( ) { 23 | echo 24 | echo "$*" 25 | echo 26 | exit 1 27 | } 28 | 29 | # OS specific support (must be 'true' or 'false'). 30 | cygwin=false 31 | msys=false 32 | darwin=false 33 | case "`uname`" in 34 | CYGWIN* ) 35 | cygwin=true 36 | ;; 37 | Darwin* ) 38 | darwin=true 39 | ;; 40 | MINGW* ) 41 | msys=true 42 | ;; 43 | esac 44 | 45 | # Attempt to set APP_HOME 46 | # Resolve links: $0 may be a link 47 | PRG="$0" 48 | # Need this for relative symlinks. 49 | while [ -h "$PRG" ] ; do 50 | ls=`ls -ld "$PRG"` 51 | link=`expr "$ls" : '.*-> \(.*\)$'` 52 | if expr "$link" : '/.*' > /dev/null; then 53 | PRG="$link" 54 | else 55 | PRG=`dirname "$PRG"`"/$link" 56 | fi 57 | done 58 | SAVED="`pwd`" 59 | cd "`dirname \"$PRG\"`/" >/dev/null 60 | APP_HOME="`pwd -P`" 61 | cd "$SAVED" >/dev/null 62 | 63 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 64 | 65 | # Determine the Java command to use to start the JVM. 66 | if [ -n "$JAVA_HOME" ] ; then 67 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 68 | # IBM's JDK on AIX uses strange locations for the executables 69 | JAVACMD="$JAVA_HOME/jre/sh/java" 70 | else 71 | JAVACMD="$JAVA_HOME/bin/java" 72 | fi 73 | if [ ! -x "$JAVACMD" ] ; then 74 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 75 | 76 | Please set the JAVA_HOME variable in your environment to match the 77 | location of your Java installation." 78 | fi 79 | else 80 | JAVACMD="java" 81 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 82 | 83 | Please set the JAVA_HOME variable in your environment to match the 84 | location of your Java installation." 85 | fi 86 | 87 | # Increase the maximum file descriptors if we can. 88 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then 89 | MAX_FD_LIMIT=`ulimit -H -n` 90 | if [ $? -eq 0 ] ; then 91 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 92 | MAX_FD="$MAX_FD_LIMIT" 93 | fi 94 | ulimit -n $MAX_FD 95 | if [ $? -ne 0 ] ; then 96 | warn "Could not set maximum file descriptor limit: $MAX_FD" 97 | fi 98 | else 99 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 100 | fi 101 | fi 102 | 103 | # For Darwin, add options to specify how the application appears in the dock 104 | if $darwin; then 105 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 106 | fi 107 | 108 | # For Cygwin, switch paths to Windows format before running java 109 | if $cygwin ; then 110 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 111 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 112 | JAVACMD=`cygpath --unix "$JAVACMD"` 113 | 114 | # We build the pattern for arguments to be converted via cygpath 115 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 116 | SEP="" 117 | for dir in $ROOTDIRSRAW ; do 118 | ROOTDIRS="$ROOTDIRS$SEP$dir" 119 | SEP="|" 120 | done 121 | OURCYGPATTERN="(^($ROOTDIRS))" 122 | # Add a user-defined pattern to the cygpath arguments 123 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 124 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 125 | fi 126 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 127 | i=0 128 | for arg in "$@" ; do 129 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 130 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 131 | 132 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 133 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 134 | else 135 | eval `echo args$i`="\"$arg\"" 136 | fi 137 | i=$((i+1)) 138 | done 139 | case $i in 140 | (0) set -- ;; 141 | (1) set -- "$args0" ;; 142 | (2) set -- "$args0" "$args1" ;; 143 | (3) set -- "$args0" "$args1" "$args2" ;; 144 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;; 145 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 146 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 147 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 148 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 149 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 150 | esac 151 | fi 152 | 153 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules 154 | function splitJvmOpts() { 155 | JVM_OPTS=("$@") 156 | } 157 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS 158 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" 159 | 160 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 161 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @if "%DEBUG%" == "" @echo off 2 | @rem ########################################################################## 3 | @rem 4 | @rem Gradle startup script for Windows 5 | @rem 6 | @rem ########################################################################## 7 | 8 | @rem Set local scope for the variables with windows NT shell 9 | if "%OS%"=="Windows_NT" setlocal 10 | 11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 12 | set DEFAULT_JVM_OPTS= 13 | 14 | set DIRNAME=%~dp0 15 | if "%DIRNAME%" == "" set DIRNAME=. 16 | set APP_BASE_NAME=%~n0 17 | set APP_HOME=%DIRNAME% 18 | 19 | @rem Find java.exe 20 | if defined JAVA_HOME goto findJavaFromJavaHome 21 | 22 | set JAVA_EXE=java.exe 23 | %JAVA_EXE% -version >NUL 2>&1 24 | if "%ERRORLEVEL%" == "0" goto init 25 | 26 | echo. 27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 28 | echo. 29 | echo Please set the JAVA_HOME variable in your environment to match the 30 | echo location of your Java installation. 31 | 32 | goto fail 33 | 34 | :findJavaFromJavaHome 35 | set JAVA_HOME=%JAVA_HOME:"=% 36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 37 | 38 | if exist "%JAVA_EXE%" goto init 39 | 40 | echo. 41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 42 | echo. 43 | echo Please set the JAVA_HOME variable in your environment to match the 44 | echo location of your Java installation. 45 | 46 | goto fail 47 | 48 | :init 49 | @rem Get command-line arguments, handling Windowz variants 50 | 51 | if not "%OS%" == "Windows_NT" goto win9xME_args 52 | if "%@eval[2+2]" == "4" goto 4NT_args 53 | 54 | :win9xME_args 55 | @rem Slurp the command line arguments. 56 | set CMD_LINE_ARGS= 57 | set _SKIP=2 58 | 59 | :win9xME_args_slurp 60 | if "x%~1" == "x" goto execute 61 | 62 | set CMD_LINE_ARGS=%* 63 | goto execute 64 | 65 | :4NT_args 66 | @rem Get arguments from the 4NT Shell from JP Software 67 | set CMD_LINE_ARGS=%$ 68 | 69 | :execute 70 | @rem Setup the command line 71 | 72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 73 | 74 | @rem Execute Gradle 75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% 76 | 77 | :end 78 | @rem End local scope for the variables with windows NT shell 79 | if "%ERRORLEVEL%"=="0" goto mainEnd 80 | 81 | :fail 82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 83 | rem the _cmd.exe /c_ return code! 84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 85 | exit /b 1 86 | 87 | :mainEnd 88 | if "%OS%"=="Windows_NT" endlocal 89 | 90 | :omega 91 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app' 2 | --------------------------------------------------------------------------------