Swift: видеозаписи одного размера, но при неправильном размере

Цель состоит в том, чтобы захватить полноэкранное видео на устройстве с помощью Swift. В приведенном ниже коде захват видео появляется в полноэкранном режиме (во время записи предварительного просмотра камеры используется полный экран), но рендеринг видео происходит с другим разрешением. Для 5S в частности, похоже, что захват происходит при 320x568, но рендеринг происходит при 320x480.

Как вы можете захватить и отобразить полноэкранное видео?

Код для видеозахвата:

private func initPBJVision() {
    // Store PBJVision in var for convenience
    let vision = PBJVision.sharedInstance()

    // Configure PBJVision
    vision.delegate = self
    vision.cameraMode = PBJCameraMode.Video
    vision.cameraOrientation = PBJCameraOrientation.Portrait
    vision.focusMode = PBJFocusMode.ContinuousAutoFocus
    vision.outputFormat = PBJOutputFormat.Preset
    vision.cameraDevice = PBJCameraDevice.Back

    // Let taps start/pause recording
    let tapHandler = UITapGestureRecognizer(target: self, action: "doTap:")
    view.addGestureRecognizer(tapHandler)

    // Log status
    print("Configured PBJVision")
}


private func startCameraPreview() {
    // Store PBJVision in var for convenience
    let vision = PBJVision.sharedInstance()

    // Connect PBJVision camera preview to <videoView>
    // -- Get preview width
    let deviceWidth = CGRectGetWidth(view.frame)
    let deviceHeight = CGRectGetHeight(view.frame)

    // -- Configure PBJVision preview layer
    let previewLayer = vision.previewLayer
    previewLayer.frame = CGRectMake(0, 0, deviceWidth, deviceHeight)
    previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill
    ...
}

Код визуализации видео:

func exportVideo(fileUrl: NSURL) {
    // Create main composition object
    let videoAsset = AVURLAsset(URL: fileUrl, options: nil)
    let mainComposition = AVMutableComposition()
    let compositionVideoTrack = mainComposition.addMutableTrackWithMediaType(AVMediaTypeVideo, preferredTrackID: CMPersistentTrackID(kCMPersistentTrackID_Invalid))
    let compositionAudioTrack = mainComposition.addMutableTrackWithMediaType(AVMediaTypeAudio, preferredTrackID: CMPersistentTrackID(kCMPersistentTrackID_Invalid))

    // -- Extract and apply video & audio tracks to composition
    let sourceVideoTrack = videoAsset.tracksWithMediaType(AVMediaTypeVideo)[0]
    let sourceAudioTrack = videoAsset.tracksWithMediaType(AVMediaTypeAudio)[0]
    do {
        try compositionVideoTrack.insertTimeRange(CMTimeRangeMake(kCMTimeZero, videoAsset.duration), ofTrack: sourceVideoTrack, atTime: kCMTimeZero)
    } catch {
        print("Error with insertTimeRange. Video error: \(error).")
    }
    do {
        try compositionAudioTrack.insertTimeRange(CMTimeRangeMake(kCMTimeZero, videoAsset.duration), ofTrack: sourceAudioTrack, atTime: kCMTimeZero)
    } catch {
        print("Error with insertTimeRange. Audio error: \(error).")
    }

    // Add text to video
    // -- Create video composition object
    let renderSize = compositionVideoTrack.naturalSize
    let videoComposition = AVMutableVideoComposition()
    videoComposition.renderSize = renderSize
    videoComposition.frameDuration = CMTimeMake(Int64(1), Int32(videoFrameRate))

    // -- Add instruction to  video composition object
    let instruction = AVMutableVideoCompositionInstruction()
    instruction.timeRange = CMTimeRangeMake(kCMTimeZero, videoAsset.duration)
    let videoLayerInstruction = AVMutableVideoCompositionLayerInstruction(assetTrack: compositionVideoTrack)
    instruction.layerInstructions = [videoLayerInstruction]
    videoComposition.instructions = [instruction]

    // -- Define video frame
    let videoFrame = CGRectMake(0, 0, renderSize.width, renderSize.height)
    print("Video Frame: \(videoFrame)")  // <-- Prints frame of 320x480 so render size already wrong here 
    ...

Ответ 1

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

Свойство videoGravity вашего previewLayer указывает, как растянуть/подгонять ваш просмотр внутри вашего слоя. Это не влияет на выход захвата.

Фактический размер кадра зависит от свойства sessionPreset вашего текущего AVCaptureSession. И, как я могу понять, прочитав репозиторий GitHub PBJVision lib, его singleton имеет для этого сеттер (называемый captureSessionPreset). Вы можете изменить его в своем методе initPBJVision.

Здесь вы можете найти возможные значения пресетов сеанса.

Ответ 2

Попробуйте SDAVAssetExportSession

Вы можете указать AVVideoWidthKey и AVVideoHeightKey Вы также можете указать профиль AVVideoProfileLevelKey, т.е. AVVideoProfileLevelH264HighAutoLevel для поддержки чего-то 4k, друг (Майкл Лоуин) отслеживает это свойство, чтобы помочь нам улучшить некоторые экспортные качества.