Android Activity中获取View和屏幕的宽高

在Activity已经启动完之后,如果需要获取一个View的宽和高,大部分初学者都试过在onCreate、onStart、onResume中获取,然而获取的结果均为 0。这是因为View的measure过程和Activity的生命周期并不是同步执行的,因此无法保证在onCreate、onStart、onResume中获取宽高时View已经测量完了,如果没有测量完,获得的宽高就是0。
我们用以下几个方法解决这个问题:

一、Activity/View#onWindowFocusChanged
onWindowFocusChanged的含义:View已经初始化完毕了,宽高已经准备好了,这个时候获取宽高是没有问题的。当Activity的当前Window获得或失去焦点时会回调此方法,也就是说当Activity暂停执行和继续执行都会回调此方法,即这个方法会被频繁调用。我们一般在第一次获取焦点时获取宽高,代码如下:

private boolean isFirstFocus = true;
@Override
public void onWindowFocusChanged(boolean hasFocus) {
    super.onWindowFocusChanged(hasFocus);
    if (hasFocus&&isFirstFocus){
        isFirstFocus = false;
        int width = view.getMeasuredWidth();
        int height = view.getMeasuredHeight();
    }
}

二、view.post(runnable)

利用 Handler 通信机制,通过post将添加一个 Runnable到message queue的队尾,当View初始化完成之后,Looper会调用此runnable,然后通知UI线程。代码如下:

@Override
protected void onStart() {
   super.onStart();
   view.post(new Runnable() {
   @Override
   public void run() {
       int width = view.getMeasuredWidth();
       int height = view.getMeasuredHeight();
   }
   });
}

三、ViewTreeObserver

当View树状态发生改变,或者View树内部的view的可见性发生改变时,onGlobalLayout会被回调,所以这也是获取宽高的一个很好的时机。伴随着View树的状态的改变,onGlobalLayout会被调用多次,因此可在第一次调用完后,移除监听事件。代码如下:

@Override
protected void onStart() {
Logger.e("onStart");
super.onStart();
ViewTreeObserver viewTreeObserver = view.getViewTreeObserver();
viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
    @Override
    public void onGlobalLayout() {
        view.removeOnLayoutChangeListener(this);
        int width = view.getMeasuredWidth();
        int height = view.getMeasuredHeight();
    }
});
}

四、View#addOnLayoutChangeListener

监听 View的onLayout()的绘制过程,一旦宽高发生变化就会回调onLayoutChange方法。因此可在第一次调用完后,移除监听事件。代码如下:

view.addOnLayoutChangeListener(new View.OnLayoutChangeListener() {
    @Override
    public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
        view.removeOnLayoutChangeListener(this);
        Logger.e("w/h:" + view.getWidth() + "-" + view.getHeight());
    }
});

有时候我们并不需要View的宽高,而是想获取屏幕的宽高,通常有以下几种方式可以获取:

方法一:通过WindowManager获取

WindowManager wm = (WindowManager) this.getSystemService(Context.WINDOW_SERVICE);
int width = wm.getDefaultDisplay().getWidth();
int height = wm.getDefaultDisplay().getHeight();

方法二:通过WindowManager获取

WindowManager wm1 = this.getWindowManager();
int width1 = wm1.getDefaultDisplay().getWidth();
int height1 = wm1.getDefaultDisplay().getHeight();

方法三:通过WindowManager获取

WindowManager manager = this.getWindowManager();
DisplayMetrics outMetrics = new DisplayMetrics();
manager.getDefaultDisplay().getMetrics(outMetrics);
int width = outMetrics.widthPixels;
int height = outMetrics.heightPixels;

方法四:通过Resources 获取

Resources resources = this.getResources();
DisplayMetrics dm = resources.getDisplayMetrics();
float density = dm.density;
int width = dm.widthPixels;
int height = dm.heightPixels;

发表回复

后才能评论