I'm still very new to swift (and programming) and I'm trying to output the CVPixelbuffer I get from ARFrame to a video in realtime (without the AR stuff on top).
I've set up the AVAssetWriter and Input and on each frame I try to append the CVPixelbuffer (converted into a CMSampleBuffer).
The code creates a file of 0 bytes and doesn't give any errors. I can't figure out what's wrong, and the documentation is rather cryptic for my level.
I'm pretty sure the timeStamp for each frame is incorrect. But I can't figure out if each frame simple needs a time, or the total time since the start of the video.
Then there is the size and the format of the cvPixelBuffer. I also can't figure out if I need to tell the AVAssetwriter what the size & format of the data is that is coming or if it figures that out by itself.
Thank you for your time.
class VideoHandler
{
var videoWriter: AVAssetWriter?
var videoWriterInput: AVAssetWriterInput?
var videoURL = URL(fileURLWithPath: "")
let videoSize = CGSize(width: 1280, height: 720)
func start()
{
videoURL = getDocumentsDirectory().appendingPathComponent("Temp.mov")
do {
try FileManager.default.removeItem(at: videoURL)
} catch {}
do {
try videoWriter = AVAssetWriter(outputURL: videoURL, fileType: AVFileType.mov)
} catch let error as NSError {
print(error)
videoWriter = nil
}
let videoSettings: [String : AnyObject] = [
AVVideoCodecKey : AVVideoCodecType.h264 as AnyObject,
AVVideoWidthKey : videoSize.width as AnyObject,
AVVideoHeightKey : videoSize.height as AnyObject,
// AVVideoCompressionPropertiesKey : [
// AVVideoAverageBitRateKey : NSInteger(1000000),
// AVVideoMaxKeyFrameIntervalKey : NSInteger(16),
// AVVideoProfileLevelKey : AVVideoProfileLevelH264BaselineAutoLevel
// ]
]
videoWriterInput = AVAssetWriterInput(mediaType: AVMediaType.video, outputSettings: videoSettings)
videoWriterInput?.expectsMediaDataInRealTime = true
//TODO: Force unwrapping... so dangerous but don't know any other way.
if videoWriter!.canAdd(videoWriterInput!)
{
videoWriter!.add(videoWriterInput!)
}
else
{
print("Can't add videoWriterInput")
}
if videoWriter?.startWriting() ?? false
{
videoWriter?.startSession(atSourceTime: CMTime.zero)
}
print("Video writing started.")
}
func addFrame(buffer: CVPixelBuffer, timeStamp: Double)
{
if videoWriterInput?.isReadyForMoreMediaData ?? false
{
videoWriterInput?.append(getCMSampleBuffer(buffer: buffer, timeStamp: timeStamp))
}
}
func getCMSampleBuffer(buffer: CVPixelBuffer, timeStamp: Double) -> CMSampleBuffer
{
// Don't know yet how to set the time and timescale stuff properly.
var info = CMSampleTimingInfo()
// info.presentationTimeStamp = CMTime.zero
info.presentationTimeStamp = CMTime(seconds: timeStamp, preferredTimescale: 1)
// info.duration = CMTime.invalid
// info.decodeTimeStamp = CMTime.invalid
var formatDesc: CMFormatDescription? = nil
CMVideoFormatDescriptionCreateForImageBuffer(
allocator: kCFAllocatorDefault,
imageBuffer: buffer,
formatDescriptionOut: &formatDesc)
var sampleBuffer: CMSampleBuffer? = nil
CMSampleBufferCreateReadyWithImageBuffer(allocator: kCFAllocatorDefault,
imageBuffer: buffer,
formatDescription: formatDesc!,
sampleTiming: &info,
sampleBufferOut: &sampleBuffer);
return sampleBuffer!
}
func stop()
{
videoWriter?.finishWriting
{
print("Video writing finished.")
}
videoWriter = nil
}
func getDocumentsDirectory() -> URL
{
let paths = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)
return paths[0]
}
}