Android使用WindowManager构造悬浮view

所属分类: 软件编程 / Android 阅读数: 878
收藏 0 赞 0 分享

一般在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有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

老生常谈Android HapticFeedback(震动反馈)

下面小编就为大家带来一篇老生常谈Android HapticFeedback(震动反馈)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

详谈OnTouchListener与OnGestureListener的区别

下面小编就为大家带来一篇详谈OnTouchListener与OnGestureListener的区别。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

Android仿知乎悬浮功能按钮FloatingActionButton效果

前段时间在看属性动画,恰巧这个按钮的效果可以用属性动画实现,下面通过本文给大家分享adroid仿知乎悬浮功能按钮FloatingActionButton效果,需要的朋友参考下吧
收藏 0 赞 0 分享

解决Android V7后自定义Toolbar、ActionBar左侧有空白问题

这篇文章主要介绍的Android V7后自定义Toolbar、ActionBar左侧有空白问题的解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android常见控件使用详解

这篇文章主要为大家详细介绍了Android常见控件的使用方法,包括ProgressBar进度条控件、AlertDialog对话框控件等,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android实现简洁的APP更新dialog数字进度条

这篇文章主要为大家详细介绍了Android实现简洁的APP更新dialog数字进度条,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android 判断当前语言环境是否是中文环境

本文主要介绍了Android 判断当前语言环境是否是中文环境的方法。具有很好的参考价值。下面跟着小编一起来看下吧
收藏 0 赞 0 分享

详谈Android中Matrix的set、pre、post的区别

下面小编就为大家带来一篇详谈Android中Matrix的set、pre、post的区别。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

Android实现登录界面记住密码的存储

这篇文章主要为大家详细介绍了Android SharedPreferrences实现登录界面记住密码的存储,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android 使用SharedPreferrences储存密码登录界面记住密码功能

Android存储方式有很多种,在这里所用的存储方式是SharedPreferrences, 其采用了Map数据结构来存储数据,以键值的方式存储,可以简单的读取与写入,下面通过实例代码给大家讲解下,需要的朋友参考下吧
收藏 0 赞 0 分享
查看更多