最近朋友想做个音乐App,让我帮忙参考下。其中歌词动态滚动的效果,正好我之前也没做过,顺便学习一下,先来个预览效果。

实现思路

歌词常见的就是lrc歌词了,我们这里也是通过解析lrc歌词文件来获取其播放参数,以实现和播放器协同。下面是我从百度音乐获取的歌词文件示例:

[ti:冰雨]
[ar:刘德华]
[al:笨小孩]
[:0.05]冰雨
[:0.94]作词:刘德华、李密 作曲:潘协庆
[:01.23]演唱:刘德华 [:01.37]
[:04.79](歌手独白)
[:17.18]我是在等待 一个女孩
[:25.18]还是在等待沉沦苦海
[:32.91]一个人静静发呆 没有人去管花谢花开
[:41.03]无法肯定的爱 左右摇摆
[:45.43]只好把心酸往深心里塞
[:49.61]我是在等待 你的回来
[:57.73]难道只换回一句活该
[:05.27]一个人静静发呆
[:09.18]两个人却有不同无奈
[:13.15]好好的一份爱 啊怎么会慢慢变坏
[:18.43]
[:20.44]冷冷的冰雨在脸上胡乱的拍
[:24.31]暖暖的眼泪跟寒雨混成一块
[:28.32]眼前的色彩忽然被掩盖
[:32.28]你的影子无情在身边徘徊
[:36.30]你就像一个刽子手把我出卖
[:40.35]我的心彷佛被剌刀狠狠地宰
[:44.36]在悬崖上的爱 谁会愿意接受最痛的意外
[:51.09]
[:26.59]我是在等待你的回来
[:35.52]难道只换回一句活该
[:42.99]一个人静静发呆
[:46.99]两个人却有不同无奈
[:51.08]好好的一份爱 啊怎么会慢慢变坏
[:56.42]
[:58.54]冷冷的冰雨在脸上胡乱的拍
[:02.41]暖暖的眼泪跟寒雨混成一块
[:06.39]眼前的色彩忽然被掩盖
[:10.31]你的影子无情在身边徘徊
[:14.23]你就像一个刽子手把我出卖
[:18.34]我的心彷佛被剌刀狠狠地宰
[:22.33]在悬崖上的爱 谁会愿意接受最痛的意外
[:28.66]
[:34.57]冷冷的冰雨在脸上胡乱的拍
[:38.33]暖暖的眼泪跟寒雨混成一块
[:42.31]眼前的色彩忽然被掩盖
[:46.32]你的影子无情在身边徘徊
[:50.27]你就像一个刽子手把我出卖
[:54.34]我的心彷佛被剌刀狠狠地宰
[:58.34]悬崖上的爱 谁会敢去采
[:02.37]还是愿意接受最痛的意外 最爱的女孩
[:08.85]
[:19.72]悬崖上的爱 谁会敢去采
[:31.84]还是愿意接受最痛的意外 最爱的女孩
[:51.75]
[:10.60]歌手独白
[:16.76]

解析lrc歌词

这可能是最常见的格式了,每行为一句歌词,[]括号内为歌词对应的时间区间,所以我们首先要做的事情就是将他们提取分离出来,分别作为时间参数数组和歌词内容数组。这里我参考了一些博客的方法,解析lrc文件的代码如下:

#import <Foundation/Foundation.h>

@interface LrcParser : NSObject

//时间
@property (nonatomic,strong) NSMutableArray *timerArray;
//歌词
@property (nonatomic,strong) NSMutableArray *wordArray; //解析歌词
-(void) parseLrc;
//解析歌词
-(void) parseLrc:(NSString*)lrc;
@end

实现代码

@implementation LrcParser

-(instancetype) init{
self=[super init];
if(self!=nil){
self.timerArray=[[NSMutableArray alloc] init];
self.wordArray=[[NSMutableArray alloc] init];
}
return self;
} -(NSString *)getLrcFile:(NSString *)lrc{
NSString* filePath=[[NSBundle mainBundle] pathForResource:lrc ofType:@"lrc"];
return [NSString stringWithContentsOfFile:filePath encoding:NSUTF8StringEncoding error:nil];
}
//测试示例
-(void)parseLrc{
[self parseLrc:[self getLrcFile:@"冰雨"]];
} -(void)parseLrc:(NSString *)lrc{
NSLog(@"%@",lrc); if(![lrc isEqual:nil]){
NSArray *sepArray=[lrc componentsSeparatedByString:@"["];
NSArray *lineArray=[[NSArray alloc] init];
for(int i=;i<sepArray.count;i++){
if([sepArray[i] length]>){
lineArray=[sepArray[i] componentsSeparatedByString:@"]"];
if(![lineArray[] isEqualToString:@"\n"]){
[self.timerArray addObject:lineArray[]]; [self.wordArray addObject:lineArray.count>?lineArray[]:@""];
}
}
}
}
}
@end

经过测试,可以将歌词顺利解析出来,下面我们要将获得歌词数据应用于控制器。

实现动态歌词页面

看了QQ音乐的滚动歌词页面后,可以知道是借助UITableView或者UIScrollView来实现的,这里我们采用UITableView来实现动态歌词界面。

- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
//歌词TableView代理
self.lrcTable.delegate=self;
self.lrcTable.dataSource=self; //解析歌词
self.lrcContent=[[LrcParser alloc] init];
[self.lrcContent parseLrc];
[self.lrcTable reloadData];
//初始化播放器
[self initPlayer];
//监听播放器状态
[NSTimer scheduledTimerWithTimeInterval:0.5 target:self selector:@selector(updateTime) userInfo:nil repeats:YES];
//载入歌词背景
UIImage *img=[UIImage imageNamed:@"wall1.jpg"]; UIImageView *bgView=[[UIImageView alloc] initWithImage:[self getBlurredImage:img]]; self.lrcTable.backgroundView=bgView; }
//cell委托
-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{ UITableViewCell *cell=[self.lrcTable dequeueReusableCellWithIdentifier:@"cell" forIndexPath:indexPath]; cell.textLabel.text=self.lrcContent.wordArray[indexPath.row]; if(indexPath.row==_currentRow) cell.textLabel.textColor = [UIColor redColor]; else cell.textLabel.textColor = [UIColor whiteColor]; cell.textLabel.textAlignment = NSTextAlignmentCenter; cell.textLabel.font = [UIFont systemFontOfSize:]; cell.backgroundColor=[UIColor clearColor]; return cell; }

这里歌词列表的背景我采用了高斯模糊的图片,高斯模糊的方法如下:

//实现高斯模糊
-(UIImage *)getBlurredImage:(UIImage *)image{
CIContext *context = [CIContext contextWithOptions:nil];
CIImage *ciImage=[CIImage imageWithCGImage:image.CGImage];
CIFilter *filter=[CIFilter filterWithName:@"CIGaussianBlur"];
[filter setValue:ciImage forKey:kCIInputImageKey];
[filter setValue:@5.0f forKey:@"inputRadius"];
CIImage *result=[filter valueForKey:kCIOutputImageKey];
CGImageRef ref=[context createCGImage:result fromRect:[result extent]];
return [UIImage imageWithCGImage:ref];
}

播放器则采用AVPlayer,其定义和初始化设置如下:

@interface ViewController : UIViewController<UITableViewDataSource,UITableViewDelegate>

@property (nonatomic,strong) AVAudioPlayer *player;

@end
-(void) initPlayer{
AVAudioSession *session=[AVAudioSession sharedInstance];
[session setActive:YES error:nil];
[session setCategory:AVAudioSessionCategoryPlayback error:nil]; [[UIApplication sharedApplication] beginReceivingRemoteControlEvents]; self.player=[[AVAudioPlayer alloc] initWithContentsOfURL:[[NSBundle mainBundle] URLForResource:@"冰雨" withExtension:@"mp3"] error:nil];
//单曲循环
self.player.numberOfLoops=;
[self.player prepareToPlay];
[self.player play]; }

这样就为应用定义了一个音乐播放器,下面要监听播放器的时间参数,来载入对应的歌词,如下:

[NSTimer scheduledTimerWithTimeInterval:0.5 target:self selector:@selector(updateTime) userInfo:nil repeats:YES];

根据时间更新UI

-(void) updateTime{
CGFloat currentTime=self.player.currentTime;
NSLog(@"%d:%d",(int)currentTime / , (int)currentTime % );
for (int i=; i<self.lrcContent.timerArray.count; i++) {
NSArray *timeArray=[self.lrcContent.timerArray[i] componentsSeparatedByString:@":"];
float lrcTime=[timeArray[] intValue]*+[timeArray[] floatValue];
if(currentTime>lrcTime){
_currentRow=i;
}else
break;
} [self.lrcTable reloadData];
[self.lrcTable scrollToRowAtIndexPath:[NSIndexPath indexPathForRow:_currentRow inSection:] atScrollPosition:UITableViewScrollPositionMiddle animated:YES];
}

最后编译运行,就会发现一个滚动歌词播放器就实现啦。

完整Demo项目地址:https://github.com/ChangweiZhang/AudioPlayerDemo

iOS开发手记-仿QQ音乐播放器动态歌词的实现的更多相关文章

  1. 基于jQuery仿QQ音乐播放器网页版代码

    基于jQuery仿QQ音乐播放器网页版代码是一款黑色样式风格的网页QQ音乐播放器样式代码.效果图如下: 在线预览   源码下载 实现的代码. html代码: <div class="m ...

  2. iOS多种刷新样式、音乐播放器、仿抖音视频、旅游App等源码

    iOS精选源码 企业级开源项目,模仿艺龙旅行App 3D立体相册,可以旋转的立方体 横竖屏切换工具,使用陀螺仪检测手机设备方向,锁屏状... Swift版Refresh(可以自定义多种样式)架构方面有 ...

  3. iOS-自定义Model转场动画-仿酷我音乐播放器效果

    周末,闲来无事,仿写了酷我音乐播放器效果: 效果图如下: 实现思路: 1.实现手势处理视图旋转 2.自定义Model动画: 1.手势是利用了一个UIPanGestureRecognizer手势: 注意 ...

  4. Android开发实战之简单音乐播放器

    最近开始学习音频相关.所以,很想自己做一个音乐播放器,于是,花了一天学习,将播放器的基本功能实现了出来.我觉得学习知识点还是蛮多的,所以写篇博客总结一下关于一个音乐播放器实现的逻辑.希望这篇博文对你的 ...

  5. iOS电商常见动画与布局、微信悬浮窗、音乐播放器、歌词解析、拖动视图等源码

    iOS精选源码 MXScroll 介绍 混合使用UIScrollView ios 电商demo(实现各种常见动画效果和页面布局) 一行代码集成微信悬浮窗 可拖动,大小的视图,可放置在屏幕边缘. 在使用 ...

  6. IOS实现多媒体音频之音乐播放器

    随着智能手机市场越来越活跃,相应的app也变得五彩缤纷,各式各样,让你的app更吸引人多媒体技术不可避免.通过对音频和视频等控制让你的app更加丰富多彩,今天和大家一起研究下基本的音频使用.本文只提供 ...

  7. 简单的HTML5音乐播放器(带歌词滚动)

      // // 0) { this.lrcArr.push(item); } } frag = document.createDocumentFragment(); for(i = 0,len = t ...

  8. jquery音乐播放器(歌词滚动版)

    好久没出来水了!!!忙忙碌碌的找工作~然后中秋节也算过了,祝各位coding们,直接觉醒第七感小宇宙,直接用心就能找到bug-_-// 最后如题这是一篇很正规的coding的文章 大概么比以前的加了个 ...

  9. Android自定义View,高仿QQ音乐歌词滚动控件!

    最近在以QQ音乐为样板做一个手机音乐播放器,源码下篇博文放出.今天我想聊的是这个QQ音乐播放器中歌词显示控件的问题,和小伙伴们一起来探讨怎么实现这个歌词滚动的效果.OK,废话不多说,先来看看效果图: ...

随机推荐

  1. SSM商城项目(五)

    1.   学习计划 1.前台系统搭建 2.商城首页展示 3.Cms系统的实现 a)         内容分类管理 b)         内容管理 4.前台内容动态展示 2.   商城首页展示 2.1. ...

  2. prometheus 表达式

    avg_over_time(my_inprogress_requests{job="mhc"}[5m] offset 3m) 返回time=1550664637开始向前偏移3分钟之 ...

  3. python--第十二天总结(Python操作 RabbitMQ、Redis、Memcache、SQLAlchemy)

    Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度 ...

  4. Linux上web服务器搭建

    安装php依赖包: yum -y install gcc gcc++ libxml2 libxml2-devel yum install gcc make gd-devel libjpeg-devel ...

  5. m文件转换c代码

    parametet.mclc; clear; load('src.mat') CZT_N = ; CZT_M = ; CZT_W = exp(-j*(*pi/)); CZT_A = exp(j**pi ...

  6. jqGrid的userData的用法!!!

    在一次项目中想从后台自定义一些返回值传回jqGrid,所以就想到了jqGrid的这个userData属性,但是真的是坑了我好惨,这里记录一下! 1.首先看说明,这个jsonReader的默认配置,us ...

  7. [leetcode]21. Merge Two Sorted Lists合并两个链表

    Merge two sorted linked lists and return it as a new list. The new list should be made by splicing t ...

  8. JAVAEE——SpringBoot配置篇:配置文件、YAML语法、文件值注入、加载位置与顺序、自动配置原理

    转载 https://www.cnblogs.com/xieyupeng/p/9664104.html @Value获取值和@ConfigurationProperties获取值比较   @Confi ...

  9. sql语句-单表查询

    一:单表查询 CREATE TABLE `Score`( `s_id` ), `c_id` ), `s_score` ), PRIMARY KEY(`s_id`,`c_id`) ); ); ); ); ...

  10. Html5学习笔记:图片上传

    图片上传是业务需求中常见的功能,最基础的是单图片的上传.比较复杂的多图片上传,都是基于单图片上传. Form表单上传 h5的原生表单上传图片,代码如下: <!DOCTYPE html> & ...