【android】利用getViewTreeObserver().addOnGlobalLayoutListener()获得一个视图的高度

本文介绍了如何在Android中使用ViewTreeObserver的OnGlobalLayoutListener接口在视图树全局布局改变时获取一个view的实际高度,同时提到了OnGlobalLayoutListener可能会被多次触发,并强调了注销监听器的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们知道在oncreate中View.getWidth和View.getHeight无法获得一个view的高度和宽度,这是因为View组件布局要在onResume回调后完成。所以现在需要使用getViewTreeObserver().addOnGlobalLayoutListener()来获得宽度或者高度。这是获得一个view的宽度和高度的方法之一。

OnGlobalLayoutListener 是ViewTreeObserver的内部类,当一个视图树的布局发生改变时,可以被ViewTreeObserver监听到,这是一个注册监听视图树的观察者(observer),在视图树的全局事件改变时得到通知。ViewTreeObserver不能直接实例化,而是通过getViewTreeObserver()获得。

除了OnGlobalLayoutListener ,ViewTreeObserver还有如下内部类:


interface  ViewTreeObserver.OnGlobalFocusChangeListe ner
当在一个视图树中的焦点状态发生改变时,所要调用的回调函数的接口类

interface  ViewTreeObserver.OnGlobalLayoutListener
当在一个视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变时,所要调用的回调函数的接口类

interface  ViewTreeObserver.OnPreDrawListener
当一个视图树将要绘制时,所要调用的回调函数的接口类

interface  ViewTreeObserver.OnScrollChangedListener
当一个视图树中的一些组件发生滚动时,所要调用的回调函数的接口类

interface  ViewTreeObserver.OnTouchModeChangeListene r
当一个视图树的触摸模式发生改变时,所要调用的回调函数的接口类


其中,我们可以利用OnGlobalLayoutListener来获得一个视图的真实高度

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. int mHeaderViewHeight;  
  2. mHeaderView.getViewTreeObserver().addOnGlobalLayoutListener(  
  3.         new OnGlobalLayoutListener() {  
  4.             @Override  
  5.             public void onGlobalLayout() {  
  6.                                                                                                                                                                                                                                         
  7.                 mHeaderViewHeight = mHeaderView.getHeight();  
  8.                 getViewTreeObserver()  
  9.                         .removeGlobalOnLayoutListener(this);  
  10.             }  
  11.         });  

但是需要注意的是 OnGlobalLayoutListener 可能会被多次触发,因此在得到了高度之后,要将 OnGlobalLayoutListener 注销掉。



转自:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/0731/1640.html

package com.ailuxi.luxitrans; import android.app.Activity; import android.graphics.Rect; import android.os.Bundle; import android.view.MotionEvent; import android.view.View; import android.view.WindowManager; import android.view.inputmethod.InputMethodManager; import android.widget.EditText; import android.widget.FrameLayout; import android.widget.ImageButton; import android.widget.Button; import androidx.appcompat.app.AppCompatActivity; public class MainActivity extends AppCompatActivity { private EditText textInputEditText; private Button recordButton; private boolean isTextMode = true; // 默认文字模式 @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); getWindow().setSoftInputMode( WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE | WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN ); setContentView(R.layout.activity_main); SoftHideKeyBoardUtil(this); } // // 在onCreate()中调用 // public static void assistActivity(Activity activity) { // SoftHideKeyBoardUtil(activity); // } private View mChildOfContent; private int usableHeightPrevious; private void SoftHideKeyBoardUtil(Activity activity) { FrameLayout content = activity.findViewById(android.R.id.content); mChildOfContent = content.getChildAt(0); mChildOfContent.getViewTreeObserver().addOnGlobalLayoutListener(this::possiblyResizeChildOfContent); } private void possiblyResizeChildOfContent() { int usableHeightNow = computeUsableHeight(); if (usableHeightNow != usableHeightPrevious) { int usableHeightSansKeyboard = mChildOfContent.getRootView().getHeight(); int heightDifference = usableHeightSansKeyboard - usableHeightNow; if (heightDifference > (usableHeightSansKeyboard / 4)) { // 键盘弹出时调整布局高度 mChildOfContent.getLayoutParams().height = usableHeightSansKeyboard - heightDifference; } else { // 键盘收起时恢复高度 mChildOfContent.getLayoutParams().height = usableHeightSansKeyboard; } mChildOfContent.requestLayout(); usableHeightPrevious = usableHeightNow; } } private int computeUsableHeight() { Rect rect = new Rect(); mChildOfContent.getWindowVisibleDisplayFrame(rect); return rect.bottom - rect.top; } } 确实生效了但是键盘弹出时布局跟键盘之间有空隙如何适配
最新发布
07-22
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值