Android使用WindowManager构造悬浮view

作者:summerpxy 时间:2022-08-03 00:43:13 

一般在android显示一个View都是通过Activity的setContentView设置的,但是还有一种方法,可以直接使用WindowManager在整个应用的最上层绘制我们需要显示的view,总体的效果类似于AlertDialog的弹出效果。

使用WindowManager构造这样的一个悬浮View也比较简单,直接通过windowmanager.addView()方法即可。


package com.gearmotion.app.windowmanagermotion;

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

public class MainActivity extends AppCompatActivity implements View.OnClickListener, View.OnTouchListener {

Button mShowBtn;
Button mHideBtn;
WindowManager mWm;
LayoutInflater mLayoutInflater;
View mWindowView;

@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 mShowBtn = (Button) this.findViewById(R.id.showbtn);
 mHideBtn = (Button) this.findViewById(R.id.hidebtn);
 mShowBtn.setOnClickListener(this);
 mHideBtn.setOnClickListener(this);
 init();
}

private void init() {
 mWm = (WindowManager) this.getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
 mLayoutInflater = LayoutInflater.from(this);
}

@Override
public void onClick(View v) {
 if (mShowBtn.hashCode() == v.hashCode()) { //显示WindowManager
  show();
 }
 if (mHideBtn.hashCode() == v.hashCode()) { //隐藏windowmanager
  hide();
 }
}

private void show() {
 mWindowView = mLayoutInflater.inflate(R.layout.item_layout, null);
 View popView = mWindowView.findViewById(R.id.root);
 //设置popView的触摸事件,以便点击空白区域的时候使悬浮view消失
 popView.setOnTouchListener(this);
 WindowManager.LayoutParams lp = new WindowManager.LayoutParams();
 //窗口类型同系统弹出框
 lp.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
 //响应输入法
 //lp.flags = WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM;
 //透明层
 lp.format = PixelFormat.TRANSPARENT;
 lp.width = WindowManager.LayoutParams.MATCH_PARENT;
 lp.height = WindowManager.LayoutParams.MATCH_PARENT;
 lp.gravity = Gravity.CENTER_VERTICAL;
 mWm.addView(mWindowView, lp);
}

private void hide() {
 if (mWindowView != null && mWindowView.getParent() != null) {
  mWm.removeView(mWindowView);
 }
}

@Override
public boolean onTouch(View v, MotionEvent event) {
 int x = (int) event.getX();
 int y = (int) event.getY();
 //获取主view的可视区域
 Rect globalRect = new Rect();
 //获取悬浮view的可视区域
 Rect tmpRect = new Rect();
 v.getGlobalVisibleRect(globalRect);
 View child = ((ViewGroup) v).getChildAt(0);
 child.getHitRect(tmpRect);
 if (!tmpRect.contains(x, y) && globalRect.contains(x, y)) {
  hide();
 }
 return true;
}
}

activity_main.xml:


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">

<Button
 android:id="@+id/showbtn"
 android:layout_width="match_parent"
 android:layout_height="50dp"
 android:gravity="center"
 android:text="show" />

<Button
 android:id="@+id/hidebtn"
 android:layout_width="match_parent"
 android:layout_height="50dp"
 android:gravity="center"
 android:text="hide" />
</LinearLayout>

item_layout.xml:


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:id="@+id/root"
android:orientation="vertical">

<TextView
 android:layout_width="match_parent"
 android:layout_height="50dp"
 android:text="I am WindowManager layout view"
 android:textSize="20sp"
 android:gravity="center"
 android:layout_gravity="center"
 android:background="#FFF8DC"
 android:textColor="#7AC5CD"/>
</LinearLayout>

实现效果如下:

Android使用WindowManager构造悬浮view

标签:Android,WindowManager,悬浮,view
0
投稿

猜你喜欢

  • Android测试方法总结

    2022-07-27 08:02:41
  • Springboot项目快速实现过滤器功能

    2023-08-11 18:30:02
  • Maven如何打入依赖中指定的部分jar包

    2023-09-22 02:50:33
  • Android自定义view实现车载可调整轨迹线

    2022-12-06 11:22:21
  • java Servlet 实现动态验证码图片示例

    2021-07-14 15:14:28
  • c#初学简单程序实例代码介绍

    2023-11-04 10:53:53
  • SpringBoot使用Maven插件进行项目打包的方法

    2022-12-21 21:07:34
  • Java中左移和右移问题图文详解

    2023-02-14 15:44:05
  • Android开发笔记之:返回键的复写onBackPressed()介绍

    2022-04-29 17:54:40
  • Android解析XML文件升级APK的方法

    2022-06-05 20:47:26
  • 浅谈Java的两种多线程实现方式

    2022-08-17 09:42:37
  • 命令提示符编译java的方法(必看篇)

    2022-01-28 08:34:38
  • JFinal实现伪静态的方法

    2023-07-17 12:11:37
  • C#中AS和IS关键字的用法

    2021-06-12 09:44:53
  • C#中datagridview使用tooltip控件显示单元格内容的方法

    2022-04-15 12:23:23
  • Java Cookie与Session实现会话跟踪详解

    2022-12-22 08:59:13
  • Android编程实现图片背景渐变切换与图层叠加效果

    2021-10-10 06:04:03
  • mybatis根据表逆向自动化生成代码的实现

    2023-03-12 15:39:02
  • SSh结合Easyui实现Datagrid的分页显示

    2022-01-14 21:46:49
  • 使用Java桥接模式打破继承束缚优雅实现多维度变化

    2023-08-23 09:00:34
  • asp之家 软件编程 m.aspxhome.com