Android LinearLayout实现自动换行效果

在我们开发过程中会经常遇见一些客户要求但是Android系统又不提供的效果,这时我们只能自己动手去实现它,或者从网络上借鉴他人的资源,本着用别人不如自己会做的心态,在此我总结了一下Android中如何实现自动换行的LinearLayout。

在本文中,说是LinearLayout其实是继承自GroupView,在这里主要重写了两个方法,onMeasure、onLayout方法,下面我对此加以介绍。(代码中使用了AttributeSet,由于时间问题不再予以介绍)。

1. onMeasure是干什么的?

在ViewGroup的创建过程中,onMeasure是在onLayout之前的,所以在此先对onMeasure进行介绍,onMeasure方法是计算子控件与父控件在屏幕中所占长宽大小的,onMeasure传入两个参数——widthMeasureSpec和heightMeasureSpec. 这两个参数指明控件可获得的空间以及关于这个空间描述的元数据.

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。

2. onLayout是干什么的?

与onMesaure相比,onLayout更加容易理解,它的作用就是调座位,就是把所有的子View根据不同的需要,通过View. layout(int l, int t, int r, int b)方法指定它所在的位置。

3. 解决问题

只要对onMeasure和onLayout加以理解,对于该篇所要实现的功能就不再难以实现,下面贴上代码,并在代码中讲解。

WaroLinearLayout.java

public class WarpLinearLayout extends ViewGroup {

  private Type mType;
  private List<WarpLine> mWarpLineGroup;

  public WarpLinearLayout(Context context) {
    this(context, null);
  }

  public WarpLinearLayout(Context context, AttributeSet attrs) {
    this(context, attrs, R.style.WarpLinearLayoutDefault);
  }

  public WarpLinearLayout(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    mType = new Type(context, attrs);
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int withMode = MeasureSpec.getMode(widthMeasureSpec);
    int withSize = MeasureSpec.getSize(widthMeasureSpec);
    int heightMode = MeasureSpec.getMode(heightMeasureSpec);
    int heightSize = MeasureSpec.getSize(heightMeasureSpec);
    int with = 0;
    int height = 0;
    int childCount = getChildCount();
    /**
     * 在调用childView。getMeasre之前必须先调用该行代码,用于对子View大小的测量
     */
    measureChildren(widthMeasureSpec, heightMeasureSpec);
    /**
     * 计算宽度
     */
    switch (withMode) {
      case MeasureSpec.EXACTLY:
        with = withSize;
        break;
      case MeasureSpec.AT_MOST:
        for (int i = 0; i < childCount; i++) {
          if (i != 0) {
            with += mType.horizontal_Space;
          }
          with += getChildAt(i).getMeasuredWidth();
        }
        with += getPaddingLeft() + getPaddingRight();
        with = with > withSize ? withSize : with;
        break;
      case MeasureSpec.UNSPECIFIED:
        for (int i = 0; i < childCount; i++) {
          if (i != 0) {
            with += mType.horizontal_Space;
          }
          with += getChildAt(i).getMeasuredWidth();
        }
        with += getPaddingLeft() + getPaddingRight();
        break;
      default:
        with = withSize;
        break;

    }
    /**
     * 根据计算出的宽度,计算出所需要的行数
     */
    WarpLine warpLine = new WarpLine();
    /**
     * 不能够在定义属性时初始化,因为onMeasure方法会多次调用
     */
    mWarpLineGroup = new ArrayList<WarpLine>();
    for (int i = 0; i < childCount; i++) {
      if (warpLine.lineWidth + getChildAt(i).getMeasuredWidth() + mType.horizontal_Space > with) {
        if (warpLine.lineView.size() == 0) {
          warpLine.addView(getChildAt(i));
          mWarpLineGroup.add(warpLine);
          warpLine = new WarpLine();
        } else {
          mWarpLineGroup.add(warpLine);
          warpLine = new WarpLine();
          warpLine.addView(getChildAt(i));
        }
      } else {
        warpLine.addView(getChildAt(i));
      }
    }
    /**
     * 添加最后一行
     */
    if (warpLine.lineView.size() > 0 && !mWarpLineGroup.contains(warpLine)) {
      mWarpLineGroup.add(warpLine);
    }
    /**
     * 计算宽度
     */
    height = getPaddingTop() + getPaddingBottom();
    for (int i = 0; i < mWarpLineGroup.size(); i++) {
      if (i != 0) {
        height += mType.vertical_Space;
      }
      height += mWarpLineGroup.get(i).height;
    }
    switch (heightMode) {
      case MeasureSpec.EXACTLY:
        height = heightSize;
        break;
      case MeasureSpec.AT_MOST:
        height = height > heightSize ? heightSize : height;
        break;
      case MeasureSpec.UNSPECIFIED:
        break;
      default:
        break;
    }
    setMeasuredDimension(with, height);
  }

  @Override
  protected void onLayout(boolean changed, int l, int t, int r, int b) {
    t = getPaddingTop();
    for (int i = 0; i < mWarpLineGroup.size(); i++) {
      int left = getPaddingLeft();
      WarpLine warpLine = mWarpLineGroup.get(i);
      int lastWidth = getMeasuredWidth() - warpLine.lineWidth;
      for (int j = 0; j < warpLine.lineView.size(); j++) {
        View view = warpLine.lineView.get(j);
        if (isFull()) {//需要充满当前行时
          view.layout(left, t, left + view.getMeasuredWidth() + lastWidth / warpLine.lineView.size(), t + view.getMeasuredHeight());
          left += view.getMeasuredWidth() + mType.horizontal_Space + lastWidth / warpLine.lineView.size();
        } else {
          switch (getGrivate()) {
            case 0://右对齐
              view.layout(left + lastWidth, t, left + lastWidth + view.getMeasuredWidth(), t + view.getMeasuredHeight());
              break;
            case 2://居中对齐
              view.layout(left + lastWidth / 2, t, left + lastWidth / 2 + view.getMeasuredWidth(), t + view.getMeasuredHeight());
              break;
            default://左对齐
              view.layout(left, t, left + view.getMeasuredWidth(), t + view.getMeasuredHeight());
              break;
          }
          left += view.getMeasuredWidth() + mType.horizontal_Space;
        }
      }
      t += warpLine.height + mType.vertical_Space;
    }
  }

  /**
   * 用于存放一行子View
   */
  private final class WarpLine {
    private List<View> lineView = new ArrayList<View>();
    /**
     * 当前行中所需要占用的宽度
     */
    private int lineWidth = getPaddingLeft() + getPaddingRight();
    /**
     * 该行View中所需要占用的最大高度
     */
    private int height = 0;

    private void addView(View view) {
      if (lineView.size() != 0) {
        lineWidth += mType.horizontal_Space;
      }
      height = height > view.getMeasuredHeight() ? height : view.getMeasuredHeight();
      lineWidth += view.getMeasuredWidth();
      lineView.add(view);
    }
  }

  /**
   * 对样式的初始化
   */
  private final static class Type {
    /*
     *对齐方式 right 0,left 1,center 2
    */
    private int grivate;
    /**
     * 水平间距,单位px
     */
    private float horizontal_Space;
    /**
     * 垂直间距,单位px
     */
    private float vertical_Space;
    /**
     * 是否自动填满
     */
    private boolean isFull;

    Type(Context context, AttributeSet attrs) {
      if (attrs == null) {
        return;
      }
      TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.WarpLinearLayout);
      grivate = typedArray.getInt(R.styleable.WarpLinearLayout_grivate, grivate);
      horizontal_Space = typedArray.getDimension(R.styleable.WarpLinearLayout_horizontal_Space, horizontal_Space);
      vertical_Space = typedArray.getDimension(R.styleable.WarpLinearLayout_vertical_Space, vertical_Space);
      isFull = typedArray.getBoolean(R.styleable.WarpLinearLayout_isFull, isFull);
    }
  }

  public int getGrivate() {
    return mType.grivate;
  }

  public float getHorizontal_Space() {
    return mType.horizontal_Space;
  }

  public float getVertical_Space() {
    return mType.vertical_Space;
  }

  public boolean isFull() {
    return mType.isFull;
  }

  public void setGrivate(int grivate) {
    mType.grivate = grivate;
  }

  public void setHorizontal_Space(float horizontal_Space) {
    mType.horizontal_Space = horizontal_Space;
  }

  public void setVertical_Space(float vertical_Space) {
    mType.vertical_Space = vertical_Space;
  }

  public void setIsFull(boolean isFull) {
    mType.isFull = isFull;
  }

  /**
   * 每行子View的对齐方式
   */
  public final static class Gravite {
    public final static int RIGHT = 0;
    public final static int LEFT = 1;
    public final static int CENTER = 2;
  }
}

attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
  <declare-styleable name="WarpLinearLayout">
    <attr name="grivate" format="enum"><!--对齐方式 !-->
      <enum name="right" value="0"></enum>
      <enum name="left" value="1"></enum>
      <enum name="center" value="2"></enum>
    </attr>
    <attr name="horizontal_Space" format="dimension"></attr>
    <attr name="vertical_Space" format="dimension"></attr>
    <attr name="isFull" format="boolean"></attr>
  </declare-styleable>
</resources>

WarpLinearLayoutDefault

<style name="WarpLinearLayoutDefault">
    <item name="grivate">left</item>
    <item name="horizontal_Space">20dp</item>
    <item name="vertical_Space">20dp</item>
    <item name="isFull">false</item>
</style>

MainActivity.java

public class MainActivity extends Activity {
  private Button btn;
  private WarpLinearLayout warpLinearLayout;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    btn = (Button) findViewById(R.id.btn);
    warpLinearLayout = (WarpLinearLayout) findViewById(R.id.warpLinearLayout);
    btn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        int n = new Random().nextInt(10) + 5;
        StringBuffer stringBuffer = new StringBuffer();
        Random random = new Random();
        Log.i("WarpLinearLayout","n="+n);
        for (int i = 0; i < n; i++) {
          stringBuffer.append((char)(65+random.nextInt(26)));
          Log.i("WarpLinearLayout", "StringBuffer=" + stringBuffer.toString());
        }
        TextView tv = new TextView(MainActivity.this);
        tv.setText(stringBuffer.toString()+"000");
        tv.setBackgroundResource(R.drawable.radius_backgroup_yellow);
        tv.setPadding(10,10,10,10);
        warpLinearLayout.addView(tv);
      }
    });
  }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:app="http://schemas.android.com/apk/res-auto"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:paddingBottom="@dimen/activity_vertical_margin"
  android:paddingLeft="@dimen/activity_horizontal_margin"
  android:paddingRight="@dimen/activity_horizontal_margin"
  android:paddingTop="@dimen/activity_vertical_margin"
  tools:context=".MainActivity">

  <Button
    android:id="@+id/btn"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:gravity="center"
    android:text="add"
    android:textSize="20dp" />

  <com.example.customview.viewgroup.WarpLinearLayout
    android:id="@+id/warpLinearLayout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_below="@id/btn"
    android:background="#FF00FF00"
    android:padding="10dp"
    app:grivate="right"
    app:horizontal_Space="10dp"
    app:isFull="false"
    app:vertical_Space="10dp"></com.example.customview.viewgroup.WarpLinearLayout>
</RelativeLayout>

运行效果图如下:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android自定View流式布局根据文字数量换行

    本文实例为大家分享了Android根据文字数量换行的具体代码,供大家参考,具体内容如下 //主页 定义数据框 package com.example.customwaterfallviewgroup; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.EditText; import java.util

  • Android LinearLayout实现自动换行

    由于前段时间项目中使用到了自动换行的线性布局,本来打算用表格布局在里面一个个的用Java代码添加ImageView的,但是添加的View控件是不确定的,因为得靠服务器的数据返回,就这样手动用Java代码画布局的方式就这样夭折了,因为在表哥布局中我无法确定一行显示多少个ImageView的数目,所以无法动态添加,最后自能自己去看看那种能够换行的线性布局了,线性布局比较不好的是不能自动换行,也就是当设置LinearLayout的orentation 设置为vertical 为竖直方向也就是只有一列,

  • 浅谈Android textview文字对齐换行的问题

    今天忽然发现android项目中的文字排版参差不齐的情况非常严重,不得不想办法解决一下. 经过研究之后,终于找到了textview自动换行导致混乱的原因了----半角字符与全角字符混乱所致!一般情况下,我们输入的数字.字母以及英文标点都是半角,所以占位无法确定. 它们与汉字的占位大大的不同,由于这个原因,导致很多文字的排版都是参差不齐的. 对此我找到了两种办法可以解决这个问题: 1. 将textview中的字符全角化. 即将所有的数字.字母及标点全部转为全角字符,使它们与汉字同占两个字节,这样就

  • Android中用StaticLayout实现文本绘制自动换行详解

    前言 使用Canvas的drawText绘制文本是不会自动换行的,即使一个很长很长的字符串,drawText也只显示一行,超出部分被隐藏在屏幕之外.可以逐个计算每个字符的宽度,通过一定的算法将字符串分割成多个部分,然后分别调用drawText一部分一部分的显示, 但是这种显示效率会很低. StaticLayout是android中处理文字换行的一个工具类, StaticLayout 已经实现了文本绘制换行处理,下面是如何使用 StaticLayout 的例子: 示例代码 package com.

  • Android LinearLayout实现自动换行效果

    在我们开发过程中会经常遇见一些客户要求但是Android系统又不提供的效果,这时我们只能自己动手去实现它,或者从网络上借鉴他人的资源,本着用别人不如自己会做的心态,在此我总结了一下Android中如何实现自动换行的LinearLayout. 在本文中,说是LinearLayout其实是继承自GroupView,在这里主要重写了两个方法,onMeasure.onLayout方法,下面我对此加以介绍.(代码中使用了AttributeSet,由于时间问题不再予以介绍). 1. onMeasure是干什

  • Android实现左右滑动效果的方法详解

    本示例演示在Android中实现图片左右滑动效果. 关于滑动效果,在Android中用得比较多,本示例实现的滑动效果是使用ViewFlipper来实现的,当然也可以使用其它的View来实现.接下来就让我们开始实现这种效果.为了方便大家理解,我们先来看一下效果图:主要效果图如下图:    接下来我们看一下程序结构图: MainActivity文件中代码: 复制代码 代码如下: package com.android.flip;import android.app.Activity;import a

  • Android实现底部弹窗效果

    本文实例为大家分享了Android实现底部弹窗效果的具体代码,供大家参考,具体内容如下 源代码地址:https://github.com/luoye123/Box 东西很简单,我就直接亮代码了: 1.activity_main.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/

  • Android实现带动画效果的可点击展开TextView

    本文为大家分享了Android实现带动画效果的可点击展开TextView 制作代码,效果图: 收起(默认)效果: 点击展开后的效果: 源码: 布局: <?xml version="1.0" encoding="utf-8"?> <LinearLayout android:id="@+id/activity_main" xmlns:android="http://schemas.android.com/apk/res/a

  • 简单实现Android弹出菜单效果

    本文实例为大家分享了Android弹出菜单效果的具体代码,供大家参考,具体内容如下 功能描述:用户单击按钮弹出菜单.当用户选择一个菜单项,会触发MenuItemClick事件并让弹出的菜单消失:如果用户在菜单外单击,则直接消失弹出的菜单.当菜单消失时,会引发DismissEvent事件(利用此事件可在菜单消失时做一些后续处理). 1.运行效果 2.添加菜单项 在Resources文件夹下添加一个menu子文件夹,然后在此子文件夹下添加一个名为demo07_popup_menu.xml的文件: <

  • Android实现蒙板效果

    本文实例为大家分享了Android实现蒙板效果的相关代码,供大家参考,具体内容如下 1.不保留标题栏蒙板的实现 效果: 原理: 1.弹窗时,设置背景窗体的透明度 2.取消弹窗时,恢复背景窗体的透明度 关键代码: private void popupWindows(List<String> list){ //产生背景变暗效果 WindowManager.LayoutParams lp=getWindow().getAttributes(); lp.alpha = 0.4f; getWindow(

  • Android 实现伸缩布局效果示例代码

    最近项目实现下面的图示的效果,本来想用listview+gridview实现,但是貌似挺麻烦的于是就用flowlayout 来addview实现添加伸缩的效果,实现也比较简单. mainActivity 布局 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

  • Android实现顶部悬浮效果

    本文实例为大家分享了Android实现顶部悬浮效果的具体代码,供大家参考,具体内容如下 效果图 布局 <?xml version="1.0" encoding="utf-8"?> <android.support.v4.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http:

  • Android GridView实现动画效果实现代码

     Android GridView实现动画效果 项目中用到的一些动画,GridView的Item依次从屏幕外飞入到相应位置,附上相关代码: MainActivity.Java package com.mundane.gridanimationdemo; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.view.View; import android.view.an

随机推荐