Android自定义控件之刻度尺控件

今天我做的是一个自定义刻度尺控件,由于项目需求需要使用刻度尺那样滑动选择,由于对自定义控件的认识还不够深入,于是花了一周多时间才把这个控件给整出来,也是呕心沥血的经历啊,也让我对自定义控件有了自己的认识,废话不多说,先上一个简单的效果图,大家可以看看,如有需求可以直接拿去使用

效果图如下:只是我的一个简单Demo,效果有点丑陋了点,希望海涵!

效果已经出来接下来就是代码部分了,一看就只是一般的控件很难实现,于是就开始了我的自定义View之旅,每次自定义完后总是会收获很多东西,如下是我的代码:

package android.tst.com.myapplication;

import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.text.Layout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.VelocityTracker;
import android.view.View;
import android.view.ViewConfiguration;
import android.widget.Scroller;

/**
 * 卷尺控件类。由于时间比较紧,只有下班后有时间,因此只实现了基本功能。<br>
 * 细节问题包括滑动过程中widget边缘的刻度显示问题等<br>
 * @version create:2014年8月26日
 */
@SuppressLint("ClickableViewAccessibility")
public class RulerView extends View {

 public interface OnValueChangeListener {
 public void onValueChange(float value);
 }

 public static final int MOD_TYPE_HALF = 2;
 public static final int MOD_TYPE_ONE = 10;
 private static final int ITEM_HALF_DIVIDER = 10;
 private static final int ITEM_ONE_DIVIDER = 10;
 private static final int ITEM_MAX_HEIGHT = 20;
 private static final int ITEM_MIN_HEIGHT = 10;
 private static final int TEXT_SIZE = 7;
 private float mDensity;
 private int mValue = 50, mMaxValue = 100, mModType = MOD_TYPE_HALF,
 mLineDivider = ITEM_HALF_DIVIDER;
 // private int mValue = 50, mMaxValue = 500, mModType = MOD_TYPE_ONE,
 // mLineDivider = ITEM_ONE_DIVIDER;

 private int mLastX, mMove;
 private int mWidth, mHeight;

 private int mMinVelocity;
 private Scroller mScroller;
 private VelocityTracker mVelocityTracker;

 private OnValueChangeListener mListener;

 @SuppressWarnings("deprecation")
 public RulerView(Context context, AttributeSet attrs) {
 super(context, attrs);

 mScroller = new Scroller(getContext());
 mDensity = getContext().getResources().getDisplayMetrics().density;
 mMinVelocity = ViewConfiguration.get(getContext())
 .getScaledMinimumFlingVelocity();
 }

 /**
 *
 * 考虑可扩展,但是时间紧迫,只可以支持两种类型效果图中两种类型
 *
 * @param value
 *   初始值
 * @param maxValue
 *   最大值
 * @param model
 *   刻度盘精度:<br>
 *   {@link MOD_TYPE_HALF}<br>
 *   {@link MOD_TYPE_ONE}<br>
 */
 public void initViewParam(int defaultValue, int maxValue, int model) {
 switch (model) {
 case MOD_TYPE_HALF:
 mModType = MOD_TYPE_HALF;
 mLineDivider = ITEM_HALF_DIVIDER;
 mValue = defaultValue * 2;
 mMaxValue = maxValue * 2;
 break;
 case MOD_TYPE_ONE:
 mModType = MOD_TYPE_ONE;
 mLineDivider = ITEM_ONE_DIVIDER;
 mValue = defaultValue;
 mMaxValue = maxValue;
 break;

 default:
 break;
 }
 invalidate();

 mLastX = 0;
 mMove = 0;
 notifyValueChange();
 }

 /**
 * 设置用于接收结果的监听器
 *
 * @param listener
 */
 public void setValueChangeListener(OnValueChangeListener listener) {
 mListener = listener;
 }

 /**
 * 获取当前刻度值
 *
 * @return
 */
 public float getValue() {
 return mValue;
 }
 public void setValue(int value){
 mValue = value;
 notifyValueChange();
 postInvalidate();
 }
 public void setValueToChange(int what) {
 mValue += what;
 notifyValueChange();
 postInvalidate();
 }

 @Override
 protected void onLayout(boolean changed, int left, int top, int right,
 int bottom) {
 mWidth = getWidth();
 mHeight = getHeight();
 super.onLayout(changed, left, top, right, bottom);
 }

 @Override
 protected void onDraw(Canvas canvas) {
 super.onDraw(canvas);
 drawScaleLine(canvas);
 // drawWheel(canvas);
 drawMiddleLine(canvas);
 }

 /**
 * 从中间往两边开始画刻度线
 *
 * @param canvas
 */
 private void drawScaleLine(Canvas canvas) {
 canvas.save();
 Paint linePaint = new Paint();
 linePaint.setStrokeWidth(2);
 linePaint.setColor(Color.rgb(141, 189, 225));

 TextPaint textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG);
 textPaint.setColor(Color.rgb(68, 135, 188));
 textPaint.setTextSize(TEXT_SIZE * mDensity);

 int width = mWidth, drawCount = 0;
 float xPosition = 0, textWidth = Layout.getDesiredWidth("0", textPaint);

 for (int i = 0; drawCount <= 4 * width; i++) {
 int numSize = String.valueOf(mValue + i).length();
 // 前
 xPosition = (width / 2 - mMove) + i * mLineDivider * mDensity;
 if (xPosition + getPaddingRight() < mWidth) {
 if ((mValue + i) % mModType == 0) {
  linePaint.setColor(Color.rgb(68, 135, 188));
  canvas.drawLine(xPosition, getPaddingTop(), xPosition,
  mDensity * ITEM_MAX_HEIGHT, linePaint);

  if (mValue + i <= mMaxValue) {
  switch (mModType) {
  case MOD_TYPE_HALF:
  canvas.drawText(
   String.valueOf((mValue + i) / 2),
   countLeftStart(mValue + i, xPosition,
   textWidth),
   getHeight() - textWidth, textPaint);
  break;
  case MOD_TYPE_ONE:
  canvas.drawText(String.valueOf(mValue + i),
   xPosition - (textWidth * numSize / 2),
   getHeight() - textWidth, textPaint);
  break;
  default:
  break;
  }
  }
 } else {
  linePaint.setColor(Color.rgb(141, 189, 225));
  // linePaint.setColor(Color.rgb(68, 135, 188));
  canvas.drawLine(xPosition, getPaddingTop(), xPosition,
  mDensity * ITEM_MIN_HEIGHT, linePaint);
 }
 }
 // 后
 xPosition = (width / 2 - mMove) - i * mLineDivider * mDensity;
 if (xPosition > getPaddingLeft()) {
 if ((mValue - i) % mModType == 0) {
  linePaint.setColor(Color.rgb(68, 135, 188));
  canvas.drawLine(xPosition, getPaddingTop(), xPosition,
  mDensity * ITEM_MAX_HEIGHT, linePaint);

  if (mValue - i >= 0) {
  switch (mModType) {
  case MOD_TYPE_HALF:
  canvas.drawText(
   String.valueOf((mValue - i) / 2),
   countLeftStart(mValue - i, xPosition,
   textWidth),
   getHeight() - textWidth, textPaint);
  break;
  case MOD_TYPE_ONE:
  canvas.drawText(String.valueOf(mValue - i),
   xPosition - (textWidth * numSize / 2),
   getHeight() - textWidth, textPaint);
  break;

  default:
  break;
  }
  }
 } else {
  linePaint.setColor(Color.rgb(141, 189, 225));
  canvas.drawLine(xPosition, getPaddingTop(), xPosition,
  mDensity * ITEM_MIN_HEIGHT, linePaint);
 }
 }

 drawCount += 2 * mLineDivider * mDensity;
 }

 canvas.restore();
 }

 /**
 * 计算没有数字显示位置的辅助方法
 *
 * @param value
 * @param xPosition
 * @param textWidth
 * @return
 */
 private float countLeftStart(int value, float xPosition, float textWidth) {
 float xp = 0f;
 if (value < 20) {
 xp = xPosition - (textWidth * 1 / 2);
 } else {
 xp = xPosition - (textWidth * 2 / 2);
 }
 return xp;
 }

 /**
 * 画中间的红色指示线、阴影等。指示线两端简单的用了两个矩形代替
 *
 * @param canvas
 */
 private void drawMiddleLine(Canvas canvas) {
 // TOOD 常量太多,暂时放这,最终会放在类的开始,放远了怕很快忘记
 int gap = 12, indexWidth = 2, indexTitleWidth = 24, indexTitleHight = 10, shadow = 6;
 String color = "#66999999";

 canvas.save();

 Paint redPaint = new Paint();
 redPaint.setStrokeWidth(indexWidth);
 redPaint.setColor(Color.RED);
 canvas.drawLine(mWidth / 2, 0, mWidth / 2, mHeight, redPaint);

 canvas.restore();
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 int action = event.getAction();
 int xPosition = (int) event.getX();

 if (mVelocityTracker == null) {
 mVelocityTracker = VelocityTracker.obtain();
 }
 mVelocityTracker.addMovement(event);

 switch (action) {
 case MotionEvent.ACTION_DOWN:

 mScroller.forceFinished(true);

 mLastX = xPosition;
 mMove = 0;
 break;
 case MotionEvent.ACTION_MOVE:
 getParent().requestDisallowInterceptTouchEvent(true);
 mMove += (mLastX - xPosition);
 changeMoveAndValue();
 break;
 case MotionEvent.ACTION_UP:
 case MotionEvent.ACTION_CANCEL:
 countMoveEnd();
 countVelocityTracker(event);
 getParent().requestDisallowInterceptTouchEvent(false);
 return false;
 // break;
 default:
 break;
 }

 mLastX = xPosition;
 return true;
 }

 private void countVelocityTracker(MotionEvent event) {
 mVelocityTracker.computeCurrentVelocity(1000);
 float xVelocity = mVelocityTracker.getXVelocity();
 if (Math.abs(xVelocity) > mMinVelocity) {
 mScroller.fling(0, 0, (int) xVelocity, 0, Integer.MIN_VALUE,
  Integer.MAX_VALUE, 0, 0);
 }
 }

 private void changeMoveAndValue() {
 int tValue = (int) (mMove / (mLineDivider * mDensity));
 if (Math.abs(tValue) > 0) {
 mValue += tValue;
 mMove -= tValue * mLineDivider * mDensity;
 if (mValue <= 0 || mValue > mMaxValue) {
 mValue = mValue <= 0 ? 0 : mMaxValue;
 mMove = 0;
 mScroller.forceFinished(true);
 }
 notifyValueChange();
 }
 postInvalidate();
 }

 private void countMoveEnd() {
 int roundMove = Math.round(mMove / (mLineDivider * mDensity));
 mValue = mValue + roundMove;
 mValue = mValue <= 0 ? 0 : mValue;
 mValue = mValue > mMaxValue ? mMaxValue : mValue;

 mLastX = 0;
 mMove = 0;

 notifyValueChange();
 postInvalidate();
 }

 private void notifyValueChange() {
 if (null != mListener) {
 if (mModType == MOD_TYPE_ONE) {
 mListener.onValueChange(mValue);
 }
 if (mModType == MOD_TYPE_HALF) {
 mListener.onValueChange(mValue / 2f);
 }
 }
 }

 @Override
 public void computeScroll() {
 super.computeScroll();
 if (mScroller.computeScrollOffset()) {
 if (mScroller.getCurrX() == mScroller.getFinalX()) { // over
 countMoveEnd();
 } else {
 int xPosition = mScroller.getCurrX();
 mMove += (mLastX - xPosition);
 changeMoveAndValue();
 mLastX = xPosition;
 }
 }
 }

 @Override
 public boolean dispatchTouchEvent(MotionEvent event) {
 getParent().requestDisallowInterceptTouchEvent(true);
 return super.dispatchTouchEvent(event);
 }
}

这是我的自定义View部分的代码,下面就是在布局中的使用了

<TextView
  android:id="@+id/tv_values"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:padding="10dp"
  android:gravity="center"
  android:textColor="@android:color/holo_red_dark"/>
 <android.tst.com.myapplication.RulerView
  android:id="@+id/rv_view"
  android:layout_width="match_parent"
  android:layout_height="60dp"/>

<LinearLayout
 android:layout_width="match_parent"
 android:orientation="horizontal"
 android:layout_height="wrap_content">

 <Button
  android:id="@+id/btn_jia"
  android:layout_width="0dp"
  android:layout_height="wrap_content"
  android:text="+"
  android:textSize="25sp"
  android:gravity="center"
  android:layout_marginRight="15dp"
  android:layout_weight="1"/>
 <Button
  android:id="@+id/btn_jian"
  android:layout_width="0dp"
  android:layout_height="wrap_content"
  android:text="-"
  android:layout_marginLeft="15dp"
  android:textSize="25sp"
  android:gravity="center"
  android:layout_weight="1"/>
</LinearLayout>

如上根据效果图,我需要一个TextView进行显示,还有就是我的自定义刻度尺控件了,接下来就是两个Button控件加减。

接下来就是在Activity中的使用了,

首先需要一个Handler进行更新TextView中的值

Handler handler = new Handler() {
  @Override
  public void handleMessage(Message msg) {
   tv_values.setText(rv_view.getValue() + "Kg");
  };
 };

其次就是初始化相关的操作了,

private void deployRulerView(){
  rv_view= (RulerView) findViewById(R.id.rv_view);
  btn_jia= (Button) findViewById(R.id.btn_jia);
  btn_jian= (Button) findViewById(R.id.btn_jian);
  tv_values= (TextView) findViewById(R.id.tv_values);
  //设置RulerView的初始值
  rv_view.setValue(60);
  //初始化刻度尺范围
  rv_view.initViewParam(60, 200, RulerView.MOD_TYPE_ONE);
  rv_view.setValueChangeListener(new RulerView.OnValueChangeListener() {
   @Override
   public void onValueChange(float value) {
    handler.sendMessage(new Message());
   }
  });
  tv_values.setText(60+"KG");
  //给两个控件添加监听事件
  btn_jia.setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View v) {
    rv_view.setValueToChange(1);
   }
  });
  btn_jian.setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View v) {
    rv_view.setValueToChange(-1);
   }
  });
 }

到这里整个过程已经完成了,如果不好的地方尽情吐槽,真个过程,最复杂的莫过于自定义中的绘制过程,但是一切的问题当你静下心好好去实现时,那一切的问题都就不存在了

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

(0)

相关推荐

  • Android实现自定义滑动刻度尺方法示例

    一 基础: 自定义View实现跟随手指滚动的刻度尺,实现了类似SeekBar的滑动选中效果.项目地址,欢迎star! UI图: 功能: 通过设置最小值跟最大值的范围,以及offset值.View将根据这些数据去计算出需要几个小刻度和几个长刻度,和每个长刻度上面显示的数值. 指针可以随意的定制. 当滑动停止后,刻度尺会根据四舍五入将距离指针最近的长刻度滑动到指针的位置. 支持范围越界回弹. 支持设置默认值. 二 实现: 先扯一下,再看别人写的控件的时候总有一种一脸懵逼的感觉,好多凌乱的变量和一大堆

  • Android实现滚动刻度尺效果

    缘起 最近在帮人做一个计步器,其中涉及到身高.体重等信息的采集:我参考了众多app的实现,觉得"乐动力"中滑动刻度的方式比较优雅.于是乎,反编译了该app,结果发现它是采用图片的方式实现的,即ScrollView内嵌了一张带刻度的图片. 个人觉得该方式太不灵活,且对美工的依赖较大,于是便想自定义一个刻度尺控件. 需求分析 绘制刻度,区分整值刻度和普通刻度 红色指针始终在刻度尺的中间,表示当前的刻度 刻度的最大值和最小值可动态设置 刻度尺的高度或宽度可设置,设置后中间刻度不变 可滑动,滑

  • Android实现滑动刻度尺效果

    最近群里的开发人员咨询怎样实现刻度尺的滑动效果去选择身高体重等信息.给个横着的效果,自己试着去改编或者修改一下,看看通过自己的能力能不能做出竖着的效果来,过两天我再把竖着的那个滑动选择效果分享出来.废话不多说了,上代码. 效果图如下: 第一步:activity_mian.xml布局: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://s

  • Android自定义控件之刻度尺控件

    今天我做的是一个自定义刻度尺控件,由于项目需求需要使用刻度尺那样滑动选择,由于对自定义控件的认识还不够深入,于是花了一周多时间才把这个控件给整出来,也是呕心沥血的经历啊,也让我对自定义控件有了自己的认识,废话不多说,先上一个简单的效果图,大家可以看看,如有需求可以直接拿去使用 效果图如下:只是我的一个简单Demo,效果有点丑陋了点,希望海涵! 效果已经出来接下来就是代码部分了,一看就只是一般的控件很难实现,于是就开始了我的自定义View之旅,每次自定义完后总是会收获很多东西,如下是我的代码: p

  • Android自定义控件之组合控件学习笔记分享

    我们来讲一下自定义组合控件,相信大家也接触过自定义组合控件吧,话不多说,直接干(哈~哈~): 大家看到这个觉得这不是很简单的吗,这不就是写个布局文件就搞定嘛,没错,确实直接上布局就行,不过,我只是用这个简单的例子来讲一下自定义组合控件的用法. 首先看看,这一行行的条目看起来都长得差不多,只是图片和文字不一样,没错,就是看中这一点,我们可以把一个条目做成一个组合控件,做为一个整体,这样不管你有几个条目,就写几个组合控件就行了. 步骤: 1.先建立组合控件的布局 myView.xml <Relati

  • Android自定义播放器控件VideoView

    介绍 最近要使用播放器做一个简单的视频播放功能,开始学习VideoView,在横竖屏切换的时候碰到了点麻烦,不过在查阅资料后总算是解决了.在写VideoView播放视频时候定义控制的代码全写在Actvity里了,写完一看我靠代码好乱,于是就写了个自定义的播放器控件,支持指定大小,可以横竖屏切换,手动左右滑动快进快退.好了,下面开始. 效果图有点卡,我也不知道为啥..... VideoView介绍 这个是我们实现视频播放最主要的控件,详细的介绍大家百度就去看,这里介绍几个常用的方法. 用于播放视频

  • C#自定义控件VS用户控件

    C#中自定义控件VS用户控件大比拼 1 自定义控件与用户控件区别 WinForm中, 用户控件(User Control):继承自 UserControl,主要用于开发 Container 控件,Container控件可以添加其他Controls控件 自定义控件(Custom Control):继承自 Control,主要用于开发windows控件的最基本的类,比如 Text,Button 控件 2 要开发自己的控件的几种方法[1] 复合控件(Composite Controls):将现有的各种

  • Android开发之TextView控件用法实例总结

    本文实例总结了Android开发之TextView控件用法.分享给大家供大家参考,具体如下: TextView控件可以向用户展现文本信息,我们可以设置该文本信息是否能编辑 1.TextView基本使用 在程序中创建TextView对象 在xml文件中布局使用 2.New Android Project-> Project name:TextView Build Target:Android 2.2 Application name:TextViewDemo Package name:com.b5

  • Android中findViewById获取控件返回为空问题怎么解决

    在Android程序中,有时候需要加载非原来activity中xml布局中的控件,来使Android程序的界面更加丰富. 我本身是在使用ViewFlipper中遇到的问题. public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); inflater=(LayoutInflater)getSystemService(LAYO

  • android开发实现列表控件滚动位置精确保存和恢复的方法(推荐)

    Android开发经常要对列表的滚动位置进行保存和恢复,网上也有很多关于此功能的方法文章,但绝大多数都只能保存恢复到某一行,对于滚动到半行的情况不能精确的恢复.也有很多文章介绍了好几种方法,也说某些方法能够精确的控制,但实际上根本不能实现.还有些介绍了很多玄乎且非常复杂的方法,但也没看到能完整实现的代码. 经过一段时间的研究测试,下面的代码可以完美的实现列表滚动位置的精确保存和恢复,而且只是在原来记忆到行位置的基础上增加了2行代码而已. 具体见下面代码和注释: //保存位置: int posit

  • Android省市区三级联动控件使用方法实例讲解

    最近有需求需要实现省市区三级联动,但是发现之前的实现不够灵活,自己做了一些优化.为了方便以后使用,抽离出来放在了github上WheelView.同时把其核心库放在了JCenter中了,可以直接引用.也可以参考项目中的Demo进行引用 下面介绍一下如何使用 如果用的是AndroidStudio那么直接在build.gradle文件中添加依赖: dependencies { compile 'chuck.WheelItemView:library:1.0.1' } 成功引入库之后,可以在需要弹出省

  • Android开发之基本控件和四种布局方式详解

    Android中的控件的使用方式和iOS中控件的使用方式基本相同,都是事件驱动.给控件添加事件也有接口回调和委托代理的方式.今天这篇博客就总结一下Android中常用的基本控件以及布局方式.说到布局方式Android和iOS还是区别挺大的,在iOS中有Frame绝对布局和AutoLayout相对布局.而在Android中的布局方式就比较丰富了,今天博客中会介绍四种常用的布局方式.先总结一下控件,然后再搞一搞基本方式,开发环境还是用的Mac下的Android Studio.开始今天的正题, 虽然A

  • Android RadioGroup和RadioButton控件简单用法示例

    本文实例讲述了Android RadioGroup和RadioButton控件简单用法.分享给大家供大家参考,具体如下: RadioGroup和RadioButton代表的是Android中单选按钮的一种控件,写个简单的代码熟悉一下: import android.app.Activity; import android.os.Bundle; import android.widget.RadioButton; import android.widget.RadioGroup; import a

随机推荐