PopWindow的理解

最近又要做一个自己的框, 本来想直接用原来的,但是觉得自己对Popwindow理解比较差,自己再看看。

原来的代码来自于网上,直接修改了部分变成自己的。

自己试试用系统的实现,增加自己的理解。


首先,先调用Popwindow

final PopupWindow mPW = new PopupWindow(this);
mPW.setContentView(LayoutInflater.from(this).inflate(R.layout.eelog, null));
mPW.showAsDropDown(v);

发现没有显示, 可能是没有设置宽和高(系统没有默认?)。

final PopupWindow mPW = new PopupWindow(this);
mPW.setContentView(LayoutInflater.from(this).inflate(R.layout.eelog, null));
mPW.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setHeight(WindowManager.LayoutParams.WRAP_CONTENT)
mPW.showAsDropDown(v);
添加后,可以显示了, 但是点击没有响应, back就直接退出了, 说明自己的Popwindow没有得到焦点。自己手动设置下焦点。

final PopupWindow mPW = new PopupWindow(this);
mPW.setContentView(LayoutInflater.from(this).inflate(R.layout.eelog, null));
mPW.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setFocusable(true);
mPW.showAsDropDown(v);

就可以获得焦点了。

但是,无论点击什么地方,自己的Popwindow都会消失,系统分辨不出点击的是里面还是外面。

看了下,自己添加了一个mPW.setOutsideTouchable(true);(系统默认是false)

意思是,可以点击外面,并且点击外面时,Popwindow会消失  , 点击里面不会消失   

【原来看网上是通过setTouchInterceptor设置消失,自己试了试貌似setTouchInterceptor的event不适合】

final PopupWindow mPW = new PopupWindow(this);
mPW.setContentView(LayoutInflater.from(this).inflate(R.layout.eelog, null));
mPW.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setFocusable(true);
mPW.setOutsideTouchable(true);
mPW.showAsDropDown(v);

这样就可以点击外面消失了。


最后,我们重要的是 对点击Popwindow的监听,在 Popwindow里面没有找到OnClick的监听,这下急了,(最开始做的都是很多item,一条或者多条的,再对每个item监听,没有做对这个mpw的监听)

看了下Popwindow的部分源码,field中的一些字段。

    private boolean mIsShowing;
    private boolean mIsDropdown;

    private View mContentView;
    private View mPopupView;
    private boolean mFocusable;
    private int mInputMethodMode = INPUT_METHOD_FROM_FOCUSABLE;
    private int mSoftInputMode = WindowManager.LayoutParams.SOFT_INPUT_STATE_UNCHANGED;
    private boolean mTouchable = true;
    private boolean mOutsideTouchable = false;
    private boolean mClippingEnabled = true;
    private int mSplitTouchEnabled = -1;
    private boolean mLayoutInScreen;
    private boolean mClipToScreen;
    private boolean mAllowScrollingAnchorParent = true;
    private boolean mLayoutInsetDecor = false;
    private boolean mNotTouchModal;

发现了的 View的contentView 。   有 

public View getContentView() {
    return mContentView;
}

方法得到ContentView。自己再试试,

final PopupWindow mPW = new PopupWindow(this);
mPW.setContentView(LayoutInflater.from(this).inflate(R.layout.eelog, null));
mPW.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
mPW.setFocusable(true);
mPW.setOutsideTouchable(true);
mPW.getContentView().setOnClickListener(new OnClickListener() {
	public void onClick(View v) {
		// TODO Auto-generated method stub
		Toast.makeText(ShowDialogActivity.this, "里面", 100).show();
		mPW.dismiss();
	}
});
mPW.showAsDropDown(v);

基本的点击事件完毕。


-----------------------------  


一些简单的事件处理完了, 下面就是怎么样来弹出Popwindow的问题了, 先了解下

mPW.showAsDropDown(v); 

的实现  (v是点击的view)


看下Popwindow, 

mPW.showAsDropDown(v);调用的是 

public void showAsDropDown(View anchor) {
    showAsDropDown(anchor, 0, 0);
}

而 showAsDropDown(anchor, 0, 0); 

public void showAsDropDown(View anchor, int xoff, int yoff) {
    if (isShowing() || mContentView == null) {
        return;
    }

    registerForScrollChanged(anchor, xoff, yoff);

    mIsShowing = true;
    mIsDropdown = true;

    WindowManager.LayoutParams p = createPopupLayout(anchor.getWindowToken());
    preparePopup(p);

    updateAboveAnchor(findDropDownPosition(anchor, p, xoff, yoff));

    if (mHeightMode < 0) p.height = mLastHeight = mHeightMode;
    if (mWidthMode < 0) p.width = mLastWidth = mWidthMode;

    p.windowAnimations = computeAnimationResource();

    invokePopup(p);
}

里面比较复杂,自己不了解,只能靠猜了  

(先判断错误,否则不给与响应return      在注册scroll的监听, 这只状态showing和dropdown都为true,装载对应的view如果有background,返回对应的容器的view,再执行动画 )


preparePopup方法

    private void preparePopup(WindowManager.LayoutParams p) {
        if (mContentView == null || mContext == null || mWindowManager == null) {
            throw new IllegalStateException("You must specify a valid content view by "
                    + "calling setContentView() before attempting to show the popup.");
        }

        if (mBackground != null) {
            final ViewGroup.LayoutParams layoutParams = mContentView.getLayoutParams();
            int height = ViewGroup.LayoutParams.MATCH_PARENT;
            if (layoutParams != null &&
                    layoutParams.height == ViewGroup.LayoutParams.WRAP_CONTENT) {
                height = ViewGroup.LayoutParams.WRAP_CONTENT;
            }

            // when a background is available, we embed the content view
            // within another view that owns the background drawable
            PopupViewContainer popupViewContainer = new PopupViewContainer(mContext);
            PopupViewContainer.LayoutParams listParams = new PopupViewContainer.LayoutParams(
                    ViewGroup.LayoutParams.MATCH_PARENT, height
            );
            popupViewContainer.setBackgroundDrawable(mBackground);
            popupViewContainer.addView(mContentView, listParams);

            mPopupView = popupViewContainer;
        } else {
            mPopupView = mContentView;
        }
        mPopupWidth = p.width;
        mPopupHeight = p.height;
    }

由于源码比较复杂,自己肯定不能纠结这里了,改动可能只能通过自己的代码来修改对应的响应了

只能通过 showAsDropDown(anchor, 0, 0);  入手了,

mPW.showAsDropDown(v);   等价于  mPW.showAsDropDown(v, 0, 0);  

现在显示是对应点击的view的左下角  为相对位置的

对应的参数  x 是向右多少像素, y 是向下多少像素


现在的问题是:

1 当点击的地方很靠下的时候,如果不能scroll,就会显示不全或者变形。

解决:  当位置比较考下的时候, 就让 y 的值为 -aaa   ,向上偏移

2 当点击的地方很靠右的时候,就会显示不全或者变形。

解决: 当位置比较靠右的时候, 就让 x的值为 -aaa, 想左偏移。


先解决1, 同理就可以解决2了

先要得到  我里面的v, 也就是点击的view的位置, 通过v.getLocationOnScreen(location); 来给对应的2维数组赋值,得到对应位置 (矩形框Rect)

这里 v 用anchor表示点击的view

int[] location 	= new int[2];
anchor.getLocationOnScreen(location);
Rect anchorRect = new Rect(location[0], 
			location[1], 
			location[0] + anchor.getWidth(), 
			location[1] + anchor.getHeight());
这样就可以得到对应的位置了,

再要得到对应的显示的Popwindow的大小

这个要看是设置的dip大小, 还是设置的background的图片了

分别是

		int height = mPW.getContentView().getMeasuredHeight();
		int width = mPW.getContentView().getMeasuredWidth();
或者

		int height = mPW.getContentView().getBackground().getIntrinsicHeight();
		int width = mPW.getContentView().getBackground().getIntrinsicWidth();

再得到屏幕的大小

		int screenWidth 	= windowManager.getDefaultDisplay().getWidth();
		int screenHeight	= windowManager.getDefaultDisplay().getHeight();



-----------------------------  有时间再继续吧







                
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值