继承ViewGroup自定义TagLayout

一、ViewGroup测量子View宽高的方法:

 Measure  measureChild   measureChildren  LayoutParams   ViewGroup.MarginLayoutParams  (后两者也可以用于在onLayout里面改变或者重新设置子View的位置)

这里推荐看一下View的生命周期方法

详解在onMeasure()方法中如何测量一个控件尺寸

二、运行效果

 

三 代码

public class TagLayout2 extends ViewGroup {
    private List<Rect> mchildRect=new ArrayList<>();

    public TagLayout2(Context context) {
        super(context);
    }

    public TagLayout2(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public TagLayout2(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    //这里可以根据需求改
    public void setData(List<String> data){
        for (final String datum : data) {
            View view = inflate(getContext(),R.layout.tag,null);
            TextView textView = view.findViewById(R.id.text);
            textView.setText(datum);
            textView.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    mClicktag.OnClickTag(datum);
                }
            });
            addView(view);
        }
    }

    private onClickTag  mClicktag;

    public void setmClicktag(onClickTag mClicktag) {
        this.mClicktag = mClicktag;
    }

    public interface onClickTag{
        void OnClickTag(String tag);
    }

    /*
    *  首先是测量子View的高 在测量整个TagLayout 高度  宽比较好获得
    *  测量  子View  的宽高
    *  当childLeft左边距 + 自身宽度 > TagLayout  自身的宽度时     换行
    *  每一行的childView高度相加  =  TagLayout 高度
    *  最后在onLayout  里面进行摆放
     */


    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        mchildRect.clear();
        //获得TagLayout  的宽度
        int layoutWidth = MeasureSpec.getSize(widthMeasureSpec);
        //TagLayout 的高度
        int layoutHeight=0;
        //获取子View总数
        int childCount=getChildCount();
        //childView  距离父布局的左边距离
        int childLeft=getPaddingLeft();
        //childView 距离父布局的顶部距离
        int childTop=getPaddingTop();
        //获取每个childView情况
        for (int i=0;i<childCount;i++){
            View childView = getChildAt(i);
            //由子View自身指定了 自己的宽高
            childView.measure(MeasureSpec.UNSPECIFIED,MeasureSpec.UNSPECIFIED);
            int childWidth = childView.getMeasuredWidth();
            int childHeight=childView.getMeasuredHeight();
            //当 childView  距离父布局的左边距离  +  childView  自身的距离 > TagLayout  的宽度时换行
            //这里面不仅完成了 换行操作 还计算了 TagLayout的高度
            //应该说换行操作 = TagLayout的高度
            if (childLeft+childWidth>layoutWidth){
                childLeft=0;  //换行后将childLeft 置为0  重新开始计算宽度
                childTop=childTop+childHeight+20;  //加20 加15是希望childView之间不要太靠近也可以用marginLayoutParams
            }
            //左上右下
            mchildRect.add(new Rect(childLeft,childTop,childLeft+childWidth,childTop+childHeight));
            //这里计算childLeft   确定每一行的子View 位置
            childLeft+=childWidth+15;
            //当是最后一个childView时  自身高度 + childTop = layoutHeight
            if (i==childCount-1){
                layoutHeight=childTop+childHeight;
            }
        }
             //最后不要忘了这里设置Taglayout 自身的宽度和高度
            setMeasuredDimension(layoutWidth,layoutHeight);
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
     int childCount = getChildCount();
     for (int i=0;i<childCount;i++){
         View childView= getChildAt(i);
         Rect lcation = mchildRect.get(i);
         childView.layout(lcation.left,lcation.top,lcation.right,lcation.bottom);
     }
    }

}
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
       TagLayout2 tagLayout = findViewById(R.id.taglayout);
       tagLayout.setData(Arrays.asList(getResources().getStringArray(R.array.tags)));
       tagLayout.setmClicktag(new TagLayout2.onClickTag() {
           @Override
           public void OnClickTag(String tag) {
               Toast.makeText(getApplicationContext(),tag,Toast.LENGTH_SHORT).show();
           }
       });
    }
}

tag标签布局 (主布局放一个自定义ViewGroup就行)

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="wrap_content"
              android:layout_height="wrap_content"
              android:background="@drawable/tag_bg"
              android:gravity="center"
              android:orientation="horizontal"
              android:paddingBottom="3dp"
              android:paddingLeft="15dp"
              android:paddingRight="15dp"
              android:paddingTop="3dp">

    <TextView
        android:id="@+id/text"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textColor="@android:color/holo_orange_dark"  用shape画一个椭圆
        android:textSize="16sp"/>
</LinearLayout>

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值