onLayout方法是ViewGroup中子View的布局方法,用于放置子View的位置。放置子View很简单,只需在重写onLayout方法,然后获取子View的实例,调用子View的layout方法实现布局。在实际开发中,一般要配合onMeasure测量方法一起使用。
onLayout方法:
@Override protected abstract void onLayout(boolean changed, int l, int t, int r, int b);
layout方法:
public void layout(int l, int t, int r, int b);
平常开发所用到RelativeLayout、LinearLayout、FrameLayout...这些都是继承ViewGroup的布局。这些布局的实现都是通过都实现ViewGroup的onLayout方法,只是实现方法不一样而已。
下面是一个自定义ViewGroup的Demo,用onLayout和layout实现子View的水平放置,间隔是20px
public class MyViewGroup extends ViewGroup { // 子View的水平间隔 private final static int padding = 20; public MyViewGroup(Context context, AttributeSet attrs) { super(context, attrs); // TODO Auto-generated constructor stub } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { // TODO Auto-generated method stub // 动态获取子View实例 for (int i = 0, size = getChildCount(); i < size; i++) { View view = getChildAt(i); // 放置子View,宽高都是100 view.layout(l, t, l + 100, t + 100); l += 100 + padding; } } }
Activity的XML布局:
<relativelayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:padding="10dp"> <com.example.layout.myviewgroup android:layout_width="match_parent" android:layout_height="100dp" android:background="#0000ff"> <view android:layout_width="match_parent" android:layout_height="match_parent" android:background="#ff0000"> <view android:layout_width="match_parent" android:layout_height="match_parent" android:background="#00ff00"> </view></view></com.example.layout.myviewgroup> </relativelayout>
PS:最关心的 只有两点 一个是参数,一个是在哪里可以重载这个函数(自定义View里面) 关于是用layout(),还是用super.onLayout()方法 具体看需求了,
下面是RecyclerView 自定义视图里面用到的一个onLayout()方法:
public class MyRecyclerView extends RecyclerView implements OnScrollListener { /** *记录当前的第一个View */ private View mCurrentView; /** * 滚动监听器 */ private OnItemScrollChangeListener mItemScrollChangeListener; /** * 设置监听器 * @param mItemScrollChangeListener */ public void setOnItemScrollChangeListener(OnItemScrollChangeListener mItemScrollChangeListener) { this.mItemScrollChangeListener = mItemScrollChangeListener; } /** * 自定义了一个接口方法,用于ScrollListener 每个item 滚动的点击响应事件 */ public interface OnItemScrollChangeListener { void onChange(View view, int position); } /** * 继承 RecyclerView 必须要重写的方法 * 构造函数,在这里设置 调用 ScrollListener的滚动事件 */ public MyRecyclerView(Context context, AttributeSet attrs) { super(context, attrs); //implements OnScrollListener 启动 this.setOnScrollListener(this); } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) {//左上角的left、top以及右下角right、bottom值 super.onLayout(changed, l, t, r, b); //重载该函数,实现的功能就下面实时监听当前的子View // 首先申明 这里是 水平混动 //需要实现的效果就是每次你滑动停止之后,srcollView 第一个View 展现在上面的大图片中 mCurrentView = getChildAt(0); if (mItemScrollChangeListener != null) { mItemScrollChangeListener.onChange(mCurrentView,getChildPosition(mCurrentView)); } } /** * implements OnScrollListener 必须要重载的方法 */ @Override public void onScrollStateChanged(int arg0) { } /** * implements OnScrollListener 必须要重载的方法 *滚动时,判断当前第一个View是否发生变化,发生才回调 */ @Override public void onScrolled(int arg0, int arg1) { View newView = getChildAt(0); if (mItemScrollChangeListener != null) { if (newView != null && newView != mCurrentView) { mCurrentView = newView ; mItemScrollChangeListener.onChange(mCurrentView,getChildPosition(mCurrentView)); } } } }只是拿其中一部分代码作为展示,
这是效果图:在任何一次的滑动的过程中,都是选择左边第一个作为上面大图片的展示,需要用到onLayout()方法