首页 > 代码库 > 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学习总结