Android 自定义控件-星级评分

2023-10-08 01:50

本文主要是介绍Android 自定义控件-星级评分,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在学习自定义控件时需要一些例子来练练手,本文这个控件就是在这种环境下产生的(可能有BUG);

这个控件设计的特点:

1,可以任意修改星星数量

2,可以星星大小会随控件大小而缩小,在控件足够大的情况可以任意设置星星大小

3,滑动监听,根据滑动距离选择星级

4,可以设置星星之间的间距和左右间距

第一步:

初始化星星图片,随便设置星星的默认宽高

private void init() {mPaint = new Paint();star = BitmapFactory.decodeResource(getResources(), R.drawable.icon_evaluate_star);starPressed = BitmapFactory.decodeResource(getResources(), R.drawable.icon_evaluate_star_pressed);starWidth = star.getWidth();starHeight = star.getHeight();}

第二步:

重写onMeasure方法,在这里说一下onMeasure方法的两个参数:

widthMeasureSpec和heightMeasureSpec:分别 代表了View宽高的:大小模式和大小数值

一个int 类型怎么能代表两个东西呢, 系统时这样规定的,采用最高两位表示模式,如下图:

最高位00表示:MeasureSpec.UNSPECIFIED : 表示在XML 中使用wrap_centent

最高位01表示:MeasureSpec.EXACTLY: 表示在XML 中使用 xxdp

最高位11表示:MeasureSpec.AT_MOST:表示在XML 中使用 match_parent

然后代码中的逻辑: 计算使用默认值时需要的实际宽高,在判断控件是否指定宽高 是的再判断是否大于实际需要宽高 小于就按比例缩小,大于就按居中显示 把多余的宽高都加左右/上下间距里具体代码,代码备注的已经很详细了

    @Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);int widthMode = MeasureSpec.getMode(widthMeasureSpec);mWidth = MeasureSpec.getSize(widthMeasureSpec);//// 实际所需要的宽 = 星星的宽 * 星星数量 + 星星之间的间距 * 间距数  + 左右间距float totalWidthSpacing = (starCount - 1) * spacing + leftSpacing + rightSpacing; // 总的间距float width = starWidth * starCount + totalWidthSpacing;switch (widthMode) {case MeasureSpec.AT_MOST:case MeasureSpec.EXACTLY:// 当实际所需的宽 大于控件所设定的宽时   应该按比例缩小实际所需要宽来满足控件所给宽if (width > mWidth) {// 计算比例float scale = mWidth / width;starWidth = starWidth * scale;spacing = spacing * scale;leftSpacing = leftSpacing * scale;rightSpacing = rightSpacing * scale;} else {// 如果实际所需宽小于 控件所给宽  那就加大左右间距  尽量保持居中效果float diff = width - mWidth;leftSpacing = leftSpacing + diff / 2;rightSpacing = rightSpacing + diff / 2;}// 重新计算totalWidthSpacing = (starCount - 1) * spacing + leftSpacing + rightSpacing; // 总的间距width = starWidth * starCount + totalWidthSpacing;mWidth = (int) (width + totalWidthSpacing);break;case MeasureSpec.UNSPECIFIED:// 未指定的情况下 我就安实际所需宽高来 做控件宽高mWidth = (int) width;break;}int heightMode = MeasureSpec.getMode(heightMeasureSpec);mHeight = MeasureSpec.getSize(heightMeasureSpec);// 实际所需高float height = starHeight + topSpacing + bottomSpacing;switch (heightMode) {case MeasureSpec.AT_MOST:case MeasureSpec.EXACTLY:// 当控件指定高时  尽可能满足指定的高if (height > mHeight) {// 当实际所需高大于指定高时  按比例缩小实际所需高float scale = mHeight / height;starHeight = starHeight * scale;topSpacing = topSpacing * scale;bottomSpacing = bottomSpacing * scale;} else {// 实际所需高小于指定高时  将多余的都加到 上下间距float diff = mHeight - height;topSpacing = topSpacing + diff / 2;bottomSpacing = bottomSpacing + diff / 2;}// 重新计算高mHeight = (int) (starHeight + topSpacing + bottomSpacing);break;case MeasureSpec.UNSPECIFIED:// 未指定的情况下 我就安实际所需宽高来 做控件宽高mHeight = (int) height;break;}// 设置宽高setMeasuredDimension(mWidth, mHeight);}

第三步 画星星 在上面我已经初始化星星的Bitmap了 

重写onDraw 方法 有starCount 来决定画星星的数量 再由星级来决定画什么样的星星。

再计算星星该画在什么位置 计算方式都在代码里里 也有详细的备注

这个主要说明一下 canvas.drawBitmap(bitmap, src, dst , mPaint); 这方法

第一个参数: 表示需要画的图

第二个参数:表示图片需要绘制的区域,可以参数可以为空, 表示绘制这张图片

第三个参数:表示图片应该被绘制在画布的什么区域,不能为空

第四个蚕食:画笔, 可以为空。

    @Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);for (int i = 0; i < starCount; i++) {Bitmap bitmap = star;if (i < level) {bitmap = starPressed;}// 表示图片需要绘制区域Rect src = new Rect(0, 0, bitmap.getWidth(), bitmap.getHeight());// 表示图片应该被绘制在的区域RectF dst = new RectF();dst.top = topSpacing ;dst.left = leftSpacing + (starWidth + spacing) * i;dst.right = dst.left + starWidth;dst.bottom = dst.top + starWidth;canvas.drawBitmap(bitmap, src, dst, mPaint);}}

第四步 重写onTouchEvent() 方法

这个说明一下 当我们手指触摸屏幕时有三种情况:

手指按下:MotionEvent.ACTION_DOWN.

手指滑动:MotionEvent.ACTION_MOVE.

手指带起:MotionEvent.ACTION_UP.

这就是我们点击屏幕时三种动作。

在这里我先划分点击有效区域,在有效距离内再根据x值除于星星的宽加星星之间的间距来知道点击了那个星星

最后再加个判断只有当星级发生改变的时候才重绘控件。因为在onTouchEvent方法执行次数太多,避免没必要的重绘

    @Overridepublic boolean onTouchEvent(MotionEvent event) {int oldLevel = level;switch (event.getAction()) {case MotionEvent.ACTION_DOWN: // 手指按下break;case MotionEvent.ACTION_MOVE: // 手指滑动float x = event.getX();float y = event.getY();// 当点击区域在  星星所在区域时 才点击有效if (y > topSpacing && y < topSpacing + starHeight) {// 根据点击位置确实星级if (x < leftSpacing ) {// 小于左边距 表示没有点到一个星星level = 0;} else {// 只要左边距肯定已经点到星星了  除于星星宽个间距即知道点击了那个星星level = (int) ((x - leftSpacing) / (starWidth + spacing)) + 1;Log.e("AAA—>", "onTouchEvent: " + level );if (oldLevel != level) {// 只有当星级发生改变时才去刷新布局 不做没必要刷新if (onLevelChangeListener != null) {onLevelChangeListener.levelChange(level);}postInvalidate();}}}break;case MotionEvent.ACTION_UP: // 手指抬起break;}return true;}

由于本人水平有限,文笔也比较糙,不喜勿喷。

源代码

这篇关于Android 自定义控件-星级评分的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/161899

相关文章

如何自定义一个log适配器starter

《如何自定义一个log适配器starter》:本文主要介绍如何自定义一个log适配器starter的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求Starter 项目目录结构pom.XML 配置LogInitializer实现MDCInterceptor

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Druid连接池实现自定义数据库密码加解密功能

《Druid连接池实现自定义数据库密码加解密功能》在现代应用开发中,数据安全是至关重要的,本文将介绍如何在​​Druid​​连接池中实现自定义的数据库密码加解密功能,有需要的小伙伴可以参考一下... 目录1. 环境准备2. 密码加密算法的选择3. 自定义 ​​DruidDataSource​​ 的密码解密3

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

WinForms中主要控件的详细使用教程

《WinForms中主要控件的详细使用教程》WinForms(WindowsForms)是Microsoft提供的用于构建Windows桌面应用程序的框架,它提供了丰富的控件集合,可以满足各种UI设计... 目录一、基础控件1. Button (按钮)2. Label (标签)3. TextBox (文本框

Android NDK版本迭代与FFmpeg交叉编译完全指南

《AndroidNDK版本迭代与FFmpeg交叉编译完全指南》在Android开发中,使用NDK进行原生代码开发是一项常见需求,特别是当我们需要集成FFmpeg这样的多媒体处理库时,本文将深入分析A... 目录一、android NDK版本迭代分界线二、FFmpeg交叉编译关键注意事项三、完整编译脚本示例四

Android与iOS设备MAC地址生成原理及Java实现详解

《Android与iOS设备MAC地址生成原理及Java实现详解》在无线网络通信中,MAC(MediaAccessControl)地址是设备的唯一网络标识符,本文主要介绍了Android与iOS设备M... 目录引言1. MAC地址基础1.1 MAC地址的组成1.2 MAC地址的分类2. android与I

Android 实现一个隐私弹窗功能

《Android实现一个隐私弹窗功能》:本文主要介绍Android实现一个隐私弹窗功能,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 效果图如下:1. 设置同意、退出、点击用户协议、点击隐私协议的函数参数2. 《用户协议》、《隐私政策》设置成可点击的,且颜色要区分出来res/l

Android实现一键录屏功能(附源码)

《Android实现一键录屏功能(附源码)》在Android5.0及以上版本,系统提供了MediaProjectionAPI,允许应用在用户授权下录制屏幕内容并输出到视频文件,所以本文将基于此实现一个... 目录一、项目介绍二、相关技术与原理三、系统权限与用户授权四、项目架构与流程五、环境配置与依赖六、完整

Android 12解决push framework.jar无法开机的方法小结

《Android12解决pushframework.jar无法开机的方法小结》:本文主要介绍在Android12中解决pushframework.jar无法开机的方法,包括编译指令、框架层和s... 目录1. android 编译指令1.1 framework层的编译指令1.2 替换framework.ja