当用户选择操作表上的选项时,我正在尝试将音频输出更改为设备。这是代码,当我选择音频去设备时,下次不会出现蓝牙。:
for input in AVAudioSession.sharedInstance().availableInputs!{
if input.portType == AVAudioSessionPortBluetoothA2DP || input.portType == AVAudioSessionPortBluetoothHFP || input.portType == AVAudioSessionPortBluetoothLE{
let bluetooth = UIAlertAction(title: input.portName, style: .default, handler: {
(alert: UIAlertAction!) -> Void in
let audioSession = AVAudioSession.sharedInstance()
do {
try audioSession.setCategory(AVAudioSessionCategoryPlayAndRecord, with: AVAudioSessionCategoryOptions.allowBluetooth)
try audioSession.setActive(true)
} catch {
fatalError("Error Setting Up bluetooth output \(input.portName)")
}
})
bluetooth.setValue(UIImage(named:"bluetooth.png"), forKey: "image")
optionMenu.addAction(bluetooth)
}
let iphomeOutput = UIAlertAction(title: "iPhone", style: .default, handler: {
(alert: UIAlertAction!) -> Void in
let audioSession = AVAudioSession.sharedInstance()
do {
do {
try audioSession.setCategory(AVAudioSessionCategoryPlayAndRecord, with: AVAudioSessionCategoryOptions.duckOthers)
try audioSession.setActive(true)
} catch {
fatalError("Error Setting Up audio output Phone")
}
try audioSession.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
} catch let error as NSError {
print("audioSession error turning off speaker: \(error.localizedDescription)")
}
})
for description in currentRoute.outputs {
if description.portType == AVAudioSessionPortHeadsetMic{
optionMenu.setValue(true, forKey: "checked")
break
}
}
optionMenu.addAction(iphomeOutput)
答案 0 :(得分:4)
我可能能够更有效地做到这一点,但这是我最终使用的:
var deviceAction = UIAlertAction()
var headphonesExist = false
let audioSession = AVAudioSession.sharedInstance()
let optionMenu = UIAlertController(title: nil, message: nil, preferredStyle: .actionSheet)
let currentRoute = audioSession.currentRoute
for input in audioSession.availableInputs!{
if input.portType == AVAudioSessionPortBluetoothA2DP || input.portType == AVAudioSessionPortBluetoothHFP || input.portType == AVAudioSessionPortBluetoothLE{
let localAction = UIAlertAction(title: input.portName, style: .default, handler: {
(alert: UIAlertAction!) -> Void in
do {
try audioSession.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
} catch let error as NSError {
print("audioSession error turning off speaker: \(error.localizedDescription)")
}
do {
try audioSession.setPreferredInput(input)
}catch _ {
print("cannot set mic ")
}
})
for description in currentRoute.outputs {
if description.portType == AVAudioSessionPortBluetoothA2DP {
localAction.setValue(true, forKey: "checked")
break
}else if description.portType == AVAudioSessionPortBluetoothHFP {
localAction.setValue(true, forKey: "checked")
break
}else if description.portType == AVAudioSessionPortBluetoothLE{
localAction.setValue(true, forKey: "checked")
break
}
}
localAction.setValue(UIImage(named:"bluetooth.png"), forKey: "image")
optionMenu.addAction(localAction)
} else if input.portType == AVAudioSessionPortBuiltInMic || input.portType == AVAudioSessionPortBuiltInReceiver {
deviceAction = UIAlertAction(title: "iPhone", style: .default, handler: {
(alert: UIAlertAction!) -> Void in
do {
try audioSession.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
} catch let error as NSError {
print("audioSession error turning off speaker: \(error.localizedDescription)")
}
do {
try audioSession.setPreferredInput(input)
}catch _ {
print("cannot set mic ")
}
})
for description in currentRoute.outputs {
if description.portType == AVAudioSessionPortBuiltInMic || description.portType == AVAudioSessionPortBuiltInReceiver {
deviceAction.setValue(true, forKey: "checked")
break
}
}
} else if input.portType == AVAudioSessionPortHeadphones || input.portType == AVAudioSessionPortHeadsetMic {
headphonesExist = true
let localAction = UIAlertAction(title: "Headphones", style: .default, handler: {
(alert: UIAlertAction!) -> Void in
do {
try audioSession.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
} catch let error as NSError {
print("audioSession error turning off speaker: \(error.localizedDescription)")
}
do {
try audioSession.setPreferredInput(input)
}catch _ {
print("cannot set mic ")
}
})
for description in currentRoute.outputs {
if description.portType == AVAudioSessionPortHeadphones {
localAction.setValue(true, forKey: "checked")
break
} else if description.portType == AVAudioSessionPortHeadsetMic {
localAction.setValue(true, forKey: "checked")
break
}
}
optionMenu.addAction(localAction)
}
}
if !headphonesExist {
optionMenu.addAction(deviceAction)
}
let speakerOutput = UIAlertAction(title: "Speaker", style: .default, handler: {
(alert: UIAlertAction!) -> Void in
do {
try audioSession.overrideOutputAudioPort(AVAudioSessionPortOverride.speaker)
} catch let error as NSError {
print("audioSession error turning on speaker: \(error.localizedDescription)")
}
})
for description in currentRoute.outputs {
if description.portType == AVAudioSessionPortBuiltInSpeaker{
speakerOutput.setValue(true, forKey: "checked")
break
}
}
speakerOutput.setValue(UIImage(named:"speaker.png"), forKey: "image")
optionMenu.addAction(speakerOutput)
let cancelAction = UIAlertAction(title: "Hide", style: .cancel, handler: {
(alert: UIAlertAction!) -> Void in
})
optionMenu.addAction(cancelAction)
self.present(optionMenu, animated: true, completion: nil)
答案 1 :(得分:0)
我用扩展名(和很少的安排)更新了FreeGor的答案
extension AVAudioSession {
func ChangeAudioOutput(presenterViewController : UIViewController) {
let CHECKED_KEY = "checked"
let IPHONE_TITLE = "iPhone"
let HEADPHONES_TITLE = "Headphones"
let SPEAKER_TITLE = "Speaker"
let HIDE_TITLE = "Hide"
var deviceAction = UIAlertAction()
var headphonesExist = false
let currentRoute = self.currentRoute
let optionMenu = UIAlertController(title: nil, message: nil, preferredStyle: .actionSheet)
for input in self.availableInputs!{
switch input.portType {
case AVAudioSessionPortBluetoothA2DP, AVAudioSessionPortBluetoothHFP, AVAudioSessionPortBluetoothLE:
let action = UIAlertAction(title: input.portName, style: .default) { (action) in
do {
// remove speaker if needed
try self.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
// set new input
try self.setPreferredInput(input)
} catch let error as NSError {
print("audioSession error change to input: \(input.portName) with error: \(error.localizedDescription)")
}
}
if currentRoute.outputs.contains(where: {return $0.portType == input.portType}){
action.setValue(true, forKey: CHECKED_KEY)
}
optionMenu.addAction(action)
break
case AVAudioSessionPortBuiltInMic, AVAudioSessionPortBuiltInReceiver:
deviceAction = UIAlertAction(title: IPHONE_TITLE, style: .default) { (action) in
do {
// remove speaker if needed
try self.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
// set new input
try self.setPreferredInput(input)
} catch let error as NSError {
print("audioSession error change to input: \(input.portName) with error: \(error.localizedDescription)")
}
}
if currentRoute.outputs.contains(where: {return $0.portType == input.portType}){
deviceAction.setValue(true, forKey: CHECKED_KEY)
}
break
case AVAudioSessionPortHeadphones, AVAudioSessionPortHeadsetMic:
headphonesExist = true
let action = UIAlertAction(title: HEADPHONES_TITLE, style: .default) { (action) in
do {
// remove speaker if needed
try self.overrideOutputAudioPort(AVAudioSessionPortOverride.none)
// set new input
try self.setPreferredInput(input)
} catch let error as NSError {
print("audioSession error change to input: \(input.portName) with error: \(error.localizedDescription)")
}
}
if currentRoute.outputs.contains(where: {return $0.portType == input.portType}){
action.setValue(true, forKey: CHECKED_KEY)
}
optionMenu.addAction(action)
break
default:
break
}
}
if !headphonesExist {
optionMenu.addAction(deviceAction)
}
let speakerOutput = UIAlertAction(title: SPEAKER_TITLE, style: .default, handler: {
(alert: UIAlertAction!) -> Void in
do {
try self.overrideOutputAudioPort(AVAudioSessionPortOverride.speaker)
} catch let error as NSError {
print("audioSession error turning on speaker: \(error.localizedDescription)")
}
})
if currentRoute.outputs.contains(where: {return $0.portType == AVAudioSessionPortBuiltInSpeaker}){
speakerOutput.setValue(true, forKey: CHECKED_KEY)
}
optionMenu.addAction(speakerOutput)
let cancelAction = UIAlertAction(title: HIDE_TITLE, style: .cancel, handler: {
(alert: UIAlertAction!) -> Void in
})
optionMenu.addAction(cancelAction)
presenterViewController.present(optionMenu, animated: true, completion: nil)
}
}