iOS音频播放(二):AudioSession

(本文转自码农人生)
前言

- extern OSStatus AudioSessionInitialize(CFRunLoopRef inRunLoop,
- CFStringRef inRunLoopMode,
- AudioSessionInterruptionListener inInterruptionListener,
- void *inClientData);
- typedef void (*AudioSessionInterruptionListener)(void * inClientData, UInt32 inInterruptionState);
- extern OSStatus AudioSessionAddPropertyListener(AudioSessionPropertyID inID,
- AudioSessionPropertyListener inProc,
- void *inClientData);
- typedef void (*AudioSessionPropertyListener)(void * inClientData,
- AudioSessionPropertyID inID,
- UInt32 inDataSize,
- const void * inData);
- //AudioSession的AudioRouteChangeReason枚举
- enum {
- kAudioSessionRouteChangeReason_Unknown = 0,
- kAudioSessionRouteChangeReason_NewDeviceAvailable = 1,
- kAudioSessionRouteChangeReason_OldDeviceUnavailable = 2,
- kAudioSessionRouteChangeReason_CategoryChange = 3,
- kAudioSessionRouteChangeReason_Override = 4,
- kAudioSessionRouteChangeReason_WakeFromSleep = 6,
- kAudioSessionRouteChangeReason_NoSuitableRouteForCategory = 7,
- kAudioSessionRouteChangeReason_RouteConfigurationChange = 8
- };
- //AVAudioSession的AudioRouteChangeReason枚举
- typedef NS_ENUM(NSUInteger, AVAudioSessionRouteChangeReason)
- {
- AVAudioSessionRouteChangeReasonUnknown = 0,
- AVAudioSessionRouteChangeReasonNewDeviceAvailable = 1,
- AVAudioSessionRouteChangeReasonOldDeviceUnavailable = 2,
- AVAudioSessionRouteChangeReasonCategoryChange = 3,
- AVAudioSessionRouteChangeReasonOverride = 4,
- AVAudioSessionRouteChangeReasonWakeFromSleep = 6,
- AVAudioSessionRouteChangeReasonNoSuitableRouteForCategory = 7,
- AVAudioSessionRouteChangeReasonRouteConfigurationChange NS_ENUM_AVAILABLE_IOS(7_0) = 8
- }
- + (BOOL)usingHeadset
- {
- #if TARGET_IPHONE_SIMULATOR
- return NO;
- #endif
- CFStringRef route;
- UInt32 propertySize = sizeof(CFStringRef);
- AudioSessionGetProperty(kAudioSessionProperty_AudioRoute, &propertySize, &route);
- BOOL hasHeadset = NO;
- if((route == NULL) || (CFStringGetLength(route) == 0))
- {
- // Silent Mode
- }
- else
- {
- /* Known values of route:
- * "Headset"
- * "Headphone"
- * "Speaker"
- * "SpeakerAndMicrophone"
- * "HeadphonesAndMicrophone"
- * "HeadsetInOut"
- * "ReceiverAndMicrophone"
- * "Lineout"
- */
- NSString* routeStr = (__bridge NSString*)route;
- NSRange headphoneRange = [routeStr rangeOfString : @"Headphone"];
- NSRange headsetRange = [routeStr rangeOfString : @"Headset"];
- if (headphoneRange.location != NSNotFound)
- {
- hasHeadset = YES;
- }
- else if(headsetRange.location != NSNotFound)
- {
- hasHeadset = YES;
- }
- }
- if (route)
- {
- CFRelease(route);
- }
- return hasHeadset;
- }
- + (BOOL)isAirplayActived
- {
- CFDictionaryRef currentRouteDescriptionDictionary = nil;
- UInt32 dataSize = sizeof(currentRouteDescriptionDictionary);
- AudioSessionGetProperty(kAudioSessionProperty_AudioRouteDescription, &dataSize, ¤tRouteDescriptionDictionary);
- BOOL airplayActived = NO;
- if (currentRouteDescriptionDictionary)
- {
- CFArrayRef outputs = CFDictionaryGetValue(currentRouteDescriptionDictionary, kAudioSession_AudioRouteKey_Outputs);
- if(outputs != NULL && CFArrayGetCount(outputs) > 0)
- {
- CFDictionaryRef currentOutput = CFArrayGetValueAtIndex(outputs, 0);
- //Get the output type (will show airplay / hdmi etc
- CFStringRef outputType = CFDictionaryGetValue(currentOutput, kAudioSession_AudioRouteKey_Type);
- airplayActived = (CFStringCompare(outputType, kAudioSessionOutputRoute_AirPlay, 0) == kCFCompareEqualTo);
- }
- CFRelease(currentRouteDescriptionDictionary);
- }
- return airplayActived;
- }
- extern OSStatus AudioSessionSetProperty(AudioSessionPropertyID inID,
- UInt32 inDataSize,
- const void *inData);
- UInt32 sessionCategory = kAudioSessionCategory_MediaPlayback;
- AudioSessionSetProperty (kAudioSessionProperty_AudioCategory,
- sizeof(sessionCategory),
- &sessionCategory);
- /* set session category */
- - (BOOL)setCategory:(NSString *)category error:(NSError **)outError;
- /* set session category with options */
- - (BOOL)setCategory:(NSString *)category withOptions: (AVAudioSessionCategoryOptions)options error:(NSError **)outError NS_AVAILABLE_IOS(6_0);
- //AudioSession的AudioSessionCategory枚举
- enum {
- kAudioSessionCategory_AmbientSound = 'ambi',
- kAudioSessionCategory_SoloAmbientSound = 'solo',
- kAudioSessionCategory_MediaPlayback = 'medi',
- kAudioSessionCategory_RecordAudio = 'reca',
- kAudioSessionCategory_PlayAndRecord = 'plar',
- kAudioSessionCategory_AudioProcessing = 'proc'
- };
- //AudioSession的AudioSessionCategory字符串
- /* Use this category for background sounds such as rain, car engine noise, etc.
- Mixes with other music. */
- AVF_EXPORT NSString *const AVAudioSessionCategoryAmbient;
- /* Use this category for background sounds. Other music will stop playing. */
- AVF_EXPORT NSString *const AVAudioSessionCategorySoloAmbient;
- /* Use this category for music tracks.*/
- AVF_EXPORT NSString *const AVAudioSessionCategoryPlayback;
- /* Use this category when recording audio. */
- AVF_EXPORT NSString *const AVAudioSessionCategoryRecord;
- /* Use this category when recording and playing back audio. */
- AVF_EXPORT NSString *const AVAudioSessionCategoryPlayAndRecord;
- /* Use this category when using a hardware codec or signal processor while
- not playing or recording audio. */
- AVF_EXPORT NSString *const AVAudioSessionCategoryAudioProcessing;
- //AudioSession的启动方法
- extern OSStatus AudioSessionSetActive(Boolean active);
- extern OSStatus AudioSessionSetActiveWithFlags(Boolean active, UInt32 inFlags);
- //AVAudioSession的启动方法
- - (BOOL)setActive:(BOOL)active error:(NSError **)outError;
- - (BOOL)setActive:(BOOL)active withFlags:(NSInteger)flags error:(NSError **)outError NS_DEPRECATED_IOS(4_0, 6_0);
- - (BOOL)setActive:(BOOL)active withOptions:(AVAudioSessionSetActiveOptions)options error:(NSError **)outError NS_AVAILABLE_IOS(6_0);

- #import <Endian.h>
- NSString * OSStatusToString(OSStatus status)
- {
- size_t len = sizeof(UInt32);
- long addr = (unsigned long)&status;
- char cstring[5];
- len = (status >> 24) == 0 ? len - 1 : len;
- len = (status >> 16) == 0 ? len - 1 : len;
- len = (status >> 8) == 0 ? len - 1 : len;
- len = (status >> 0) == 0 ? len - 1 : len;
- addr += (4 - len);
- status = EndianU32_NtoB(status); // strings are big endian
- strncpy(cstring, (char *)addr, len);
- cstring[len] = 0;
- return [NSString stringWithCString:(char *)cstring encoding:NSMacOSRomanStringEncoding];
- }
- static void MyAudioSessionInterruptionListener(void *inClientData, UInt32 inInterruptionState)
- {
- AudioSessionInterruptionType interruptionType = kAudioSessionInterruptionType_ShouldNotResume;
- UInt32 interruptionTypeSize = sizeof(interruptionType);
- AudioSessionGetProperty(kAudioSessionProperty_InterruptionType,
- &interruptionTypeSize,
- &interruptionType);
- NSDictionary *userInfo = @{MyAudioInterruptionStateKey:@(inInterruptionState),
- MyAudioInterruptionTypeKey:@(interruptionType)};
- [[NSNotificationCenter defaultCenter] postNotificationName:MyAudioInterruptionNotification object:nil userInfo:userInfo];
- }
- - (void)interruptionNotificationReceived:(NSNotification *)notification
- {
- UInt32 interruptionState = [notification.userInfo[MyAudioInterruptionStateKey] unsignedIntValue];
- AudioSessionInterruptionType interruptionType = [notification.userInfo[MyAudioInterruptionTypeKey] unsignedIntValue];
- [self handleAudioSessionInterruptionWithState:interruptionState type:interruptionType];
- }
- - (void)handleAudioSessionInterruptionWithState:(UInt32)interruptionState type:(AudioSessionInterruptionType)interruptionType
- {
- if (interruptionState == kAudioSessionBeginInterruption)
- {
- //控制UI,暂停播放
- }
- else if (interruptionState == kAudioSessionEndInterruption)
- {
- if (interruptionType == kAudioSessionInterruptionType_ShouldResume)
- {
- OSStatus status = AudioSessionSetActive(true);
- if (status == noErr)
- {
- //控制UI,继续播放
- }
- }
- }
- }
iOS音频播放(二):AudioSession的更多相关文章
- iOS音频播放 (二):AudioSession 转
原文出处 :http://msching.github.io/blog/2014/07/08/audio-in-ios-2/ 前言 本篇为<iOS音频播放>系列的第二篇. 在实施前一篇中所 ...
- IOS 音频播放
iOS音频播放 (一):概述 前言 从事音乐相关的app开发也已经有一段时日了,在这过程中app的播放器几经修改我也因此对于iOS下的音频播放实现有了一定的研究.写这个系列的博客目的一方面希望能够抛砖 ...
- iOS音频播放(一):概述
(本文转自码农人生) 前言 从事音乐相关的app开发也已经有一段时日了,在这过程中app的播放器几经修改,我也因此对于iOS下的音频播放实现有了一定的研究.写这个 系列的博客目的一方面希望能够抛砖引玉 ...
- iOS音频播放 (五):AudioQueue
码农人生 ChengYin's coding life 主页 Blog 分类 Categories 归档 Archives 关于 About Weibo GitHub RSS Where there ...
- iOS音频播放、录音、视频播放、拍照、视频录制
随着移动互联网的发展,如今的手机早已不是打电话.发短信那么简单了,播放音乐.视频.录音.拍照等都是很常用的功能.在iOS中对于多媒体的支持是非常强大的,无论是音视频播放.录制,还是对麦克风.摄像头的操 ...
- iOS音频播放概述
在iOS系统中apple对音频播放需要的操作进行了封装并提供了不同层次的接口 下面对其中的中高层接口进行功能说明: Audio File Services:读写音频数据,可以完成播放流程中的第2步: ...
- iOS音频播放 (四):AudioFile 转
原文出处 : http://msching.github.io/blog/2014/07/19/audio-in-ios-4/ 前言 接着第三篇的AudioStreamFile这一篇要来聊一下Audi ...
- iOS音频播放之AudioQueue(一):播放本地音乐
AudioQueue简单介绍 AudioStreamer说明 AudioQueue具体解释 AudioQueue工作原理 AudioQueue主要接口 AudioQueueNewOutput Audi ...
- iOS音频播放 (三):AudioFileStream 转
原文出处 :http://msching.github.io/blog/2014/07/09/audio-in-ios-3/ 前言 本来说好是要在第三篇中讲AudioFileStream和AudioQ ...
随机推荐
- hdu 4751 Divide Groups(dfs染色 或 2-sat)
Problem Description This year is the 60th anniversary of NJUST, and to make the celebration more c ...
- Python 面向对象基础
面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数. 面向过程的程序设计把计算机 ...
- hdu 4930 Fighting the Landlords--2014 Multi-University Training Contest 6
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4930 Fighting the Landlords Time Limit: 2000/1000 MS ...
- [KMP求最小循环节][HDU3746][Cyclic Nacklace]
题意 给你个字符串,问在字符串末尾还要添加几个字符,使得字符串循环2次以上. 解法 无论这个串是不是循环串 i-next[i] 都能求出它的最小循环节 代码: /* 思路:kmp+字符串的最小循环节问 ...
- .net 链接oracle
虽然EF6都快要出来了,但是对于Oracle数据库,仍然只能用DB first和Model First来编程,不能用Code First真是一个很大的遗憾啊. 好了,废话少说,我们来看看EF中是如何用 ...
- html标签之meta标签
1 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> ...
- 【.NetRemoting-2】2015.09.17
[Remoting架构] [1]是.NetFramework的一个重要组成 [2]框架的两个重要特性 [A]基本实现[B]可扩展/可定制 [各个组成部分] [1][客户端,客户端应用程序域] [组成] ...
- 深入理解JavaWeb技术内幕(一)
最近在看许令波的<深入理解JavaWeb技术内幕>.整理了一些笔记.想做一个系列,这篇是系列的第一篇,讲Web请求. B/S架构 最常见的架构方式. 优点: 1.客户端使用统一(此处的统一 ...
- spring下配置dbcp,c3p0,proxool[转]
不管通过何种持久化技术,都必须通过数据连接访问数据库,在Spring中,数据连接是通过数据源获得的.在以往的应用中,数据源一般是Web应用服务器提供的.在Spring中,你不但可以通过JNDI获取应用 ...
- Android识别图片中脸部信息
在Android开发中,大部分应用都是以用户为第一位,用户都有自己的个人中心,用来展示自己的信息,头像无疑是展示自己最直观的方式,随着各种政策的出台,实名认证,真人头像变得尤为重要,如果要求上传真人头 ...