android:duplicateParentState

本文详细解析了Android中ViewGroup的子View如何通过`android:duplicateParentState`属性来同步父View的状态变化,包括状态改变的传递机制、`refreshDrawableState()`和`drawableStateChanged()`方法的作用,以及如何影响子View的背景绘制和重绘过程。

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

子控件 childView ,以View为例;

父控件 parantView 以ViewGroup为例;

childView设置了android:duplicateParentState 或 setDuplicateParentStateEnabled 为true

实际上是调用

setFlags(enabled ?DUPLICATE_PARENT_STATE : 0, DUPLICATE_PARENT_STATE);把标志位保存到mViewFlags中.

当 parantView状态改变,比如setSelected 或 setPressed,会调用refreshDrawableState();

    public void refreshDrawableState() {

        mPrivateFlags |= PFLAG_DRAWABLE_STATE_DIRTY;

        drawableStateChanged();


        ViewParent parent = mParent;

        if (parent !=null) {

            parent.childDrawableStateChanged(this);

        }

    }

可以看到,它里面其实调用了drawableStateChanged();

    protected void drawableStateChanged() {

        super.drawableStateChanged();


        if ((mGroupFlags &FLAG_NOTIFY_CHILDREN_ON_DRAWABLE_STATE_CHANGE) != 0) {

            if ((mGroupFlags &FLAG_ADD_STATES_FROM_CHILDREN) != 0) {

                thrownew IllegalStateException("addStateFromChildren cannot be enabled if a"

                        + " child has duplicateParentState set to true");

            }


            final View[] children =mChildren;

            final intcount = mChildrenCount;


            for (inti = 0; i < count;i++) {

                final View child = children[i];

                if ((child.mViewFlags & DUPLICATE_PARENT_STATE) != 0) {

   //在这里调用了childView的refreshDrawableState();

                    child.refreshDrawableState();

                }

            }

        }

    }

parantView是通过标志位FLAG_NOTIFY_CHILDREN_ON_DRAWABLE_STATE_CHANGE来判断执行refreshDrawableState();

而标志位是在addViewInner里面设置的,子view如果设置了DUPLICATE_PARENT_STATE,父view则会设置FLAG_NOTIFY_CHILDREN_ON_DRAWABLE_STATE_CHANGE.

private void addViewInner(Viewchild, int index, LayoutParamsparams,boolean preventRequestLayout) {

...

        if ((child.mViewFlags & DUPLICATE_PARENT_STATE) == DUPLICATE_PARENT_STATE) {

            mGroupFlags |=FLAG_NOTIFY_CHILDREN_ON_DRAWABLE_STATE_CHANGE;

        }


...

}

childView的refreshDrawableState()和parantView一样,也调用了drawableStateChanged()方法

    protected void drawableStateChanged() {

        Drawable d = mBackground;

        if (d != null && d.isStateful()) {

            d.setState(getDrawableState());

        }

    }

setState会调用onStateChange,以StateListDrawable为例,onStateChange会调用selectDrawable()改变当前的mCurrDrawable 和调用invalidateSelf();

    public void invalidateSelf() {

        final Callback callback = getCallback();

        if (callback !=null) {

            callback.invalidateDrawable(this);

        }

    }

callback回到view的invalidateDrawable()方法

    public void invalidateDrawable(Drawabledrawable) {

        if (verifyDrawable(drawable)) {

            final Rect dirty =drawable.getBounds();

            final intscrollX = mScrollX;

            final intscrollY = mScrollY;


            invalidate(dirty.left +scrollX, dirty.top +scrollY,

                    dirty.right +scrollX, dirty.bottom +scrollY);

        }

    }

请求view重新绘制.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值