Android 自動換行的LinearLayout

在咱們開發過程當中會常常碰見一些客戶要求可是Android系統又不提供的效果,這時咱們只能本身動手去實現它,或者從網絡上借鑑他人的資源,本着用別人不如本身會作的心態,在此我總結了一下Android中如何實現自動換行的LinearLayout。html

在本文中,說是LinearLayout實際上是繼承自GroupView,在這裏主要重寫了兩個方法,onMeasure、onLayout方法,下面我對此加以介紹。(代碼中使用了AttributeSet,因爲時間問題再也不予以介紹)。java

1.     onMeasure是幹什麼的?

在ViewGroup的建立過程當中,onMeasure是在onLayout以前的,因此在此先對onMeasure進行介紹,onMeasure方法是計算子控件與父控件在屏幕中所佔長寬大小的,onMeasure傳入兩個參數——widthMeasureSpec和heightMeasureSpec. 這兩個參數指明控件可得到的空間以及關於這個空間描述的元數據.android

int withMode = MeasureSpec.getMode(widthMeasureSpec);
int withSize = MeasureSpec.getSize(widthMeasureSpec);
int heightMode = MeasureSpec.getMode(heightMeasureSpec);
int heightSize = MeasureSpec.getSize(heightMeasureSpec);網絡

Mode有3種模式分別是UNSPECIFIED, EXACTLY和AT_MOST,若是是AT_MOST,Size表明的是最大可得到的空間;若是是EXACTLY,Size表明的是精確的尺寸;若是是UNSPECIFIED,就是你想要多少就有多少。通過代碼測試就知道,當咱們設置width或height爲fill_parent時,容器在佈局時調用子 view的measure方法傳入的模式是EXACTLY,由於子view會佔據剩餘容器的空間,因此它大小是肯定的。而當設置爲 wrap_content時,容器傳進去的是AT_MOST, 表示子view的大小最可能是多少,這樣子view會根據這個上限來設置本身的尺寸。當子view的大小設置爲精確值時,容器傳入的是EXACTLY。app

2.     onLayout是幹什麼的?

與onMesaure相比,onLayout更加容易理解,它的做用就是調座位,就是把全部的子View根據不一樣的須要,經過View. layout(int l, int t, int r, int b)方法指定它所在的位置。dom

3.     解決問題

只要對onMeasure和onLayout加以理解,對於該篇所要實現的功能就再也不難以實現,下面貼上代碼,並在代碼中講解。ide

WaroLinearLayout.java佈局

 

[java] view plain copy測試

  1. public class WarpLinearLayout extends ViewGroup {  
  2.   
  3.     private Type mType;  
  4.     private List<WarpLine> mWarpLineGroup;  
  5.   
  6.     public WarpLinearLayout(Context context) {  
  7.         this(context, null);  
  8.     }  
  9.   
  10.     public WarpLinearLayout(Context context, AttributeSet attrs) {  
  11.         this(context, attrs, R.style.WarpLinearLayoutDefault);  
  12.     }  
  13.   
  14.     public WarpLinearLayout(Context context, AttributeSet attrs, int defStyleAttr) {  
  15.         super(context, attrs, defStyleAttr);  
  16.         mType = new Type(context, attrs);  
  17.     }  
  18.   
  19.     @Override  
  20.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
  21.         int withMode = MeasureSpec.getMode(widthMeasureSpec);  
  22.         int withSize = MeasureSpec.getSize(widthMeasureSpec);  
  23.         int heightMode = MeasureSpec.getMode(heightMeasureSpec);  
  24.         int heightSize = MeasureSpec.getSize(heightMeasureSpec);  
  25.         int with = 0;  
  26.         int height = 0;  
  27.         int childCount = getChildCount();  
  28.         /** 
  29.          * 在調用childView。getMeasre以前必須先調用該行代碼,用於對子View大小的測量 
  30.          */  
  31.         measureChildren(widthMeasureSpec, heightMeasureSpec);  
  32.         /** 
  33.          * 計算寬度 
  34.          */  
  35.         switch (withMode) {  
  36.             case MeasureSpec.EXACTLY:  
  37.                 with = withSize;  
  38.                 break;  
  39.             case MeasureSpec.AT_MOST:  
  40.                 for (int i = 0; i < childCount; i++) {  
  41.                     if (i != 0) {  
  42.                         with += mType.horizontal_Space;  
  43.                     }  
  44.                     with += getChildAt(i).getMeasuredWidth();  
  45.                 }  
  46.                 with += getPaddingLeft() + getPaddingRight();  
  47.                 with = with > withSize ? withSize : with;  
  48.                 break;  
  49.             case MeasureSpec.UNSPECIFIED:  
  50.                 for (int i = 0; i < childCount; i++) {  
  51.                     if (i != 0) {  
  52.                         with += mType.horizontal_Space;  
  53.                     }  
  54.                     with += getChildAt(i).getMeasuredWidth();  
  55.                 }  
  56.                 with += getPaddingLeft() + getPaddingRight();  
  57.                 break;  
  58.             default:  
  59.                 with = withSize;  
  60.                 break;  
  61.   
  62.         }  
  63.         /** 
  64.          * 根據計算出的寬度,計算出所須要的行數 
  65.          */  
  66.         WarpLine warpLine = new WarpLine();  
  67.         /** 
  68.          * 不可以在定義屬性時初始化,由於onMeasure方法會屢次調用 
  69.          */  
  70.         mWarpLineGroup = new ArrayList<WarpLine>();  
  71.         for (int i = 0; i < childCount; i++) {  
  72.             if (warpLine.lineWidth + getChildAt(i).getMeasuredWidth() + mType.horizontal_Space > with) {  
  73.                 if (warpLine.lineView.size() == 0) {  
  74.                     warpLine.addView(getChildAt(i));  
  75.                     mWarpLineGroup.add(warpLine);  
  76.                     warpLine = new WarpLine();  
  77.                 } else {  
  78.                     mWarpLineGroup.add(warpLine);  
  79.                     warpLine = new WarpLine();  
  80.                     warpLine.addView(getChildAt(i));  
  81.                 }  
  82.             } else {  
  83.                 warpLine.addView(getChildAt(i));  
  84.             }  
  85.         }  
  86.         /** 
  87.          * 添加最後一行 
  88.          */  
  89.         if (warpLine.lineView.size() > 0 && !mWarpLineGroup.contains(warpLine)) {  
  90.             mWarpLineGroup.add(warpLine);  
  91.         }  
  92.         /** 
  93.          * 計算寬度 
  94.          */  
  95.         height = getPaddingTop() + getPaddingBottom();  
  96.         for (int i = 0; i < mWarpLineGroup.size(); i++) {  
  97.             if (i != 0) {  
  98.                 height += mType.vertical_Space;  
  99.             }  
  100.             height += mWarpLineGroup.get(i).height;  
  101.         }  
  102.         switch (heightMode) {  
  103.             case MeasureSpec.EXACTLY:  
  104.                 height = heightSize;  
  105.                 break;  
  106.             case MeasureSpec.AT_MOST:  
  107.                 height = height > heightSize ? heightSize : height;  
  108.                 break;  
  109.             case MeasureSpec.UNSPECIFIED:  
  110.                 break;  
  111.             default:  
  112.                 break;  
  113.         }  
  114.         setMeasuredDimension(with, height);  
  115.     }  
  116.   
  117.     @Override  
  118.     protected void onLayout(boolean changed, int l, int t, int r, int b) {  
  119.         t = getPaddingTop();  
  120.         for (int i = 0; i < mWarpLineGroup.size(); i++) {  
  121.             int left = getPaddingLeft();  
  122.             WarpLine warpLine = mWarpLineGroup.get(i);  
  123.             int lastWidth = getMeasuredWidth() - warpLine.lineWidth;  
  124.             for (int j = 0; j < warpLine.lineView.size(); j++) {  
  125.                 View view = warpLine.lineView.get(j);  
  126.                 if (isFull()) {//須要充滿當前行時  
  127.                     view.layout(left, t, left + view.getMeasuredWidth() + lastWidth / warpLine.lineView.size(), t + view.getMeasuredHeight());  
  128.                     left += view.getMeasuredWidth() + mType.horizontal_Space + lastWidth / warpLine.lineView.size();  
  129.                 } else {  
  130.                     switch (getGrivate()) {  
  131.                         case 0://右對齊  
  132.                             view.layout(left + lastWidth, t, left + lastWidth + view.getMeasuredWidth(), t + view.getMeasuredHeight());  
  133.                             break;  
  134.                         case 2://居中對齊  
  135.                             view.layout(left + lastWidth / 2, t, left + lastWidth / 2 + view.getMeasuredWidth(), t + view.getMeasuredHeight());  
  136.                             break;  
  137.                         default://左對齊  
  138.                             view.layout(left, t, left + view.getMeasuredWidth(), t + view.getMeasuredHeight());  
  139.                             break;  
  140.                     }  
  141.                     left += view.getMeasuredWidth() + mType.horizontal_Space;  
  142.                 }  
  143.             }  
  144.             t += warpLine.height + mType.vertical_Space;  
  145.         }  
  146.     }  
  147.   
  148.     /** 
  149.      * 用於存放一行子View 
  150.      */  
  151.     private final class WarpLine {  
  152.         private List<View> lineView = new ArrayList<View>();  
  153.         /** 
  154.          * 當前行中所須要佔用的寬度 
  155.          */  
  156.         private int lineWidth = getPaddingLeft() + getPaddingRight();  
  157.         /** 
  158.          * 該行View中所須要佔用的最大高度 
  159.          */  
  160.         private int height = 0;  
  161.   
  162.         private void addView(View view) {  
  163.             if (lineView.size() != 0) {  
  164.                 lineWidth += mType.horizontal_Space;  
  165.             }  
  166.             height = height > view.getMeasuredHeight() ? height : view.getMeasuredHeight();  
  167.             lineWidth += view.getMeasuredWidth();  
  168.             lineView.add(view);  
  169.         }  
  170.     }  
  171.   
  172.     /** 
  173.      * 對樣式的初始化 
  174.      */  
  175.     private final static class Type {  
  176.         /* 
  177.          *對齊方式 right 0,left 1,center 2 
  178.         */  
  179.         private int grivate;  
  180.         /** 
  181.          * 水平間距,單位px 
  182.          */  
  183.         private float horizontal_Space;  
  184.         /** 
  185.          * 垂直間距,單位px 
  186.          */  
  187.         private float vertical_Space;  
  188.         /** 
  189.          * 是否自動填滿 
  190.          */  
  191.         private boolean isFull;  
  192.   
  193.         Type(Context context, AttributeSet attrs) {  
  194.             if (attrs == null) {  
  195.                 return;  
  196.             }  
  197.             TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.WarpLinearLayout);  
  198.             grivate = typedArray.getInt(R.styleable.WarpLinearLayout_grivate, grivate);  
  199.             horizontal_Space = typedArray.getDimension(R.styleable.WarpLinearLayout_horizontal_Space, horizontal_Space);  
  200.             vertical_Space = typedArray.getDimension(R.styleable.WarpLinearLayout_vertical_Space, vertical_Space);  
  201.             isFull = typedArray.getBoolean(R.styleable.WarpLinearLayout_isFull, isFull);  
  202.         }  
  203.     }  
  204.   
  205.     public int getGrivate() {  
  206.         return mType.grivate;  
  207.     }  
  208.   
  209.     public float getHorizontal_Space() {  
  210.         return mType.horizontal_Space;  
  211.     }  
  212.   
  213.     public float getVertical_Space() {  
  214.         return mType.vertical_Space;  
  215.     }  
  216.   
  217.     public boolean isFull() {  
  218.         return mType.isFull;  
  219.     }  
  220.   
  221.     public void setGrivate(int grivate) {  
  222.         mType.grivate = grivate;  
  223.     }  
  224.   
  225.     public void setHorizontal_Space(float horizontal_Space) {  
  226.         mType.horizontal_Space = horizontal_Space;  
  227.     }  
  228.   
  229.     public void setVertical_Space(float vertical_Space) {  
  230.         mType.vertical_Space = vertical_Space;  
  231.     }  
  232.   
  233.     public void setIsFull(boolean isFull) {  
  234.         mType.isFull = isFull;  
  235.     }  
  236.   
  237.     /** 
  238.      * 每行子View的對齊方式 
  239.      */  
  240.     public final static class Gravite {  
  241.         public final static int RIGHT = 0;  
  242.         public final static int LEFT = 1;  
  243.         public final static int CENTER = 2;  
  244.     }  
  245. }  

attrs.xmlthis

 

 

[html] view plain copy

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <resources>  
  3.     <declare-styleable name="WarpLinearLayout">  
  4.         <attr name="grivate" format="enum"><!--對齊方式 !-->  
  5.             <enum name="right" value="0"></enum>  
  6.             <enum name="left" value="1"></enum>  
  7.             <enum name="center" value="2"></enum>  
  8.         </attr>  
  9.         <attr name="horizontal_Space" format="dimension"></attr>  
  10.         <attr name="vertical_Space" format="dimension"></attr>  
  11.         <attr name="isFull" format="boolean"></attr>  
  12.     </declare-styleable>  
  13. </resources>  

 

 

WarpLinearLayoutDefault

[html] view plain copy

  1. <style name="WarpLinearLayoutDefault">  
  2.        <item name="grivate">left</item>  
  3.        <item name="horizontal_Space">20dp</item>  
  4.        <item name="vertical_Space">20dp</item>  
  5.        <item name="isFull">false</item>  
  6.    </style>  


MainActivity.java

 

 

[java] view plain copy

  1. public class MainActivity extends Activity {  
  2.     private Button btn;  
  3.     private WarpLinearLayout warpLinearLayout;  
  4.   
  5.     @Override  
  6.     protected void onCreate(Bundle savedInstanceState) {  
  7.         super.onCreate(savedInstanceState);  
  8.         setContentView(R.layout.activity_main);  
  9.         btn = (Button) findViewById(R.id.btn);  
  10.         warpLinearLayout = (WarpLinearLayout) findViewById(R.id.warpLinearLayout);  
  11.         btn.setOnClickListener(new View.OnClickListener() {  
  12.             @Override  
  13.             public void onClick(View v) {  
  14.                 int n = new Random().nextInt(10) + 5;  
  15.                 StringBuffer stringBuffer = new StringBuffer();  
  16.                 Random random = new Random();  
  17.                 Log.i("WarpLinearLayout","n="+n);  
  18.                 for (int i = 0; i < n; i++) {  
  19.                     stringBuffer.append((char)(65+random.nextInt(26)));  
  20.                     Log.i("WarpLinearLayout", "StringBuffer=" + stringBuffer.toString());  
  21.                 }  
  22.                 TextView tv = new TextView(MainActivity.this);  
  23.                 tv.setText(stringBuffer.toString()+"000");  
  24.                 tv.setBackgroundResource(R.drawable.radius_backgroup_yellow);  
  25.                 tv.setPadding(10,10,10,10);  
  26.                 warpLinearLayout.addView(tv);  
  27.             }  
  28.         });  
  29.     }  
  30. }  

activity_main.xml

 

 

[html] view plain copy

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     xmlns:app="http://schemas.android.com/apk/res-auto"  
  4.     xmlns:tools="http://schemas.android.com/tools"  
  5.     android:layout_width="match_parent"  
  6.     android:layout_height="match_parent"  
  7.     android:paddingBottom="@dimen/activity_vertical_margin"  
  8.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  9.     android:paddingRight="@dimen/activity_horizontal_margin"  
  10.     android:paddingTop="@dimen/activity_vertical_margin"  
  11.     tools:context=".MainActivity">  
  12.   
  13.     <Button  
  14.         android:id="@+id/btn"  
  15.         android:layout_width="match_parent"  
  16.         android:layout_height="wrap_content"  
  17.         android:gravity="center"  
  18.         android:text="add"  
  19.         android:textSize="20dp" />  
  20.   
  21.     <com.example.customview.viewgroup.WarpLinearLayout  
  22.         android:id="@+id/warpLinearLayout"  
  23.         android:layout_width="match_parent"  
  24.         android:layout_height="wrap_content"  
  25.         android:layout_below="@id/btn"  
  26.         android:background="#FF00FF00"  
  27.         android:padding="10dp"  
  28.         app:grivate="right"  
  29.         app:horizontal_Space="10dp"  
  30.         app:isFull="false"  
  31.         app:vertical_Space="10dp"></com.example.customview.viewgroup.WarpLinearLayout>  
  32. </RelativeLayout>  

 

 

運行效果圖以下:

相關文章
相關標籤/搜索