网站首页 编程语言 正文
本文实例为大家分享了Android自定义View实现标签流效果的具体代码,供大家参考,具体内容如下
一、概述
Android自定义View实现标签流效果,一行放不下时会自动换行,用户可以自己定义单个标签的样式,可以选中和取消,可以监听单个标签的点击事件,功能还算强大,可以满足大部分开发需求,值得推荐,效果图如下:

二、实现代码
1.自定义View
定义属性文件
<declare-styleable name="FlowTagView"> <attr name="lineSpacing" format="dimension" /> <attr name="tagSpacing" format="dimension" /> <!-- 是否是固定布局 --> <attr name="isFixed" format="boolean" /> <attr name="columnSize" format="integer" /> </declare-styleable>
FlowTagConfig.java
package com.czhappy.effectdemo.flowtag;
import android.content.Context;
import android.content.res.TypedArray;
import android.util.AttributeSet;
import com.czhappy.effectdemo.R;
/**
* Description:
* User: chenzheng
* Date: 2017/2/17 0017
* Time: 10:23
*/
public class FlowTagConfig {
private static final int DEFAULT_LINE_SPACING = 5;//默认行间距
private static final int DEFAULT_TAG_SPACING = 10;//各个标签之间的默认距离
private static final int DEFAULT_FIXED_COLUMN_SIZE = 3; //默认列数
private int lineSpacing;
private int tagSpacing;
private int columnSize;
private boolean isFixed;
public FlowTagConfig(Context context,AttributeSet attrs){
TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.FlowTagView);
try {
lineSpacing = a.getDimensionPixelSize(R.styleable.FlowTagView_lineSpacing, DEFAULT_LINE_SPACING);
tagSpacing = a.getDimensionPixelSize(R.styleable.FlowTagView_tagSpacing, DEFAULT_TAG_SPACING);
columnSize = a.getInteger(R.styleable.FlowTagView_columnSize, DEFAULT_FIXED_COLUMN_SIZE);
isFixed = a.getBoolean(R.styleable.FlowTagView_isFixed,false);
} finally {
a.recycle();
}
}
public int getLineSpacing() {
return lineSpacing;
}
public void setLineSpacing(int lineSpacing) {
this.lineSpacing = lineSpacing;
}
public int getTagSpacing() {
return tagSpacing;
}
public void setTagSpacing(int tagSpacing) {
this.tagSpacing = tagSpacing;
}
public int getColumnSize() {
return columnSize;
}
public void setColumnSize(int columnSize) {
this.columnSize = columnSize;
}
public boolean isFixed() {
return isFixed;
}
public void setIsFixed(boolean isFixed) {
this.isFixed = isFixed;
}
}FlowTagView.java
package com.czhappy.effectdemo.flowtag;
import android.content.Context;
import android.database.DataSetObserver;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
/**
* Description:
* User: chenzheng
* Date: 2017/2/17 0017
* Time: 10:23
*/
public class FlowTagView extends ViewGroup {
private int mLineSpacing;//行间距
private int mTagSpacing;//各个标签之间的距离
private BaseAdapter mAdapter;
private TagItemClickListener mListener;
private DataChangeObserver mObserver;
public FlowTagView(Context context) {
super(context);
init(context, null, 0);
}
public FlowTagView(Context context, AttributeSet attrs) {
super(context, attrs);
init(context, attrs, 0);
}
public FlowTagView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init(context, attrs, defStyle);
}
private void init(Context context, AttributeSet attrs, int defStyle) {
//获取属性
FlowTagConfig config = new FlowTagConfig(context, attrs);
mLineSpacing = config.getLineSpacing();
mTagSpacing = config.getTagSpacing();
}
private void drawLayout() {
if (mAdapter == null || mAdapter.getCount() == 0) {
return;
}
this.removeAllViews();
for (int i = 0; i < mAdapter.getCount(); i++) {
View view = mAdapter.getView(i,null,null);
final int position = i;
view.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (mListener != null) {
mListener.itemClick(position);
}
}
});
this.addView(view);
}
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int wantHeight = 0;
int wantWidth = resolveSize(0, widthMeasureSpec);
int paddingLeft = getPaddingLeft();
int paddingRight = getPaddingRight();
int paddingTop = getPaddingTop();
int paddingBottom = getPaddingBottom();
int childLeft = paddingLeft;
int childTop = paddingTop;
int lineHeight = 0;
//固定列的数量所需要的代码
for (int i = 0; i < getChildCount(); i++) {
final View childView = getChildAt(i);
LayoutParams params = childView.getLayoutParams();
childView.measure(
getChildMeasureSpec(widthMeasureSpec, paddingLeft + paddingRight, params.width),
getChildMeasureSpec(heightMeasureSpec, paddingTop + paddingBottom, params.height)
);
//获取单个tag的宽高
int childHeight = childView.getMeasuredHeight();
int childWidth = childView.getMeasuredWidth();
lineHeight = Math.max(childHeight, lineHeight);
//超过长度的新起一行
if (childLeft + childWidth + paddingRight > wantWidth) {
childLeft = paddingLeft;
childTop += mLineSpacing + childHeight;
lineHeight = childHeight;
}
childLeft += childWidth + mTagSpacing;
}
wantHeight += childTop + lineHeight + paddingBottom;
setMeasuredDimension(wantWidth, resolveSize(wantHeight, heightMeasureSpec));
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
//固定列的数量所需要的代码
int width = r - l;
int paddingLeft = getPaddingLeft();
int paddingTop = getPaddingTop();
int paddingRight = getPaddingRight();
int childLeft = paddingLeft;
int childTop = paddingTop;
int lineHeight = 0;
for (int i = 0; i < getChildCount(); i++) {
final View childView = getChildAt(i);
if (childView.getVisibility() == View.GONE) {
continue;
}
int childWidth = childView.getMeasuredWidth();
int childHeight = childView.getMeasuredHeight();
lineHeight = Math.max(childHeight, lineHeight);
if (childLeft + childWidth + paddingRight > width) {
childLeft = paddingLeft;
childTop += mLineSpacing + lineHeight;
lineHeight = childHeight;
}
childView.layout(childLeft, childTop, childLeft + childWidth, childTop + childHeight);
childLeft += childWidth + mTagSpacing;
}
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
}
@Override
public LayoutParams generateLayoutParams(AttributeSet attrs) {
return new LayoutParams(this.getContext(), attrs);
}
public void setAdapter(BaseAdapter adapter){
if (mAdapter == null){
mAdapter = adapter;
if (mObserver == null){
mObserver = new DataChangeObserver();
mAdapter.registerDataSetObserver(mObserver);
}
drawLayout();
}
}
public void setItemClickListener(TagItemClickListener mListener) {
this.mListener = mListener;
}
/**
* 单击监听接口
*/
public interface TagItemClickListener {
void itemClick(int position);
}
class DataChangeObserver extends DataSetObserver {
@Override
public void onChanged() {
FlowTagView.this.drawLayout();
}
@Override
public void onInvalidated() {
super.onInvalidated();
}
}
}2.测试类
FlowTagActivity.java
package com.czhappy.effectdemo.activity;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import com.czhappy.effectdemo.R;
import com.czhappy.effectdemo.adapter.EvaluateAdapter;
import com.czhappy.effectdemo.flowtag.FlowTagView;
import com.czhappy.effectdemo.model.Evaluate;
import java.util.ArrayList;
import java.util.List;
/**
* Description:
* User: chenzheng
* Date: 2017/2/17 0017
* Time: 11:47
*/
public class FlowTagActivity extends AppCompatActivity {
private FlowTagView mContainer;
private EvaluateAdapter adapter;
private List<Evaluate> chooseList = new ArrayList<Evaluate>();
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_flowtag);
initView();
initData();
}
private void initData() {
List<Evaluate> list = new ArrayList();
Evaluate e1 = new Evaluate("热情", "1");
Evaluate e2 = new Evaluate("服务周到", "2");
Evaluate e3 = new Evaluate("一般", "3");
Evaluate e4 = new Evaluate("技术活杠杠的", "4");
Evaluate e5 = new Evaluate("专业精通", "5");
Evaluate e6 = new Evaluate("只会吹牛逼", "6");
Evaluate e7 = new Evaluate("地下第一仅此一家", "7");
list.add(e1);
list.add(e2);
list.add(e3);
list.add(e4);
list.add(e5);
list.add(e6);
list.add(e7);
adapter.setItems(list);
}
private void initView() {
mContainer = (FlowTagView) this.findViewById(R.id.container);
adapter = new EvaluateAdapter(this);
mContainer.setAdapter(adapter);
mContainer.setItemClickListener(new FlowTagView.TagItemClickListener() {
@Override
public void itemClick(int position) {
Evaluate e = (Evaluate) adapter.getItem(position);
e.is_choosed = !e.is_choosed;
if(e.is_choosed){
chooseList.add(e);
}else{
chooseList.remove(e);
}
adapter.notifyDataSetChanged();
}
});
}
}EvaluateAdapter.java
package com.czhappy.effectdemo.adapter;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import com.czhappy.effectdemo.R;
import com.czhappy.effectdemo.model.Evaluate;
import java.util.ArrayList;
import java.util.List;
/**
* Description:
* User: chenzheng
* Date: 2017/2/17 0017
* Time: 11:43
*/
public class EvaluateAdapter extends BaseAdapter {
private Context context;
private LayoutInflater mInflater;
private List<Evaluate> list;
public EvaluateAdapter(Context context) {
this.context = context;
this.mInflater = LayoutInflater.from(context);
this.list = new ArrayList<Evaluate>();
}
public List<Evaluate> getList(){
return list;
}
public void setItems(List<Evaluate> list){
this.list = list;
notifyDataSetChanged();
}
@Override
public int getCount() {
return list == null ? 0 : list.size();
}
@Override
public Object getItem(int position) {
return list.get(position);
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
holder = new ViewHolder();
convertView = mInflater.inflate(
R.layout.evaluate_grid_item, null);
holder.evaluate_tv = (TextView)convertView.findViewById(R.id.evaluate_tv);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
final Evaluate ee = (Evaluate) getItem(position);
holder.evaluate_tv.setText(ee.getName());
if(ee.is_choosed){
holder.evaluate_tv.setBackgroundResource(R.drawable.bg_round_corner_line_orange);
}else{
holder.evaluate_tv.setBackgroundResource(R.drawable.bg_round_corner_line_gray);
}
return convertView;
}
private final class ViewHolder {
private TextView evaluate_tv;
}
}布局文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <com.czhappy.effectdemo.flowtag.FlowTagView android:id="@+id/container" android:layout_width="match_parent" android:layout_height="wrap_content" android:padding="10dp" app:tagSpacing="10dp" app:lineSpacing="10dp"/> </LinearLayout>
bg_round_corner_line_orange.xml
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android" > <solid android:color="#ffffff" /> <corners android:radius="5dp" /> <stroke android:width="0.5dp" android:color="#FF6700"/> </shape>
bg_round_corner_line_gray.xml
<?xml version="1.0" encoding="utf-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android" > <solid android:color="#ffffff" /> <corners android:radius="5dp" /> <stroke android:width="0.5dp" android:color="#cccccc"/> </shape>
原文链接:https://blog.csdn.net/chenzheng8975/article/details/55511498
相关推荐
- 2024-01-31 Mybatis plus并发更新时的问题
- 2023-02-09 Golang实现Trie(前缀树)的示例_Golang
- 2022-08-20 Docker容器修改端口映射的实现_docker
- 2022-03-19 linux修改文件所属用户和组的实例方法_Linux
- 2022-05-12 adb 导出android目录下的文件
- 2022-05-18 Python学习之自定义异常详解_python
- 2022-12-10 Qt显示QImage图像在label上,并保持自适应大小问题_C 语言
- 2022-04-12 Qt实现实时鼠标绘制图形_C 语言
- 最近更新
-
- window11 系统安装 yarn
- 超详细win安装深度学习环境2025年最新版(
- Linux 中运行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存储小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基础操作-- 运算符,流程控制 Flo
- 1. Int 和Integer 的区别,Jav
- spring @retryable不生效的一种
- Spring Security之认证信息的处理
- Spring Security之认证过滤器
- Spring Security概述快速入门
- Spring Security之配置体系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置权
- redisson分布式锁中waittime的设
- maven:解决release错误:Artif
- restTemplate使用总结
- Spring Security之安全异常处理
- MybatisPlus优雅实现加密?
- Spring ioc容器与Bean的生命周期。
- 【探索SpringCloud】服务发现-Nac
- Spring Security之基于HttpR
- Redis 底层数据结构-简单动态字符串(SD
- arthas操作spring被代理目标对象命令
- Spring中的单例模式应用详解
- 聊聊消息队列,发送消息的4种方式
- bootspring第三方资源配置管理
- GIT同步修改后的远程分支
