Подтвердить что ты не робот

Как получить переднюю камеру, заднюю камеру и аудио с помощью AVCaptureDeviceDiscoverySession

До выхода iOS 10 я использовал следующий код, чтобы получить видео и аудиозапись для моего видеомагнитофона:

 for device in AVCaptureDevice.devices()
 {
     if (device as AnyObject).hasMediaType( AVMediaTypeAudio )
     {
         self.audioCapture = device as? AVCaptureDevice
     }
     else if (device as AnyObject).hasMediaType( AVMediaTypeVideo )
     {
         if (device as AnyObject).position == AVCaptureDevicePosition.back
         {
             self.backCameraVideoCapture = device as? AVCaptureDevice
         }
         else
         {
             self.frontCameraVideoCapture = device as? AVCaptureDevice
         }
     }
 }

Когда iOS 10 наконец вышел, я получил следующее предупреждение, когда я запускал свой код. Обратите внимание, что мой видеомагнитофон все еще работал плавно в течение 2 недель.

'devices()' устарел в iOS 10.0: вместо этого используйте AVCaptureDeviceDiscoverySession.

Когда я запускал свой код этим утром, мой видеомагнитофон переставал работать. xCode8 не дает мне никаких ошибок, но предварительный просмотр для захвата камеры полностью белый. Когда я начинаю запись, я получаю следующую ошибку:

Ошибка домена = AVFoundationErrorDomain Code = -11800 "Операция не может быть завершена" UserInfo = {NSLocalizedDescription = Операция не может быть выполнена, NSUnderlyingError = 0x17554440 {Ошибка домена = NSOSStatusErrorDomain Code = -12780 "(null)" } NSLocalizedFailureReason = Произошла неизвестная ошибка (-12780)}

Я считаю, что это связано с тем, что я использую устаревший подход AVCaptureDevice.devices(). Следовательно, мне было интересно, как использовать AVCaptureDeviceDiscoverySession вместо этого?

Спасибо за вашу помощь заранее!

4b9b3361

Ответ 1

Вы можете получить фронтальную камеру со следующим:

AVCaptureDevice.default(.builtInWideAngleCamera, for: AVMediaType.video, position: .front)

Задняя камера:

AVCaptureDevice.default(.builtInWideAngleCamera, for: AVMediaType.video, position: .back)

И микрофон:

AVCaptureDevice.default(.builtInMicrophone, for: AVMediaType.audio, position: .unspecified)

Ответ 2

Swift 4, iOS 10+ и Xcode 10.1 заменяет

if let cameraID = AVCaptureDevice.default(.builtInWideAngleCamera, for: .video, position: .front)?.localizedName {
       //cameraID = "Front Camera"
}

с реализацией AVCaptureDevice.DiscoverySession

if let cameraID = AVCaptureDevice.DiscoverySession.init(deviceTypes: [AVCaptureDevice.DeviceType.builtInWideAngleCamera], mediaType: .video, position: .front).devices.first?.localizedName{
        //cameraID = "Front Camera"
} 

Необходимо обернуть его проверкой #available (iOS 10, *).

Ответ 3

Здесь мой код (Swift 3), чтобы получить положение камеры:

// Find a camera with the specified AVCaptureDevicePosition, returning nil if one is not found
func cameraWithPosition(_ position: AVCaptureDevicePosition) -> AVCaptureDevice?
{
    if let deviceDescoverySession = AVCaptureDeviceDiscoverySession.init(deviceTypes: [AVCaptureDeviceType.builtInWideAngleCamera],
                                                          mediaType: AVMediaTypeVideo,
                                                          position: AVCaptureDevicePosition.unspecified) {

        for device in deviceDescoverySession.devices {
            if device.position == position {
                return device
            }
        }
    }

    return nil
}

Если вы хотите, вы также можете получить новые типы устройств с iPhone 7+ (двойная камера), изменив массив deviceTypes.

Здесь хорошо читается: https://forums.developer.apple.com/thread/63347

Ответ 5

Swift 3

Для выбора задней камеры: (также вы можете изменить .back по мере необходимости)

Для выбора другого типа устройства просто добавьте его внутри [] (i.e:

[deviceTypeCamera, AVCaptureDeviceType.builtInMicrophone]

(или создать приватное разрешение... как я сделал в коде с задней камерой)

 private let position = AVCaptureDevicePosition.back
 private let deviceTypeBackCamera = AVCaptureDeviceType.builtInWideAngleCamera

 private func selectCaptureDevice() -> AVCaptureDevice? {
    return AVCaptureDeviceDiscoverySession(deviceTypes: [deviceTypeBackCamera], mediaType: AVMediaTypeVideo, position: position).devices.first

 }

Ответ 6

пример: iOS 11 Swift 4

override func viewDidLoad() {
    super.viewDidLoad()

    // Get the back-facing camera for capturing videos

    // AVCaptureDevice.default(.builtInWideAngleCamera, for: AVMediaType.video, position: .back)
    let deviceDiscoverySession = AVCaptureDevice.default(.builtInWideAngleCamera, for: AVMediaType.video, position: .back)

   guard let captureDevice = deviceDiscoverySession else {
       print("Failed to get the camera device")
       return
   }

    do {
        // Get an instance of the AVCaptureDeviceInput class using the previous device object.
        let input = try AVCaptureDeviceInput(device: captureDevice)

        // Set the input device on the capture session.
        captureSession.addInput(input)

    } catch {
        // If any error occurs, simply print it out and don't continue any more.
        print(error)
        return
    }

    // Initialize the video preview layer and add it as a sublayer to the viewPreview view layer.
    videoPreviewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
    videoPreviewLayer?.videoGravity = AVLayerVideoGravity.resizeAspectFill
    videoPreviewLayer?.frame = view.layer.bounds
    view.layer.addSublayer(videoPreviewLayer!)

    // Start video capture.
    captureSession.startRunning()

Ответ 7

Попробуйте ввести код, чтобы получить идентификатор камеры:

NSString *cameraID = nil;

NSArray *captureDeviceType = @[AVCaptureDeviceTypeBuiltInWideAngleCamera];
AVCaptureDeviceDiscoverySession *captureDevice = 
              [AVCaptureDeviceDiscoverySession 
                discoverySessionWithDeviceTypes:captureDeviceType 
                mediaType:AVMediaTypeVideo 
                position:AVCaptureDevicePositionUnspecified];

cameraID = [captureDevice.devices.lastObject localizedName];

Ответ 8

Для моего приложения для захвата видео я использую следующий код, чтобы получить микрофон, переднюю и заднюю камеру, и я протестировал этот код с iOS 7 до 10.0.2.

        var frontCamera : AVCaptureDevice?
        var rearCamera : AVCaptureDevice?

        captureSession = AVCaptureSession()

        let devices = AVCaptureDevice.devicesWithMediaType(AVMediaTypeVideo)

        let audioDevices = AVCaptureDevice.devicesWithMediaType(AVMediaTypeAudio)

        for mic in audioDevices {
            audioDevice = mic as? AVCaptureDevice
            audioCapturePossible = true
        }

        for device in devices {
            if device.position == AVCaptureDevicePosition.Front {
                frontCamera = device as? AVCaptureDevice
                hasFrontCamera = true
            }
            else if device.position == AVCaptureDevicePosition.Back {
                rearCamera = device as? AVCaptureDevice
                hasRearCamera = true
            }

        }

Ответ 9

Swift 4 (xCode 10.1)

Это то, что сработало для меня в последней версии Swift. Я не видел этот ответ, и мне потребовалось некоторое время, чтобы разобраться, вот как получить фронтальную камеру.

 if let device = AVCaptureDevice.defaultDevice(withDeviceType: .builtInWideAngleCamera , mediaType: AVMediaTypeVideo, position: .front)  {
    //Do the camera thing here..
}

Ответ 10

05/2019:

 //video
        self.session = AVCaptureSession()
        guard
            let videoDeviceInput = try? AVCaptureDeviceInput(device: device!),
            self.session!.canAddInput(videoDeviceInput)
            else { return }
        self.session!.addInput(videoDeviceInput)

  //audio
        guard
            let audioDeviceInput = try? AVCaptureDeviceInput(device: mic!),
            self.session!.canAddInput(audioDeviceInput)
            else { return }
        self.session!.addInput(audioDeviceInput)