用PopupWindow实现自定义溢出(overflow)菜单

关注微信号:javalearns   随时随地学Java

或扫一扫

随时随地学Java


Action Bar的Action放不下时,系统会将其收集在overflow中。

用hierarchyviewer查看系统自己生成的Overflow,发现它本身就是popupWindow。

所以我们也可以用popUpWindow来写自己的overflow实现更多功能,做出像微信一样的效果。

第一次写,废话有点多,还望多包涵。

效果(GIF演示在文章底部):

最右边的Action(那个三点菜单)是自己添加的Action,使用了Android开发包里的图标ic_action_overflow.png,可到官网下载。

首先在Item中添加Action,为了演示,添加了一个Submenu


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
     xmlns:tools= "http://schemas.android.com/tools"
     tools:context= "com.example.popupwindowoverflow.MainActivity" >
<item
         android:id= "@+id/action_new"
         android:orderInCategory= "1"
         android:title= "SubMenu"
         android:icon= "@drawable/ic_action_new"
         app:showAsAction= "always" >
     <menu>
         <item android:id= "@+id/submenu1"
             android:title= "Accept"
             android:titleCondensed= "Accept"
             android:icon= "@drawable/ic_action_accept" />
          <item android:id= "@+id/submenu2"
             android:title= "Cancel"
             android:titleCondensed= "Cancel"
             android:icon= "@drawable/ic_action_cancel" />
          <item android:id= "@+id/submenu3"
             android:title= "Unread"
             android:titleCondensed= "Unread"
             android:icon= "@drawable/ic_action_unread" />
     </menu>
</item>
<item
         android:id= "@+id/action_overflow"
         android:orderInCategory= "2"
         android:title= "PopupWindow"
         android:icon= "@drawable/ic_action_overflow"
         app:showAsAction= "always" />
</menu>

监听ID为action_overflow的Action,创建popupWindow弹出自己的overflow。

1
2
3
4
5
6
7
8
9
10
11
12
public boolean onOptionsItemSelected(MenuItem item) {
         // Handle action bar item clicks here. The action bar will
         // automatically handle clicks on the Home/Up button, so long
         // as you specify a parent activity in AndroidManifest.xml.   
         int id = item.getItemId();
         switch (id) {
         case R.id.action_overflow:
             popUpMyOverflow(); //弹出自定义overflow
             return true ;
         }
         return super .onOptionsItemSelected(item);
     }

下面介绍popUpMyOverflow()方法,就是通过它弹出了我们的overflow,自定义overflow的布局文件就是R.layout.action_overflow_popwindow,这里就不贴出来啦。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public void popUpMyOverflow() {
         /**
          * 定位PopupWindow,让它恰好显示在Action Bar的下方。 通过设置Gravity,确定PopupWindow的大致位置。
          * 首先获得状态栏的高度,再获取Action bar的高度,这两者相加设置y方向的offset样PopupWindow就显示在action
          * bar的下方了。 通过dp计算出px,就可以在不同密度屏幕统一X方向的offset.但是要注意不要让背景阴影大于所设置的offset,
          * 否则阴影的宽度为offset.
          */
         // 获取状态栏高度
         Rect frame = new Rect();
         getWindow().getDecorView().getWindowVisibleDisplayFrame(frame);
//        状态栏高度:frame.top
         int xOffset = frame.top+getActionBar().getHeight()-25; //减去阴影宽度,适配UI.
         int yOffset = Dp2Px( this , 5f); //设置x方向offset为5dp
         View parentView = getLayoutInflater().inflate(R.layout.activity_main,
                 null );
         View popView = getLayoutInflater().inflate(
                 R.layout.action_overflow_popwindow, null );
         PopupWindow popWind = new PopupWindow(popView,
                 LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, true ); //popView即popupWindow的布局,ture设置focusAble.
                                                     
         //必须设置BackgroundDrawable后setOutsideTouchable(true)才会有效。这里在XML中定义背景,所以这里设置为null;
         popWind.setBackgroundDrawable( new BitmapDrawable(getResources(),
                 (Bitmap) null ));
         popWind.setOutsideTouchable( true ); //点击外部关闭。
         popWind.setAnimationStyle(android.R.style.Animation_Dialog);    //设置一个动画。
         //设置Gravity,让它显示在右上角。
         popWind.showAtLocation(parentView, Gravity.RIGHT | Gravity.TOP,
                 yOffset, xOffset);
//点击事件。
Button btn1 = (Button) popView.findViewById(R.id.btn_popwindow_1); btn1.setOnClickListener(this); Button btn2 = (Button) popView.findViewById(R.id.btn_popwindow_2); btn2.setOnClickListener(this);
     }

在android中,为了适配不同屏幕密度和尺寸,android用了Dp单位,但是在Java代码中多是接受px单位的尺寸,所以这里要转换一下。

Dp转换Px的方法。

1
2
3
4
public int Dp2Px(Context context, float dp) {
     final float scale = context.getResources().getDisplayMetrics().density;
     return (int) (dp * scale + 0.5f);
}

好的,现在我们有了所有要显示自定义Overflow的东西了!运行你的app吧。

最终效果:

.................... 【.........阅读全文】

Java免费学习   Java自学网 http://www.javalearns.com

关注微信号:javalearns   随时随地学Java

或扫一扫

随时随地学Java


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当您需要更高级的弹窗样式和交互时,可以使用 PopupWindow 来创建自定义布局的弹窗。下面是使用 PopupWindow 创建自定义布局的步骤: 1. 创建自定义布局文件:首先,创建一个 XML 文件来定义您的自定义布局。例如,您可以创建一个名为 `custom_popup.xml` 的文件,并在其中定义您希望显示的布局。 2. 实例化 PopupWindow:在您的 Activity 或 Fragment 中,实例化 PopupWindow 对象。 ```java LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE); View customView = inflater.inflate(R.layout.custom_popup, null); PopupWindow popupWindow = new PopupWindow(customView, ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT); ``` 3. 设置 PopupWindow 属性:根据需要,设置 PopupWindow 的属性,例如背景、动画效果、焦点等。 ```java popupWindow.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT)); popupWindow.setFocusable(true); // 设置动画效果 popupWindow.setAnimationStyle(R.style.PopupAnimation); ``` 4. 设置布局中的控件和事件:通过 `customView` 获取布局中的控件,并设置相应的事件监听器。 ```java Button button = customView.findViewById(R.id.button); button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 处理点击事件 } }); ``` 5. 显示弹窗:使用 `showAtLocation()` 或 `showAsDropDown()` 方法显示弹窗。`showAtLocation()` 方法可以显示在指定的位置,而 `showAsDropDown()` 方法则可以显示在某个视图的下方。 ```java View anchorView = findViewById(R.id.anchor_view); // 锚点视图 popupWindow.showAsDropDown(anchorView); // 或者使用 showAtLocation() 方法 ``` 这样,您就可以使用 PopupWindow 创建自定义布局的弹窗。希望对您有所帮助!如果有任何问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值