自定义popWindow(高仿qq加号弹出框)

1) 弹出菜单的封装PopMenu

这里自定义的PopMenu类就是自定义的一个容器类,这个容器就是我们用来放置popWindow的相关的内容。

package tongchuang.com.test.app.utils.GZXPopWindowView;

/**
 * Created by Administrator on 2016/5/25 0025.
 */
import android.content.Context;
import android.graphics.drawable.ColorDrawable;
import android.view.KeyEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.PopupWindow;

import java.util.ArrayList;

public abstract class PopMenu {

    private Context mContext;
    private ArrayList<Item> mItemList;
    private ArrayAdapter<Item> mAdapter;
    private OnItemSelectedListener mListener;
    private ListView mListView;
    private PopupWindow mPopupWindow;

    public PopMenu(Context context) {
        mContext = context;
        mItemList = new ArrayList<Item>(2);
        View view = onCreateView(context);
        view.setFocusableInTouchMode(true);
        mAdapter = onCreateAdapter(context, mItemList);
        mListView = findListView(view);
        mListView.setAdapter(mAdapter);
        mListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                Item item = mAdapter.getItem(position);
                if (mListener != null) {
                    mListener.selected(view, item, position);
                }
                mPopupWindow.dismiss();
            }
        });
        view.setOnKeyListener(new View.OnKeyListener() {
            @Override
            public boolean onKey(View v, int keyCode, KeyEvent event) {
                if (keyCode == KeyEvent.KEYCODE_MENU && mPopupWindow.isShowing()) {
                    mPopupWindow.dismiss();
                    return true;
                }
                return false;
            }
        });
        mPopupWindow = new PopupWindow(view, ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT, true);
        mPopupWindow.setBackgroundDrawable(new ColorDrawable(0x00000000));
    }

    protected abstract View onCreateView(Context context);

    protected abstract ListView findListView(View view);

    protected abstract ArrayAdapter<Item> onCreateAdapter(Context context, ArrayList<Item> itemList);

    public void addItem(String text, int id) {
        mItemList.add(new Item(text, id));
        mAdapter.notifyDataSetChanged();
    }

    public void addItem(int resId, int id) {
        addItem(mContext.getString(resId), id);
    }

    public void showAsDropDown(View parent) {
        mPopupWindow.showAsDropDown(parent);
    }

    public void dismiss() {
        mPopupWindow.dismiss();
    }

    public void setOnItemSelectedListener(OnItemSelectedListener listener) {
        mListener = listener;
    }

    public boolean isShowing() {
        return mPopupWindow.isShowing();
    }

    public static class Item {
        public String text;
        public int id;

        public Item(String text, int id) {
            this.text = text;
            this.id = id;
        }

        @Override
        public String toString() {
            return text;
        }
    }

    public static interface OnItemSelectedListener {

        public void selected(View view, Item item, int position);
    }
}

在这个类中主要实现的功能主要包含的是:一是返回并实现这个弹出菜单,也就是这里的view;二是对于popwindow中每一个item都是使用的是listView来进行呈现的,也就是使用适配的原理实现的;三是对于一些方法进行封装;四是在这个PopMenu类中存在着三个抽象方法,这3个抽象方法就是实现自定义PopMenu类。

2) 自定义MyPopMenu

在需要产生popWindowActivity类中,使用的就是MyPopWindow类,该类继承了PopMenu类,在MyPopWindow类中主要实现的就是重写PopMenu类中的3个抽象方法。

package tongchuang.com.test.app.utils.GZXPopWindowView;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;

import java.util.ArrayList;

import tongchuang.com.test.R;

/**
 * Created by Administrator on 2016/5/25 0025.
 */
public class MyPopWindow extends PopMenu{
    public MyPopWindow(Context context) {
        super(context);
    }

    @Override
    protected ListView findListView(View view) {
        return (ListView) view.findViewById(R.id.menu_listview);
    }

    @Override
    protected View onCreateView(Context context) {
        View view = LayoutInflater.from(context).inflate(R.layout.popwindow_pop, null);
        return view;
    }

    @Override
    protected ArrayAdapter<Item> onCreateAdapter(Context context, ArrayList<Item> items) {
        return new ArrayAdapter<Item>(context, R.layout.popwindow_pop_item, items);
    }
}

在这个MyPopWindow类中,主要实现3个抽象方法,第一个抽象方法返回的就是自定义的ListView,使用的是protected ListView findListView(View view) 这个方法;第二个抽象方法,protected View onCreateView(Context context) ,就是返回的是当前存在着自定义的ListViewxml布局文件;第三个抽象方法,protected ArrayAdapter<Item> onCreateAdapter(Context context, ArrayList<Item> items) ,就是使用的是ListView的子item

3) 自定义ListView

package tongchuang.com.test.app.utils.GZXPopWindowView;
import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.widget.ListView;
/**
 * Created by Administrator on 2016/5/25 0025.
 */
public class PopShowListView extends ListView{

    public PopShowListView(Context context) {
        super(context);
    }

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

    public PopShowListView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int width = 0;
        for (int i = 0; i < getChildCount(); i++) {
            View child = getChildAt(i);
            child.measure(MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED), heightMeasureSpec);
            int w = child.getMeasuredWidth();
            if (w > width) width = w;
        }

        widthMeasureSpec = MeasureSpec.makeMeasureSpec(width + getPaddingLeft() + getPaddingRight(), MeasureSpec.EXACTLY);

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }
}

自定义完成ListView后,此时就可以在xml中进行使用了。

4) popWindow_pop.xml主布局文件

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

    <ImageView
        android:id="@+id/head"
        android:src="@drawable/toparrow"
        android:layout_gravity="right"
        android:layout_width="wrap_content"
        android:contentDescription="@null"
        android:layout_marginRight="18dp"
        android:layout_height="wrap_content"
/>

    <tongchuang.com.test.app.utils.GZXPopWindowView.PopShowListView
        android:id="@+id/menu_listview"
        android:padding="6dp"
        android:focusable="true"
        android:background="#FFFFFF"
        android:layout_width="wrap_content"
        android:cacheColorHint="@android:color/transparent"
        android:layout_height="wrap_content"
/>
</LinearLayout>

在这个布局文件中主要是存在两个控件,一个是ImageView,一个是自定义的ListView,第一个ImageView控件实现的是,如下图所示,当popWindow弹出后,此时有一个白色的三角是指向点击的弹出popwindow的控件,而下面的所显示的item就是通过自定义控件来完成的。

 

5) Popwindow_pop_item.Xmlitem布局

<?xml version="1.0" encoding="utf-8"?>

<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:textSize="16dp"
    android:gravity="center"
    android:layout_gravity="center"
    android:paddingLeft="20dp"
    android:paddingTop="6dp"
    android:singleLine="true"
    android:paddingBottom="6dp"
    android:paddingRight="20dp"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
/>

 

6) Activity类中使用自定义的PopWindow

7) private static final int USER_SEARCH = 0;
private static final int USER_ADD = 1;
private MyPopWindow mMenu;

private void initMenu() {
    $.toast("点击了第0",0);
    mMenu = new MyPopWindow(getBaseContext());
    mMenu.addItem("aaaa", USER_SEARCH);
    mMenu.addItem("bbbb", USER_ADD);
    mMenu.setOnItemSelectedListener(new PopMenu.OnItemSelectedListener() {
        @Override
        public void selected(View view, PopMenu.Item item, int position) {
            switch (item.id) {
                case USER_SEARCH:
                  $.toast("点击了第一个",0);
                    break;
                case USER_ADD:
                    $.toast("点击了第二个",0);
                    break;
            }
        }
    });

    mMenu.showAsDropDown(right_image);
}

这就是在Activity类中使用popwindow的方法,由于这里只是在popwindow中添加了2条数据所以这里的switch中只是存在着两个case,而且还要注意的是,当我们需要弹出popwindow的时候,此时必须加上mMenu.showAsDropDown(right_image);而且这个方法的定义是showAsDropDown(View parent);所以的是,这里的参数View实际上表示的就是我们需要点击的View,这也就是这里的popWindow的父级View

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值