android之自定义ViewGroup实现自动换行布局

泡在网上的日子 / 文 发表于2013-03-05 22:54 次阅读 ViewGroup,android

viewgroup简单说就是可以装view的view.

ViewGroup其实是一个视图容器,而他本身也是一个view,因为他是一个容器,所以很多布局控件都是由继承他才得以实现的,比如LinearLayout,FrameLayout等布局。

如果我们需要一个苛刻的布局,比如能根据子view的宽度自动换行的布局,那么我们通过自定义ViewGroup来实现是最好的方法。

 

今天遇到一个问题,就是需要一个可以自动根据一行中view的宽度自动换行的布局,网上找了下,没有相关的例子,但是找到了思路:自定义一个viewgroup,然后在onlayout文件里面自动检测view的右边缘的横坐标值,和你的view的parent view的况度判断是否换行显示view就可以了。因为代码比较简单,就不多说了:

public class MyViewGroup extends ViewGroup { 
  
    private final static String TAG = "MyViewGroup"; 
  
    private final static int VIEW_MARGIN = 2; 
  
    public MyViewGroup(Context context) { 
  
        super(context); 
  
    } 
  
    @Override 
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
  
        Log.d(TAG, "widthMeasureSpec = " + widthMeasureSpec 
                + " heightMeasureSpec" + heightMeasureSpec); 
  
        for (int index = 0; index < getChildCount(); index++) { 
  
            final View child = getChildAt(index); 
  
            // measure 
  
            child.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED); 
  
        } 
  
        super.onMeasure(widthMeasureSpec, heightMeasureSpec); 
  
    } 
  
    @Override 
    protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) { 
  
        Log.d(TAG, "changed = " + arg0 + " left = " + arg1 + " top = " + arg2 
                + " right = " + arg3 + " botom = " + arg4); 
  
        final int count = getChildCount(); 
  
        int row = 0;// which row lay you view relative to parent 
  
        int lengthX = arg1; // right position of child relative to parent 
  
        int lengthY = arg2; // bottom position of child relative to parent 
  
        for (int i = 0; i < count; i++) { 
  
            final View child = this.getChildAt(i); 
  
            int width = child.getMeasuredWidth(); 
  
            int height = child.getMeasuredHeight(); 
  
            lengthX += width + VIEW_MARGIN; 
  
            lengthY = row * (height + VIEW_MARGIN) + VIEW_MARGIN + height 
                    + arg2; 
  
            // if it can't drawing on a same line , skip to next line 
  
            if (lengthX > arg3) { 
  
                lengthX = width + VIEW_MARGIN + arg1; 
  
                row++; 
  
                lengthY = row * (height + VIEW_MARGIN) + VIEW_MARGIN + height 
                        + arg2; 
  
            } 
  
            child.layout(lengthX - width, lengthY - height, lengthX, lengthY); 
  
        } 
  
    } 
  
}

 这里有个地方要注意,那就要明白ViewGroup的绘图流程:ViewGroup绘制包括两个步骤:1.measure 2.layout

  在两个步骤中分别调用回调函数:1.onMeasure() 2.onLayout()

  1.onMeasure() 在这个函数中,ViewGroup会接受childView的请求的大小,然后通过childView的 measure(newWidthMeasureSpec, heightMeasureSpec)函数存储到childView中,以便childView的getMeasuredWidth() andgetMeasuredHeight() 的值可以被后续工作得到。

  2.onLayout() 在这个函数中,ViewGroup会拿到childView的getMeasuredWidth() andgetMeasuredHeight(),用来布局所有的childView。

  3.View.MeasureSpec 与 LayoutParams 这两个类,是ViewGroup与childView协商大小用的。其中,View.MeasureSpec是ViewGroup用来部署 childView用的, LayoutParams是childView告诉ViewGroup 我需要多大的地方。

  4.在View 的onMeasure的最后要调用setMeasuredDimension()这个方法存储View的大小,这个方法决定了当前View的大小。


  效果图:

2011112423260436.png

收藏 赞 (13) 踩 (3)
上一篇:Android中的WiFi开发简明教程
Android提供了android.net.wifi包供我们操作WiFi。首先介绍一下几个类: WifiManager :提供了管理WiFI连接的大部分API ScanResult :已经检测出的接入点(包括接入点的地址、名称、身份认证、频率、信号强度) WifiConfiguration :WiFi连接的网络配置(包
下一篇:Android 4.0日历(calendar)源码分析之月视图
1.日期跳转问题: 这个问题是在处理点击today按钮之后,新建日程的日期不是当天,而是当月的1号这个故障时发现的。 对于总是所在月的1号,这个很好处理,因为google原生就是这样设计的。只需在SimpleDayPickerFragment.java文件中将goto()函数的 mFirstDay