iOS开发拓展篇—音频处理(音乐播放器5)

实现效果:

一、半透明滑块的设置

 /**
*拖动滑块
*/
- (IBAction)panSlider:(UIPanGestureRecognizer *)sender { //1.获得挪动的距离
CGPoint t=[sender translationInView:sender.view];
//把挪动清零
[sender setTranslation:CGPointZero inView:sender.view]; //2.控制滑块和进度条的frame
CGFloat sliderMaxX=self.view.width-self.slider.width;
self.slider.x+=t.x;
//控制滑块的frame,不让其越界
if(self.slider.x<)
{
self.slider.x=;
}else if (self.slider.x>sliderMaxX)
{
self.slider.x=sliderMaxX;
}
//设置进度条的宽度
self.progressView.width=self.slider.center.x; //3.设置时间值
double progress=self.slider.x/sliderMaxX;
//当前的时间值=音乐的时长*当前的进度值
NSTimeInterval time=self.player.duration*progress;
[self.slider setTitle:[self strWithTime:time] forState:UIControlStateNormal]; //设置拖拽进度的X的值
self.currentTimeView.x=self.slider.x;
[self.currentTimeView setTitle:self.slider.currentTitle forState:UIControlStateNormal]; //4.如果开始拖动,那么就停止定时器
if (sender.state==UIGestureRecognizerStateBegan) {
//停止定时器
[self removeCurrentTime]; //设置拖拽进度
//显示
self.currentTimeView.hidden=NO;
self.currentTimeView.y=self.currentTimeView.superview.height--self.currentTimeView.height; }else if(sender.state==UIGestureRecognizerStateEnded)
{
//隐藏
self.currentTimeView.hidden=YES;
//设置播放器播放的时间
self.player.currentTime=time;
#warning 如果正在播放,才需要添加定时器
// if (self.player.isPlaying) {
//开启定时器
[self addCurrentTimeTimer];
// }
}
}

裁剪圆角的细节处理:

  

二、播放或暂停、上一首、下一首的实现

 //上一首
- (IBAction)previous {
//1.在开始播放之前,禁用一切的app点击事件
UIWindow *window=[[UIApplication sharedApplication].windows lastObject];
window.userInteractionEnabled=NO; //2.重置当前歌曲
[self resetPlayingMusic]; //3.获得上一首歌曲
[YYMusicTool setPlayingMusic:[YYMusicTool previousMusic]]; //4.播放上一首歌曲
[self starPlayingMusic]; //5.回复window的点击为可用
window.userInteractionEnabled=YES;
}
//下一首
- (IBAction)next {
//1.在开始播放之前,禁用一切的app点击事件
UIWindow *window=[[UIApplication sharedApplication].windows lastObject];
window.userInteractionEnabled=NO; //2.重置当前歌曲
[self resetPlayingMusic]; //3.获得下一首歌曲
[YYMusicTool setPlayingMusic:[YYMusicTool nextMusic]]; //4.播放下一首歌曲
[self starPlayingMusic]; //5.回复window的点击为可用
window.userInteractionEnabled=YES;
} //继续或暂停播放
- (IBAction)playOrPause {
if (self.playOrPauseButton.isSelected) {//暂停
self.playOrPauseButton.selected=NO;
//暂停播放
[YYAudioTool pauseMusic:self.playingMusic.filename];
//停掉定时器
[self removeCurrentTime];
}else
{
self.playOrPauseButton.selected=YES;
//继续播放
[YYAudioTool playMusic:self.playingMusic.filename];
//开启定时器
[self addCurrentTimeTimer];
}
}
说明:播放和暂停按钮的图片设置在两种状态下并不一样,设置播放按钮的状态
 
 
三、对存在的bug进行改进
拖拽还存在问题(定时器的问题)
  
更好的方法时在添加定时器的地方进行更细的控制:
  

 /**
* 添加一个定时器
*/
-(void)addCurrentTimeTimer
{
//如果当前没有在播放,那么就直接返回
if (self.player.isPlaying==NO) return; //在添加一个定时器之前,先把以前的定时器移除
[self removeCurrentTime]; //提前先调用一次进度更新,以保证定时器的工作时及时的
[self updateCurrentTime]; //创建一个定时器,每一秒钟调用一次
self.CurrentTimeTimer=[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(updateCurrentTime) userInfo:nil repeats:YES];
//把定时器加入到运行时中
[[NSRunLoop mainRunLoop]addTimer:self.CurrentTimeTimer forMode:NSRunLoopCommonModes];
}

四、补充

完整的代码如下:

 //
// YYPlayingViewController.m
// 20-音频处理(音乐播放器1)
//
// Created by apple on 14-8-13.
// Copyright (c) 2014年 yangyong. All rights reserved.
// #import "YYPlayingViewController.h"
#import "YYMusicTool.h"
#import "YYMusicModel.h"
#import "YYAudioTool.h" @interface YYPlayingViewController ()
//显示拖拽进度
@property (weak, nonatomic) IBOutlet UIButton *currentTimeView;
//进度条
@property (weak, nonatomic) IBOutlet UIView *progressView;
//滑块
@property (weak, nonatomic) IBOutlet UIButton *slider;
@property (weak, nonatomic) IBOutlet UIImageView *iconView;
@property (weak, nonatomic) IBOutlet UILabel *songLabel;
@property (weak, nonatomic) IBOutlet UILabel *singerLabel;
//当前播放的音乐的时长
@property (weak, nonatomic) IBOutlet UILabel *durationLabel;
//正在播放的音乐
@property(nonatomic,strong)YYMusicModel *playingMusic;
//音乐播放器对象
@property(nonatomic,strong)AVAudioPlayer *player;
//定时器
@property(nonatomic,strong)NSTimer *CurrentTimeTimer;
- (IBAction)exit;
- (IBAction)tapProgressBg:(UITapGestureRecognizer *)sender;
- (IBAction)panSlider:(UIPanGestureRecognizer *)sender;
- (IBAction)previous;
- (IBAction)playOrPause;
- (IBAction)next;
@property (weak, nonatomic) IBOutlet UIButton *playOrPauseButton; @end @implementation YYPlayingViewController -(void)viewDidLoad
{
[super viewDidLoad]; //裁剪圆角
self.currentTimeView.layer.cornerRadius=; }
#pragma mark-公共方法
-(void)show
{
//1.禁用整个app的点击事件
UIWindow *window=[UIApplication sharedApplication].keyWindow;
window.userInteractionEnabled=NO; //2.添加播放界面
//设置View的大小为覆盖整个窗口
self.view.frame=window.bounds;
//设置view显示
self.view.hidden=NO;
//把View添加到窗口上
[window addSubview:self.view]; //3.检测是否换了歌曲
if (self.playingMusic!=[YYMusicTool playingMusic]) {
[self resetPlayingMusic];
} //4.使用动画让View显示
self.view.y=self.view.height;
[UIView animateWithDuration:0.25 animations:^{
self.view.y=;
} completion:^(BOOL finished) { //设置音乐数据
[self starPlayingMusic];
window.userInteractionEnabled=YES;
}];
} #pragma mark-私有方法
//重置正在播放的音乐
-(void)resetPlayingMusic
{
//1.重置界面数据
self.iconView.image=[UIImage imageNamed:@"play_cover_pic_bg"];
self.songLabel.text=nil;
self.singerLabel.text=nil; //2.停止播放
[YYAudioTool stopMusic:self.playingMusic.filename];
//把播放器进行清空
self.player=nil; //3.停止定时器
[self removeCurrentTime]; //4.设置音乐播放按钮的状态
self.playOrPauseButton.selected=NO;
}
//开始播放音乐数据
-(void)starPlayingMusic
{
//1.设置界面数据 //如果当前播放的音乐就是传入的音乐,那么就直接返回
if (self.playingMusic==[YYMusicTool playingMusic])
{
//把定时器加进去
[self addCurrentTimeTimer];
return;
}
//存取音乐
self.playingMusic=[YYMusicTool playingMusic];
self.iconView.image=[UIImage imageNamed:self.playingMusic.icon];
self.songLabel.text=self.playingMusic.name;
self.singerLabel.text=self.playingMusic.singer; //2.开始播放
self.player = [YYAudioTool playMusic:self.playingMusic.filename]; //3.设置时长
//self.player.duration; 播放器正在播放的音乐文件的时间长度
self.durationLabel.text=[self strWithTime:self.player.duration]; //4.添加定时器
[self addCurrentTimeTimer]; //5.设置音乐播放按钮的状态
self.playOrPauseButton.selected=YES;
} /**
*把时间长度-->时间字符串
*/
-(NSString *)strWithTime:(NSTimeInterval)time
{
int minute=time / ;
int second=(int)time % ;
return [NSString stringWithFormat:@"%d:%d",minute,second];
} #pragma mark-定时器控制
/**
* 添加一个定时器
*/
-(void)addCurrentTimeTimer
{
//如果当前没有在播放,那么就直接返回
if (self.player.isPlaying==NO) return; //在添加一个定时器之前,先把以前的定时器移除
[self removeCurrentTime]; //提前先调用一次进度更新,以保证定时器的工作时及时的
[self updateCurrentTime]; //创建一个定时器,每一秒钟调用一次
self.CurrentTimeTimer=[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(updateCurrentTime) userInfo:nil repeats:YES];
//把定时器加入到运行时中
[[NSRunLoop mainRunLoop]addTimer:self.CurrentTimeTimer forMode:NSRunLoopCommonModes];
}
/**
*移除一个定时器
*/
-(void)removeCurrentTime
{
[self.CurrentTimeTimer invalidate]; //把定时器清空
self.CurrentTimeTimer=nil;
} /**
* 更新播放进度
*/
-(void)updateCurrentTime
{
//1.计算进度值
double progress=self.player.currentTime/self.player.duration; //2.计算滑块的x值
// 滑块的最大的x值
CGFloat sliderMaxX=self.view.width-self.slider.width;
self.slider.x=sliderMaxX*progress;
//设置滑块上的当前播放时间
[self.slider setTitle:[self strWithTime:self.player.currentTime] forState:UIControlStateNormal]; //3.设置进度条的宽度
self.progressView.width=self.slider.center.x; } #pragma mark-内部的按钮监听方法
//返回按钮
- (IBAction)exit { //0.移除定时器
[self removeCurrentTime];
//1.禁用整个app的点击事件
UIWindow *window=[UIApplication sharedApplication].keyWindow;
window.userInteractionEnabled=NO; //2.动画隐藏View
[UIView animateWithDuration:0.25 animations:^{
self.view.y=window.height;
} completion:^(BOOL finished) {
window.userInteractionEnabled=YES;
//设置view隐藏能够节省一些性能
self.view.hidden=YES;
}];
} /**
*点击了进度条
*/
- (IBAction)tapProgressBg:(UITapGestureRecognizer *)sender {
//获取当前单击的点
CGPoint point=[sender locationInView:sender.view];
//切换歌曲的当前播放时间
self.player.currentTime=(point.x/sender.view.width)*self.player.duration;
//更新播放进度
[self updateCurrentTime];
}
/**
*拖动滑块
*/
- (IBAction)panSlider:(UIPanGestureRecognizer *)sender { //1.获得挪动的距离
CGPoint t=[sender translationInView:sender.view];
//把挪动清零
[sender setTranslation:CGPointZero inView:sender.view]; //2.控制滑块和进度条的frame
CGFloat sliderMaxX=self.view.width-self.slider.width;
self.slider.x+=t.x;
//控制滑块的frame,不让其越界
if(self.slider.x<)
{
self.slider.x=;
}else if (self.slider.x>sliderMaxX)
{
self.slider.x=sliderMaxX;
}
//设置进度条的宽度
self.progressView.width=self.slider.center.x; //3.设置时间值
double progress=self.slider.x/sliderMaxX;
//当前的时间值=音乐的时长*当前的进度值
NSTimeInterval time=self.player.duration*progress;
[self.slider setTitle:[self strWithTime:time] forState:UIControlStateNormal]; //设置拖拽进度的X的值
self.currentTimeView.x=self.slider.x;
[self.currentTimeView setTitle:self.slider.currentTitle forState:UIControlStateNormal]; //4.如果开始拖动,那么就停止定时器
if (sender.state==UIGestureRecognizerStateBegan) {
//停止定时器
[self removeCurrentTime]; //设置拖拽进度
//显示
self.currentTimeView.hidden=NO;
self.currentTimeView.y=self.currentTimeView.superview.height--self.currentTimeView.height; }else if(sender.state==UIGestureRecognizerStateEnded)
{
//隐藏
self.currentTimeView.hidden=YES;
//设置播放器播放的时间
self.player.currentTime=time;
#warning 如果正在播放,才需要添加定时器
// if (self.player.isPlaying) {
//开启定时器
[self addCurrentTimeTimer];
// }
}
} //上一首
- (IBAction)previous {
//1.在开始播放之前,禁用一切的app点击事件
UIWindow *window=[[UIApplication sharedApplication].windows lastObject];
window.userInteractionEnabled=NO; //2.重置当前歌曲
[self resetPlayingMusic]; //3.获得上一首歌曲
[YYMusicTool setPlayingMusic:[YYMusicTool previousMusic]]; //4.播放上一首歌曲
[self starPlayingMusic]; //5.回复window的点击为可用
window.userInteractionEnabled=YES;
}
//下一首
- (IBAction)next {
//1.在开始播放之前,禁用一切的app点击事件
UIWindow *window=[[UIApplication sharedApplication].windows lastObject];
window.userInteractionEnabled=NO; //2.重置当前歌曲
[self resetPlayingMusic]; //3.获得下一首歌曲
[YYMusicTool setPlayingMusic:[YYMusicTool nextMusic]]; //4.播放下一首歌曲
[self starPlayingMusic]; //5.回复window的点击为可用
window.userInteractionEnabled=YES;
}
//继续或暂停播放
- (IBAction)playOrPause {
if (self.playOrPauseButton.isSelected) {//暂停
self.playOrPauseButton.selected=NO;
//暂停播放
[YYAudioTool pauseMusic:self.playingMusic.filename];
//停掉定时器
[self removeCurrentTime];
}else
{
self.playOrPauseButton.selected=YES;
//继续播放
[YYAudioTool playMusic:self.playingMusic.filename];
//开启定时器
[self addCurrentTimeTimer];
}
} @end

iOS开发拓展篇—音频处理(音乐播放器5)的更多相关文章

  1. iOS开发——高级篇——音频、音乐播放(封装类)

    一.简介 简单来说,音频可以分为2种音效又称“短音频”,通常在程序中的播放时长为1~2秒在应用程序中起到点缀效果,提升整体用户体验 音乐比如游戏中的“背景音乐”,一般播放时间较长 播放音频可以使用框架 ...

  2. iOS开发拓展篇—音频处理(音乐播放器1)

    iOS开发拓展篇—音频处理(音乐播放器1) 说明:该系列文章通过实现一个简单的音乐播放器来介绍音频处理的相关知识点,需要重点注意很多细节的处理. 一.调整项目的结构,导入必要的素材 调整后的项目结构如 ...

  3. iOS开发拓展篇—音频处理(音乐播放器2)

    iOS开发拓展篇—音频处理(音乐播放器2) 说明:该文主要介绍音乐播放界面的搭建. 一.跳转 1.跳转到音乐播放界面的方法选择 (1)使用模态跳转(又分为手动的和自动的) (2)使用xib并设置跳转 ...

  4. iOS开发拓展篇—音频处理(音乐播放器3)

    iOS开发拓展篇—音频处理(音乐播放器3) 说明:这篇文章主要介绍音频工具类和播放工具类的封装. 一.控制器间数据传递 1.两个控制器之间数据的传递 第一种方法:self.parentViewCont ...

  5. iOS开发拓展篇—音频处理(音乐播放器4)

    iOS开发拓展篇—音频处理(音乐播放器4) 说明:该文主要介绍音乐播放器实现过程中的一些细节控制. 实现的效果: 一.完整的代码 YYPlayingViewController.m文件 // // Y ...

  6. iOS开发拓展篇—音频处理(音乐播放器6)

    iOS开发拓展篇—音频处理(音乐播放器6) 一.图片处理 说明: Aspect表示按照原来的宽高比进行缩放. Aspectfit表示按照原来的宽高比缩放,要求看到全部图片,后果是不能完全覆盖窗口,会留 ...

  7. iOS开发手记-仿QQ音乐播放器动态歌词的实现

    最近朋友想做个音乐App,让我帮忙参考下.其中歌词动态滚动的效果,正好我之前也没做过,顺便学习一下,先来个预览效果. 实现思路 歌词常见的就是lrc歌词了,我们这里也是通过解析lrc歌词文件来获取其播 ...

  8. iOS开发拓展篇—音乐的播放

    iOS开发拓展篇—音乐的播放 一.简单说明 音乐播放用到一个叫做AVAudioPlayer的类,这个类可以用于播放手机本地的音乐文件. 注意: (1)该类(AVAudioPlayer)只能用于播放本地 ...

  9. iOS开发拓展篇—封装音频文件播放工具类

    iOS开发拓展篇—封装音频文件播放工具类 一.简单说明 1.关于音乐播放的简单说明 (1)音乐播放用到一个叫做AVAudioPlayer的类 (2)AVAudioPlayer常用方法 加载音乐文件 - ...

随机推荐

  1. struts2-通配符映射(基本没啥卵用)和动态调用

    通配符 使用*代表任意字符 一般在action的name中使用*,并可以使用多个 可以使用{通配符的序号}引用对应的通配符所代表的值,序号从1开始 {0}代表整个URI 匹配规则 首先完全匹配,没有完 ...

  2. html5,导航

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  3. 关于学习Knockoutjs--入门(二)

    这两天终于闲一丢丢了,可以有多点时间学习一下拉.接下来要写到的还是Knockoutjs. Knockout是建立在以下3个核心功能之上的: 1. 属性监控与依赖跟踪 2. 声明式绑定 3. 模版机制 ...

  4. Android生成一维码

    BitmapUtil.java里面添加个方法 /** * 用于将给定的内容生成成一维码 注:目前生成内容为中文的话将直接报错,要修改底层jar包的内容 * * @param content 将要生成一 ...

  5. shell调试

    sh -x set -x 中间是要调试的代码 set +x

  6. 对于JQuery的一些见解

    jQuery是什么?(了解)  www.github.com jQuery 其实就是一堆的js函数,是普通的js,只不过应用广泛,形成了行业标准. 参考书:锋利的jQuery 学习参考:http:// ...

  7. jenkins对结果进行断言问题

    TextFinder plugin插件 Jenkins在判定使用shell scripts完成build成功与否的时候,是根据shell最终的返回值是否为零来判定的:零即成功,非零即失败.这点判定事实 ...

  8. llinux 压缩 解压

    1.zip  1) 将文件夹 mydir 压缩为 mydir.zip zip -r mydir.zip mydir 2) 将文件 one.two 压缩到 ot.zip zip -r ot.zip on ...

  9. (转)MySQL命令行--导入导出数据库

    MySQL命令行导出数据库:   1,进入MySQL目录下的bin文件夹:cd MySQL中到bin文件夹的目录 如我输入的命令行:cd C:\Program Files\MySQL\MySQL Se ...

  10. 用JavaBean实现数据库的连接和关闭,在jsp页面输出数据库中student表中学生的信息

    package com.hanqi.test; import java.sql.*; public class XveSheng { Connection conn; Statement st; Re ...