Android自定义控件RatingBar调整字体大小

发布时间 - 2026-01-11 00:26:26    点击率:

项目需要,做一个可以调整字体大小的控件,能在滑动或点击时改变选中的位置,效果图如下:

这是一个类似于RatingBar的控件,然而配置RatingBar的样式难以实现这样的效果,如选中的图案和上面的文字对齐。因此,有必要写一个自定义View来实现。

思路如下:

  • 新建一个TextRatingBar继承View类
  • 在onDraw()方法中绘制元素:文字、横线、短竖线和圆形标记
  • 重写onMeasure()方法,控制整体大小和边界
  • 在onTouchEvent()方法中处理ACTION_DOWN和ACTION_MOVE事件,调用invalidate()方法引起View的重绘,以更新视图
  • 定义一个用户选中某个字体的回调接口,以通知外部处理,比如去设置字体大小。

TextRatingBar类代码如下:

package cc.rome753.demo.view;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;

/**
 * Created by rome753 on 2017/3/10
 */

public class TextRatingBar extends View{

 //paddingLeft
 private int mLeft;
 //paddingTop
 private int mTop;
 //当前rating
 private int mRating;
 //总raring数
 private int mCount;
 //rating文字
 private String[] texts = {"小","中","大","超大"};
 //相邻raring的距离
 private int mUnitSize;
 //bar到底部的距离
 private int mYOffset;
 //小竖条的一半长度
 private int mMarkSize;

 Paint paint = new Paint();

 public TextRatingBar(Context context) {
 this(context, null);
 }

 public TextRatingBar(Context context, AttributeSet attrs) {
 this(context, attrs, 0);
 }

 public TextRatingBar(Context context, AttributeSet attrs, int defStyleAttr) {
 super(context, attrs, defStyleAttr);
 mCount = 4;
 mRating = 0;
 mMarkSize = 3;
 }

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
 super.onMeasure(widthMeasureSpec, heightMeasureSpec);
 Log.i("test", getMeasuredWidth() + " " + getMeasuredHeight());
 mLeft = (getPaddingLeft()+getPaddingRight())/2;
 mTop = getPaddingTop();
 int barWidth = getMeasuredWidth() - 2 * mLeft;
 mUnitSize = barWidth/(mCount - 1);
 mYOffset = getMeasuredHeight() - getPaddingBottom();
 }

 @Override
 protected void onDraw(Canvas canvas) {
 paint.setStrokeWidth(2);
 paint.setColor(Color.RED);
 canvas.drawLine(mLeft,mYOffset,mLeft+mRating*mUnitSize,mYOffset,paint);
 for(int i=0;i<mCount;i++){
  paint.setColor(Color.RED);
  canvas.drawLine(mLeft+i*mUnitSize,mYOffset-mMarkSize,mLeft+i*mUnitSize,mYOffset+mMarkSize,paint);
  paint.setColor(mRating == i ? Color.RED : Color.BLACK);
  paint.setTextSize(30);
  paint.setTextAlign(Paint.Align.CENTER);
  canvas.drawText(texts[i],mLeft+i*mUnitSize,mTop,paint);
 }
 paint.setColor(Color.GRAY);
 canvas.drawLine(mLeft+mRating*mUnitSize,mYOffset,mLeft+(mCount-1)*mUnitSize,mYOffset,paint);
 canvas.drawCircle(mLeft+mRating*mUnitSize,mYOffset,10,paint);

 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 if(event.getAction() == MotionEvent.ACTION_DOWN || event.getAction() == MotionEvent.ACTION_MOVE){
  float x = event.getX();
  for(int i=0;i<mCount;i++){
  float distance = mLeft+i*mUnitSize - x;
  if(Math.abs(distance) < 100){
   setRating(i);
   if(onRatingListener != null){
   onRatingListener.onRating(mRating);
   }
   break;
  }
  }
 }
 return true;
 }

 public void setRating(int rating) {
 mRating = rating;
 invalidate();
 }

 private OnRatingListener onRatingListener;

 public void setOnRatingListener(OnRatingListener onRatingListener) {
 this.onRatingListener = onRatingListener;
 }

 interface OnRatingListener{
 void onRating(int rating);
 }
}

几个要点:

  • onDraw()中的绘制要注意周围的预留空间,防止最左边的字体只显示一半,或滑动条下面没有一点空白的预留空间,导致用户不好划。因此在计算每一个绘制坐标时额外加上这些。
  • 触摸事件是判断当前触摸点离哪个rating点最近,要加上左右临界值Math.abs(distance),用户点击或划动在distance范围内就算发生了onRating()事件。

没有用到自定义属性,使用时直接放到布局中,周围加上padding就行了。

 <cc.rome753.demo.view.TextRatingBar
 android:paddingTop="20dp"
 android:paddingLeft="40dp"
 android:paddingRight="40dp"
 android:paddingBottom="35dp"
 android:layout_width="match_parent"
 android:layout_height="70dp" />

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # RatingBar  # Android开发之自定义星星评分控件RatingBar用法示例  # Android星级评分条控件RatingBar使用详解  # Android RatingBar星星评分控件实例代码  # Android开发手册RatingBar星级评分控件实例  # 自定义  # 临界值  # 几个  # 能在  # 这是一个  # 要注意  # 做一个  # 只显示  # 有必要  # 重写  # 来实现  # 类似于  # 回调  # 大家多多  # 新建一个  # 发生了  # 就行了  # 划动  # util  # Paint 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 微信小程序 canvas开发实例及注意事项  EditPlus中的正则表达式 实战(2)  Laravel怎么使用Intervention Image库处理图片上传和缩放  Linux系统命令中tree命令详解  Laravel如何创建自定义中间件?(Middleware代码示例)  为什么php本地部署后css不生效_静态资源加载失败修复技巧【技巧】  如何快速生成橙子建站落地页链接?  PythonWeb开发入门教程_Flask快速构建Web应用  SQL查询语句优化的实用方法总结  微信小程序 五星评分(包括半颗星评分)实例代码  HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】  Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布  Laravel如何生成URL和重定向?(路由助手函数)  如何用wdcp快速搭建高效网站?  Claude怎样写结构化提示词_Claude结构化提示词写法【教程】  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  Laravel怎么实现验证码功能_Laravel集成验证码库防止机器人注册  如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?  奇安信“盘古石”团队突破 iOS 26.1 提权  购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?  Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  Laravel中间件起什么作用_Laravel Middleware请求生命周期与自定义详解  详解jQuery停止动画——stop()方法的使用  高性价比服务器租赁——企业级配置与24小时运维服务  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  济南网站建设制作公司,室内设计网站一般都有哪些功能?  Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】  如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)  齐河建站公司:营销型网站建设与SEO优化双核驱动策略  详解Oracle修改字段类型方法总结  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  详解MySQL数据库的安装与密码配置  如何用5美元大硬盘VPS安全高效搭建个人网站?  Win11怎样安装网易有道词典_Win11安装词典教程【步骤】  javascript中数组(Array)对象和字符串(String)对象的常用方法总结  Java解压缩zip - 解压缩多个文件或文件夹实例  Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧  如何获取上海专业网站定制建站电话?  如何用腾讯建站主机快速创建免费网站?  laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法  Thinkphp 中 distinct 的用法解析  网站页面设计需要考虑到这些问题  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  大学网站设计制作软件有哪些,如何将网站制作成自己app?  怎样使用JSON进行数据交换_它有什么限制  如何用狗爹虚拟主机快速搭建网站?  如何在腾讯云服务器快速搭建个人网站?  如何在Windows服务器上快速搭建网站?  简单实现Android文件上传