Android UI开发第十六篇——分享一个popuwindow实例

来源:互联网 发布:java求100内的素数 编辑:程序博客网 时间:2024/05/16 17:44
        PopupWindow在android.widget包下,弹出窗口的形式展示。官方文档对该控件的描述是:“一个弹出窗口控件,可以用来显示任意视图(View),而且会浮动在当前 活动(activity)的顶部”。PopupWindow可以让我们实现多种自定义控件,例如:menu、alertdialog等弹窗似的View。

UI开发第三篇——popupwindow 中简单介绍了一些简单方法,这一篇分享一个实例。看效果:

            

 

 

    

 

实现中使用的  PopupWindow。这里做了简单封装,其中有三个类组成:PopuItem、PopuJar、PopupWindows。

public class PopuItem {private Drawable icon;private Bitmap thumb;private String title;private int actionId = -1;    private boolean selected;    private boolean sticky;    /**     * Constructor     *      * @param actionId  Action id for case statements     * @param title     Title     * @param icon      Icon to use     */    public PopuItem(int actionId, String title, Drawable icon) {        this.title = title;        this.icon = icon;        this.actionId = actionId;    }        /**     * Constructor     */    public PopuItem() {        this(-1, null, null);    }        /**     * Constructor     *      * @param actionId  Action id of the item     * @param title     Text to show for the item     */    public PopuItem(int actionId, String title) {        this(actionId, title, null);    }        /**     * Constructor     *      * @param icon {@link Drawable} action icon     */    public PopuItem(Drawable icon) {        this(-1, null, icon);    }        /**     * Constructor     *      * @param actionId  Action ID of item     * @param icon      {@link Drawable} action icon     */    public PopuItem(int actionId, Drawable icon) {        this(actionId, null, icon);    }/** * Set action title *  * @param title action title */public void setTitle(String title) {this.title = title;}/** * Get action title *  * @return action title */public String getTitle() {return this.title;}/** * Set action icon *  * @param icon {@link Drawable} action icon */public void setIcon(Drawable icon) {this.icon = icon;}/** * Get action icon * @return  {@link Drawable} action icon */public Drawable getIcon() {return this.icon;} /**     * Set action id     *      * @param actionId  Action id for this action     */    public void setActionId(int actionId) {        this.actionId = actionId;    }        /**     * @return  Our action id     */    public int getActionId() {        return actionId;    }        /**     * Set sticky status of button     *      * @param sticky  true for sticky, pop up sends event but does not disappear     */    public void setSticky(boolean sticky) {        this.sticky = sticky;    }        /**     * @return  true if button is sticky, menu stays visible after press     */    public boolean isSticky() {        return sticky;    }    /** * Set selected flag; *  * @param selected Flag to indicate the item is selected */public void setSelected(boolean selected) {this.selected = selected;}/** * Check if item is selected *  * @return true or false */public boolean isSelected() {return this.selected;}/** * Set thumb *  * @param thumb Thumb image */public void setThumb(Bitmap thumb) {this.thumb = thumb;}/** * Get thumb image *  * @return Thumb image */public Bitmap getThumb() {return this.thumb;}}


 

public class PopuJar extends PopupWindows implements OnDismissListener {private View mRootView;private ImageView mArrowUp;private ImageView mArrowDown;private LayoutInflater mInflater;private ViewGroup mTrack;private ScrollView mScroller;private OnPopuItemClickListener mItemClickListener;private OnDismissListener mDismissListener;private List<PopuItem> PopuItems = new ArrayList<PopuItem>();private boolean mDidAction;private int mChildPos;    private int mInsertPos;    private int mAnimStyle;    private int mOrientation;    private int rootWidth=0;        public static final int HORIZONTAL = 0;    public static final int VERTICAL = 1;        public static final int ANIM_GROW_FROM_LEFT = 1;public static final int ANIM_GROW_FROM_RIGHT = 2;public static final int ANIM_GROW_FROM_CENTER = 3;public static final int ANIM_REFLECT = 4;public static final int ANIM_AUTO = 5;    /**     * Constructor for default vertical layout     *      * @param context  Context     */    public PopuJar(Context context) {        this(context, VERTICAL);    }    /**     * Constructor allowing orientation override     *      * @param context    Context     * @param orientation Layout orientation, can be vartical or horizontal     */    public PopuJar(Context context, int orientation) {        super(context);                mOrientation = orientation;                mInflater  = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);        if (mOrientation == HORIZONTAL) {            setRootViewId(R.layout.popup_horizontal);        } else {            setRootViewId(R.layout.popup_vertical);        }        mAnimStyle = ANIM_AUTO;        mChildPos = 0;    }    /**     * Get action item at an index     *      * @param index  Index of item (position from callback)     *      * @return  Action Item at the position     */    public PopuItem getPopuItem(int index) {        return PopuItems.get(index);    }    /** * Set root view. *  * @param id Layout resource id */public void setRootViewId(int id) {mRootView= (ViewGroup) mInflater.inflate(id, null);mTrack = (ViewGroup) mRootView.findViewById(R.id.tracks);mArrowDown = (ImageView) mRootView.findViewById(R.id.arrow_down);mArrowUp = (ImageView) mRootView.findViewById(R.id.arrow_up);mScroller= (ScrollView) mRootView.findViewById(R.id.scroller);//This was previously defined on show() method, moved here to prevent force close that occured//when tapping fastly on a view to show quickaction dialog.//Thanx to zammbi (github.com/zammbi)mRootView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));setContentView(mRootView);}/** * Set animation style *  * @param mAnimStyle animation style, default is set to ANIM_AUTO */public void setAnimStyle(int mAnimStyle) {this.mAnimStyle = mAnimStyle;}/** * Set listener for action item clicked. *  * @param listener Listener */public void setOnPopuItemClickListener(OnPopuItemClickListener listener) {mItemClickListener = listener;}/** * Add action item *  * @param action  {@link PopuItem} */public void addPopuItem(PopuItem action) {PopuItems.add(action);String title = action.getTitle();Drawable icon = action.getIcon();View container;if (mOrientation == HORIZONTAL) {            container = mInflater.inflate(R.layout.action_item_horizontal, null);        } else {            container = mInflater.inflate(R.layout.action_item_vertical, null);        }ImageView img = (ImageView) container.findViewById(R.id.iv_icon);TextView text = (TextView) container.findViewById(R.id.tv_title);if (icon != null) {img.setImageDrawable(icon);} else {img.setVisibility(View.GONE);}if (title != null) {text.setText(title);} else {text.setVisibility(View.GONE);}final int pos =  mChildPos;final int actionId = action.getActionId();container.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {if (mItemClickListener != null) {                    mItemClickListener.onItemClick(PopuJar.this, pos, actionId);                }                if (!getPopuItem(pos).isSticky()) {                  mDidAction = true;                                    dismiss();                }}});container.setFocusable(true);container.setClickable(true); if (mOrientation == HORIZONTAL && mChildPos != 0) {            View separator = mInflater.inflate(R.layout.horiz_separator, null);                        RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.FILL_PARENT);                        separator.setLayoutParams(params);            separator.setPadding(5, 0, 5, 0);                        mTrack.addView(separator, mInsertPos);                        mInsertPos++;        }mTrack.addView(container, mInsertPos);mChildPos++;mInsertPos++;}/** * Show quickaction popup. Popup is automatically positioned, on top or bottom of anchor view. *  */public void show (View anchor) {preShow();int xPos, yPos, arrowPos;mDidAction = false;int[] location = new int[2];anchor.getLocationOnScreen(location);Rect anchorRect = new Rect(location[0], location[1], location[0] + anchor.getWidth(), location[1]                 + anchor.getHeight());//mRootView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));mRootView.measure(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);int rootHeight = mRootView.getMeasuredHeight();if (rootWidth == 0) {rootWidth= mRootView.getMeasuredWidth();}int screenWidth = mWindowManager.getDefaultDisplay().getWidth();int screenHeight= mWindowManager.getDefaultDisplay().getHeight();//automatically get X coord of popup (top left)if ((anchorRect.left + rootWidth) > screenWidth) {xPos = anchorRect.left - (rootWidth-anchor.getWidth());xPos = (xPos < 0) ? 0 : xPos;arrowPos = anchorRect.centerX()-xPos;} else {if (anchor.getWidth() > rootWidth) {xPos = anchorRect.centerX() - (rootWidth/2);} else {xPos = anchorRect.left;}arrowPos = anchorRect.centerX()-xPos;}int dyTop= anchorRect.top;int dyBottom= screenHeight - anchorRect.bottom;boolean onTop= (dyTop > dyBottom) ? true : false;if (onTop) {if (rootHeight > dyTop) {yPos = 15;LayoutParams l = mScroller.getLayoutParams();l.height= dyTop - anchor.getHeight();} else {yPos = anchorRect.top - rootHeight;}} else {yPos = anchorRect.bottom;if (rootHeight > dyBottom) { LayoutParams l = mScroller.getLayoutParams();l.height= dyBottom;}}showArrow(((onTop) ? R.id.arrow_down : R.id.arrow_up), arrowPos);setAnimationStyle(screenWidth, anchorRect.centerX(), onTop);mWindow.showAtLocation(anchor, Gravity.NO_GRAVITY, xPos, yPos);}/** * Set animation style *  * @param screenWidth screen width * @param requestedX distance from left edge * @param onTop flag to indicate where the popup should be displayed. Set TRUE if displayed on top of anchor view *   and vice versa */private void setAnimationStyle(int screenWidth, int requestedX, boolean onTop) {int arrowPos = requestedX - mArrowUp.getMeasuredWidth()/2;switch (mAnimStyle) {case ANIM_GROW_FROM_LEFT:mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Left : R.style.Animations_PopDownMenu_Left);break;case ANIM_GROW_FROM_RIGHT:mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Right : R.style.Animations_PopDownMenu_Right);break;case ANIM_GROW_FROM_CENTER:mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Center : R.style.Animations_PopDownMenu_Center);break;case ANIM_REFLECT:mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Reflect : R.style.Animations_PopDownMenu_Reflect);break;case ANIM_AUTO:if (arrowPos <= screenWidth/4) {mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Left : R.style.Animations_PopDownMenu_Left);} else if (arrowPos > screenWidth/4 && arrowPos < 3 * (screenWidth/4)) {mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Center : R.style.Animations_PopDownMenu_Center);} else {mWindow.setAnimationStyle((onTop) ? R.style.Animations_PopUpMenu_Right : R.style.Animations_PopDownMenu_Right);}break;}}/** * Show arrow *  * @param whichArrow arrow type resource id * @param requestedX distance from left screen */private void showArrow(int whichArrow, int requestedX) {        final View showArrow = (whichArrow == R.id.arrow_up) ? mArrowUp : mArrowDown;        final View hideArrow = (whichArrow == R.id.arrow_up) ? mArrowDown : mArrowUp;        final int arrowWidth = mArrowUp.getMeasuredWidth();        showArrow.setVisibility(View.VISIBLE);                ViewGroup.MarginLayoutParams param = (ViewGroup.MarginLayoutParams)showArrow.getLayoutParams();               param.leftMargin = requestedX - arrowWidth / 2;                hideArrow.setVisibility(View.INVISIBLE);    }/** * Set listener for window dismissed. This listener will only be fired if the quicakction dialog is dismissed * by clicking outside the dialog or clicking on sticky item. */public void setOnDismissListener(PopuJar.OnDismissListener listener) {setOnDismissListener(this);mDismissListener = listener;}@Overridepublic void onDismiss() {if (!mDidAction && mDismissListener != null) {mDismissListener.onDismiss();}}/** * Listener for item click * */public interface OnPopuItemClickListener {public abstract void onItemClick(PopuJar source, int pos, int actionId);}/** * Listener for window dismiss *  */public interface OnDismissListener {public abstract void onDismiss();}}


 

 

public class PopupWindows {protected Context mContext;protected PopupWindow mWindow;protected View mRootView;protected Drawable mBackground = null;protected WindowManager mWindowManager;/** * Constructor. *  * @param context Context */public PopupWindows(Context context) {mContext= context;mWindow = new PopupWindow(context);mWindow.setTouchInterceptor(new OnTouchListener() {@Overridepublic boolean onTouch(View v, MotionEvent event) {if (event.getAction() == MotionEvent.ACTION_OUTSIDE) {mWindow.dismiss();return true;}return false;}});mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);}/** * On dismiss */protected void onDismiss() {}/** * On show */protected void onShow() {}/** * On pre show */protected void preShow() {if (mRootView == null) throw new IllegalStateException("setContentView was not called with a view to display.");onShow();if (mBackground == null) mWindow.setBackgroundDrawable(new BitmapDrawable());else mWindow.setBackgroundDrawable(mBackground);mWindow.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);mWindow.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);mWindow.setTouchable(true);mWindow.setFocusable(true);mWindow.setOutsideTouchable(true);mWindow.setContentView(mRootView);}/** * Set background drawable. *  * @param background Background drawable */public void setBackgroundDrawable(Drawable background) {mBackground = background;}/** * Set content view. *  * @param root Root view */public void setContentView(View root) {mRootView = root;mWindow.setContentView(root);}/** * Set content view. *  * @param layoutResID Resource id */public void setContentView(int layoutResID) {LayoutInflater inflator = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);setContentView(inflator.inflate(layoutResID, null));}/** * Set listener on window dismissed. *  * @param listener */public void setOnDismissListener(PopupWindow.OnDismissListener listener) {mWindow.setOnDismissListener(listener);  }/** * Dismiss the popup window. */public void dismiss() {mWindow.dismiss();}}

 

 

Popu调用时在onCreate使用如下:

   

PopuItem userItem = new PopuItem(ID_USER, "用户", getResources().getDrawable(R.drawable.child_image));PopuItem grounpItem = new PopuItem(ID_GROUNP, "群组", getResources().getDrawable(R.drawable.user_group));          //use setSticky(true) to disable PopuJar dialog being dismissed after an item is clicked        userItem.setSticky(true);//create PopuJar. Use PopuJar.VERTICAL or PopuJar.HORIZONTAL param to define layout final PopuJar mPopu = new PopuJar(this, PopuJar.VERTICAL);//add action items into PopuJarmPopu.addPopuItem(userItem);mPopu.addPopuItem(grounpItem);



显示popu:

mPopu.show(v); //v表示显示在那个view下面

 

 

 源码:

http://download.csdn.net/detail/xyz_lmn/4674374

参考:

http://code.google.com/p/simple-quickactions/

/**
* @author 张兴业
*  http://blog.csdn.net/xyz_lmn
*  iOS入门群:83702688
*  android开发进阶群:241395671
*  我的新浪微博:@张兴业TBOW
*/

原创粉丝点击