안 드 로 이 드 고수 의 길 그래 픽 시스템 (6) requestLayout 의 절차

더 읽 기
View 가 requestLayout 를 호출 할 때 현재 View 에 FORCE 를 설정 합 니 다.라 유 트 태그.뷰 파 렌 트 에 레이아웃 을 요청 합 니 다.이렇게 해서 이 View 부터 위로 계속 requestLayout.결국 뷰 루트 임 플 에 도 착 했 습 니 다.ViewParent 는 현재 전송 체인 입 니 다.[직책 체인 디자인 모델 참조]
 
첫걸음
ViewRootImpl 에서 레이아웃 을 요청 한 것 을 발 견 했 습 니 다.그럼 measure 방법 을 호출 하 겠 습 니 다.
measure 방법 현재 View 에 FORCE 가 있 는 지 확인라 유 트 태그.
있 으 면 재 measure 를 한다.그리고 태그 LAYOUT 설정REQUIRED。
public final void measure(int widthMeasureSpec, int heightMeasureSpec) {
        if ((mPrivateFlags & FORCE_LAYOUT) == FORCE_LAYOUT ||
                widthMeasureSpec != mOldWidthMeasureSpec ||
                heightMeasureSpec != mOldHeightMeasureSpec) {

            // first clears the measured dimension flag
            mPrivateFlags &= ~MEASURED_DIMENSION_SET;

            if (ViewDebug.TRACE_HIERARCHY) {
                ViewDebug.trace(this, ViewDebug.HierarchyTraceType.ON_MEASURE);
            }

            // measure ourselves, this should set the measured dimension flag back
            onMeasure(widthMeasureSpec, heightMeasureSpec);

            // flag not set, setMeasuredDimension() was not invoked, we raise
            // an exception to warn the developer
            if ((mPrivateFlags & MEASURED_DIMENSION_SET) != MEASURED_DIMENSION_SET) {
                throw new IllegalStateException("onMeasure() did not set the"
                        + " measured dimension by calling"
                        + " setMeasuredDimension()");
            }

            mPrivateFlags |= LAYOUT_REQUIRED;
        }

        mOldWidthMeasureSpec = widthMeasureSpec;
        mOldHeightMeasureSpec = heightMeasureSpec;
    }

 
 
두 번 째 단계.
 다음 layot 방법 에서 이 표 시 를 판단 합 니 다.이 표시 가 트 루 라면
그럼 꼭 onLayout 를 호출 하 겠 습 니 다.
onLayout 호출 후 LAYOUT 청소REQUIRED 태그.
layot 호출 후 FORCE 제거라 유 트 태그.
 @SuppressWarnings({"unchecked"})
    public void layout(int l, int t, int r, int b) {
        int oldL = mLeft;
        int oldT = mTop;
        int oldB = mBottom;
        int oldR = mRight;
        boolean changed = setFrame(l, t, r, b);
        if (changed || (mPrivateFlags & LAYOUT_REQUIRED) == LAYOUT_REQUIRED) {
            if (ViewDebug.TRACE_HIERARCHY) {
                ViewDebug.trace(this, ViewDebug.HierarchyTraceType.ON_LAYOUT);
            }

            onLayout(changed, l, t, r, b);
            mPrivateFlags &= ~LAYOUT_REQUIRED;

            ListenerInfo li = mListenerInfo;
            if (li != null && li.mOnLayoutChangeListeners != null) {
                ArrayList listenersCopy =
                        (ArrayList)li.mOnLayoutChangeListeners.clone();
                int numListeners = listenersCopy.size();
                for (int i = 0; i < numListeners; ++i) {
                    listenersCopy.get(i).onLayoutChange(this, l, t, r, b, oldL, oldT, oldR, oldB);
                }
            }
        }
        mPrivateFlags &= ~FORCE_LAYOUT;
    }

 
물론 이 과정 에서 형제 나 아버지 View 의 크기 에 영향 을 미 쳤 다 면 형제 나 아버지 View 도 layot / onLayout 를 호출 할 것 이다.requestLayout 를 호출 했 든 안 했 든지정 한 MeasureSpec 도 이 로 인해 변화 가 생 겼 다 면,
그럼 measure / on Measure 도 호출 됩 니 다.
 
상기 분석 을 통 해 requestlayot 를 호출 하면 measure 와 onMeasure, layot, onlayout, draw onDraw 가 모두 호출 되 는 것 으로 나 타 났 다.
 
많은 경우 에 requestLayout 는 호출 될 필요 가 없습니다.예 를 들 어 AbsoluteLayout 에 있 는 childView 를 자 리 를 옮 깁 니 다.현재 AbsoluteLayout 를 다시 배치 한 다음 invalidate 방법 으로 다시 그 릴 수 있 습 니 다.현재 View 에서 위로 올 라 가 는 전체 View 트 리 구조 가 아니 라 layout, onLayout, measure, onMeasure 를 다시 해 야 합 니 다.
이 럴 때 어 떡 하지?
 
한 가지 방법 은 onLayout 를 직접 호출 하 는 것 이다.그리고 invalidate 를 호출 하여 다시 그립 니 다.그리 기 효율 을 높 일 수 있 음 이 분명 하 다.부모 View 의 layot 구현 중 레이아웃 을 알 리 는 listener 입 니 다.그러나 listener 를 받 을 수 없 기 때문에 onlayout 을 호출 할 때 이 를 알 릴 수 없 는 것 도 이러한 결함 이다.
 
 
 
 
 

좋은 웹페이지 즐겨찾기