Android 自定义尺子

开始之前bb几句,虽然现在已经有类似的开源项目,但如果想真正理解它,还是要自己去手写一下,这不,趁着这两天空闲,公司项目不忙,就自己写了个demo,基本功能已完成,可能会有bug,回头再优化。

先说一下基本思路:

1.先创建一个类,继承View,先把刻度画出来。

2.再创建一个类,继承ViewGroup,这个类主要是控制内部尺子的滚动。

下面直接贴代码:

RulerView:

package com.test.demotestobject.view

import android.content.Context
import android.graphics.Canvas
import android.graphics.Color
import android.graphics.Paint
import android.graphics.Path
import android.util.AttributeSet
import android.util.Log
import android.view.View
import androidx.annotation.Nullable

open class RulerView : View{

    private lateinit var mPaint : Paint

    private lateinit var path: Path
    private  var color:String  = "#FF4081"
    //尺子Y轴水平基点
    private  var startYDraw : Float=0f
    //矫正刻度起始位置
    private  var firststep : Float  = 0f
    //每个刻度起始位置
    private  var cachStep: Float  = 0f
    //尺子长度
    private  var ruleSize  = 501
    //一屏幕的刻度数
    private  var  viewAll = 60
    private lateinit var numListener:NumListener

    private  var isScrowComplent = false

    constructor(context: Context?) : super(context) {
        initView()
    }

    constructor(context:Context , @Nullable attrs: AttributeSet): super(context,attrs) {
        initView()
    }

    constructor(context:Context , @Nullable attrs: AttributeSet,defStyleAttr:Int) : super(context,attrs,defStyleAttr){
        initView()
    }



    private fun initView() {
        mPaint =  Paint()
        path =  Path()
        mPaint.style=(Paint.Style.FILL)
        mPaint.strokeWidth=(2f)
        mPaint.color=(Color.parseColor(color))
        mPaint.textAlign=(Paint.Align.CENTER)
        mPaint.textSize=(14f)
        mPaint.isAntiAlias=true
        mPaint.isFilterBitmap=true
    }


    fun setScrollComplete(isScrollComplete:Boolean) {
        this.isScrowComplent = isScrollComplete
        invalidate()
    }

    /**
     * 滚动时调用(给RuleViewGroup调用,不要私自调用)
     *
     * @param changeX
     */
    fun setChangeX(changeX:Float) {
        isLastComplete=false
        firststep += changeX
        invalidate()
    }

    /**
     * 置为中点使用(给RuleViewGroup调用,不要私自调用)
     *
     * @param changeX
     */
    var isLastComplete: Boolean=false

    fun setCenterChangeX(changeX:Float) {
        firststep += changeX
        if(!isLastComplete) {
            invalidate()
        }
    }

    /**
     * 手动输入时,设置当前刻度值
     *
     * @param rule
     */
    fun setNowRule(rule:Int ) {
        firststep = (width / 2 - (width / viewAll * rule)).toFloat()
        invalidate()
    }


    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)

        startYDraw = ((height / 2 - 20).toFloat())

        path.moveTo((width / 2).toFloat(), (startYDraw))
        path.lineTo(((width / 2) + 10).toFloat(), (startYDraw) - 30)
        path.lineTo(((width / 2) - 10).toFloat(), (startYDraw) - 30)
        path.lineTo((width / 2).toFloat(), (startYDraw))
        canvas.drawPath(path, mPaint)


        //当起始位置超过中点时,则切换为中点,这样起始位置就不会跨过中点
        if (firststep > width / 2) {
            firststep = (width / 2).toFloat()
        }


        //-((width/viewAll)*(ruleSize-1)-(width/2))的计算结果为终点滑到中点时,起始位置的结果,
        // 控制器不要超过此点,能保证终点不划过中点
        if (firststep <= -((width / viewAll) * (ruleSize - 1) - (width / 2))) {
            firststep = (-((width / viewAll) * (ruleSize - 1) - (width / 2))).toFloat()
        }

        cachStep = firststep
        for ( i in 0..ruleSize) {
            Log.e("测试cacheStep",cachStep.toString())

            if(cachStep>width){
                break
            }

            if(cachStep>=0 || cachStep<=width) {
                if ((i % 5) == 0) {
                    canvas.drawLine(cachStep, startYDraw, cachStep, (startYDraw) + 80, mPaint)
                    canvas.drawText(i.toString(), cachStep, (startYDraw) + 120, mPaint)
                } else {
                    canvas.drawLine(cachStep, startYDraw, cachStep, (startYDraw) + 55, mPaint)
                }

                //当前刻度在中点
                if (cachStep <= (width / 2) && (cachStep + (width / viewAll)) >= (width / 2)) {
                    numListener.getnum(i)

                    //滚动完成
                    if (isScrowComplent) {
                        if (cachStep + (width / viewAll / 2) >= (width / 2)) {
                            setCenterChangeX((width / 2) - cachStep)
                        } else {
                            setCenterChangeX((width / 2) - (width / viewAll) - cachStep)
                        }
                        isLastComplete=true
                    }
                }
            }

            cachStep += (width / viewAll)

        }
    }

    public fun setNumListener(numListener : NumListener ) {
        this.numListener = numListener
    }

    public interface NumListener {
        fun getnum(num:Int)
    }

}

RulerViewGroup:

package com.test.demotestobject.view

import android.content.Context
import android.util.AttributeSet
import android.util.Log
import android.view.MotionEvent
import android.view.VelocityTracker
import android.widget.RelativeLayout


open class RulerViewGroup : RelativeLayout, Runnable {
    private var mVelocityTracker: VelocityTracker? = null

    //手指离开时的滚动速度
    private var velocityX = 0f

    //当前的触摸点的X值
    private var nowtouchX = 0f

    //加速度
    private val speedAdd = 2

    //单位时间
    private val unitTime = 5

    //手指滑动方向
    private var isLeft = false

    constructor(context: Context?) : super(context) {
        intView()
    }

    constructor(context: Context?, attrs: AttributeSet?) : super(context, attrs) {
        intView()
    }

    constructor(context: Context?, attrs: AttributeSet?, defStyleAttr: Int) : super(
        context,
        attrs,
        defStyleAttr
    ) {
        intView()
    }

    private fun intView() {}
    override fun onTouchEvent(event: MotionEvent): Boolean {
        if (mVelocityTracker == null) {
            mVelocityTracker = VelocityTracker.obtain()
        }
        var eventAddedToVelocityTracker = false
        val vtev = MotionEvent.obtain(event)
        when (event.action) {
            MotionEvent.ACTION_DOWN -> nowtouchX = event.x
            MotionEvent.ACTION_MOVE -> {
                (getChildAt(0) as RulerView).setScrollComplete(false)
                try {
                    (getChildAt(0) as RulerView).setChangeX(event.x - nowtouchX)
                    Log.e("ChangeX", (event.x - nowtouchX).toString() + "")
                    nowtouchX = event.x
                } catch (e: Exception) {
                    Log.e("RuleViewGroup", "布局错误")
                }


                //计算实时滑动速度
                mVelocityTracker!!.addMovement(vtev)
                eventAddedToVelocityTracker = true
                mVelocityTracker!!.computeCurrentVelocity(unitTime, 1000f)
                velocityX = mVelocityTracker!!.getXVelocity(event.getPointerId(0))

                //用于run方法判断加速度的正负
                isLeft = velocityX < 0
            }
            MotionEvent.ACTION_UP ->                 //启动惯性滚动
                postDelayed(this, unitTime.toLong())
            MotionEvent.ACTION_CANCEL -> releaseVelocityTracker()
            else -> {
            }
        }
        if (!eventAddedToVelocityTracker) {
            mVelocityTracker!!.addMovement(vtev)
        }
        vtev.recycle()
        return true
    }

    override fun run() {
        if (isLeft) {
            velocityX += speedAdd
            if (velocityX >= 0) {
                velocityX = 0f
            }
        } else {
            velocityX -= speedAdd
            if (velocityX <= 0) {
                velocityX = 0f
            }
        }
        try {
            (getChildAt(0) as RulerView).setChangeX(velocityX)
        } catch (e: Exception) {
            Log.e("RuleViewGroup", "布局错误")
        }
        if (velocityX != 0f) {
            postDelayed(this, unitTime.toLong())
        } else {
            try {
                (getChildAt(0) as RulerView).setScrollComplete(true)
            } catch (e: Exception) {
                Log.e("RuleViewGroup", "布局错误")
            }
        }
    }

    //释放VelocityTracker
    private fun releaseVelocityTracker() {
        if (null != mVelocityTracker) {
            mVelocityTracker!!.clear()
            mVelocityTracker!!.recycle()
            mVelocityTracker = null
        }
    }
}

下面是布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

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

        <EditText
            android:id="@+id/num"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:hint="请输入刻度值"
            android:inputType="number"
            android:text=""
            android:textSize="20dp" />

        <TextView
            android:id="@+id/setnum"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:inputType="number"
            android:text=""
            android:textSize="30dp" />

        <Button
            android:id="@+id/submit"
            android:layout_width="100dp"
            android:layout_height="wrap_content"
            android:text="提交" />

    </LinearLayout>


    <com.example.myapplication.touch.RulerViewGroup
        android:layout_width="match_parent"
        android:layout_height="200dp">

        <com.example.myapplication.touch.RulerView
            android:id="@+id/ruler"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true" />

    </com.example.myapplication.touch.RulerViewGroup>
</LinearLayout>

以上代码放到项目里可以直接用

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容