iOS从零开始学习直播之音频4.歌词
上一篇讲了歌曲的切换,这一篇主要讲歌词部分的实现。
先看效果图。当歌手唱到这句歌词时候,我们要标记出来,这里显示字体为黄色。

1.获取歌词
一般歌词都是一个链接。类似于“http://musicdata.baidu.com/data2/lrc/131707548/131707548.lrc” ,我们从服务器下载下来,显示到页面上。每句歌词后面都有“\n”,前面都有时间。这个链接有可能直接下载,看不到下面的效果。换个浏览器多试几次就可以了,尽量使用谷歌浏览器。

#pragma mark - 获取歌词
- (void)getAlbumLrc {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
// 异步并发下载歌曲
NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:_currentModel.lrclink]];
// 二进制转为字符串
NSString *allLrcStr = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
// 分割字符串
NSArray *lrcArray = [allLrcStr componentsSeparatedByString:@"\n"];
// 添加到数组中
[self.lrcArr removeAllObjects];
for (NSString *lrc in lrcArray) {
FHLrcModel *lrcModel = [FHLrcModel allocLrcModelWithLrc:lrc];
[self.lrcArr addObject:lrcModel];
}
// 主线程更新UI
dispatch_async(dispatch_get_main_queue(), ^{
[_lrcTableView reloadData];
});
});
}
我用FHLrcModel管理每句歌词。
FHLrcModel.h
#import <Foundation/Foundation.h>
@interface FHLrcModel : NSObject
@property (nonatomic, copy) NSString *lrc; // 歌词
@property (nonatomic, assign) int presenTime; //显示这句歌词的时间
@property (nonatomic, assign) bool isPresent; // 当前显示的是否是这句歌词
// 实例化方法
+ (instancetype)allocLrcModelWithLrc: (NSString *)lrc;
@end
FHLrcModel.m
#import "FHLrcModel.h"
@implementation FHLrcModel
+ (instancetype)allocLrcModelWithLrc: (NSString *)lrc {
FHLrcModel *model =[FHLrcModel new];
// 把歌词和时间分割开
NSArray *array = [lrc componentsSeparatedByString:@"]"];
// 处理时间 [00:01.70] =》 1.70
NSString *timeStr;
if ([array[0] length] >8) {
timeStr = [array[0] substringWithRange:NSMakeRange(1, 8)];
}
NSArray *timeArr = [timeStr componentsSeparatedByString:@":"];
if (timeArr.count > 1) {
model.presenTime = (int)[timeArr[0] floatValue] * 60 + [timeArr[1] intValue];
}
// 如果没有歌词 就换行
if (array.count > 1) {
model.lrc = array[1];
}else {
model.lrc = @"\n";
}
return model;
}
@end
2.显示歌词
显示歌词显然用UITableView了,这个很简单。实例化UITableView的方法我就不贴出来了,我只贴它的代理部分的代码。
#pragma mark - UITableView
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
return self.lrcArr.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell *cell = [_lrcTableView dequeueReusableCellWithIdentifier:@"LrcCell" forIndexPath:indexPath];
// 去掉点击效果
cell.selectionStyle = UITableViewCellSelectionStyleNone;
cell.backgroundColor = [UIColor clearColor];
FHLrcModel *lrcmodel = self.lrcArr[indexPath.row];
cell.textLabel.text = lrcmodel.lrc;
cell.textLabel.textAlignment = NSTextAlignmentCenter;
if (lrcmodel.isPresent) {
// 当前歌词显示黄色
cell.textLabel.textColor = [UIColor yellowColor];
}else {
cell.textLabel.textColor = [UIColor whiteColor];
}
return cell;
}
3.同步歌词
这部分挺有难度。我走了很多弯路。后来我想了一个办法,当前播放时间和歌词数组里的第一个元素去比较,比较成功了,再去比较第二个元素。这样就避免了用for循环每一次都比较全部,这样节省内存和时间。这样做是因为我发现了歌词的时间从上往下是不断增加的,没有歌词复用的情况。至于怎么监听播放进度可以看我前面的播放。
#pragma mark - 添加监听播放进度的观察者
- (void)addTimePlayProgerssObserver {
__block UISlider *weakPregressSlider = _playerSlider;
__weak UILabel *waekCurrentLabel = _currentLabel;
__block int weakRow = _row;// 来记录当前歌词显示到第几行
__weak typeof(self) weakSelf = self;
self.timePlayProgerssObserver = [self.avPlayer addPeriodicTimeObserverForInterval:CMTimeMake(1.0, 1.0) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {
// 当前播放的时间
float current = CMTimeGetSeconds(time);
// ***更新歌词***
if (weakRow < weakSelf.lrcArr.count) {
FHLrcModel *model = weakSelf.lrcArr[weakRow];
// 比较时间 比较成功了刷新TabelView
if (model.presenTime == (int)current) {
[weakSelf reloadTabelViewWithRow:weakRow];
weakRow++;
}
}
// 总时间
float total = CMTimeGetSeconds(weakSelf.avPlayer.currentItem.duration);
// 更改当前播放时间
NSString *currentSStr = [weakSelf FormatTime: (int)current % 60];
waekCurrentLabel.text = [NSString stringWithFormat:@"%d:%@",(int)current / 60,currentSStr];
// 更新播放进度条
weakPregressSlider.value = current / total;
}];
}
备注: 主要看更新歌词部分,其他可以略过。
#pragma mark - 更新歌词
- (void)reloadTabelViewWithRow:(int)row {
// 找到cell
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:row inSection:0];
UITableViewCell *cell = [_lrcTableView cellForRowAtIndexPath:indexPath];
// 字体变色
cell.textLabel.textColor = [UIColor yellowColor];
// 当前歌词滑动到TableView中间
[_lrcTableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionMiddle animated:YES];
// 上一句变为白色 如果是第一句就没有上一句,所以不操作
if (row > 0) {
UITableViewCell *preCell = [_lrcTableView cellForRowAtIndexPath:[NSIndexPath indexPathForRow:row - 1 inSection:0]];
preCell.textLabel.textColor = [UIColor whiteColor];
}
}
这样就可以把当前演唱的歌词实时标记出来了,还可以随着演唱往上滑动。这个一定有其他的方法,大家可以和我探讨,只有交流才能知道自己的不足,才能不断进步。项目地址GitUp ,欢迎下载。
iOS从零开始学习直播之音频4.歌词的更多相关文章
- iOS从零开始学习直播之音频1.播放本地音频文件
现在直播越来越火,俨然已经成为了下一个红海.作为一个资深码农(我只喜欢这样称呼自己,不喜欢别人这样称呼我),我必须赶上时代的潮流,开始研究视频直播.发现视屏直播类的文章上来就讲拉流.推流.采集.美 ...
- iOS从零开始学习直播之音频3.歌曲切换
上周迟到了,周末去参加OSC源创会了,还是有点启发的.但这不是重点,重点是 上一篇我只是实现了一首歌曲的在线播放,这肯定是不够的.这一篇博客主要是实现了多首歌曲的顺序播放以及上一首和下一首切换. ...
- iOS从零开始学习直播之音频2.后台播放和在线播放
本篇主要讲音频的后台播放和在线播放. 后台播放 上一篇写的工程运行之后程序退至后台,发现运行不了,歌停止了,这显然不行,音乐后台播放是标配啊.今天就来讲一下后台播放. 1.在plist文件里,告诉 ...
- iOS从零开始学习直播之2.采集
直播的采集由采集的设备(摄像头.话筒)不同分为视频采集和音频采集,本篇文章会分别介绍. 1.采集步骤 1.创建捕捉会话(AVCaptureSession),iOS调用相机和话筒之前都需要创建捕 ...
- iOS从零开始学习直播之1.播放
对于直播来说,客户端主要做两件事情,推流和播放.今天先讲播放. 播放流程 1.拉流:服务器已有直播内容,从指定地址进行拉取的过程.其实就是向服务器请求数据. 2.解码:对视屏数据进行解压缩. 3. ...
- iOS从零开始学习直播之3.美颜
任何一款直播软件都必须进行美颜,不然哪来的那么多美女,所以技术改变世界,不只是说说而已.美颜在采集的时候就得就行,让主播实时看到直播的效果. 1.美颜原理 其实美颜的本质就是美白和磨皮,分别通 ...
- iOS 直播-获取音频(视频)数据
iOS 直播-获取音频(视频)数据 // // ViewController.m // capture-test // // Created by caoxu on 16/6/3. // Copyri ...
- ios网络学习------6 json格式数据的请求处理
ios网络学习------6 json格式数据的请求处理 分类: IOS2014-06-30 20:33 471人阅读 评论(3) 收藏 举报 #import "MainViewContro ...
- 从零开始学习CocoaPods安装和使用
从零开始学习CocoaPods安装和使用 转载: Code4App原创:http://code4app.com/article/cocoapods-install-usage http://m.i ...
随机推荐
- Unity插件之plyGame教程:DiaQ对话系统
本文为孤月蓝风编写,转载请注明出处:http://fengyu.name/?cat=game&id=296 DiaQ是plyGame旗下的一款对话及任务系统.拥有可视化的对话及任务编辑器,能够 ...
- CSS3变形记(上):千变万化的Div
传统上,css就是用来对网页进行布局和渲染网页样式的.然而,css3的出现彻底打破了这一格局.了解过css3的人都知道,css3不但可以对网页进行布局和渲染样式,还可以绘制一些图形.对元素进行2D和3 ...
- 分享一个与ABP配套使用的代码生成器源码
点这里进入ABP系列文章总目录 分享一个与ABP配套使用的代码生成器源码 真对不起关注我博客的朋友, 因最近工作很忙, 很久没有更新博客了.以前答应把自用的代码生成器源码共享出来, 也一直没有时间整理 ...
- WCF学习之旅—第三个示例之一(二十七)
一.前言 通过前面二十几个章节的学习,我们知道了什么是WCF:WCF中的A.B.C:WCF的传输模式:WCF的寄宿方式:WCF的异常处理.本文综合应用以上知识点,一步一步写一个小的WCF应用程序——书 ...
- JavaScript权威设计--Window对象(简要学习笔记十三)
1.Window对象是所有客户端JavaScript特性和API的主要接入点. Window对象中的一个重要属性是document,它引用Document对象. JavaScript程序可以通过Doc ...
- 在C#里面给PPT添加注释
平常开会或者做总结报告的时候我们通常都会用到PowerPoint演示文稿,我们可以在单个幻灯片或者全部幻灯片里面添加注释,这样观众可以从注释内容里面获取更多的相关信息. 有些朋友不清楚如何在幻灯片里面 ...
- 计算机网络学习笔记--网络层之IP地址与子网
IPv4地址: 我们知道在网络层(TCP/IP体系结构的网际互联层),最重要的一个协议就是IP协议,现在正处于IPv4和IPv6的过渡时期,但目前来说,IPv4仍为主流,所以主要讲Ipv4. IP地址 ...
- 你真的会玩SQL吗?和平大使 内连接、外连接
你真的会玩SQL吗?系列目录 你真的会玩SQL吗?之逻辑查询处理阶段 你真的会玩SQL吗?和平大使 内连接.外连接 你真的会玩SQL吗?三范式.数据完整性 你真的会玩SQL吗?查询指定节点及其所有父节 ...
- NSIS 打包脚本基础
简介 NSIS(Nullsoft Scriptable Install System)是一个开源的 Windows 系统下安装程序制作程序.它提供了安装.卸载.系统设置.文件解压缩等功能.这如其名字所 ...
- android手机旋转屏幕时让GridView的列数与列宽度自适应
无意中打开了一年前做过的一个android应用的代码,看到里面实现的一个小功能点(如题),现写篇文章做个笔记.当时面临的问题是,在旋转屏幕的时候需要让gridview的列数与宽度能自适应屏幕宽度,每个 ...