Android PopupWindow使用方法小結

来源:http://www.cnblogs.com/jzyhywxz/archive/2017/06/17/7039503.html
-Advertisement-
Play Games

前幾天要用到PopupWindow,一時竟想不起來怎麼用,趕緊上網查了查,自己寫了個demo,併在此記錄一下PopupWindow的用法。 使用場景 PopupWindow,顧名思義,就是彈窗,在很多場景下都可以見到它。例如ActionBar/Toolbar的選項彈窗,一組選項的容器,或者列表等集合 ...


前幾天要用到PopupWindow,一時竟想不起來怎麼用,趕緊上網查了查,自己寫了個demo,併在此記錄一下PopupWindow的用法。

使用場景

PopupWindow,顧名思義,就是彈窗,在很多場景下都可以見到它。例如ActionBar/Toolbar的選項彈窗,一組選項的容器,或者列表等集合的視窗等等。

基本用法

使用PopupWindow很簡單,可以總結為三個步驟:

  1. 創建PopupWindow對象實例;
  2. 設置背景、註冊事件監聽器和添加動畫;
  3. 顯示PopupWindow。

其中,第二步是可選的(不過基本上都要進行第二步的設置)。下麵是一個簡單的例子:

 1     // 用於PopupWindow的View
 2     View contentView=LayoutInflater.from(context).inflate(layoutRes, null, false);
 3     // 創建PopupWindow對象,其中:
 4     // 第一個參數是用於PopupWindow中的View,第二個參數是PopupWindow的寬度,
 5     // 第三個參數是PopupWindow的高度,第四個參數指定PopupWindow能否獲得焦點
 6     PopupWindow window=new PopupWindow(contentView, 100, 100, true);
 7     // 設置PopupWindow的背景
 8     window.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
 9     // 設置PopupWindow是否能響應外部點擊事件
10     window.setOutsideTouchable(true);
11     // 設置PopupWindow是否能響應點擊事件
12     window.setTouchable(true);
13     // 顯示PopupWindow,其中:
14     // 第一個參數是PopupWindow的錨點,第二和第三個參數分別是PopupWindow相對錨點的x、y偏移
15     window.showAsDropDown(anchor, xoff, yoff);
16     // 或者也可以調用此方法顯示PopupWindow,其中:
17     // 第一個參數是PopupWindow的父View,第二個參數是PopupWindow相對父View的位置,
18     // 第三和第四個參數分別是PopupWindow相對父View的x、y偏移
19     // window.showAtLocation(parent, gravity, x, y);

每個方法的作用都寫在註解里了,相信大家都能看懂。不過這裡要註意這兩行:

1     window.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
2     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進行簡單的封裝:

 1 public abstract class CommonPopupWindow {
 2     protected Context context;
 3     protected View contentView;
 4     protected PopupWindow mInstance;
 5 
 6     public CommonPopupWindow(Context c, int layoutRes, int w, int h) {
 7         context=c;
 8         contentView=LayoutInflater.from(c).inflate(layoutRes, null, false);
 9         initView();
10         initEvent();
11         mInstance=new PopupWindow(contentView, w, h, true);
12         initWindow();
13     }
14 
15     public View getContentView() { return contentView; }
16     public PopupWindow getPopupWindow() { return mInstance; }
17 
18     protected abstract void initView();
19     protected abstract void initEvent();
20 
21     protected void initWindow() {
22         mInstance.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
23         mInstance.setOutsideTouchable(true);
24         mInstance.setTouchable(true);
25     }
26 
27     public void showBashOfAnchor(View anchor, LayoutGravity layoutGravity, int xmerge, int ymerge) {
28         int[] offset=layoutGravity.getOffset(anchor, mInstance);
29         mInstance.showAsDropDown(anchor, offset[0]+xmerge, offset[1]+ymerge);
30     }
31 
32     public void showAsDropDown(View anchor, int xoff, int yoff) {
33         mInstance.showAsDropDown(anchor, xoff, yoff);
34     }
35 
36     public void showAtLocation(View parent, int gravity, int x, int y) {
37         mInstance.showAtLocation(parent, gravity, x, y);
38     }
39 }

這裡我們要實現的就是“showBashOfAnchor”方法,其中有一個“LayoutGravity”類型的參數,這就是控制PopupWindow相對錨點位置的對象。下麵來定義“LayoutGravity”:

 1 public static class LayoutGravity {
 2     private int layoutGravity;
 3     // waring, don't change the order of these constants!
 4     public static final int ALIGN_LEFT=0x1;
 5     public static final int ALIGN_ABOVE=0x2;
 6     public static final int ALIGN_RIGHT=0x4;
 7     public static final int ALIGN_BOTTOM=0x8;
 8     public static final int TO_LEFT=0x10;
 9     public static final int TO_ABOVE=0x20;
10     public static final int TO_RIGHT=0x40;
11     public static final int TO_BOTTOM=0x80;
12     public static final int CENTER_HORI=0x100;
13     public static final int CENTER_VERT=0x200;
14 
15     public LayoutGravity(int gravity) {
16         layoutGravity=gravity;
17     }
18 
19     public int getLayoutGravity() { return layoutGravity; }
20     public void setLayoutGravity(int gravity) { layoutGravity=gravity; }
21 
22     public void setHoriGravity(int gravity) {
23         layoutGravity&=(0x2+0x8+0x20+0x80+0x200);
24         layoutGravity|=gravity;
25     }
26     public void setVertGravity(int gravity) {
27         layoutGravity&=(0x1+0x4+0x10+0x40+0x100);
28         layoutGravity|=gravity;
29     }
30 
31     public boolean isParamFit(int param) {
32         return (layoutGravity & param) > 0;
33     }
34 
35     public int getHoriParam() {
36         for(int i=0x1; i<=0x100; i=i<<2)
37             if(isParamFit(i))
38                 return i;
39         return ALIGN_LEFT;
40     }
41 
42     public int getVertParam() {
43         for(int i=0x2; i<=0x200; i=i<<2)
44             if(isParamFit(i))
45                 return i;
46         return TO_BOTTOM;
47     }
48 
49     public int[] getOffset(View anchor, PopupWindow window) {
50         int anchWidth=anchor.getWidth();
51         int anchHeight=anchor.getHeight();
52 
53         int winWidth=window.getWidth();
54         int winHeight=window.getHeight();
55         View view=window.getContentView();
56         if(winWidth<=0)
57             winWidth=view.getWidth();
58         if(winHeight<=0)
59             winHeight=view.getHeight();
60 
61         int xoff=0;
62         int yoff=0;
63 
64         switch (getHoriParam()) {
65             case ALIGN_LEFT:
66                 xoff=0; break;
67             case ALIGN_RIGHT:
68                 xoff=anchWidth-winWidth; break;
69             case TO_LEFT:
70                 xoff=-winWidth; break;
71             case TO_RIGHT:
72                 xoff=anchWidth; break;
73             case CENTER_HORI:
74                 xoff=(anchWidth-winWidth)/2; break;
75             default:break;
76         }
77         switch (getVertParam()) {
78             case ALIGN_ABOVE:
79                 yoff=-anchHeight; break;
80             case ALIGN_BOTTOM:
81                 yoff=-winHeight; break;
82             case TO_ABOVE:
83                 yoff=-anchHeight-winHeight; break;
84             case TO_BOTTOM:
85                 yoff=0; break;
86             case CENTER_VERT:
87                 yoff=(-winHeight-anchHeight)/2; break;
88             default:break;
89         }
90         return new int[]{ xoff, yoff };
91     }
92 }

這裡的主要方法就是“getOffset”,它會根據水平和垂直方向的gravity決定PopupWindow相對錨點的位置。

使用“LayoutGravity”時,可以通過“setHoriGravity”和“setVertGravity”方法設置水平和垂直方向的gravity,或者新建一個“LayoutGravity”對象。

下麵是一個demo:

 

使用setAnimationStyle方法添加動畫

上面我們提到了為PopupWindow設置背景和註冊事件監聽器,現在我們再來為PopupWindow添加動畫。

這裡的動畫是指PopupWindow出現和消失時的動畫。預設是直接彈出和消失,這樣難免讓用戶有一種突兀的感覺;如果PopupWindow能夠“滑入”屏幕和“滑出”屏幕(或者其他方式),用戶體驗會更好。

為PopupWindow添加動畫可以調用`setAnimationStyle`方法,該方法只有一個參數,就是指定動畫的樣式,因此我們需要定義動畫資源和樣式資源。

下麵是一個“滑入滑出”動畫:

 1 <!-- res/anim/translate_in.xml -->
 2 <?xml version="1.0" encoding="utf-8"?>
 3 <set xmlns:android="http://schemas.android.com/apk/res/android">
 4     <translate
 5         android:fromXDelta="0"
 6         android:toXDelta="0"
 7         android:fromYDelta="100%"
 8         android:toYDelta="0"
 9         android:duration="200" >
10     </translate>
11 </set>

 

 1 <!-- res/anim/translate_out.xml -->
 2 <?xml version="1.0" encoding="utf-8"?>
 3 <set xmlns:android="http://schemas.android.com/apk/res/android">
 4     <translate
 5         android:fromXDelta="0"
 6         android:toXDelta="0"
 7         android:fromYDelta="0"
 8         android:toYDelta="100%"
 9         android:duration="200" >
10     </translate>
11 </set>

然後定義“滑動”動畫樣式:

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

現在我們就可以為PopupWindow添加“滑動”動畫了:

1 window.setAnimationStyle(R.style.animTranslate);

我們來看下效果:


PS:這裡由於動畫的時間太短(200ms),另外轉GIF的時候可能截取的頻率有點低,導致滑動效果不是很明顯,建議自己運行demo查看

現在PopupWindow的出現/消失已經不是那麼突兀了。不過,當彈窗出現後,發現彈窗和背景不是很容易區分,如果此時彈窗的背景能“變暗”就好了。

沒問題,我們可以在彈窗出現後讓背景變暗,併在彈窗消失後讓背景還原:

 1     window.setOnDismissListener(new PopupWindow.OnDismissListener() {
 2         @Override
 3         public void onDismiss() {
 4             WindowManager.LayoutParams lp=getWindow().getAttributes();
 5             lp.alpha=1.0f;
 6             getWindow().clearFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
 7             getWindow().setAttributes(lp);
 8         }
 9     });
10 
11     window.showAtLocation(activityPopup, Gravity.BOTTOM, 0, 0);
12     WindowManager.LayoutParams lp=getWindow().getAttributes();
13     lp.alpha=0.3f;
14     getWindow().addFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
15     getWindow().setAttributes(lp);

現在再來看下效果:



現在PopupWindow就比較明顯了。

另外,我們還實現了透明度、縮放和旋轉三種動畫樣式,實現方式和上述大同小異,這裡就不再贅述。

源代碼

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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、BUG-In android7 phone can not slide above 註:Android 7.0以上,iScroll滑動緩慢遲鈍問題解決 What browser are you using? There was a fix to iScroll's handling of pas ...
  • 介紹 vue schart 是使用vue.js封裝了sChart.js圖表庫的一個小組件。支持vue.js 1.x & 2.x 倉庫地址: "https://github.com/lin xin/vue schart" sChart.js 作為一個小型簡單的圖表庫,沒有過多的圖表類型,只包含了柱狀圖 ...
  • 本來是在看阮大神寫的ajax教程,突然發現點擊目錄文字會跳轉到相對應的文本內容,於是乎激發了我的興趣。 這個究竟怎麼做的,剛開始看的時候一知半解,找度娘就是:“點擊跳轉到頁面指定位置”,找了下,原來專業術語叫:錨點。 度娘真是個博大精深的地方,有著多種的方法可以使用。 最簡單的一種: 設置a標簽的錨 ...
  • 參考資料:ios模式詳解,runtime完整總結 類和對象 Objective-C語言是一門動態語言,它將很多靜態語言在編譯和鏈接時期做的事放到了運行時來處理。這種動態語言的優勢在於:我們寫代碼時更具靈活性,如我們可以把消息轉發給我們想要的對象,或者隨意交換一個方法的實現等。 這種特性意味著Obje ...
  • 這篇是講Glide源碼中into方法的實現原理,可以說with和load方法只是做了前期的初始化配置工作,而真正意義上的圖片載入就是在into方法中實現的,所以該方法的複雜程度是可以想象的,還是依照我之前的寫作習慣,一步步的分析,不留下任何的盲點給大家帶來困惑,那麼下麵就開始吧。 ...
  • 我們都知道系統要確定View的大小,首先得先獲得MeasureSpec,再通過MeasureSpec來決定View的大小。 MeasureSpec(32為int值)由兩部分組成: SpecMode(高2位):測量模式。 SpecSize(低30位):某種測量模式下的規格大小。 SpecMode有3類 ...
  • 在你的程式沒有發佈(release)到AppStore的時候,你可以使用TestFlight去邀請用戶測試你的app,進而收集有用的反饋信息。 ...
  • 創建動畫 協議代理 屬性 設置動畫組件Item的動力屬性 UIDynamicItemBehavior 為動畫組件添加具體行為 吸引行為 UISnapBehavior 重力行為 UIGravityBehavior 碰撞行為 UICollisionBehavior 作用力行為 UIPushBehavio ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...