Android PopupWindow使用方法小结

前几天要用到PopupWindow,一时竟想不起来怎么用,赶紧上网查了查,自己写了个demo,并在此记录一下PopupWindow的用法。

使用场景

PopupWindow,顾名思义,就是弹窗,在很多场景下都可以见到它。例如ActionBar/Toolbar的选项弹窗,一组选项的容器,或者列表等集合的窗口等等。

基本用法

使用PopupWindow很简单,可以总结为三个步骤:

  • 创建PopupWindow对象实例;
  • 设置背景、注册事件监听器和添加动画;
  • 显示PopupWindow。

其中,第二步是可选的(不过基本上都要进行第二步的设置)。下面是一个简单的例子:

// 用于PopupWindow的View
 View contentView=LayoutInflater.from(context).inflate(layoutRes, null, false);
 // 创建PopupWindow对象,其中:
 // 第一个参数是用于PopupWindow中的View,第二个参数是PopupWindow的宽度,
 // 第三个参数是PopupWindow的高度,第四个参数指定PopupWindow能否获得焦点
 PopupWindow window=new PopupWindow(contentView, 100, 100, true);
 // 设置PopupWindow的背景
 window.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
 // 设置PopupWindow是否能响应外部点击事件
 window.setOutsideTouchable(true);
 // 设置PopupWindow是否能响应点击事件
 window.setTouchable(true);
 // 显示PopupWindow,其中:
 // 第一个参数是PopupWindow的锚点,第二和第三个参数分别是PopupWindow相对锚点的x、y偏移
 window.showAsDropDown(anchor, xoff, yoff);
 // 或者也可以调用此方法显示PopupWindow,其中:
 // 第一个参数是PopupWindow的父View,第二个参数是PopupWindow相对父View的位置,
 // 第三和第四个参数分别是PopupWindow相对父View的x、y偏移
 // window.showAtLocation(parent, gravity, x, y);

每个方法的作用都写在注解里了,相信大家都能看懂。不过这里要注意这两行:

window.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
window.setOutsideTouchable(true);

只有同时设置PopupWindow的背景和可以响应外部点击事件,它才能“真正”响应外部点击事件。也就是说,当你点击PopupWindow的外部或者按下“Back”键时,PopupWindow才会消失。

使用showAsDropDown方法显示PopupWindow

通常情况下,调用showAsDropDown方法后PopupWindow将会在锚点的左下方显示(drop down)。但是,有时想让PopupWindow在锚点的上方显示,或者在锚点的中间位置显示,此时就需要用到showAsDropDown方法的xoff和yoff参数了。

这里我们的目的不仅包括上面提到的两种情况(锚点上方或锚点中部),而是囊括了水平和垂直方向各5种显示方式:

水平方向:

ALIGN_LEFT:在锚点内部的左边;
ALIGN_RIGHT:在锚点内部的右边;
CENTER_HORI:在锚点水平中部;
TO_RIGHT:在锚点外部的右边;
TO_LEFT:在锚点外部的左边。

垂直方向:
ALIGN_ABOVE:在锚点内部的上方;
ALIGN_BOTTOM:在锚点内部的下方;
CENTER_VERT:在锚点垂直中部;
TO_BOTTOM:在锚点外部的下方;
TO_ABOVE:在锚点外部的上方。

下面来看张图:

我们先定义一个类对PopupWindow进行简单的封装:

public abstract class CommonPopupWindow {
 protected Context context;
 protected View contentView;
 protected PopupWindow mInstance;

 public CommonPopupWindow(Context c, int layoutRes, int w, int h) {
  context=c;
  contentView=LayoutInflater.from(c).inflate(layoutRes, null, false);
  initView();
  initEvent();
  mInstance=new PopupWindow(contentView, w, h, true);
  initWindow();
 }

 public View getContentView() { return contentView; }
 public PopupWindow getPopupWindow() { return mInstance; }

 protected abstract void initView();
 protected abstract void initEvent();

 protected void initWindow() {
  mInstance.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
  mInstance.setOutsideTouchable(true);
  mInstance.setTouchable(true);
 }

 public void showBashOfAnchor(View anchor, LayoutGravity layoutGravity, int xmerge, int ymerge) {
  int[] offset=layoutGravity.getOffset(anchor, mInstance);
  mInstance.showAsDropDown(anchor, offset[0]+xmerge, offset[1]+ymerge);
 }

 public void showAsDropDown(View anchor, int xoff, int yoff) {
  mInstance.showAsDropDown(anchor, xoff, yoff);
 }

 public void showAtLocation(View parent, int gravity, int x, int y) {
  mInstance.showAtLocation(parent, gravity, x, y);
 }
}

这里我们要实现的就是“showBashOfAnchor”方法,其中有一个“LayoutGravity”类型的参数,这就是控制PopupWindow相对锚点位置的对象。下面来定义“LayoutGravity”:

public static class LayoutGravity {
 private int layoutGravity;
 // waring, don't change the order of these constants!
 public static final int ALIGN_LEFT=0x1;
 public static final int ALIGN_ABOVE=0x2;
 public static final int ALIGN_RIGHT=0x4;
 public static final int ALIGN_BOTTOM=0x8;
 public static final int TO_LEFT=0x10;
 public static final int TO_ABOVE=0x20;
 public static final int TO_RIGHT=0x40;
 public static final int TO_BOTTOM=0x80;
 public static final int CENTER_HORI=0x100;
 public static final int CENTER_VERT=0x200;

 public LayoutGravity(int gravity) {
  layoutGravity=gravity;
 }

 public int getLayoutGravity() { return layoutGravity; }
 public void setLayoutGravity(int gravity) { layoutGravity=gravity; }

 public void setHoriGravity(int gravity) {
  layoutGravity&=(0x2+0x8+0x20+0x80+0x200);
  layoutGravity|=gravity;
 }
 public void setVertGravity(int gravity) {
  layoutGravity&=(0x1+0x4+0x10+0x40+0x100);
  layoutGravity|=gravity;
 }

 public boolean isParamFit(int param) {
  return (layoutGravity & param) > 0;
 }

 public int getHoriParam() {
  for(int i=0x1; i<=0x100; i=i<<2)
   if(isParamFit(i))
    return i;
  return ALIGN_LEFT;
 }

 public int getVertParam() {
  for(int i=0x2; i<=0x200; i=i<<2)
   if(isParamFit(i))
    return i;
  return TO_BOTTOM;
 }

 public int[] getOffset(View anchor, PopupWindow window) {
  int anchWidth=anchor.getWidth();
  int anchHeight=anchor.getHeight();

  int winWidth=window.getWidth();
  int winHeight=window.getHeight();
  View view=window.getContentView();
  if(winWidth<=0)
   winWidth=view.getWidth();
  if(winHeight<=0)
   winHeight=view.getHeight();

  int xoff=0;
  int yoff=0;

  switch (getHoriParam()) {
   case ALIGN_LEFT:
    xoff=0; break;
   case ALIGN_RIGHT:
    xoff=anchWidth-winWidth; break;
   case TO_LEFT:
    xoff=-winWidth; break;
   case TO_RIGHT:
    xoff=anchWidth; break;
   case CENTER_HORI:
    xoff=(anchWidth-winWidth)/2; break;
   default:break;
  }
  switch (getVertParam()) {
   case ALIGN_ABOVE:
    yoff=-anchHeight; break;
   case ALIGN_BOTTOM:
    yoff=-winHeight; break;
   case TO_ABOVE:
    yoff=-anchHeight-winHeight; break;
   case TO_BOTTOM:
    yoff=0; break;
   case CENTER_VERT:
    yoff=(-winHeight-anchHeight)/2; break;
   default:break;
  }
  return new int[]{ xoff, yoff };
 }
}

这里的主要方法就是“getOffset”,它会根据水平和垂直方向的gravity决定PopupWindow相对锚点的位置。

使用“LayoutGravity”时,可以通过“setHoriGravity”和“setVertGravity”方法设置水平和垂直方向的gravity,或者新建一个“LayoutGravity”对象。

下面是一个demo:

使用setAnimationStyle方法添加动画

上面我们提到了为PopupWindow设置背景和注册事件监听器,现在我们再来为PopupWindow添加动画。

这里的动画是指PopupWindow出现和消失时的动画。默认是直接弹出和消失,这样难免让用户有一种突兀的感觉;如果PopupWindow能够“滑入”屏幕和“滑出”屏幕(或者其他方式),用户体验会更好。

为PopupWindow添加动画可以调用`setAnimationStyle`方法,该方法只有一个参数,就是指定动画的样式,因此我们需要定义动画资源和样式资源。

下面是一个“滑入滑出”动画:

<!-- res/anim/translate_in.xml -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
 <translate
  android:fromXDelta="0"
  android:toXDelta="0"
  android:fromYDelta="100%"
  android:toYDelta="0"
  android:duration="200" >
 </translate>
</set> 
<!-- res/anim/translate_out.xml -->
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
 <translate
  android:fromXDelta="0"
  android:toXDelta="0"
  android:fromYDelta="0"
  android:toYDelta="100%"
  android:duration="200" >
 </translate>
</set>

然后定义“滑动”动画样式:

<!-- res/values/styles.xml -->
 <style name="animTranslate">
  <item name="android:windowEnterAnimation">@anim/translate_in</item>
  <item name="android:windowExitAnimation">@anim/translate_out</item>
</style>

现在我们就可以为PopupWindow添加“滑动”动画了:

window.setAnimationStyle(R.style.animTranslate);

我们来看下效果:

PS:这里由于动画的时间太短(200ms),另外转GIF的时候可能截取的频率有点低,导致滑动效果不是很明显,建议自己运行demo查看

现在PopupWindow的出现/消失已经不是那么突兀了。不过,当弹窗出现后,发现弹窗和背景不是很容易区分,如果此时弹窗的背景能“变暗”就好了。

没问题,我们可以在弹窗出现后让背景变暗,并在弹窗消失后让背景还原:

window.setOnDismissListener(new PopupWindow.OnDismissListener() {
  @Override
  public void onDismiss() {
   WindowManager.LayoutParams lp=getWindow().getAttributes();
   lp.alpha=1.0f;
   getWindow().clearFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
   getWindow().setAttributes(lp);
  }
 });

 window.showAtLocation(activityPopup, Gravity.BOTTOM, 0, 0);
 WindowManager.LayoutParams lp=getWindow().getAttributes();
 lp.alpha=0.3f;
 getWindow().addFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
 getWindow().setAttributes(lp);

现在再来看下效果:

现在PopupWindow就比较明显了。

另外,我们还实现了透明度、缩放和旋转三种动画样式,实现方式和上述大同小异,这里就不再赘述。

源代码

上述所有代码(包括未给出的)都已上传到GitHub:
https://github.com/jzyhywxz/PopupWindow

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android popupwindow简单使用方法介绍

    先看下效果 1.首页 package com.yskj.jh.demopopupwindow; import android.content.Context; import android.graphics.drawable.BitmapDrawable; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.LayoutInflater; import and

  • Android Popupwindow弹出窗口的简单使用方法

    本文实例为大家分享了Android Popupwindow弹出窗口的具体代码,供大家参考,具体内容如下 代码很简单,没有和别的控件连用.布局自己随意定义,我的这个是最基础的,就直接上代码啦! 在MainActivity里 import android.content.Context; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.LayoutInflat

  • Android中使用PopupWindow 仿微信点赞和评论弹出

    微信朋友圈的点赞和评论功能,有2个组成部分:左下角的"更多"按钮:点击该按钮后弹出的对话框: PopupWindow,弹出框使用PopupWindow实现,这是点赞和评论的载体,具体要涉及 PopupWindow 点击非窗口位置和再次点击消失以及显示位置的问题(根据相应更多按钮的位置确定 PopupWindow 的显示位置 package com.example.cmm.helloworld; import android.app.AlertDialog; import android

  • Android自定义弹出窗口PopupWindow使用技巧

    PopupWindow是Android上自定义弹出窗口,使用起来很方便. PopupWindow的构造函数为 复制代码 代码如下: public PopupWindow(View contentView, int width, int height, boolean focusable) contentView为要显示的view,width和height为宽和高,值为像素值,也可以是MATCHT_PARENT和WRAP_CONTENT. focusable为是否可以获得焦点,这是一个很重要的参数

  • Android中PopupWindow使用方法详解

    参考原文Android PopupWindow用法解析进行学习,通过实例及PopupWindow源码分析了PopupWindow的使用.文章最后的"补充Case: 弹窗不消失,但是事件向下传递"很赞. 不过,源码已经发生了变化,文章中提到的PopupViewContainer类,在目前的源码(Android6.0)中使用的是PopupBackgroundView和PopupDecorView共同完成的. 而在6.0版本的PopupWindow的preparePopup方法中,无论是否s

  • Android PopupWindow使用实例

    示例效果如下:   MainActivity.xml package sn.qdj.popupwindowdemo; import android.support.v7.app.ActionBarActivity; import android.os.Bundle; import android.view.Gravity; import android.view.View; import android.view.View.OnClickListener; import android.widg

  • PopupWindow使用方法详解

    学习了Android PopupWindow的使用技巧和[Android UI设计与开发]7.底部菜单栏(四)PopupWindow 实现显示仿腾讯新闻底部弹出菜单,然后自己进行了一下研究,写一个总结,方便以后学习. 效果图: 1.PopupWindow的布局: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.androi

  • Android组件popupwindow使用方法详解

    先看效果: 现在很多的应用效果都需要做的炫些,像UC,以及天天静听,效果很炫的,源码已经对外开放了,有兴趣的可以去研究下的 上源码 main.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fi

  • Android 使用PopupWindow实现弹出更多的菜单实例详解

    最近想要做一个弹出更多的菜单,而原生的弹出菜单却不是我们想要的效果,所以必然要自定义菜单咯.本人也是借鉴网上的资料进行封装的,感觉还蛮不错的. 原生的菜单如下图: 自定义之后的效果图: 是不是看到这里之后,对比可知,原生的效果不太理想,所以还是再自己定义吧! 1.PopupWindow可以说是一个浮动在Activity之上的容器,通常用来显示自定义的视图.弹出菜单的封装PopMenuMore /** * 对弹出菜单的封装. * http://blog.csdn.net/maosidiaoxian

  • android使用PopupWindow实现页面点击顶部弹出下拉菜单

    实现此功能没有太多的技术难点,主要通过PopupWindow方法,同时更进一步加深了PopupWindow的使用,实现点击弹出一个自定义的view,view里面可以自由设计,比较常用的可以放一个listview. demo中我只是一个点击展示,简单的使用了fade in out的动画效果,也没有精美的图片资源,看着也丑,不过这么短的时间,让你掌握一个很好用的技术,可以自己扩展,不很好么? 废话不说了,直接上代码: MainActivity.java public class MainActivi

随机推荐