Android自定义View-简约风歌词控件
前言
最近重构了之前的音乐播放器(音乐播放器的源码地址在文章底部),添加了许多功能,比如歌词,下载功能等。这篇文章就让我们聊聊歌词控件的实现(歌词控件也已经开源,地址也在文章底部),先上效果图,如果感觉海星,就继续瞧下去!
看到这里,估计你对这个控件还有点感兴趣的吧,那接下来就让我们来瞧瞧实现这个歌词控件需要做些什么!(如果想直接使用就直接点击文末中的开源库地址,里面会有添加依赖库的说明)
一、 歌词解析
首先,我们得知道正常的歌词格式是怎样的,大概是长这个样子:
1 | [ti:喜欢你] |
从上面可以看出这种格式前面是开始时间,从左往右一一对应分,秒,毫秒,后面就是歌词。所以我们要创建一个实体类来保存每一句的歌词信息。
1.歌词实体类LrcBean
1 | public class LrcBean { |
每句歌词,我们需要开始时间,结束时间和歌词这些信息,那么你就会有疑问了?上面提到的歌词格式好像只有歌词开始时间,那我们怎么知道结束时间呢?其实很简单,这一句歌词的开始时间就是上一句歌词的结束时间。有了歌词实体类,我们就得开始对歌词进行解析了!
2. 解析歌词工具类LrcUtil
1 | public class LrcUtil { |
相信上面的代码和注释已经将这个歌词解析解释的挺明白了,需要注意的是上面对i=5,也就是歌词真正开始的第一句做了特殊处理,因为i=5这句有可能是很长的,假设i=5是“光年之外 (《太空旅客(Passengers)》电影中国区主题曲) - G.E.M. 邓紫棋 (Gem Tang)”这句歌词,如果我们不做特殊处理,在后面绘制的时候,就会发现这句歌词会超过屏幕大小,很影响美观,所以我们只截取歌曲名和演唱者,有些说明直接省略掉了。解析好了歌词,接下来就是重头戏-歌词绘制!
二、歌词绘制
歌词绘制就涉及到了自定义View的知识,所以还未接触自定义View的小伙伴需要先去看看自定View的基础知识。歌词绘制的主要工作主要由下面几部分构成:
- 为歌词控件设置自定义属性,在构造方法中获取并设置自定义属性的默认值
- 初始化两支画笔。分别是歌词普通画笔,歌词高亮画笔。
- 获取当前播放歌词的位置
- 画歌词,根据当前播放歌词的位置来决定用哪支画笔画
- 歌词随歌曲播放同步滑动
- 重新绘制
1.设置自定View属性,在代码中设置默认值
在res文件中的values中新建一个attrs.xml文件,然后定义歌词的自定义View属性
1 |
|
这里只自定义了歌词颜色,歌词高亮颜色,歌词大小,歌词行间距的属性,可根据自己需要自行添加。
然后在Java代码中,设置默认值。
1 | private int lrcTextColor;//歌词颜色 |
2. 初始化两支画笔
1 | private void init() { |
我们把初始化的方法放到了构造方法中,这样就可以避免在重绘时再次初始化。另外由于我们把init方法只放到了第三个构造方法中,所以在上面两个构造方法需要将super改成this,这样就能保证哪个构造方法都能执行init方法
1 | public LrcView(Context context) { |
3. 重复执行onDraw方法
因为后面的步骤都是在onDraw方法中执行的,所以我们先贴出onDraw方法中的代码
1 |
|
1.获得控件的测量后的宽高
1 | private int width, height;//屏幕宽高 |
为什么要获得控件的宽高呢?因为在下面我们需要画歌词,画歌词时需要画的位置,这时候就需要用到控件的宽高了。
2. 得到当前歌词的位置
1 | private List<LrcBean> lrcBeanList;//歌词集合 |
我们根据当前播放的歌曲时间来遍历歌词集合,从而判断当前播放的歌词的位置。细心的你可能会发现在currentPosition = 0中有个curTime>10 60 1000的判断,这是因为在实际使用中发现当player还未播放时,这时候得到的curTime会很大,所以才有了这个判断(因为正常的歌曲不会超过10分钟)。
在这个方法我们会发现出现了歌词集合和播放器,你可能会感到困惑,这些不是还没赋值吗?困惑就对了,所以我们需要提供外部方法来给外部传给歌词控件歌词集合和播放器。
1 | //将歌词集合传给到这个自定义View中 |
外部方法中setLrc的参数必须是前面提到的标准歌词格式的字符串形式,这样我们就能利用上文的解析工具类LrcUtil中的解析方法将字符串解析成歌词集合。
3. 画歌词
1 | private void drawLrc(Canvas canvas) { |
知道了当前歌词的位置就很容易画歌词了。遍历歌词集合,如果是当前歌词,则用高亮的画笔画,其它歌词就用普通画笔画。这里需注意的是两支画笔画的位置公式都是一样的,坐标位置为x=宽的一半,y=高的一半+当前位置*行间距。随着当前位置的变化,就能画出上下句歌词来。所以其实绘制出来后你会发现歌词是从控件的正中央开始绘制的,这是为了方便与下面歌词同步滑动功能配合。
4. 歌词同步滑动
1 | //歌词滑动 |
如果不实现弹性滑动的话,只要判断当前播放歌曲的时间是否大于当前位置歌词的结束时间,然后进行scrollTo(0,(int)currentPosition * lineSpacing)滑动即可。但是为了实现弹性滑动,我们需要将一次滑动分成若干次小的滑动并在一个时间段内完成,所以我们动态设置y的值,由于不断重绘,就能实现在0.5秒内完成View的滑动,这样就能实现歌词同步弹性滑动。
500其实就是0.5s,因为在这里currentTime和startTime的单位都是ms
1 | float y = (currentTime - startTime) > 500 ? currentPosition * lineSpacing : lastPosition * lineSpacing + (currentPosition - lastPosition) * lineSpacing * ((currentTime - startTime) / 500f); |
5.不断重绘
通过不断重绘才能实现歌词同步滑动,这里每隔0.1s进行重绘
1 | postInvalidateDelayed(100);//延迟0.1s刷新 |
你以为这样就结束了吗?其实还没有,答案下文揭晓!
三 、使用
然后我们兴高采烈的在xml中,引用这个自定义View
LrcView前面的名称为你建这个类的完整包名
1 | <com.example.library.view.LrcView |
在Java代码中给这个自定义View传入标准歌词字符串和播放器。
1 | lrcView.setLrc(lrc).setPlayer(player); |
点击运行,满心期待自己的成果,接着你就会一脸懵逼,what?怎么是一片空白,什么也没有!其实这时候你重新理一下上面歌词绘制的流程,就会发现问题所在。首先我们的自定义View控件引用到布局中时是先执行onDraw方法的,所以当你调用setLrc和setPlayer方法后,是不会再重新调用onDraw方法的,等于你并没有传入歌词字符串和播放器,所以当然会显示一片空白
解决方法:我们在刚才自定义View歌词控件中添加一个外部方法来调用onDraw,刚好这个invalidate()就能够重新调用onDraw方法
1 | public LrcView draw() { |
然后我们在主代码中,在调用setLrc和setPlayer后还得调用draw方法
1 | lrcView.setLrc(lrc).setPlayer(player).draw(); |
这样我们节约风的歌词控件就大功告成了。
相关源码地址
如果觉得不错的话,欢迎大家来star!