Android流式布局

总结流式布局经验,比较简单,就是先获取当前容器的宽高,遍历所有的子view,根据子view的宽度和子view间的距离来计算是否超出当前容器的宽度,如果超出宽度的话记录当前行的子view,和当前行的高度,如果不需要换行,就宽度累加,获取最大高度。获取到总高度和每行的子view后,开始设置每个子view的位置,为每个view设置布局属性


import java.util.ArrayList;
import java.util.List;

import android.content.Context;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;

public class CustomFlowLayout extends ViewGroup {

	public CustomFlowLayout(Context context, AttributeSet attrs, int defStyle) {
		super(context, attrs, defStyle);
	}

	public CustomFlowLayout(Context context, AttributeSet attrs) {
		this(context, attrs,0);
	}

	public CustomFlowLayout(Context context) {
		this(context,null);
	}
	
	@Override
	protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
		
		//获取宽度和高度的值和模式
		int sizeWidth = MeasureSpec.getSize(widthMeasureSpec);
		int modeWidth = MeasureSpec.getMode(widthMeasureSpec);
		
		int sizeHeight = MeasureSpec.getSize(heightMeasureSpec);
		int modeHeight = MeasureSpec.getMode(heightMeasureSpec);
		
		//wrap_content
		int width=0;
		int height=0;
		
		//记录每一行的宽度和高度
		int lineWidth = 0;
		int lineHeight = 0;
		
		//得到内部元素的个数
		int cCount = getChildCount();
		for(int i=0;i<cCount;i++){
			View child = getChildAt(i);
			//测量子View的宽和高
			measureChild(child, widthMeasureSpec, heightMeasureSpec);
			//重写了generateLayoutParams,获取到的是子控件布局的属性,可以获取属性中margin的值
			MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); 
			//子View占据的宽度
			int childWidth = child.getMeasuredWidth()+lp.leftMargin+lp.rightMargin;
			//子view占据的高度
			int childHeight = child.getMeasuredHeight()+lp.topMargin+lp.bottomMargin;
			
			//大于sizewidth换行
			if(lineWidth+childWidth>sizeWidth){
				//对比得到�?大的宽度
				width = Math.max(width, lineWidth);
				//重置记录linewidth的宽度
				lineWidth = childWidth;
				height += lineHeight;
				lineHeight = childHeight;
			}else{
				//未换行
				//叠加行宽
				lineWidth+=childWidth;
				//得到当前最大的高度
				lineHeight=Math.max(lineHeight, childHeight);
			}
		
			if(i==cCount-1){//最后一个控件
				width = Math.max(lineWidth, width);
				height +=lineHeight;
			}
		}
		
		//wrap_content
//		if(modeWidth==MeasureSpec.AT_MOST){
//			setMeasuredDimension(width, height);
//		}else{
//			setMeasuredDimension(sizeWidth, sizeHeight);
//		}
		//以上注释部分可以用下面代码替换
		
		//如果模式是EXACTLY(宽高为精确值) 用传入的控件宽高,否则使用自己测量的其控件宽高
		setMeasuredDimension(
				modeWidth==MeasureSpec.EXACTLY?sizeWidth:width, 
				modeHeight==MeasureSpec.EXACTLY?sizeHeight:height);
	}
	
	/**
	 * 存储所有的view
	 */
	private List<List<View>> mAllViews = new ArrayList<List<View>>();
	/**
	 * 每一行的高度
	 */
	private List<Integer> mLineHeight = new ArrayList<Integer>();
	
	/**
	 * 设置子view的宽高布局
	 */
	@Override
	protected void onLayout(boolean changed, int l, int t, int r, int b) {
		
		mAllViews.clear();
		mLineHeight.clear();
		//当前viewgroup的宽度
		int width = getWidth();
		
		//记录行宽度
		int lineWidth = 0;
		int lineHeight = 0;
		List<View> lineViews = new ArrayList<View>();
		int cCount = getChildCount();//子view的个数
		for(int i=0;i<cCount-1;i++){
			View child = getChildAt(i);
			MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
			//获取子view的宽度
			int childWidth = child.getMeasuredWidth();
			int childHeight = child.getMeasuredHeight();
		
			if(childWidth+lineWidth+lp.leftMargin+lp.rightMargin>width){
				//记录lineHeight
				mLineHeight.add(lineHeight);
				//记录当前的views
				mAllViews.add(lineViews);
				
				//重置下一行宽度
				lineWidth=0;
				lineHeight = childHeight+lp.topMargin+lp.bottomMargin;
				//重置views集合
				lineViews = new ArrayList<View>();
			}else{
				lineWidth+=childWidth+lp.leftMargin+lp.rightMargin;
				lineHeight = Math.max(lineHeight, childHeight+lp.topMargin+lp.bottomMargin);
				lineViews.add(child);
			}
		}//for  end
		
		//处理最后一个
		mLineHeight.add(lineHeight);
		mAllViews.add(lineViews);
		
		//设置子view的位置
		int left = 0;
		int top = 0;
		//行数
		int lineNum = mAllViews.size();
		
		for(int i=0;i<lineNum;i++){
			//当前行的所有的view
			lineViews = mAllViews.get(i);
			//当前控件
			lineHeight = mLineHeight.get(i);
			
			for(int j=0;j<lineViews.size();j++){
				View child = lineViews.get(j);
				if(child.getVisibility()==View.GONE){
					continue;
				}
				MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
				
				int lc = left + lp.leftMargin;
				int tc = top + lp.topMargin;
				int rc = lc + child.getMeasuredWidth();
				int bc = tc + child.getMeasuredHeight();
				
				//为子view进行布局
				child.layout(lc, tc, rc, bc);
				
				left+=child.getMeasuredWidth()+lp.leftMargin+lp.rightMargin;
			}
			//遍历完一行left设为0;高度增加lineheight
			left=0;
			top += lineHeight;
		}
	}
	
	/**
	 * 与当前viewgroup对应的布局
	 */
	@Override
	public LayoutParams generateLayoutParams(AttributeSet attrs) {
		return new MarginLayoutParams(getContext(),attrs);
	}

}



测试Activity:

import com.example.testcustomviewgroup.R;

import android.os.Bundle;
import android.app.Activity;
import android.view.ViewGroup.MarginLayoutParams;
import android.widget.Button;

public class MainActivity extends Activity {

	CustomFlowLayout vg_test;
	String[] text = {"pulltofresh","visiable","hello world","linearlayout","goodidea","test","abcd","free","Android"}; 
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		vg_test = (CustomFlowLayout) findViewById(R.id.vg_test);
		initData();
	}
	private void initData() {
		for(int i=0;i<text.length;i++){
			Button btnButton = new Button(this);
			MarginLayoutParams lp = new MarginLayoutParams(MarginLayoutParams.WRAP_CONTENT, MarginLayoutParams.WRAP_CONTENT);
			btnButton.setText(text[i]);
			vg_test.addView(btnButton,lp);
		}
	}

}

结果显示:

### 回答1: Android中的流式布局是一种常用的布局方式,它可以根据内容的大小和数量自动调整控件的位置和大小,使得界面能够自适应屏幕的宽度。通常在需要展示多个标签、图片或文字等的场景下使用。 流式布局的特点是将内容按照先后顺序从左到右排列,当一行的宽度不足以容纳下一个控件时,会自动换行。这种布局方式能够节省空间,提高界面的可读性和美观性。 在Android中,可以使用FlowLayout这个第三方库来实现流式布局。使用FlowLayout的步骤如下:首先在项目的build.gradle文件中添加依赖,然后在布局文件中将根布局设置为FlowLayout,并在其中添加需要展示的控件,可以通过调整控件的属性来定义布局的样式和排列方式。 流式布局可以动态调整控件的位置和大小,可以通过设置权重来控制每个控件在水平方向上的占比,也可以设置边距来调整控件之间的间隔。另外,流式布局还可以为每个控件设置点击事件和长按事件,方便实现更丰富的交互效果。 总之,流式布局是一种灵活且强大的布局方式,可以有效地解决多个控件在界面上排列不下或排列不美观的问题,同时也能够提高界面的可读性和用户体验。在开发Android应用时,如果遇到需要展示多个标签、图片或文字等的场景,流式布局是一个很好的选择。 ### 回答2: Android流式布局是一种灵活的布局方式,用于在屏幕上动态自适应地显示一系列视图。它能够根据子视图的大小和屏幕大小自动调整子视图的位置和宽度。这种布局方式适用于显示不规则大小的子视图,尤其适用于显示标签、图片、标签云等。 Android流式布局可以通过使用LinearLayout或GridLayout来实现。在LinearLayout中,可以设置orientation属性为horizontal或vertical来实现水平或垂直流式布局。在GridLayout中,可以通过设置列数来控制每行显示的子视图数量。 Android流式布局的优点是可以根据屏幕的大小和方向自动调整子视图的布局,使得页面在不同设备上都能够良好地显示。同时,它也提供了更好的用户体验,因为用户可以在不同屏幕上以不同的方式查看和交互。 然而,Android流式布局也存在一些限制。由于其自适应特性,子视图的大小和位置可能会受到限制。此外,较复杂的布局可能会导致性能问题,因为在布局过程中需要进行多次测量和计算。因此,在使用流式布局时,需要谨慎处理子视图的大小和数量,以提高性能并避免布局过于复杂。 总结来说,Android流式布局是一种灵活而自适应的布局方式,适用于显示不规则大小的子视图。它可以根据屏幕的大小和方向自动调整子视图的布局,并提供更好的用户体验。然而,需要注意处理子视图的大小和数量,以提高性能并避免布局过于复杂。 ### 回答3: Android流式布局(Flow Layout)是一种动态适应屏幕宽度的布局方式,主要用于解决在屏幕上按行排列多个子视图的问题。 在传统的线性布局中,如果视图超出屏幕宽度,就会自动换行,但是每一行只会放置一个子视图。而在流式布局中,子视图会根据屏幕宽度自动换行,并且每一行可以放置多个子视图,适应屏幕不同宽度的设备。 流式布局的使用非常方便,只需要将子视图添加到流式布局中即可。它提供了一些属性来控制子视图在布局中的排列方式,比如子视图之间的间距、子视图的对齐方式等。此外,流式布局还可以通过设置权重属性,实现子视图的均匀分布或者按比例分布。 流式布局在一些场景下非常有用,比如在标签云、瀑布流展示等需要动态调整子视图排列的情况下。相比于其他布局方式,流式布局可以更好地利用屏幕空间,提高用户体验。 总之,Android流式布局是一种动态适应屏幕宽度的布局方式,可以方便地排列多个子视图,并提供了一些属性来控制子视图的排列方式和样式。它的使用简单灵活,适用于多种场景,可以有效地提高用户界面的可用性和美观性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值