Android 悬浮窗口

一.创建悬浮窗口步骤
    1.实现一个ViewGroup类,作为悬浮窗口的界面类,以便在里面重写onInterceptTouchEvent和onTouchEvent方法,实现移动界面的目的.
      在本例中实现了一个FloatLayer类,可以作为通用的类,使用时需要传入WindowManager对象以实现移动窗口.

// FloatLayer ~
package com.example.hellofloatingwnd; import static com.ahai.util.DebugMessage.d;
import android.content.Context;
import android.graphics.PixelFormat;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.WindowManager;
import android.widget.RelativeLayout; public class FloatLayer extends RelativeLayout { // flags: 须设置成 FLAG_NOT_FOCUSABLE, 否则悬浮窗口下面的窗口不能取得焦点, 无法响应触摸事件
// type: 值低的窗口在值高的下层,相同的 type值,后创建的窗口显示在先创建的窗口上面.
// 对应的type需要相应的权限,否则会报异常 BadTokenException.
// 对于权限 android.permission.SYSTEM_ALERT_WINDOW 可使用以下几个值:
// TYPE_PHONE, TYPE_SYSTEM_ALERT, TYPE_TOAST, TYPE_SYSTEM_OVERLAY
// 其中 TYPE_TOAST, TYPE_SYSTEM_OVERLAY 不能响应触摸事件
public static class FloatLayoutParams extends WindowManager.LayoutParams { public FloatLayoutParams() {
super(TYPE_PHONE, FLAG_NOT_FOCUSABLE, PixelFormat.RGBA_8888);
gravity = Gravity.LEFT | Gravity.TOP;
} public FloatLayoutParams(int type) {
super(type, FLAG_NOT_FOCUSABLE, PixelFormat.RGBA_8888);
gravity = Gravity.LEFT | Gravity.TOP;
} public FloatLayoutParams(int xpos, int ypos) {
super(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, xpos,
ypos, TYPE_PHONE, FLAG_NOT_FOCUSABLE, PixelFormat.RGBA_8888);
gravity = Gravity.LEFT | Gravity.TOP;
} public FloatLayoutParams(int xpos, int ypos, int type) {
super(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, xpos,
ypos, type, FLAG_NOT_FOCUSABLE, PixelFormat.RGBA_8888);
gravity = Gravity.LEFT | Gravity.TOP;
}
} private WindowManager mWindowManager;
private int mStatusBarHeight;
private int mMoveX;
private int mMoveY; public FloatLayer(Context context, AttributeSet attrs, int defStyleAttr,
int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
mStatusBarHeight = getStatusBarHeight();
} public FloatLayer(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mStatusBarHeight = getStatusBarHeight();
} public FloatLayer(Context context, AttributeSet attrs) {
super(context, attrs);
mStatusBarHeight = getStatusBarHeight();
} public FloatLayer(Context context) {
super(context);
mStatusBarHeight = getStatusBarHeight();
} public void setWindowManager(WindowManager windowManager) {
mWindowManager = windowManager;
} /** 取得系统状态栏的高度 */
private int getStatusBarHeight() {
int statusBarHeight = 0;
int resourceId = getResources().getIdentifier("status_bar_height",
"dimen", "android");
if (resourceId > 0) {
statusBarHeight = getResources().getDimensionPixelSize(resourceId);
}
// d("statusBarHeight=" + statusBarHeight);
return statusBarHeight;
} @Override
public boolean onInterceptTouchEvent(MotionEvent event) { final int action = event.getAction();
if (action == MotionEvent.ACTION_MOVE) {
if (handleMoveEvent(event))
return true;
} else if (action == MotionEvent.ACTION_DOWN) {
mMoveX = (int) event.getRawX();
mMoveY = (int) event.getRawY();
}
return super.onInterceptTouchEvent(event);
} @Override
public boolean onTouchEvent(MotionEvent event) { final int action = event.getAction();
if (action == MotionEvent.ACTION_MOVE) {
if (handleMoveEvent(event))
return true;
} else if (action == MotionEvent.ACTION_DOWN) {
mMoveX = (int) event.getRawX();
mMoveY = (int) event.getRawY();
}
return super.onTouchEvent(event);
} private boolean handleMoveEvent(MotionEvent event) { try {
if (mWindowManager != null) { // 通过以下消息可知getLayoutParams得到的对象即为 addView 传入的 LayoutParams 对象
// d("class:" + getLayoutParams().getClass());
final int x = (int) event.getRawX();
final int y = (int) event.getRawY(); int[] location = new int[2];
getLocationOnScreen(location);
FloatLayoutParams layoutParams = (FloatLayoutParams) getLayoutParams();
layoutParams.x = location[0] + (x - mMoveX);
layoutParams.y = location[1] + (y - mMoveY) - mStatusBarHeight;
mWindowManager.updateViewLayout(this, layoutParams); mMoveX = x;
mMoveY = y;
return true;
}
} catch (Exception e) {
d("", e);
} return false;
}
}

2.在res/layout中创建一个布局文件,实现界面布局.如float_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<com.example.hellofloatingwnd.FloatLayer xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content" > <ImageButton
android:id="@+id/mBtnHide"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:layout_centerHorizontal="true"
android:background="@drawable/hide_button_selector" /> <TextView
android:id="@+id/mTvHello"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/mBtnHide"
android:layout_centerHorizontal="true"
android:text="@string/hello"
android:textColor="#21d" /> </com.example.hellofloatingwnd.FloatLayer>

3.取得WindowManager对象.
        在 Activity 中可以通过以下方法取得, 其中前面的3个方法实际取得的是当前Activity的应用程序窗口对象,在Activity销毁等情况下,
    WindowManager对象也就不存在了,需要将悬浮窗口移除,否则会报错.
        WindowManager windowManager;
        windowManager = getWindow().getWindowManager();
        windowManager = getWindowManager();
        windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
        windowManager = (WindowManager) getApplication().getSystemService(WINDOW_SERVICE);
        在 Service 中, 以下两个方法均可, 在 onDestory 中将悬浮窗口移除即可.
        windowManager = (WindowManager) getApplication().getSystemService(WINDOW_SERVICE);
        windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
   
    4.创建View并显示. 通过以下两个行代码完成:
        mFloatView = (FloatLayer) inflater.inflate(R.layout.float_layout, null);
        windowManager.addView(mFloatView, layoutParams);
        其中layoutParams是实现悬浮窗口的关键,窗口的配置及移动都通过其指定.
        LayoutParams需使用WindowManager.LayoutParams对象或继承自该类重写.
        移动窗口通过下面的代码实现:
            mWindowManager.updateViewLayout(mFloatView, layoutParams);
        layoutParams最重要的两个参数是flags和type:
        flags: 须设置成 FLAG_NOT_FOCUSABLE, 否则悬浮窗口下面的窗口不能取得焦点,不能响应触摸事件.
        type: 值低的窗口在值高的下层,相同的 type值,后创建的窗口显示在先创建的窗口上面.
              对应的type需要相应的权限,否则会报异常 BadTokenException.
        对于权限 android.permission.SYSTEM_ALERT_WINDOW, type可使用以下几个值:
            TYPE_PHONE, TYPE_SYSTEM_ALERT, TYPE_TOAST, TYPE_SYSTEM_OVERLAY
        其中 TYPE_TOAST, TYPE_SYSTEM_OVERLAY 不能响应触摸事件

  在Service中实现

  

package com.example.hellofloatingwnd;

import android.app.Application;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.ImageButton;
import android.widget.Toast; public class FloatingService extends Service { private FloatLayer mFloatView;
private WindowManager mWindowManager;
private ImageButton mBtnHide; @Override
public void onCreate() {
super.onCreate(); Application app = getApplication(); mWindowManager = (WindowManager) app.getSystemService(WINDOW_SERVICE);
LayoutInflater inflater = LayoutInflater.from(app); mFloatView = (FloatLayer) inflater.inflate(R.layout.float_layout, null);
mFloatView.setWindowManager(mWindowManager); mBtnHide = (ImageButton) mFloatView.findViewById(R.id.mBtnHide);
mBtnHide.setOnClickListener(mClickListener); FloatLayer.FloatLayoutParams layoutParams;
layoutParams = new FloatLayer.FloatLayoutParams(10, 100);
mWindowManager.addView(mFloatView, layoutParams);
} @Override
public void onDestroy() {
super.onDestroy();
if (mFloatView != null) {
mWindowManager.removeView(mFloatView);
mFloatView = null;
}
} @Override
public IBinder onBind(Intent arg0) {
return null;
} private OnClickListener mClickListener = new OnClickListener() {
@Override
public void onClick(View view) {
if (view.getId() == R.id.mBtnHide) {
Toast.makeText(getApplicationContext(),
"on float button clicked.", Toast.LENGTH_SHORT).show();
FloatingService.this.stopSelf();
}
}
};
}

  在Activity中实现

package com.example.hellofloatingwnd;

import android.app.Activity;
import android.app.Application;
import android.content.Intent;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.TextView; public class MainActivity extends Activity { private Button mBtnStart;
private Button mBtnStop; private FloatLayer mFloatView; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); mBtnStart = (Button) findViewById(R.id.mBtnStart);
mBtnStop = (Button) findViewById(R.id.mBtnStop); mBtnStart.setOnClickListener(mClickListener);
mBtnStop.setOnClickListener(mClickListener); WindowManager windowManager;
// windowManager = getWindow().getWindowManager();
windowManager = getWindowManager();
// windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
// windowManager = (WindowManager)
// getApplication().getSystemService(WINDOW_SERVICE); Application application = getApplication();
LayoutInflater inflater = LayoutInflater.from(application); mFloatView = (FloatLayer) inflater.inflate(R.layout.float_layout, null);
mFloatView.setWindowManager(windowManager);
TextView textView = (TextView) mFloatView.findViewById(R.id.mTvHello);
textView.setText("This create by activity."); FloatLayer.FloatLayoutParams layoutParams;
layoutParams = new FloatLayer.FloatLayoutParams(50, 200);
windowManager.addView(mFloatView, layoutParams);
} @Override
protected void onDestroy() {
// TODO Auto-generated method stub
super.onDestroy();
if (mFloatView != null) {
WindowManager windowManager = getWindow().getWindowManager();
windowManager.removeView(mFloatView);
mFloatView = null;
}
} private OnClickListener mClickListener = new OnClickListener() { @Override
public void onClick(View view) {
if (view.getId() == R.id.mBtnStart) {
Intent intent = new Intent(MainActivity.this,
FloatingService.class);
startService(intent);
} else if (view.getId() == R.id.mBtnStop) {
Intent intent = new Intent(MainActivity.this,
FloatingService.class);
stopService(intent);
}
}
};
}

5.在AndroidManifest.xml文件中添加权限,在layoutParams中配置不同的type类型需要不同的权限.
        <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

Android 悬浮窗口的更多相关文章

  1. Android悬浮窗口的实现

    效果图:(悬浮框可拖动) 在项目开发中有一个需求:弹出悬浮窗后,响应悬浮窗的事件再弹出对话框,但是对话框怎么也不显示.也就是说在弹出悬浮框的同时,不能再弹出对话框,可能的原因: 1.悬浮框的焦点在最前 ...

  2. android悬浮窗口的一些说明

    1.xml文件里的权限申请 <uses-permission android:name="android.permission.SYSTEM_OVERLAY_WINDOW" ...

  3. android悬浮窗口

    悬浮窗原理 做过悬浮窗功能的人都知道, 要想显示悬浮窗, 要有一个服务运行在后台, 通过getSystemService(Context.WINDOW_SERVICE)拿到WindowManager, ...

  4. Android悬浮窗及其拖动事件

    主页面布局很简单,只有一个RelativelyLayout <?xml version="1.0" encoding="utf-8"?> <R ...

  5. Android中悬浮窗口

    调用WindowManager,并设置WindowManager.LayoutParams的相关属性,通过WindowManager的addView方法创建View,这样产生出来的View根据Wind ...

  6. Android中悬浮窗口的实现原理和示例代码

    用了我一个周末的时间,个中愤懑就不说了,就这个问题,我翻遍全球网络没有一篇像样的资料,现在将实现原理简单叙述如下: 调用WindowManager,并设置WindowManager.LayoutPar ...

  7. Android之悬浮窗口实现(WindowManager)

    工作中遇到一些项目需要把窗体显示在最上层,像来电弹窗显示电话号码等信息.拦截短信信息显示给用户或者游戏中实现声音的调节,我们想这些数据放在最上层,activity就满足不了我们的需求了,有些开发者使用 ...

  8. Android 类似360悬浮窗口实现源码

    当我们在手机上安装360安全卫士时,手机屏幕上时刻都会出现一个小浮动窗口,点击该浮动窗口可跳转到安全卫士的操作界面,而且该浮动窗口不受其他activity的覆盖影响仍然可见(多米音乐也有相关的和主界面 ...

  9. Android WindowManager和WindowManager.LayoutParams的使用以及实现悬浮窗口的方法

    1.理清概念 我们使用过Dialog和PopupWindow,还有Toast,它们都显示在Activity之上.那么我们首先需要理解的是android中是如何去绘制这些UI的呢?这里我只讲我所理解的, ...

随机推荐

  1. Python版本切换和Pip安装

    Python版本切换 现在常用的linux系统中都会默认携带python运行环境,在ubuntu 16.04 和centos 7.3中携带有Python 2.7 和Python3.5两个版本, 默认使 ...

  2. poj 3468 (区间修改 区间查询)

    A Simple Problem with Integers Time Limit: 5000MS   Memory Limit: 131072K Total Submissions:147133   ...

  3. ZOJ 2760 How Many Shortest Path(最短路径+最大流)

    Description Given a weighted directed graph, we define the shortest path as the path who has the sma ...

  4. Code obfuscatio (翻译!)

    Description Kostya likes Codeforces contests very much. However, he is very disappointed that his so ...

  5. 2019寒假训练营第三次作业part2 - 实验题

    热身题 服务器正在运转着,也不知道这个技术可不可用,万一服务器被弄崩了,那损失可不小. 所以, 决定在虚拟机上试验一下,不小心弄坏了也没关系.需要在的电脑上装上虚拟机和linux系统 安装虚拟机(可参 ...

  6. MFC加速键

    添加的函数:    } // 解释说明    MFC对话框不自动处理加速键,你必须自己编写代码来做这件事情.为了理解弄清楚这是为什么,让我们回首Windows开发的历程,在使用C和原始的Windows ...

  7. C#控件DropDownList下拉列表默认打开

    c#中的控件DropDownList要实现默打开确实不容易,之前也是想过页面上的点击之后就打开了,那直接模拟点击不就行了,试过后大失所望,根本没有效果. 于是网上找到了一个例子能实现IE浏览器下的打开 ...

  8. 软工网络15个人作业4-alpha阶段个人总结(201521123059 叶文柠)

    一.个人总结 (1) 类别 具体技能和面试问题 现在回答 毕业找工作时 语言 最拿手的计算机语言之一,代码量多少? 感觉自己没有最拿手的语言,而且拿手的在计算机网络这方面的,所以在软件变成这方面的代码 ...

  9. 硬盘引导扇区、多分区图、不通硬盘的LINUX逻辑分区数量

    主要启动记录区(Master Boot Record,MBR):可以安装开机管理程序的地方,有446byte 分割表(Paritition table):记录整块硬盘分割的状态,有64bytes 下面 ...

  10. PHP实现大文件分割上传与分片上传

    转载:http://www.zixuephp.com/phpstudy/phpshilie/20170829_43029.html 服务端为什么不能直接传大文件?跟php.ini里面的几个配置有关 u ...