在Android开发中,我们经常会用到对商家或者商品的评价,运用星星进行打分。然而在Android系统中自带的打分控件,ratingBar特别不好用,间距和大小无法改变。所以,我就自定义了一个特别好用的打分控件。在项目中可以直接使用,特别简单。下面直接上图:
效果图
实现原理
其实就是自定义view继承LinearLayout ,然后里面动态加了五个ImageView。
实现代码,有详细的注释
在attrs中声明的可以在xml中设置的变量
<declare-styleable name="ratingBar"> <!--尺寸值--> <attr name="starImageSize" format="dimension" /> <!--星星间距--> <attr name="starPadding" format="dimension" /> <!--星星总数--> <attr name="starCount" format="integer" /> <!--空白的星星资源文件值--> <attr name="starEmpty" format="reference" /> <!--满星资源文件值--> <attr name="starFill" format="reference" /> <!--半星资源文件值--> <attr name="starHalf" format="reference" /> <!--是否可点击boolean值--> <attr name="clickable" format="boolean" /> <!--当前进度float值--> <attr name="starStep" format="float" /> <!--每次进度方式的值,整星还是半星--> <attr name="stepSize"> <enum name="Half" value="0" /> <enum name="Full" value="1" /> </attr> </declare-styleable>
ratingBar源码
import android.content.Context; import android.content.res.TypedArray; import android.graphics.drawable.Drawable; import android.util.AttributeSet; import android.view.View; import android.widget.ImageView; import android.widget.LinearLayout; import com.kejiang.yuandl.R; import java.math.BigDecimal; /** * Created by dylan on 2015/6/11. * 自定义打分控件ratingBar * 可以自定义星星大小和间距 * Correction clickEvent from Xml */ public class ratingBar extends LinearLayout { /** * 是否可点击 */ private boolean mClickable; /** * 星星总数 */ private int starCount; /** * 星星的点击事件 */ private Onratingchangelistener onratingchangelistener; /** * 每个星星的大小 */ private float starImageSize; /** * 每个星星的间距 */ private float starPadding; /** * 星星的显示数量,支持小数点 */ private float starStep; /** * 空白的默认星星图片 */ private Drawable starEmptyDrawable; /** * 选中后的星星填充图片 */ private Drawable starFillDrawable; /** * 半颗星的图片 */ private Drawable starHalfdrawable; /** * 每次点击星星所增加的量是整个还是半个 */ private StepSize stepSize; /** * 设置半星的图片资源文件 * * @param starHalfdrawable */ public void setStarHalfdrawable(Drawable starHalfdrawable) { this.starHalfdrawable = starHalfdrawable; } /** * 设置满星的图片资源文件 * * @param starFillDrawable */ public void setStarFillDrawable(Drawable starFillDrawable) { this.starFillDrawable = starFillDrawable; } /** * 设置空白和默认的图片资源文件 * * @param starEmptyDrawable */ public void setStarEmptyDrawable(Drawable starEmptyDrawable) { this.starEmptyDrawable = starEmptyDrawable; } /** * 设置星星是否可以点击操作 * * @param clickable */ public void setClickable(boolean clickable) { this.mClickable = clickable; } /** * 设置星星点击事件 * * @param onratingchangelistener */ public void setonratingchangelistener(Onratingchangelistener onratingchangelistener) { this.onratingchangelistener = onratingchangelistener; } /** * 设置星星的大小 * * @param starImageSize */ public void setStarImageSize(float starImageSize) { this.starImageSize = starImageSize; } public void setStepSize(StepSize stepSize) { this.stepSize = stepSize; } /** * 构造函数 * 获取xml中设置的资源文件 * * @param context * @param attrs */ public ratingBar(Context context,AttributeSet attrs) { super(context,attrs); setorientation(LinearLayout.HORIZONTAL); TypedArray mTypedArray = context.obtainStyledAttributes(attrs,R.styleable.ratingBar); starImageSize = mTypedArray.getDimension(R.styleable.ratingBar_starImageSize,20); starPadding = mTypedArray.getDimension(R.styleable.ratingBar_starPadding,10); starStep = mTypedArray.getFloat(R.styleable.ratingBar_starStep,1.0f); stepSize = StepSize.fromStep(mTypedArray.getInt(R.styleable.ratingBar_stepSize,1)); starCount = mTypedArray.getInteger(R.styleable.ratingBar_starCount,5); starEmptyDrawable = mTypedArray.getDrawable(R.styleable.ratingBar_starEmpty); starFillDrawable = mTypedArray.getDrawable(R.styleable.ratingBar_starFill); starHalfdrawable = mTypedArray.getDrawable(R.styleable.ratingBar_starHalf); mClickable = mTypedArray.getBoolean(R.styleable.ratingBar_clickable,true); mTypedArray.recycle(); for (int i = 0; i < starCount; ++i) { final ImageView imageView = getStarImageView(); imageView.setimageDrawable(starEmptyDrawable); imageView.setonClickListener( new OnClickListener() { @Override public void onClick(View v) { if (mClickable) { //浮点数的整数部分 int fint = (int) starStep; BigDecimal b1 = new BigDecimal(Float.toString(starStep)); BigDecimal b2 = new BigDecimal(Integer.toString(fint)); //浮点数的小数部分 float fPoint = b1.subtract(b2).floatValue(); if (fPoint == 0) { fint -= 1; } if (indexOfChild(v) > fint) { setStar(indexOfChild(v) + 1); } else if (indexOfChild(v) == fint) { if (stepSize == StepSize.Full) {//如果是满星 就不考虑半颗星了 return; } //点击之后默认每次先增加一颗星,再次点击变为半颗星 if (imageView.getDrawable().getCurrent().getConstantState().equals(starHalfdrawable.getConstantState())) { setStar(indexOfChild(v) + 1); } else { setStar(indexOfChild(v) + 0.5f); } } else { setStar(indexOfChild(v) + 1f); } } } } ); addView(imageView); } setStar(starStep); } /** * 设置每颗星星的参数 * * @return */ private ImageView getStarImageView() { ImageView imageView = new ImageView(getContext()); LinearLayout.LayoutParams layout = new LinearLayout.LayoutParams( Math.round(starImageSize),Math.round(starImageSize));//设置每颗星星在线性布局的大小 layout.setMargins(0,Math.round(starPadding),0);//设置每颗星星在线性布局的间距 imageView.setLayoutParams(layout); imageView.setAdjustViewBounds(true); imageView.setScaleType(ImageView.ScaleType.CENTER_CROP); imageView.setimageDrawable(starEmptyDrawable); imageView.setMinimumWidth(10); imageView.setMaxHeight(10); return imageView; } /** * 设置星星的个数 * * @param rating */ public void setStar(float rating) { if (onratingchangelistener != null) { onratingchangelistener.onratingChange(rating); } this.starStep = rating; //浮点数的整数部分 int fint = (int) rating; BigDecimal b1 = new BigDecimal(Float.toString(rating)); BigDecimal b2 = new BigDecimal(Integer.toString(fint)); //浮点数的小数部分 float fPoint = b1.subtract(b2).floatValue(); //设置选中的星星 for (int i = 0; i < fint; ++i) { ((ImageView) getChildAt(i)).setimageDrawable(starFillDrawable); } //设置没有选中的星星 for (int i = fint; i < starCount; i++) { ((ImageView) getChildAt(i)).setimageDrawable(starEmptyDrawable); } //小数点默认增加半颗星 if (fPoint > 0) { ((ImageView) getChildAt(fint)).setimageDrawable(starHalfdrawable); } } /** * 操作星星的点击事件 */ public interface Onratingchangelistener { /** * 选中的星星的个数 * * @param ratingCount */ void onratingChange(float ratingCount); } /** * 星星每次增加的方式整星还是半星,枚举类型 * 类似于View.GONE */ public enum StepSize { Half(0),Full(1); int step; StepSize(int step) { this.step = step; } public static StepSize fromStep(int step) { for (StepSize f : values()) { if (f.step == step) { return f; } } throw new IllegalArgumentException(); } } } 在xml中的用法 <com.kejiang.yuandl.view.ratingBar android:id="@+id/rb" android:layout_width="360dp" android:layout_height="50dp" app:starCount="5" app:starEmpty="@mipmap/star_grey" app:starFill="@mipmap/star_yellow" app:starHalf="@mipmap/star_half_yellow" app:starImageSize="40dp" app:starPadding="20dp" app:starStep="1.5" app:stepSize="Half"></com.kejiang.yuandl.view.ratingBar>
在Activity中的设置
ratingBar ratingBar= (ratingBar) findViewById(R.id.rb); ratingBar.setClickable(true);//设置可否点击 ratingBar.setStar(2.5f);//设置显示的星星个数 ratingBar.setStepSize(ratingBar.StepSize.Half);//设置每次点击增加一颗星还是半颗星 ratingBar.setonratingchangelistener(new ratingBar.Onratingchangelistener() { @Override public void onratingChange(float ratingCount) {//点击星星变化后选中的个数 Log.d("ratingBar","ratingBar-Count="+ratingCount); } });
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。