这几天十一放假,在群里非常活跃,很多朋友问如何实现android中listview的圆角功能,像Iphone设置里面的tableView如 如下效果:
其实这个功能实现也很简单,只是很多朋友没有仔细的去了解android布局的相关知识,这里我们使用了android中的shade的圆角功能来实现的。
java代码很简单,就一个activity,一个listview。listview中要判断item的位置,第一条,最后一条和中间的item是不一样的。代码如下:
AndroidlistviewActivity.java
package com.yangfuhai.listviewdemo; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.widget.ListView; /** * @title 圆角listview的实现 * @description 圆角listview的实现 * @company 探索者网络工作室(www.tsz.net) * @author michael Young (www.YangFuhai.com) * @version 1.0 * @created 2012-10-3 */ public class AndroidlistviewActivity extends Activity { ListView mListView; ListViewAdapter mListViewAdapter; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); mListView = (ListView) findViewById(R.id.listview); mListViewAdapter = new ListViewAdapter(this); mListView.setAdapter(mListViewAdapter); } /** * 添加 按钮执行的操作 * @param view */ static int i = 0; public void add(View view) { mListViewAdapter.addData(" ----item --- "+i+" ---"); mListViewAdapter.notifyDataSetChanged(); i++; } /** * 删除按钮执行的操作 * @param view */ public void del(View view) { mListViewAdapter.delData(); mListViewAdapter.notifyDataSetChanged(); if(i>0) i--; } }
适配器 ListViewAdapter.java
package com.yangfuhai.listviewdemo; import java.util.ArrayList; import java.util.List; import android.content.Context; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.TextView; /** * @title 圆角listview的实现 适配器 * @description ListViewAdapter listview的适配器 * @company 探索者网络工作室(www.tsz.net) * @author michael Young (www.YangFuhai.com) * @version 1.0 * @created 2012-10-3 */ public class ListViewAdapter extends BaseAdapter { private List<String> datas = new ArrayList<String>(); //数据 private Context mContext; public ListViewAdapter(Context c) { this.mContext = c; } public void addData(String strData){ if(strData!=null) datas.add(strData); } public void delData(){ if(datas.size() > 0) datas.remove(datas.size() - 1); } @Override public int getCount() { return datas.size(); } @Override public Object getItem(int arg0) { return datas.get(arg0); } @Override public long getItemId(int position) { return position; } /** * listview中要判断item的位置,第一条,最后一条和中间的item是不一样的。 */ @Override public View getView(int position, View convertView, ViewGroup parent) { View view = null ; if(datas.size()>1){//listView 数据是两条以上 if(position == 0){ //第一条数据 view = View.inflate(mContext, R.layout.list_item_top, null); }else if(position == datas.size() - 1){ //最后一条数据 view = View.inflate(mContext, R.layout.list_item_bottom, null); }else{ //中间的数据 view = View.inflate(mContext, R.layout.list_item_middle, null); } }else{ //只有一条数据 view = View.inflate(mContext, R.layout.list_item_single, null); } ((TextView)view.findViewById(R.id.title)).setText(datas.get(position));//设置文本样式 return view; } }
布局文件main.xml
<?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="fill_parent" android:orientation="vertical" > <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="请点击添加删除 查看效果" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" > <Button android:id="@+id/buttonAdd" android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="add" android:text="添加" /> <Button android:id="@+id/buttonDel" android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="del" android:text="删除" /> </LinearLayout> <ListView android:id="@+id/listview" android:layout_marginLeft="10dip" android:layout_marginRight="10dip" android:layout_width="fill_parent" android:layout_height="wrap_content" /> </LinearLayout>
上面的代码很简单,没有什么可讲的。主要讲的是listview每个item的样式文件
listview的item有四个布局文件,分别是:只有一个item时候的样式,多个item时候上边item的样式,下边item的样式,中间item的样式。 布局文件和背景文件对应关系如下图所示:
listview的item布局样式如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" style="@style/list_item_middle" android:layout_width="fill_parent" android:layout_height="wrap_content" android:minHeight="60dip" > <TextView android:id="@+id/title" style="@style/content_page_large_text" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="title" /> </LinearLayout>
这四个布局文件中唯一不同的只是他们的stype属性(style="@style/list_item_middle")不同,也就是他们的背景不同。
下面我们先贴出布局文件背景文件的shade代码,然后再仔细的讲解背景这些文件里面代码的意思。
background_view_rounded_bottom.xml
<?xml version="1.0" encoding="UTF-8"?> <inset xmlns:android="http://schemas.android.com/apk/res/android" android:insetBottom="1.0px" android:insetLeft="1.0px" android:insetRight="1.0px" android:insetTop="1.0px" > <selector> <item android:state_pressed="true"> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_pressed" android:startColor="@color/base_start_color_pressed" /> <corners android:bottomLeftRadius="10.0dip" android:bottomRightRadius="10.0dip" android:radius="2.0dip" android:topLeftRadius="0.0dip" android:topRightRadius="0.0dip" /> </shape> </item> <item> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_default" android:startColor="@color/base_start_color_default" /> <corners android:bottomLeftRadius="11.0dip" android:bottomRightRadius="11.0dip" android:radius="2.0dip" android:topLeftRadius="0.0dip" android:topRightRadius="0.0dip" /> </shape> </item> </selector> </inset>
background_view_rounded_middle.xml:
<?xml version="1.0" encoding="UTF-8"?> <inset xmlns:android="http://schemas.android.com/apk/res/android" android:insetBottom="0.0px" android:insetLeft="1.0px" android:insetRight="1.0px" android:insetTop="1.0px" > <selector> <item android:state_pressed="true"> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_pressed" android:startColor="@color/base_start_color_pressed" /> <corners android:radius="0.0dip" /> </shape> </item> <item> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_default" android:startColor="@color/base_start_color_default" /> <corners android:radius="0.0dip" /> </shape> </item> </selector> </inset>
background_view_rounded_single.xml :
<?xml version="1.0" encoding="UTF-8"?> <inset xmlns:android="http://schemas.android.com/apk/res/android" android:insetBottom="1.0px" android:insetLeft="1.0px" android:insetRight="1.0px" android:insetTop="0.0px" > <selector> <item android:state_pressed="true"> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_pressed" android:startColor="@color/base_start_color_pressed" /> <corners android:radius="11.0dip" /> </shape> </item> <item> <shape> <stroke android:width="1.0px" android:color="@color/rounded_container_border" /> <gradient android:angle="270.0" android:endColor="@color/base_end_color_default" android:startColor="@color/base_start_color_default" /> <corners android:radius="10.0dip" /> </shape> </item> </selector> </inset>
background_view_rounded_top.xml :
<?xml version="1.0" encoding="UTF-8"?> <inset xmlns:android="http://schemas.android.com/apk/res/android" android:insetBottom="0.0px" android:insetLeft="1.0px" android:insetRight="1.0px" android:insetTop="1.0px" > <selector> <item android:state_pressed="true"> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_pressed" android:startColor="@color/base_start_color_pressed" /> <corners android:bottomLeftRadius="0.0dip" android:bottomRightRadius="0.0dip" android:radius="2.0dip" android:topLeftRadius="10.0dip" android:topRightRadius="10.0dip" /> </shape> </item> <item> <shape> <gradient android:angle="270.0" android:endColor="@color/base_end_color_default" android:startColor="@color/base_start_color_default" /> <corners android:bottomLeftRadius="0.0dip" android:bottomRightRadius="0.0dip" android:radius="2.0dip" android:topLeftRadius="11.0dip" android:topRightRadius="11.0dip" /> </shape> </item> </selector> </inset>
我们拿 background_view_rounded_top.xml (最后一个资源文件)来给大家讲解 里面的每个属性的含义:
inset:这种资源指向一个InsetDrawable对象,它能够用指定的距离嵌入到另一个可绘制资源中。它的属性有:
android:drawable="@drawable/drawable_resource"
android:insetTop="dimension"
android:insetRight="dimension"
android:insetBottom="dimension"
android:insetLeft="dimension"
其中android:drawable是要嵌入的图片资源,android:insetXXX是嵌入位置。
selector:是一种样式选择器,它用来指导某个view(button,textview,edittext等)的不同状态(比如:正常的状态,获得焦点的状态,按下的状态 等)对应的不同资源。
shade gradient:颜色渐变
android:startColor和android:endColor分别为渐变的起始颜色和结束颜色
android:angle是渐变角度,必须为45的整数倍 。
渐变模式有两种:
android:type="linear",线性渐变,白话就是渐变是从一头到另一头的颜色渐变过程。
android:type="radial",径向渐变,白话就是渐变是从中心到四周的过程。径向渐变需要指定渐变半径android:gradientRadius="50"。
shade corners:圆角 (这篇文章主要就是用到了它)
android:radius为角的弧度,值越大角越圆。
shade 除了gradient和corners以外,还有stroke(描边),solid (实心),padding等(具体大家可以看android的帮助文档)。
通过以上的代码,我们实现了效果如下图:
完毕
源码下载:http://download.csdn.net/detail/michael_yy/4614701 (免积分下载)
转载请注明出处。http://blog.csdn.net/michael_yy/article/details/8038653
或http://www.yangfuhai.com/topic/48.html (杨福海的博客)