iOS 录音,播放并上传

2023-05-16

1.界面布局,以及相关功能

点击中间开始录音,点击左上角播放或暂停播放,点击右上角移除文件


2.定义相关属性


#import "SendVoiceController.h"

#import#import "RecordVoiceView.h"

#import "lame.h"

#import "PlayVoiceView.h"

#define cafFilePathName @"myRecordForCaf.caf"

#define mp3FilePathName @"myRecordForMp3.mp3"

@interface SendVoiceController ()//录音存储路径

@property (nonatomic, strong)NSURL *tmpFile;

//录音

@property (nonatomic, strong)AVAudioRecorder *recorder;

//播放

@property (nonatomic, strong)AVAudioPlayer *player;

//录音动画

@property(nonatomic,strong)RecordVoiceView *recordview;

@property(strong,nonatomic)PlayVoiceView *playview;

//录音计时器

@property(nonatomic,strong)NSTimer *recordTimer;

//录音秒数

@property(strong,nonatomic)NSTimer *timer;

@property(nonatomic,assign)int index;

@property(nonatomic,assign)int duration;

@end

@implementation SendVoiceController

3.创建audiosession

-(void)setAVRecorder{

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

urlStr=[urlStr stringByAppendingPathComponent:cafFilePathName];

self.tmpFile = [NSURL fileURLWithPath:urlStr];

//设置后台播放

AVAudioSession *session = [AVAudioSession sharedInstance];

NSError *sessionError;

//AVAudioSessionCategoryPlayAndRecord,这个确保了既能录音也能播放

[session setCategory:AVAudioSessionCategoryPlayAndRecord error:&sessionError];

//判断后台有没有播放

if (session == nil) {

NSLog(@"Error creating sessing:%@", [sessionError description]);

} else {

[session setActive:YES error:nil];

}

}

4.中间图片手势代理方法

//长按事件的实现方法

- (void) handleTableviewCellLongPressed:(UILongPressGestureRecognizer *)gestureRecognizer {

if (gestureRecognizer.state ==

UIGestureRecognizerStateBegan) {

NSLog(@"UIGestureRecognizerStateBegan");

[self recorderPressing];

}

if (gestureRecognizer.state ==

UIGestureRecognizerStateChanged) {

NSLog(@"UIGestureRecognizerStateChanged");

}

if (gestureRecognizer.state == UIGestureRecognizerStateEnded) {

NSLog(@"UIGestureRecognizerStateEnded");

[self recorderEnd];

}

}


5.创建AVAudioRecorder并开始录音

-(void)recorderPressing{

NSDictionary *settingdic = [NSDictionary dictionaryWithObjectsAndKeys:

[NSNumber numberWithFloat:8000],                  AVSampleRateKey, // 电话所用采样率

[NSNumber numberWithInt:kAudioFormatLinearPCM],    AVFormatIDKey,

[NSNumber numberWithInt:2],                      AVNumberOfChannelsKey,

[NSNumber numberWithInt:16],                      AVLinearPCMBitDepthKey,

[NSNumber numberWithInt:AVAudioQualityMin],      AVEncoderAudioQualityKey,

nil];

//开始录音,将所获取到得录音存到文件里

//开始录音,将所获取到得录音存到文件里

self.recorder = [[AVAudioRecorder alloc] initWithURL:_tmpFile settings:settingdic error:nil];

self.recorder.meteringEnabled = YES;

self.recorder.delegate=self;

//准备记录录音

[self.recorder prepareToRecord];

//启动或者恢复记录的录音文件

[self.recorder record];

self.player = nil;

//以下是我仿照卫星发语音的动画效果

self.recordview.hidden=NO;

self.recordTimer=[NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(recordTimerChange) userInfo:nil repeats:YES];

}

6.中间图片长按取消的同时关闭录音

-(void)recorderEnd{

self.duration=self.index;

self.index=0;

//停止录音

if (self.recordTimer) {

[self.recordTimer invalidate];

self.recordTimer=nil;

}

[self.recorder stop];

self.recordview.hidden=YES;

self.recorder = nil;

self.playBtn.hidden=NO;

self.removeBtn.hidden=NO;

self.curveLineImage.hidden=NO;

NSError *playError;

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

urlStr=[urlStr stringByAppendingPathComponent:mp3FilePathName];

NSURL *url = [NSURL fileURLWithPath:urlStr];

self.player = [[AVAudioPlayer alloc] initWithContentsOfURL:url error:&playError];

//当播放录音为空, 打印错误信息

if (self.player == nil) {

NSLog(@"Error crenting player: %@", [playError description]);

}

self.player.delegate = self;

//    [self.player play];

}


7.与其同时触发avaudiorecord代理方法

这里面的主要作用是把录音的caf文件转成mp3文件,用于上传到服务器,其中的方法是来自于Libmo3lame库

#pragma mark---录音代理

- (void)audioRecorderDidFinishRecording:(AVAudioRecorder *)recorder successfully:(BOOL)flag{

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

NSString *cafFilePath =[urlStr stringByAppendingPathComponent:cafFilePathName];    //caf文件路径

NSString *mp3FilePath = [urlStr stringByAppendingPathComponent:mp3FilePathName]; ;//存储mp3文件的路径

NSFileManager * fileManager=[ NSFileManager defaultManager ];

if ([fileManager fileExistsAtPath:cafFilePath]){

float msize= [[fileManager attributesOfItemAtPath:cafFilePath error:nil] fileSize]/(1024.0*1024.0);

NSLog(@"caf文件大小%lf",msize);

}else{

float msize= [[fileManager attributesOfItemAtPath:mp3FilePathName error:nil] fileSize]/(1024.0*1024.0);

NSLog(@"mp3文件大小%lf",msize);

}

if ([fileManager removeItemAtPath :mp3FilePath error : nil ])

{

NSLog ( @" 删除 " );

}

@try {

int read, write;

FILE *pcm = fopen ([cafFilePath cStringUsingEncoding : 1 ], "rb" );  //source 被 转换的音频文件位置

if (pcm == NULL )

{

NSLog ( @"file not found" );

}

else

{

fseek (pcm, 4 * 1024 , SEEK_CUR );                                  //skip file header

FILE *mp3 = fopen ([mp3FilePath cStringUsingEncoding : 1 ], "wb" );  //output 输出生成的 Mp3 文件位置

const int PCM_SIZE = 8192 ;

const int MP3_SIZE = 8192 ;

short int pcm_buffer[PCM_SIZE* 2 ];

unsigned char mp3_buffer[MP3_SIZE];

lame_t lame = lame_init ();

lame_set_num_channels (lame, 2 ); // 设置 1 为单通道,默认为 2 双通道

lame_set_in_samplerate (lame,  8000.0 ); //11025.0

//lame_set_VBR(lame, vbr_default);

lame_set_brate (lame, 16);

lame_set_mode (lame, 3 );

lame_set_quality (lame, 2 ); /* 2=high 5 = medium 7=low 音 质 */

lame_init_params (lame);

do {

read = fread (pcm_buffer, 2 * sizeof ( short int ), PCM_SIZE, pcm);

if (read == 0 )

write = lame_encode_flush (lame, mp3_buffer, MP3_SIZE);

else

write = lame_encode_buffer_interleaved (lame, pcm_buffer, read, mp3_buffer, MP3_SIZE);

fwrite (mp3_buffer, write, 1 , mp3);

} while (read != 0 );

lame_close (lame);

fclose (mp3);

fclose (pcm);

return  ;

}

return  ;

}

@catch (NSException *exception) {

NSLog ( @"%@" ,[exception description ]);

return  ;

}

@finally {

NSLog ( @" 执行完成 " );

}

}

8.左上角按钮事件

-(IBAction)playerClick:(id)sender{

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

NSString * path=[urlStr stringByAppendingPathComponent:cafFilePathName];

NSFileManager* manager = [NSFileManager defaultManager];

if ([manager fileExistsAtPath:path]){

float msize= [[manager attributesOfItemAtPath:path error:nil] fileSize]/(1024.0*1024.0);

NSLog(@"文件大小%lf",msize);

}

else{

_player=nil;

}

//判断是否正在播放,如果正在播放

if ([self.player isPlaying]) {

//暂停播放

[_player pause];

[self.playBtn  setImage:[UIImage imageNamed:@"voice_play"] forState:UIControlStateNormal];

//按钮显示为播放

} else {

//开始播放

[_player play];

[self.playBtn  setImage:[UIImage imageNamed:@"voice_pause"] forState:UIControlStateNormal];

}

}


9.右上角按钮事件

-(IBAction)removeClick:(id)sender{

NSIndexPath *indexpath=[NSIndexPath indexPathForItem:6 inSection:0];

self.playBtn.hidden=YES;

self.removeBtn.hidden=YES;

self.curveLineImage.hidden=YES;

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

NSString *cafFilePath =[urlStr stringByAppendingPathComponent:cafFilePathName];    //caf文件路径

NSString *mp3FilePath = [urlStr stringByAppendingPathComponent:mp3FilePathName]; ;//存储mp3文件的路径

NSFileManager * fileManager=[ NSFileManager defaultManager ];

if ([fileManager removeItemAtPath :mp3FilePath error : nil ])

{

NSLog ( @" 删除 mp3" );

}

if ([fileManager removeItemAtPath :cafFilePath error : nil ])

{

NSLog ( @" 删除 caf" );

}

}

10.上传到服务器

主要的逻辑是把MP3文件转成nsdata,然后通过base64string 转成字符串传给后台

NSString *url=[[HJInterfaceManager sharedInstance]addRichInfo];

NSMutableDictionary *mdic=[[NSMutableDictionary alloc]initWithCapacity:0];

NSString *urlStr=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

NSString *mp3FilePath = [urlStr stringByAppendingPathComponent:mp3FilePathName]; ;//

//    NSString *imagePath = [[NSBundle mainBundle] pathForResource:@"1" ofType:@"png"];

NSData *filedata=[NSData dataWithContentsOfFile:mp3FilePath];

SelfDataModel *selfModel=[SelfDataModel returnModelBySelectFMDB];

//        * @param familyId

//        * @param userId

//        * @param content

//        * @param richContent

//        * @param type

//        * @param sendTime

NSDate *date=[NSDate date];

NSDateFormatter *form=[[NSDateFormatter alloc]init];

[form setDateFormat:@"YYYY-MM-dd HH:mm:ss"];

NSString *datestr=[form stringFromDate:date ];

if (!self.isMyself) {

[mdic setObject:self.family.idNum forKey:@"familyId"];

}

[mdic setObject:selfModel.idNum forKey:@"userId"];

[mdic setObject:@"" forKey:@"content"];

[mdic setObject:[NSString stringWithFormat:@"%d",self.duration] forKey:@"duration"];

[mdic setObject:[filedata base64Encoding] forKey:@"richContents"];

[mdic setObject:@"SOUND" forKey:@"type"];

[mdic setObject:datestr forKey:@"sendTime"];

[HJHttpManager PostRequestWithUrl:url param:mdic finish:^(NSData *data) {

NSDictionary *dic=(NSDictionary *)data;

if ([dic[@"status"] isEqualToString:@"S"]) {

[MBProgressHUD showSuccess:@"发送成功"];

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

[self.navigationController popViewControllerAnimated:YES];

});

}else{

[MBProgressHUD showError:dic[@"message"]];

}

} failed:^(NSError *error) {

NSLog(@"请求失败");

}];


小帖士:

以下是监听声音分贝的方法

#pragma mark----监听声音分贝

/* 该方法确实会随环境音量变化而变化,但具体分贝值是否准确暂时没有研究 */

- (void)recordTimerChange {

//    [self.recorder updateMeters];//刷新音量数据

//    double lowPassResults = pow(10, (0.05 * [self.recorder peakPowerForChannel:0]));

//    NSLog(@"大小:%lf",lowPassResults);

self.recordview.timeLab.text=[NSString stringWithFormat:@"%d s",self.index];

self.index++;

[self.recorder updateMeters];

float  level;                // The linear 0.0 .. 1.0 value we need.

float  minDecibels = -80.0f; // Or use -60dB, which I measured in a silent room.

float  decibels    = [self.recorder averagePowerForChannel:0];

if (decibels < minDecibels)

{

level = 0.0f;

}

else if (decibels >= 0.0f)

{

level = 1.0f;

}

else

{

float  root            = 2.0f;

float  minAmp          = powf(10.0f, 0.05f * minDecibels);

float  inverseAmpRange = 1.0f / (1.0f - minAmp);

float  amp            = powf(10.0f, 0.05f * decibels);

float  adjAmp          = (amp - minAmp) * inverseAmpRange;

level = powf(adjAmp, 1.0f / root);

}

float ben=level*120;

if (ben<20) {

[self.recordview.imageview setImage:[UIImage imageNamed:@"order_voice_1"]];

}

else if (ben<35) {

[self.recordview.imageview setImage:[UIImage imageNamed:@"order_voice_2"]];

}

else{

[self.recordview.imageview setImage:[UIImage imageNamed:@"order_voice_3"]];

}

NSLog(@"大小%lf",ben);

}


本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

iOS 录音,播放并上传 的相关文章

  • TestFlight 提供反馈按钮

    我正在使用 iOS 8 的最新 testflight 版本 我将自己添加为内部测试人员 现在当我使用 testflight 打开应用程序时 我找不到反馈按钮 如果有人有任何线索 请告诉我 您在 Testflight 应用程序中提供反馈 打开
  • 为什么在尝试编译此代码时会收到错误“错误:未知类型名称'虚拟'”?

    Code struct IRenderingEngine virtual void Initialize int width int height 0 virtual void Render const 0 virtual void Upd
  • 从xcode上触摸屏的坐标获取ImageView的像素数据?

    单击视图并从视图内的图像获取正确的像素数据似乎存在问题 func handleTap gestureRecognizer UIGestureRecognizer print You tapped at gestureRecognizer l
  • 使用远程图像创建 MSSticker

    我正在尝试找出使用网络上托管的图像创建 MSStickers 的方法 我可以使用本地图像创建 MSStickers 例如 NSString imagePath NSBundle mainBundle pathForResource imag
  • UIStackView分布均匀填充

    所以 我有一个UIStackView其中包含四 4 UIViews 如果我删除其中一 1 个UIViews 其他三 3 个将填满UIStackView 我的问题 如何添加最大高度UIView这样它就不会填满整个空间UIStackView即使
  • 无法从 iOS 中的框架访问 .nib(XIB) 文件

    我已经从现有的代码库中创建了一个框架 并尝试在新的代码库中使用它 这很好用 但是当我尝试访问属于我的框架包的一部分的 nib 文件时 我的应用程序崩溃了 这是我用来访问视图控制器 XIB 文件的代码 testViewController c
  • 使用 NSString 进行 UTF8 解码

    我是 Objective C 新手 尝试使用以下示例将格式错误的 UTF8 编码 NSString 转换为格式良好的字符串苹果文档 http developer apple com library mac documentation Coc
  • 如何从 ContentView 外部显示 SwiftUI 警报?

    我正在构建 Swift 应用程序 并试图找出如何显示警报 我有一个单独的 swift 文件正在执行一些计算 并且在某些条件下我希望它向用户显示警报 基本上告诉他们出了问题 然而 我见过的大多数例子都要求警报在ContentView或以其他方
  • `navigator.geolocation.getCurrentPosition()` 在 iOS PWA 上挂起

    我有这个片段 const getCurrentPosition gt new Promise
  • 从 UIPickerView 的选定行设置 UIButton 的标题

    详细场景是这样的 我使用循环创建 10 个按钮并设置 0 9 的标签 点击每个按钮时 我将调用 UIPickerView 在其中加载来自不同数组的数据 到这里我就得到了预期的结果 但我希望 pickerView 中选定的行应设置为相应按钮的
  • 如何制作带有 SWIFT 图像的弹出窗口

    我想知道如何制作类似于此示例的弹出窗口 原始窗口充满了按钮 选择这些按钮后将拉出我想要使用的图像 我会简单地创建一个可重用的UIView组件以及作为子视图所需的一切 例如UIImageView为了你的形象 UILabel or a UIBu
  • ios swift parse:从 3 个类收集数据

    我有这样的结构 User CardSet 带有指向 User objectId 的指针 user 和 col name 带有点 cards 的卡片到 Card Set objectId 和列 name 我想选择所有卡数据 包括当前用户的卡集
  • iOS:生成pdf时绘制文本时如何设置字体?

    我在ios应用程序中使用drawpdf函数生成pdf 同时调用nsobject类中的drawtext函数 它根据我指定的框架和字符串清楚地绘制文本 我的代码是 void drawText NSString textToDraw inFram
  • iOS 7 上 Safari 浏览器的用户代理

    我只想在带有 Safari 浏览器的 iPhone 和 iPod 中打开我的网站 对于 Chrome Dolphin 等任何其他浏览器 它不应该打开 但目前我从几乎所有设备获得相同的用户代理 对于Safari User Agent Stri
  • 调整 UIImage 的大小而不将其完全加载到内存中?

    我正在开发一个应用程序 用户可以在其中尝试加载非常非常大的图像 这些图像首先在表格视图中显示为缩略图 我的原始代码会在大图像上崩溃 因此我重写它以首先将图像直接下载到磁盘 是否有一种已知的方法可以调整磁盘上图像的大小 而无需通过以下方式将其
  • 贴纸包会在模拟器上使 iMessage 崩溃,但在 iPhone 上不会崩溃

    按照 Apple 的在线说明和视频在 Xcode 中创建了一个贴纸包 所有图像的尺寸均正确且远低于文件大小阈值 如果我在我的实体 iPhone 上构建并运行贴纸包 一切都会完美运行 如果我在模拟器上构建并运行贴纸包 对于任何模拟的 iPho
  • watchOS 错误:控制器接口描述中的未知属性

    我将 WKInterfacePicker 添加到情节提要中 并将其连接到界面控制器中的 IBOutlet 运行应用程序时 它在控制台中显示一条错误消息 控制器的接口描述 watchPicker 中的未知属性 Code interface I
  • Objective-C 中发送给对象的消息可以被监听或者打印出来吗? [复制]

    这个问题在这里已经有答案了 可能的重复 Objective C 中拦截方法调用 https stackoverflow com questions 1618474 intercept method call in objective c 如
  • 像 TraceGL 一样分析 Objective C 中的代码路径?

    TraceGL 是一个非常简洁的项目 它允许 JS 程序员跟踪 Javascript 中的代码路径 它看起来像这样 我想为 Objective C 构建类似的东西 我知道运行时使跟踪方法调用变得相当容易 但是我如何跟踪控制流 例如 在上面的
  • 隐藏选项卡栏项目并对齐其他选项卡项目

    在我的应用程序中 我有 4 个选项卡栏项目 我正在 XIB 文件中添加这 4 个选项卡栏项目 最初我必须显示 3 个选项卡栏项目 同步后我必须在我的应用程序中显示第 4 个选项卡栏项目 因此 为此 我使用以下代码隐藏第四个选项卡栏项目 se

随机推荐