时间:2021-05-20
概述:在学习HenCoder的过程中,有一期是模仿优秀自定义View,有一个项目是仿即刻的点赞,后来原作者在点评中提到,需要将文字和图片分开来写,并且模仿者的动画实现由点杂乱。所以决定重现实现下文字部分的效果。并拓展了更多功能。最后说一句本文基于kotlin实现。不明白的地方请在评论区指出。
即刻原效果:个人效果:
分析
从效果图容易看出,图中的功能主要分为两个部分:
拓展的功能包括:文字变换模式(全部和部分) 改变文字和未改变文字的间隔和颜色,文字始终位于中心位置。
一 文字的绘制
对文字绘制还不熟悉的同学请参考HenCoder系列文章,这里只对怎么实现居中的作一下说明。
1 水平居中
水平居中的绘制按文字变换模式分为两种
全部改变时:
控件宽度的一半减去文字宽度的一半 即是文字开始绘制的位置
部分改变时
计算每部分文字起始位置
2 垂直居中
垂直居中的实现,最重要的是需要计算文字基线在垂直方向的位置 计算公式就不在这里解释了
var fontMetrics = mPaint.fontMetrics// 文字基线y轴坐标 为了 让文字 垂直居中val baseLineY = height / 2 - fontMetrics.top / 2 - fontMetrics.bottom / 2二 动画的实现
可以看到 我们默认是没有点赞的,然后点一下就赞,再点一下 取消点赞。所以思路是这样的 首先绘制居中文字,然后在控件看不到的下方再绘制一遍,然后根据平移动画完成这个效果,这个动画是通过属性动画实现的。
// 为了显示效果 根据是否是全部改变 设置不同的绘制方式if (mChangeMode === 0) {mPaint.color = mChangedTextColorcanvas.drawText(array[1], width / 2.toFloat() - halfTextWidth(array[1]), baseLineY + yOffset, mPaint)canvas.drawText(array[2], width / 2.toFloat() - halfTextWidth(array[2]), baseLineY + height / 2 + +halfOfTextHeight + yOffset, mPaint)} else if (mChangeMode === 1) {/ 获取部分改变的模式时的绘制文字其实起始位置startX = width / 2.toFloat() - (2 * halfTextWidth(array[0]) + mTextSpace + 2 * halfTextWidth(array[1])) / 2mPaint.color = mNoChangeTextColorcanvas.drawText(array[0], startX, baseLineY, mPaint)mPaint.color = mChangedTextColorcanvas.drawText(array[1], startX + 2 * halfTextWidth(array[0]) + mTextSpace, baseLineY + yOffset, mPaint)canvas.drawText(array[2], startX + 2 * halfTextWidth(array[0]) + mTextSpace, baseLineY + height / 2 + +halfOfTextHeight + yOffset, mPaint) }可以看到 在设置绘制垂直方向的位置的时候,都加入了一个 yOffset 的变量,通过改变这个属性的值也显示动画,那个这个值的最大值很明显就是 文字高度的一半加上控件高度的一半。
halfOfTextHeight = (fontMetrics.bottom - fontMetrics.top) / 2textOffset = (halfOfTextHeight + height / 2)自定义属性动画必须添加的 set get 方法
@Suppress("unused") fun setYOffset(yOffset: Float) { this.yOffset = yOffset invalidate() } @Suppress("unused") fun getYOffset() = yOffset最后提供给外界跳用的方法
fun show() { hasThumbs = if (hasThumbs) { val animator = ObjectAnimator.ofFloat(this, "yOffset", -textOffset, 0f) animator.duration = 500 animator.start() false } else { val animator = ObjectAnimator.ofFloat(this, "yOffset", 0f, -textOffset) animator.duration = 500 animator.start() true } } // 调用 val tv: ThumbsView = findViewById(R.id.thumbsView1) as ThumbsView tv.show()三 源码
github地址:源码点我直达
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
通过自定义view实现仿iOS实现滑动两端的点选择时间的效果效果图自定义的view代码publicclassRing_Slide2extendsView{pri
本文实例为大家分享了Android自定义view仿iOS弹出框的具体代码,供大家参考,具体内容如下运行效果图自定义对话框的使用,仿照ios。从底部弹出,类似po
前言Android开发中,常常自定义View实现自己想要的效果,当然自定义View也是Android开发中比较难的部分,涉及到的知识有Canvas(画布),Pa
本文实例为大家分享了Android自定义View画圆的具体代码,供大家参考,具体内容如下引入布局自定义View的java类,继承ViewpublicclassV
前言:Android开发中,自定义View实现自己想要的效果已成为一项必备的技能,当然自定义View也是Android开发中比较难的部分,涉及到的知识有Canv