有時需要在 onCreate()
方法中獲取某個 View 組件的寬度和高度,而直接調用 getWidth()
、getHeight()
、getMeasuredWidth()
、getMeasuredHeight()
方法只會得到 0。
實現方法
一、使用 View.measure 測量 View
該方法測量的寬度和高度可能與視圖繪制完成后的真實的寬度和高度不一致。
int width = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.UNSPECIFIED);
int height = View.MeasureSpec.makeMeasureSpec(0,
View.MeasureSpec.UNSPECIFIED);
view.measure(width, height);
view.getMeasuredWidth(); // 獲取寬度
view.getMeasuredHeight(); // 獲取高度
二、使用 ViewTreeObserver. OnPreDrawListener 監聽事件
在視圖將要繪制時調用該監聽事件,會被調用多次,因此獲取到視圖的寬度和高度后要移除該監聽事件。
view.getViewTreeObserver().addOnPreDrawListener(
new ViewTreeObserver.OnPreDrawListener() {
@Override
public boolean onPreDraw() {
view.getViewTreeObserver().removeOnPreDrawListener(this);
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
return true;
}
});
三、使用 ViewTreeObserver. OnGlobalLayoutListener 監聽事件
在布局發生改變或者某個視圖的可視狀態發生改變時調用該事件,會被多次調用,因此需要在獲取到視圖的寬度和高度后執行 remove 方法移除該監聽事件。
view.getViewTreeObserver().addOnGlobalLayoutListener(
new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
if (Build.VERSION.SDK_INT >= 16) {
view.getViewTreeObserver()
.removeOnGlobalLayoutListener(this);
}
else {
view.getViewTreeObserver()
.removeGlobalOnLayoutListener(this);
}
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
}
});
四、重寫 View 的 onSizeChanged 方法
在視圖的大小發生改變時調用該方法,會被多次調用,因此獲取到寬度和高度后需要考慮禁用掉代碼。
該實現方法需要繼承 View,且多次被調用,不建議使用。
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
}
五、重寫 View 的 onLayout 方法
該方法會被多次調用,獲取到寬度和高度后需要考慮禁用掉代碼。
該實現方法需要繼承 View,且多次被調用,不建議使用。
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
}
六、使用 View.OnLayoutChangeListener 監聽事件(API >= 11)
在視圖的 layout 改變時調用該事件,會被多次調用,因此需要在獲取到視圖的寬度和高度后執行 remove 方法移除該監聽事件。
view.addOnLayoutChangeListener(
new View.OnLayoutChangeListener() {
@Override
public void onLayoutChange(View v, int l, int t, int r, int b,
int oldL, int oldT, int oldR, int oldB) {
view.removeOnLayoutChangeListener(this);
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
}
});
七、使用 View.post() 方法
Runnable 對象中的方法會在 View 的 measure、layout 等事件完成后觸發。
UI 事件隊列會按順序處理事件,在 setContentView() 被調用后,事件隊列中會包含一個要求重新 layout 的 message,所以任何 post 到隊列中的 Runnable 對象都會在 Layout 發生變化后執行。
該方法只會執行一次,且邏輯簡單,建議使用。
view.post(new Runnable() {
@Override
public void run() {
view.getWidth(); // 獲取寬度
view.getHeight(); // 獲取高度
}
});
參考:
stackoverflow: getWidth() and getHeight() of View returns 0