首页 > 代码库 > Android PopupWindow学习总结

Android PopupWindow学习总结

Android中有一种可以放置在任何位置的一种弹出窗口PopupWindow,比Dialog更加灵活方便,但是作者在使用过程中遇到过几个问题,今天来总结一下。

首先先来说一下怎么来放置一个PopupWindow,PopupWindow总共提供了三个方法:

showAsDropDown(View anchor):相对某个控件的位置(正左下方),无偏移

showAsDropDown(View anchor, int xoff, int yoff):相对某个控件的位置,有偏移

showAtLocation(View parent, int gravity, int x, int y):相对于父控件的位置(例如正中央Gravity.CENTER,下方Gravity.BOTTOM等),可以设置偏移或无偏移

这几个方法的区别详细大家从字面意思上也能理解大概了,这边就不细说了,不清楚的可以自己测试一下。


下面主要讲讲作者本人在使用过程中遇到的一些问题

1、PopupWindow创建方法

第一种创建方法:

LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
// 引入窗口配置文件
View view = inflater.inflate(R.layout.pop_view, null);
// 创建PopupWindow对象
/***创建模式一****/
pop = new PopupWindow(view, LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
//设置点击窗口外边窗口消失
pop.setOutsideTouchable(true);
// 设置此参数获得焦点,否则无法点击
pop.setFocusable(true);

第二种创建方法:

LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
// 引入窗口配置文件
View view = inflater.inflate(R.layout.pop_view, null);
// 创建PopupWindow对象
/***创建模式二****/
pop = new PopupWindow(this);
pop.setContentView(view);
pop.setWidth(LayoutParams.WRAP_CONTENT);
pop.setHeight(LayoutParams.WRAP_CONTENT);
//设置点击窗口外边窗口消失
pop.setOutsideTouchable(true);
// 设置此参数获得焦点,否则无法点击
pop.setFocusable(true);

这两种方式有什么区别呢,这个区别我也是在很偶然的情况下发现的,如果你代码中不加这句代码:

pop.setBackgroundDrawable(new BitmapDrawable());

那么这两种方式就有明显区别了:按照第一种方法创建,则出现点击其他区域PopupWindow不会消失问题,按照第二种方式创建,则PopupWindow会出现灰色边框

那 pop.setBackgroundDrawable(new BitmapDrawable());那句代码又是其什么作用呢,就是为PopupWindow设置一个背景,因为按照第二种方式创建PopupWindow,PopupWindow会默认加上一个灰色的背景框,如果想去掉这个背景框,就可以使用这句代码,但是,到了这边就出现第二个问题:

2、setBackgroundDrawable()问题

可能有人觉得既然是去掉背景框,就可以用setBackgroundDrawable(null)来做,这也是作者一开始的做法,但是按照第二种方式创建PopupWindow,如果采用setBackgroundDrawable(null)这种写法,就会出现点击其他区域PopupWindow不会消失问题,Oh,MyGod,这是为什么呢,这也是作者一直疑惑的地方,国内国外转了一圈也没看到有人回答这个问题的原因,只说了解决方法,就是使用setBackgroundDrawable(new BitmapDrawable())这样的写法就可以避免这种问题,OK,没有办法,我也只能知其然不知其所以然了

3、获取PopupWindow宽高问题

作者在创建PopupWindow的时候没有给PopupWindow设置固定的宽高,于是在获取PopupWindow宽高的时候问题就出现了,我使用LayoutParams.WRAP_CONTENT来设置PopupWindow的宽高,本意是想让PopupWindow自适应大小,但是当我使用下面两句代码获取PopupWindow的宽高的时候,得到的值竟然是-2,why?

System.out.println("pop宽度="+pop.getWidth());
System.out.println("pop高度="+pop.getHeight());

后来查了一圈才发现原来返回的就是LayoutParams.WRAP_CONTENT的值-2,而没有返回PopupWindow展现时的宽度和高度值,看来这个PopupWindow是你设置时给他什么值,他就返回什么值,虽然你设置了LayoutParams.WRAP_CONTENT,但是你获取的时候PopupWindow并不是返回他展现的最终值,而仅仅是返回你给他的值,因此我就想,我先将所要展现的布局的大小算出来,再设置给PopupWindow,那得到的就是正确的值了,不多说,上代码:

//如果想获取PopupWindow对象实际布局大小的值,则可以采用如下代码:
view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
int viewWidth = view.getMeasuredWidth();
int viewHeight = view.getMeasuredHeight();
pop.setWidth(viewWidth);
pop.setHeight(viewHeight);

问题讲的差不多了,这些看似是小问题,但是当时出现的时候还是费了作者不少时间来排查和解决,接下来就给出我的全部测试代码:

activity_main.xml

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

    <Button
        android:id="@+id/btn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="dianji" />

</LinearLayout>

MainActivity

package com.example.popupwindowdemo;

import android.app.Activity;
import android.content.Context;
import android.graphics.drawable.BitmapDrawable;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup.LayoutParams;
import android.widget.Button;
import android.widget.PopupWindow;

public class MainActivity extends Activity {
    /** Called when the activity is first created. */
    
    private PopupWindow pop;
    
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
//        LayoutInflater inflater = LayoutInflater.from(this);
        LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        // 引入窗口配置文件
        View view = inflater.inflate(R.layout.pop_view, null);
        // 创建PopupWindow对象
        
        /***创建模式一****/
//        pop = new PopupWindow(view, LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        
        /***创建模式二****/
        pop = new PopupWindow(this);
        pop.setContentView(view);
//      //如果PopupWindow对象设置的宽度或者高度都为LayoutParams.WRAP_CONTENT,则getWidth()返回的就是LayoutParams.WRAP_CONTENT的值,即-2
//        pop.setWidth(LayoutParams.WRAP_CONTENT);
//        pop.setHeight(LayoutParams.WRAP_CONTENT);
//        
      //如果想获取PopupWindow对象实际布局大小的值,则可以采用如下代码:
        view.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
        int viewWidth = view.getMeasuredWidth();
        int viewHeight = view.getMeasuredHeight();
        
        pop.setWidth(viewWidth);
        pop.setHeight(viewHeight);
        
        System.out.println("pop宽度="+pop.getWidth());
        System.out.println("pop高度="+pop.getHeight());
        
        // 此参数必须设置,如果不设置,则在模式一下,则出现点击其他区域PopupWindow不会消失问题,在模式二下,PopupWindow会出现灰色边框
        pop.setBackgroundDrawable(new BitmapDrawable());
//        pop.setBackgroundDrawable(null);
        //设置点击窗口外边窗口消失
        pop.setOutsideTouchable(true);
        // 设置此参数获得焦点,否则无法点击
        pop.setFocusable(true);
        
        Button btn = (Button) findViewById(R.id.btn);
        btn.setOnClickListener(new OnClickListener() {
            
            @Override
            public void onClick(View v) {
                if(pop.isShowing()) {
                    // 隐藏窗口,如果设置了点击窗口外小时即不需要此方式隐藏
                    pop.dismiss();
                } else {
                    // 显示窗口
                    pop.showAsDropDown(v);
                }
                
            }
        });
    }
}




Android PopupWindow学习总结