如何同时使用蓝牙麦克风和内置扬声器?

时间:2020-03-31 00:14:29

标签: ios swift iphone bluetooth microphone

我正在尝试在iOS中完成一个非常简单的任务。我想从蓝牙麦克风获取音频,然后使用内置扬声器同时播放音频。

这就是我现在所拥有的。该应用程序允许用户选择扬声器和麦克风来源。但是,每当我选择一个内置扬声器时,该应用程序都会切换到内置麦克风。每当我选择蓝牙麦克风时,该应用程序就会使用蓝牙扬声器。我无法单独使用蓝牙麦克风和内置扬声器。

import UIKit
import AVFoundation
import AVKit

class ViewController: UIViewController {

    var engine = AVAudioEngine()
    let player = AVAudioPlayerNode()
    let audioSession = AVAudioSession()
    var routePickerView = AVRoutePickerView(frame: CGRect(x: 0, y: 0, width: 0, height: 50))
    let bus = 0
    var isRunning = false

    @IBOutlet weak var viewHolder: UIStackView!

    override func viewDidLoad() {
        super.viewDidLoad()

        setUpAVRoutePicker()
        setUpAVSession()
    }

    func setUpAVRoutePicker() {
        viewHolder.addArrangedSubview(routePickerView)
    }

    func setUpAVSession() {
        do {
            try audioSession.setCategory(AVAudioSession.Category.playAndRecord, options: [.defaultToSpeaker, .allowBluetooth, .allowBluetoothA2DP, .allowAirPlay])
            try audioSession.setMode(AVAudioSession.Mode.default)
            try audioSession.overrideOutputAudioPort(AVAudioSession.PortOverride.speaker)
            try audioSession.setActive(true)
        } catch {
            print("Error setting up AV session!")
            print(error)
        }
    }

    @IBAction func start(_ sender: AnyObject) {
        isRunning = !isRunning
        sender.setTitle(isRunning ? "Stop" : "Start", for: .normal)

        if isRunning {
            engine.attach(player)

            let inputFormat = engine.inputNode.inputFormat(forBus: bus)

            engine.connect(player, to: engine.mainMixerNode, format: inputFormat)

            engine.inputNode.installTap(onBus: bus, bufferSize: 512, format: inputFormat) { (buffer, time) -> Void in
                self.player.scheduleBuffer(buffer)
            }

            do {
                try engine.start()
            } catch {
                print("Engine start error")
                print(error)
                return
            }
            player.play()
        } else {
            engine.inputNode.removeTap(onBus: bus)
            engine.stop()
            player.stop()
        }

    }

    @IBAction func onInputBtnClicked(_ sender: AnyObject) {
        let controller = UIAlertController(title: "Select Input", message: "", preferredStyle: UIAlertController.Style.actionSheet)

        for input in audioSession.availableInputs ?? [] {
            controller.addAction(UIAlertAction(title: input.portName, style: UIAlertAction.Style.default, handler: { action in
            do {
                try self.audioSession.setPreferredInput(input)
            } catch {
                print("Setting preferred input error")
                print(error)
            }
            }))
        }
        present(controller, animated: true, completion: nil)
    }


}

似乎无法实现(https://stackoverflow.com/a/24519938/6308776),这有点疯狂。我知道可以在Android上实现如此简单的任务,但对于iOS来说似乎是不可能的。有人有什么想法吗?

p / s:我想使用蓝牙耳机,但我不想通过wifi进行传输。

1 个答案:

答案 0 :(得分:0)

从Apple:

https://forums.developer.apple.com/thread/62954

...如果应用程序正在使用setPreferredInput选择蓝牙 HFP输入,输出应自动更改为蓝牙 与该输入相对应的HFP输出。此外,选择一个 使用MPVolumeView的路线选择器的蓝牙HFP输出应 自动将输入更改为蓝牙HFP输入 与该输出相对应。换句话说,输入和 输出应始终在选择的同一蓝牙HFP设备上结束 对于输入/输出,即使仅设置了输入或输出 个别地。这是预期的行为,但是如果没有发生 我们肯定想知道它。