1. 程式人生 > >Android封裝之PopupWindow(一) (帶demo)

Android封裝之PopupWindow(一) (帶demo)

一、前言

Android的對話方塊有兩種:PopupWindow和AlertDialog。它們的不同點在於:

AlertDialog的位置固定,但是PopupWindow的位置可以隨意,
AlertDialog是非阻塞執行緒的,而PopupWindow是阻塞執行緒的。

PopupWindow的位置按照有無偏移分,可以分為偏移和無偏移兩種;按照參照物的不同,可以分為相對於某個控制元件(Anchor錨)和相對於父控制元件。具體如下

showAsDropDown(View anchor):相對某個控制元件的位置(正左下方),無偏移
showAsDropDown(View anchor, int xoff, int yoff):相對某個控制元件的位置,有偏移
showAtLocation(View parent, int gravity, int x, int y):相對於父控制元件的位置(例如正中央Gravity.CENTER,下方Gravity.BOTTOM等),可以設定偏移或無偏移

常用的方法

setContentView(View contentView):設定PopupWindow顯示的View

getContentView():獲得PopupWindow顯示的View

showAsDropDown(View anchor):相對某個控制元件的位置(正左下方),無偏移

showAsDropDown(View anchor, int xoff, int yoff):相對某個控制元件的位置,有偏移

showAtLocation(View parent, int gravity, int x, int y)

: 相對於父控制元件的位置(例如正中央

Gravity.CENTER,下方Gravity.BOTTOM等),可以設定偏移或無偏移 PS:parent這個引數只要是activity中的view就可以了!

setWidth/setHeight:設定寬高,也可以在構造方法那裡指定好寬高, 除了可以寫具體的值,還可以用
WRAP_CONTENT或MATCH_PARENT, popupWindow的width和height屬性直接和第一層View相對應。

setFocusable(true):設定焦點,PopupWindow彈出後,所有的觸屏和物理按鍵都由PopupWindows 處理。其他任何事件的響應都必須發生在PopupWindow消失之後,(home 等系統層面的事件除外)。 比如這樣一個PopupWindow出現的時候,按back鍵首先是讓PopupWindow消失,第二次按才是退出 activity,準確的說是想退出activity你得首先讓PopupWindow消失,因為不併是任何情況下按back PopupWindow都會消失,必須在PopupWindow設定了背景的情況下 。

setAnimationStyle(int):設定動畫效果

二、常用在哪些地方呢?

下面的場景是最常見的。

這裡寫圖片描述

三、程式碼

package com.xuhong.popupwindowsutils_sample;


import android.app.Activity;
import android.content.Context;
import android.graphics.Rect;
import android.graphics.drawable.ColorDrawable;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.PopupWindow;

public class PopupWindowUtils extends PopupWindow implements View.OnClickListener {

    private View mMenuView;

    private OnPopWindowClickListener listener;

    private Activity activity;

      PopupWindowUtils(Activity activity, OnPopWindowClickListener listener) {
        this.activity =activity;
        initView(activity, listener);
    }

     public void show(){
        Rect rect = new Rect();
          /*
           * getWindow().getDecorView()得到的View是Window中的最頂層View,可以從Window中獲取到該View,
           * 然後該View有個getWindowVisibleDisplayFrame()方法可以獲取到程式顯示的區域,
           * 包括標題欄,但不包括狀態列。
           */
        activity.getWindow().getDecorView().getWindowVisibleDisplayFrame(rect);
        int winHeight = activity.getWindow().getDecorView().getHeight();
        this.showAtLocation(activity.getWindow().getDecorView(), Gravity.BOTTOM, 0, winHeight - rect.bottom);
    }

    private void initView(Activity activity, OnPopWindowClickListener listener) {
        //設定按鈕監聽
        this.listener = listener;
        initViewSetting(activity);
        //設定SelectPicPopupWindow的View
        this.setContentView(mMenuView);
        //設定SelectPicPopupWindow彈出窗體的寬
        this.setWidth(ViewGroup.LayoutParams.MATCH_PARENT);
        //設定SelectPicPopupWindow彈出窗體的高
        this.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT);
        //設定SelectPicPopupWindow彈出窗體可點選
        this.setFocusable(true);
        //設定SelectPicPopupWindow彈出窗體動畫效果
        this.setAnimationStyle(R.style.dialog_style);
        //例項化一個ColorDrawable顏色為半透明
        ColorDrawable dw = new ColorDrawable(0xb0000000);
        //設定SelectPicPopupWindow彈出窗體的背景
        this.setBackgroundDrawable(dw);
        //mMenuView新增OnTouchListener監聽判斷獲取觸屏位置如果在選擇框外面則銷燬彈出框
        mMenuView.setOnTouchListener(new View.OnTouchListener() {
            public boolean onTouch(View v, MotionEvent event) {
                int height = mMenuView.findViewById(R.id.pop_layout).getTop();
                int y = (int) event.getY();
                if (event.getAction() == MotionEvent.ACTION_UP) {
                    if (y < height) {
                        dismiss();
                    }
                }
                return true;
            }
        });
    }

    //彈窗(關於、修改資料、退出登入)
    private void initViewSetting(Activity context) {

        Button btnAbout, btnExit, btnUpdate, btnCancel;
        LayoutInflater inflater = (LayoutInflater) context
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        mMenuView = inflater.inflate(R.layout.layout_popwindow_dialog_setting, null);

        btnAbout = (Button) mMenuView.findViewById(R.id.btn_setting_about);
        btnExit = (Button) mMenuView.findViewById(R.id.btn_setting_exit);
        btnUpdate = (Button) mMenuView.findViewById(R.id.btn_setting_update);
        btnCancel = (Button) mMenuView.findViewById(R.id.btn_setting_cancel);

        btnUpdate.setOnClickListener(this);
        btnCancel.setOnClickListener(this);
        btnExit.setOnClickListener(this);
        btnAbout.setOnClickListener(this);
    }
    @Override
    public void onClick(View v) {
        listener.onPopWindowClickListener(v);
        dismiss();
    }
    //介面
    public interface OnPopWindowClickListener {
        void onPopWindowClickListener(View view);
    }
}

在activity呼叫

package com.xuhong.popupwindowsutils_sample;

import android.graphics.Rect;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.Gravity;
import android.view.View;

public class MainActivity extends AppCompatActivity implements PopupWindowUtils.OnPopWindowClickListener {

    private static final String TAG ="MainActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    public void btn_open(View view) {
        //一行程式碼
        new PopupWindowUtils(this, this).show();
    }
    @Override
    public void onPopWindowClickListener(View view) {
        switch (view.getId()) {
            case R.id.btn_setting_about:
                Log.i(TAG,"關於");
                break;
            case R.id.btn_setting_exit:
                Log.i(TAG,"退出");
                break;
            case R.id.btn_setting_update:
                Log.i(TAG,"更新");
                break;
            case R.id.btn_setting_cancel:
                Log.i(TAG,"取消");
                break;
        }
    }
}

後續新增內容:我還會新增介面popupWindows顯示的位置。