自定義PopupWindow的超強(qiáng)使用

先看效果稠集,在TitleBar上面有一個圖片的控件,當(dāng)點擊了圖片出現(xiàn)了一個可以擴(kuò)展的PopupWindow瓢宦,同時可以通過點擊事件去做相對應(yīng)的操作

jdfw.gif
所需要的類的:Item adapter listview 還有一個管理的ExtendPopupWindow
Paste_Image.png

1斥季、為了可擴(kuò)展性的話锦溪,我們需要在PopupWindow中加載的是一個可以動態(tài)設(shè)置的item,其實呢减余,可以在item中設(shè)置過多的屬性,以滿足實際的項目需要(比如說惩系,可以設(shè)置一個紅點位岔,提醒用戶這里面有更新的東西),這里我只設(shè)置了一個title堡牡,就是和上面的git圖中的一個文字的tiitle抒抬,

package com.example.popupwindow;

import android.content.Context;

/**
 * @author shiming
 * @time 2017/5/11 18:37
 * @desc  可以擴(kuò)展的item
 */

public class PopupItem {
    private String title;
    private int tag;

    public int getIcon() {
        return icon;
    }

    public void setIcon(int icon) {
        this.icon = icon;
    }

    private int icon;
    private Context context;
    public Context getContext() {
        return context;
    }

    public void setContext(Context context) {
        this.context = context;
    }

    public int getTag() {
        return tag;
    }

    public void setTag(int tag) {
        this.tag = tag;
    }

    public PopupItem(int tag, String title) {
        this.tag = tag;
        this.title = title;
    }
    public PopupItem(Context context,String title, int tag, int icon) {
        this(tag,title);
        this.icon=icon;
        this.context = context;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }
}

2、PopupMenuExtendAdapter的創(chuàng)建

1晤柄、R.layout.nim_popup_menu_list_item擦剑,layout布局,可以通過自定義芥颈,這里就是簡單的使用的l惠勒,使用了ViewHolder 去管理
2、通過對數(shù)據(jù)的獲取PopupItem item = mPopupItemList.get(position);設(shè)置layout應(yīng)該顯示的內(nèi)容

package com.example.popupwindow;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.TextView;

import java.util.List;

/**
 * @author shiming
 * @time 2017/5/11 19:08
 * @desc  可以動態(tài)的設(shè)置很多
 */

public class PopupMenuExtendAdapter extends BaseAdapter {
    private Context mContext;
    private List<PopupItem> mPopupItemList;
    private final LayoutInflater mInflater;

    public PopupMenuExtendAdapter(Context context, List<PopupItem> popupItemList) {
        mContext = context;
        mPopupItemList = popupItemList;
        //通過這個方法去拿到 infalter
        mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    @Override
    public int getCount() {
        return mPopupItemList.size();
    }

    @Override
    public Object getItem(int position) {
        return mPopupItemList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder cache;
        if (convertView == null) {
             cache =  new ViewHolder();
            convertView = mInflater.inflate(R.layout.nim_popup_menu_list_item, null);
            cache.icon = (ImageView) convertView.findViewById(R.id.popup_menu_icon);
            cache.title = (TextView) convertView.findViewById(R.id.popup_menu_title);
            convertView.setTag(cache);
        } else {
           cache = (ViewHolder) convertView.getTag();
        }
        PopupItem item = mPopupItemList.get(position);
        cache.title.setText(item.getTitle());
        if (item.getIcon() != 0) {
            cache.icon.setVisibility(View.VISIBLE);
            cache.icon.setImageResource(item.getIcon());
        } else {
            cache.icon.setVisibility(View.GONE);
        }
        // 下面代碼實現(xiàn)數(shù)據(jù)綁定
        return convertView;
    }

    private final class ViewHolder {

        public ImageView icon;

        public TextView title;
    }

}

3爬坑、PopupMenuExtendListview的創(chuàng)建纠屋,其實也是簡單的擴(kuò)張了listview,對listview的寬度重新的測量盾计,代碼如下

package com.example.popupwindow;

import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ListView;

/**
 * @author shiming
 * @time 2017/5/8 19:56
 * @desc popup  Menu  為了可以擴(kuò)展 這里繼承了 listview
 */

public class PopupMenuExtendListview extends ListView {

    public PopupMenuExtendListview(Context context) {
        this(context, null);
    }

    public PopupMenuExtendListview(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    /**
     * 1.View本身大小多少售担,這由onMeasure()決定;
     * 2.View在ViewGroup中的位置如何署辉,這由onLayout()決定族铆;
     * 3.繪制View,onDraw()定義了如何繪制這個View哭尝。
     */
    /**
     * View本身大小多少哥攘,這由onMeasure()決定
     * @param widthMeasureSpec 代模式的32 位
     * @param heightMeasureSpec  代模式的32位
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int width = measureWidthByChilds() + getPaddingLeft() + getPaddingRight();
        //重新的測量這個 精確的模式下面
        super.onMeasure(MeasureSpec.makeMeasureSpec(width,MeasureSpec.EXACTLY), heightMeasureSpec);
    }

    /**
     * 測量listview中的孩子的寬度
     *
     * @return width
     */
    private int measureWidthByChilds() {
        int width = 0;
        View view = null;
        for (int i = 0; i < getAdapter().getCount(); i++) {
            view = getAdapter().getView(i, view, this);
            if (view != null) {
                //為什么要傳入的是0
                // *設(shè)置與此視圖相關(guān)聯(lián)的布局參數(shù)。這些供應(yīng)
                //*參數(shù)刚夺,以指定此視圖應(yīng)該如何
                // *設(shè)置献丑。有許多子viewgroup.layoutparams,這些
                // *對應(yīng)的ViewGroup負(fù)責(zé)不同的子類
                view.setLayoutParams(new ViewGroup.LayoutParams(0, 0));
                // MeasureSpec.UNSPECIFIED是未指定尺寸侠姑,這種情況不多创橄,
                // 一般都是父控件是AdapterView,通過measure方法傳入的模式莽红。
                view.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);
                if (view.getMeasuredWidth()>width){
                    width=view.getMeasuredWidth();
                }
            }
        }
        return width;
    }
}

在寫代碼的時候如果我把view.setLayoutParams(new ViewGroup.LayoutParams(0, 0));給去掉了妥畏,只去view.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);我會發(fā)現(xiàn)始終測出來的值不對邦邦,但是我也沒有想通為什么要這么去設(shè)置,這個是看別人講解醉蚁,也不太明白燃辖。記得以前實現(xiàn)過一個動畫,先要去mease(0,0)网棍;然后去執(zhí)行才不會出錯黔龟,哎哎,先給個todo

4滥玷、關(guān)鍵的交互類氏身,ExtendPopupWindow詳情

1、構(gòu)造方法傳入?yún)?shù),MenuItemClickListener是內(nèi)部的接口惑畴,專門是做來點擊的事件蛋欣。

  //當(dāng)擴(kuò)張到一定的地步是不是可以滑動
    private boolean scroll = false;
    public ExtendPopupWindow(Context context, List<PopupItem> items,  MenuItemClickListener listener) {
        this(context,items,false,listener);
    }

    public ExtendPopupWindow(Context context, List<PopupItem> items,  boolean scroll, MenuItemClickListener listener) {
        this.context = context;
        this.items = items;
        this.scroll = scroll;
        this.listener = listener;
        init();
    }
  public interface MenuItemClickListener {
         void onItemClick(PopupItem item);
    }

2、init()方法做什么

第一步初始化我們的rootview

   if (rootView == null) {
            rootView = LayoutInflater.from(context).inflate(R.layout.nim_popup_menu_layout, null);
            ListView listView = (ListView) rootView.findViewById(R.id.popmenu_listview);
            listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {

                @Override
                public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                    if (listener != null) {
                        popWindow.dismiss();
                        listener.onItemClick(items.get(position));
                    }
                }
            });
            adapter = new PopupMenuExtendAdapter(context, items);
            listView.setAdapter(adapter);
        }
        // focusableInTouchMode.這個屬性的意思一如字面所述,就是在進(jìn)入觸摸輸入模式后,該控件是否還有獲得焦點的能力.
        // 可以簡單的理解為,用戶一旦開始通過點擊屏幕的方式輸入,手機(jī)就進(jìn)入了"touch mode".
        // focusableInTouchMode這種屬性,多半是設(shè)給EditText這種即使在TouchMode下,依然需要獲取焦點的控件
        rootView.setFocusableInTouchMode(true);
        //監(jiān)聽返回的按鈕如贷,然后 隱藏pop
        rootView.setOnKeyListener(new View.OnKeyListener() {
            @Override
            public boolean onKey(View v, int keyCode, KeyEvent event) {
                if (keyCode == KeyEvent.KEYCODE_MENU && popWindow.isShowing()
                        && event.getAction() == KeyEvent.ACTION_DOWN) {
                    popWindow.dismiss();
                    return true;
                }
                return false;
            }
        });

以上的rootview其實是一個layout的高度為15dp陷虎,其中嵌套了一個我們封裝的popupwindow的listview,在給他設(shè)置了一個setOnKeyListener監(jiān)聽杠袱,判斷了按下rootview尚猿,判斷三種情況的成立的話pupwindow就必須的dismiss掉,KEYCODE_MENU其實就是早期的菜單鍵楣富,return true谊路;表示攔截這個操作

第二步做什么:初始化我們popwindow

  //初始化我們的popuwindow
        if (popWindow == null) {
            popWindow = new PopupWindow(context);
            popWindow.setContentView(rootView);
            popWindow.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
            if (scroll) {
                popWindow.setHeight(context.getApplicationContext().getResources().getDisplayMetrics().heightPixels * 2 / 3);
            } else {
                popWindow.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
            }
            popWindow.setTouchable(true);
            popWindow.setBackgroundDrawable(new BitmapDrawable());
            popWindow.setOnDismissListener(new PopupWindow.OnDismissListener() {
                @Override
                public void onDismiss() {
                }
            });
        }

以上需要說明的是: popWindow.setBackgroundDrawable(new BitmapDrawable());我們要給他一個bitmap,要不然會有問題菩彬,安卓有很多的地方有這個需求缠劝,估計理解為安卓的內(nèi)部的毛病。

假想個問題骗灶,這既然是一個交互的類惨恭,那么我們還需要做什么,好吧耙旦,我也咩想到脱羡,想到,最后看別人的代碼免都,哈哈…………

數(shù)據(jù)的交互锉罐,有adapter的對吧,so,暴露方法告訴adapter的數(shù)據(jù)變了
    public void notifyData() {
        adapter.notifyDataSetChanged();
    }
pop不可能在初始化頁面了绕娘,就開始顯示了吧脓规,so,砸門的控制show()险领,判null,是否showing,這里有個關(guān)鍵的東西侨舆,我們設(shè)置了可以滑動秒紧,同時要判斷一下我們設(shè)備的屏幕的方向。
   if (popWindow == null) {
            return;
        }
        if (popWindow.isShowing()) {
            popWindow.dismiss();
        } else {
            if (scroll) {// 當(dāng)可以滾動時挨下,橫豎屏切換時熔恢,重新確定高度
                Configuration configuration = context.getResources().getConfiguration();
                int ori = configuration.orientation;
                if (ori == Configuration.ORIENTATION_LANDSCAPE) {
                    popWindow.setHeight(context.getApplicationContext().getResources().getDisplayMetrics().widthPixels* 2 / 3);
                } else {
                    popWindow.setHeight(context.getApplicationContext().getResources().getDisplayMetrics().heightPixels  * 2 / 3);
                }
            }
            popWindow.setFocusable(true);
            popWindow.showAsDropDown(v, -10, 0);
        }

Configuration類專門描述手機(jī)設(shè)備上的配置信息,這些配置信息既包括用戶特定的配置項臭笆,也包括系統(tǒng)的動態(tài)設(shè)備配置叙淌。 int ori = configuration.orientation;判斷是否是橫豎屏,我記得沒錯的話愁铺,橫豎屏的話凿菩,我們的activity是需要重新的走生命周期,只有onnewintent的這個才不會走帜讲,是橫的話,為了顯示的更加人性化椒拗,我們要橫的2/3似将。 popWindow.showAsDropDown(v, -10, 0);顯示在哪個控件下面,這個v是我們的控件iew蚀苛,偏移了-10.y沒有動在验,這些可以動態(tài)的設(shè)置

5、Activity中的代碼如下

package com.example.popupwindow;

import android.content.Context;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {

    private ArrayList<PopupItem> menuItemList;
    private ExtendPopupWindow popupMenu;
    private View mImg;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mImg = findViewById(R.id.img);
        mImg.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                initPopuptWindow(MainActivity.this, mImg);
            }
        });
    }

    /**
     * 窗體泄漏造成的原因堵未,是我們的view雖然已經(jīng)有了這個腋舌,但是我們在沒有通過點擊事件
     * 或者其他的動作,去開啟這個popwindow的話渗蟹,我們的窗體就會泄漏
     * 哈哈
     *
     */
    @Override
    protected void onResume() {
        super.onResume();
//        initPopuptWindow(this, mImg);
    }

    private void initPopuptWindow(Context context, View view) {
        if (popupMenu == null) {
            menuItemList = new ArrayList<>();
            popupMenu = new ExtendPopupWindow(context, menuItemList, listener);
        }
        menuItemList.clear();
        menuItemList.addAll(getMoreMenuItems(context));
        popupMenu.notifyData();
        popupMenu.show(view);
    }

    private static List<PopupItem> getMoreMenuItems(Context context) {
        List<PopupItem> moreMenuItems = new ArrayList<PopupItem>();
        moreMenuItems.add(new PopupItem(context,
                "feiji  0", 1,0));
        moreMenuItems.add(new PopupItem(context,
                "feiji  1", 2,0));
        moreMenuItems.add(new PopupItem(context,
                "feiji  2", 3,0));
        moreMenuItems.add(new PopupItem(context,
                "feiji  3", 4,0));
        return moreMenuItems;
    }

    private ExtendPopupWindow.MenuItemClickListener listener = new ExtendPopupWindow.MenuItemClickListener() {

        @Override
        public void onItemClick(final PopupItem item) {
            switch (item.getTag()) {
                case 1:
                    Toast.makeText(MainActivity.this, "feiji  0",Toast.LENGTH_SHORT).show();
                    break;
                case 2:
                    Toast.makeText(MainActivity.this, "feiji  1",Toast.LENGTH_SHORT).show();
                    break;
                case 3:
                    Toast.makeText(MainActivity.this, "feiji  2",Toast.LENGTH_SHORT).show();
                    break;
                case 4:
                    Toast.makeText(MainActivity.this, "feiji  3",Toast.LENGTH_SHORT).show();
                    break;
            }
        }
    };
}

以上只說明一點:popupWindow块饺,不可能在初始化就顯示了,只能通過事件來顯示雌芽,要不然會報錯窗體泄漏

謝謝 授艰!

github地址:https://github.com/Shimingli/PopupWindow

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市世落,隨后出現(xiàn)的幾起案子淮腾,更是在濱河造成了極大的恐慌,老刑警劉巖屉佳,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谷朝,死亡現(xiàn)場離奇詭異,居然都是意外死亡武花,警方通過查閱死者的電腦和手機(jī)圆凰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來体箕,“玉大人送朱,你說我怎么就攤上這事娘荡。” “怎么了驶沼?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵炮沐,是天一觀的道長。 經(jīng)常有香客問我回怜,道長大年,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任玉雾,我火速辦了婚禮翔试,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘复旬。我一直安慰自己垦缅,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布驹碍。 她就那樣靜靜地躺著壁涎,像睡著了一般。 火紅的嫁衣襯著肌膚如雪志秃。 梳的紋絲不亂的頭發(fā)上怔球,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天,我揣著相機(jī)與錄音浮还,去河邊找鬼竟坛。 笑死,一個胖子當(dāng)著我的面吹牛钧舌,可吹牛的內(nèi)容都是我干的担汤。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼洼冻,長吁一口氣:“原來是場噩夢啊……” “哼漫试!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起碘赖,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤驾荣,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后普泡,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體播掷,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年撼班,在試婚紗的時候發(fā)現(xiàn)自己被綠了歧匈。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡砰嘁,死狀恐怖件炉,靈堂內(nèi)的尸體忽然破棺而出勘究,到底是詐尸還是另有隱情,我是刑警寧澤斟冕,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布口糕,位于F島的核電站,受9級特大地震影響磕蛇,放射性物質(zhì)發(fā)生泄漏景描。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一秀撇、第九天 我趴在偏房一處隱蔽的房頂上張望超棺。 院中可真熱鬧,春花似錦呵燕、人聲如沸棠绘。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽氧苍。三九已至,卻和暖如春霍衫,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侯养。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工敦跌, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人逛揩。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓柠傍,卻偏偏與公主長得像,于是被迫代替她去往敵國和親辩稽。 傳聞我的和親對象是個殘疾皇子惧笛,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,104評論 25 707
  • ¥開啟¥ 【iAPP實現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開一個線程,因...
    小菜c閱讀 6,408評論 0 17
  • 一逞泄、適用場景 ListViewListview是一個很重要的組件患整,它以列表的形式根據(jù)數(shù)據(jù)的長自適應(yīng)展示具體內(nèi)容,用...
    Geeks_Liu閱讀 10,668評論 1 28
  • 最近這段時間的調(diào)理,我看清了太多的東西喷众。瞬間明白了各谚,該放手的就不要強(qiáng)迫自己緊抓不放。我有情到千,你有情昌渤,亦為感情...
    逃避現(xiàn)實閱讀 990評論 0 0
  • 今天,爸爸媽媽陪我去參觀了一個有趣的科技館憔四,就在濱江新城不遠(yuǎn)的地方膀息。 進(jìn)入科技館般眉,在第一層有一個奇...
    冰風(fēng)冷雨_71b3閱讀 205評論 0 1