android编程之悬浮窗体

时间:2021-08-23 04:56:08

用过手机360和QQ手机管家等一些软件的朋友,会发现,在这些应用中,会出现一个悬浮窗体,例如QQ手机管家中打电话的场景:
android编程之悬浮窗体

这种窗体除了会显示外,还可以移动它的位置,并且一直显示。除了关闭当前程序外,窗口不会主动消失。其实,它的使用原理也很简单,就是借用了WindowManager这个管理类来实现的。
注意:要在AndroidManifest.xml中添加使用权限:

  1. <uses-permission
  2. android:name="android.permission.SYSTEM_ALERT_WINDOW" />

这里,我采用代码布局的方式,模仿了一下QQ这个界面效果:

  1. import android.content.Context;
  2. import android.widget.ImageView;
  3. import android.widget.LinearLayout;
  4. import android.widget.TextView;
  5. public class DesktopLayout extends LinearLayout {
  6. public DesktopLayout(Context context) {
  7. super(context);
  8. setOrientation(LinearLayout.HORIZONTAL);
  9. LayoutParams mLayoutParams = new LayoutParams(
  10. LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
  11. setLayoutParams(mLayoutParams);
  12. // 显示的ICON
  13. ImageView mImageView = new ImageView(context);
  14. mImageView.setImageResource(R.drawable.icon);
  15. addView(mImageView, mLayoutParams);
  16. // 显示的文字
  17. TextView mTextView = new TextView(context);
  18. mTextView.setText("Hello");
  19. mTextView.setTextSize(30);
  20. addView(mTextView, mLayoutParams);
  21. }
  22. }

接下来,在activity中让它显示出来。首先要设置一下WindowManager.LayoutParams:

  1. // 取得系统窗体
  2. mWindowManager = (WindowManager) getApplicationContext()
  3. .getSystemService("window");
  4. // 窗体的布局样式
  5. mLayoutParams = new WindowManager.LayoutParams();
  6. // 设置窗体显示类型——TYPE_SYSTEM_ALERT(系统提示)
  7. mLayoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
  8. // 设置窗体焦点及触摸:
  9. // FLAG_NOT_FOCUSABLE(不能获得按键输入焦点)
  10. mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
  11. // 设置显示的模式
  12. mLayoutParams.format = PixelFormat.RGBA_8888;
  13. // 设置对齐的方法
  14. mLayoutParams.gravity = Gravity.TOP | Gravity.LEFT;
  15. // 设置窗体宽度和高度
  16. mLayoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
  17. mLayoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
  18. // 设置窗体显示的位置,否则在屏幕中心显示
  19. mLayoutParams.x = 50;
  20. mLayoutParams.y = 50;

显示窗体与关闭窗体的方法:

  1. mWindowManager.addView(mDesktopLayout, mLayoutParams);
  2. mWindowManager.removeView(mDesktopLayout);

以下是activity的原代码,这里设计了一个双击关闭窗体的效果:

  1. import android.app.Activity;
  2. import android.graphics.PixelFormat;
  3. import android.os.Bundle;
  4. import android.view.Gravity;
  5. import android.view.MotionEvent;
  6. import android.view.View;
  7. import android.view.View.OnClickListener;
  8. import android.view.View.OnTouchListener;
  9. import android.view.WindowManager;
  10. import android.widget.Button;
  11. public class DeskTip extends Activity {
  12. private WindowManager mWindowManager;
  13. private WindowManager.LayoutParams mLayoutParams;
  14. private DesktopLayout mDesktopLayout;
  15. private long starttime;
  16. /**
  17. * 创建悬浮窗体
  18. */
  19. private void createDesktopLayout() {
  20. mDesktopLayout = new DesktopLayout(this);
  21. mDesktopLayout.setOnTouchListener(new OnTouchListener() {
  22. public boolean onTouch(View v, MotionEvent event) {
  23. onActionMove(event);
  24. return true;
  25. }
  26. });
  27. }
  28. /**
  29. * 设置WindowManager
  30. */
  31. private void createWindowManager() {
  32. // 取得系统窗体
  33. mWindowManager = (WindowManager) getApplicationContext()
  34. .getSystemService("window");
  35. // 窗体的布局样式
  36. mLayoutParams = new WindowManager.LayoutParams();
  37. // 设置窗体显示类型——TYPE_SYSTEM_ALERT(系统提示)
  38. mLayoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
  39. // 设置窗体焦点及触摸:
  40. // FLAG_NOT_FOCUSABLE(不能获得按键输入焦点)
  41. mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
  42. // 设置显示的模式
  43. mLayoutParams.format = PixelFormat.RGBA_8888;
  44. // 设置对齐的方法
  45. mLayoutParams.gravity = Gravity.TOP | Gravity.LEFT;
  46. // 设置窗体宽度和高度
  47. mLayoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
  48. mLayoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
  49. // 设置窗体显示的位置,否则在屏幕中心显示
  50. mLayoutParams.x = 50;
  51. mLayoutParams.y = 50;
  52. }
  53. private void onActionMove(MotionEvent event) {
  54. if (event.getAction() == MotionEvent.ACTION_DOWN) {
  55. long end = System.currentTimeMillis() - starttime;
  56. // 双击的间隔在 200ms 到 500ms 之间
  57. if (end > 200 && end < 500) {
  58. closeDesk();
  59. return;
  60. }
  61. starttime = System.currentTimeMillis();
  62. }
  63. mLayoutParams.x = (int) (event.getRawX() - (mDesktopLayout.getWidth()));
  64. mLayoutParams.y = (int) (event.getRawY() - (mDesktopLayout.getHeight()));
  65. mWindowManager.updateViewLayout(mDesktopLayout, mLayoutParams);
  66. }
  67. /**
  68. * 显示DesktopLayout
  69. */
  70. private void showDesk() {
  71. mWindowManager.addView(mDesktopLayout, mLayoutParams);
  72. finish();
  73. }
  74. /**
  75. * 关闭DesktopLayout
  76. */
  77. private void closeDesk() {
  78. mWindowManager.removeView(mDesktopLayout);
  79. finish();
  80. }
  81. public void onCreate(Bundle savedInstanceState) {
  82. super.onCreate(savedInstanceState);
  83. setContentView(R.layout.main);
  84. createWindowManager();
  85. createDesktopLayout();
  86. Button btn = (Button) findViewById(R.id.btn);
  87. btn.setOnClickListener(new OnClickListener() {
  88. public void onClick(View v) {
  89. showDesk();
  90. }
  91. });
  92. }
  93. }

显示的效果:
android编程之悬浮窗体

悬浮窗体例子

http://blog.csdn.net/xyz_fly/article/details/7546096