今天在使用自定义view时,之前一直只知道父布局负责margin,子布局负责padding。在使用自定义viewgroup时候也没有考虑过margin这个,今天突然想试试,发现了通过child.marginTop
获取到的为0,这个时候去看这是一个kotlin的扩展方法,如下
inline val View.marginTop: Int
get() = (layoutParams as? MarginLayoutParams)?.topMargin ?: 0
所以原因多半是来源于获取到的layoutParams并非是MarginLayouParams,然后查看网上的一些博客,都是默认获取到layoutParams强转为MarginLayoutParams然后去活动margin的,遂调试了一下,发现获取到的并非MarginLayouParams,所以无赖只能去查看源码。
想到xml都是通过inflate去解析然后生成对应的view的,所以去查看LayoutInflate的inflate方法中
public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
View view = tryInflatePrecompiled(resource, res, root, attachToRoot);
if (view != null) {
return view;
}
}
可以看到调用了tryInflatePrecompiled方法去解析xml获取这个view,在其中有生成对应的layoutParams的方法然后添加给view
ViewGroup.LayoutParams params = root.generateLayoutParams(attrs);
if (attachToRoot) {
root.addView(view, params);
} else {
view.setLayoutParams(params);
}
这个attachToRoot参数也很熟悉了,就是是否添加到父布局中还是只应用他的布局参数。而另外的这个generateLayoutParams方法就只会生成默认的layoutParams参数,所以自定义的viewgroup需要重写这个方法,不然就不能使用margin
public LayoutParams generateLayoutParams(AttributeSet attrs) {
return new LayoutParams(getContext(), attrs);
}
所以一般可以直接重写这个方法,然后返回MarginLayoutParams即可,在MarginLayoutParams类中也可以看到对传入的attrs进行解析获取margin的过程。
另外还有在viewgroup中使用ondraw去绘制需要的界面时,需要在初始化函数中设置setWillNotDraw(false)
否则是不会调用onDraw方法的