I'm trying to make movie from still image, for that i use custom compositor with custom instructions. Actually i get movie and length is as expected, but after one frame (5 second) i get black screen(for another frames)
What i did is in my demo project:
https://github.com/satishVekariya/MovideMakerDemo
class StoryComposer {
var slide:[ImageResource] = []
init() {
}
private var needRebuildComposition = true
private var needRebuildVideoComposition = true
private(set) var composition: AVMutableComposition?
private(set) var videoComposition: AVMutableVideoComposition?
private(set) var addedTrackIds:[CMPersistentTrackID] = []
}
extension StoryComposer {
func composeForAVPlayerItem() -> (AVPlayerItem, AVMutableComposition, AVMutableVideoComposition) {
let composition = compose()
let playerItem = AVPlayerItem(asset: composition)
let video = composeVideoComposition()
playerItem.videoComposition = video
return (playerItem, composition, video!)
}
}
extension StoryComposer {
@discardableResult func compose() -> AVMutableComposition {
if let oldComposition = self.composition, !needRebuildComposition {
return oldComposition
}
let composition = AVMutableComposition(urlAssetInitializationOptions: [AVURLAssetPreferPreciseDurationAndTimingKey: true])
var startTime = CMTime(seconds: 0, preferredTimescale: 1)
for (_, item) in slide.enumerated() {
let track = slide.first!.trackInfo(for: .video, at: 0).track
let emptyVideoTrack = composition.addMutableTrack(withMediaType: .video, preferredTrackID: kCMPersistentTrackID_Invalid)
if let id = emptyVideoTrack?.trackID {
addedTrackIds.append(id)
}
do {
emptyVideoTrack?.removeTimeRange(CMTimeRange(start: startTime, duration: item.duration))
try emptyVideoTrack?.insertTimeRange(CMTimeRangeMake(start: startTime, duration: CMTime(seconds: 1, preferredTimescale: 30)), of: track, at: startTime)
emptyVideoTrack?.scaleTimeRange(CMTimeRange(start: startTime, duration: CMTime(seconds: 1, preferredTimescale: 30)), toDuration: item.duration)
} catch {
print("\(Self.self) : \(#function) error - \(error)")
}
startTime = CMTimeAdd(startTime, item.duration)
}
self.composition = composition
return composition
}
}
extension StoryComposer {
public func composeVideoComposition() -> AVMutableVideoComposition? {
if let videoComposition = self.videoComposition, !needRebuildVideoComposition {
return videoComposition
}
compose()
var videoCompositionInstruction = [VideoCompositionInstruction]()
var startTime = CMTime(seconds: 0, preferredTimescale: 1)
for (index, item) in slide.enumerated() {
let trackId:CMPersistentTrackID = Int32(index) + 1
let layerInstr = VideoCompositionLayerInstruction(trackID: trackId)
layerInstr.resource = item
let instr = VideoCompositionInstruction(theSourceTrackIDs: [layerInstr.trackID as NSValue], forTimeRange: CMTimeRangeMake(start: startTime, duration: item.duration))
instr.layerInstructions = [layerInstr]
videoCompositionInstruction.append(instr)
startTime = CMTimeAdd(startTime, item.duration)
}
let videoComposition = AVMutableVideoComposition()
videoComposition.frameDuration = CMTime(value: 1, timescale: 30)
videoComposition.customVideoCompositorClass = VideoCompositor.self
videoComposition.instructions = videoCompositionInstruction
videoComposition.renderSize = CGSize(width: 480, height: 480)
self.videoComposition = videoComposition
return videoComposition
}
}
Problem in this line of code:
try emptyVideoTrack?.insertTimeRange(CMTimeRangeMake(start: startTime, duration: CMTime(seconds: 1, preferredTimescale: 30)), of: track, at: startTime)
In your case timeRange
should be start with zero, So change to below code:
try emptyVideoTrack?.insertTimeRange(CMTimeRangeMake(start: .zero, duration: CMTime(seconds: 1, preferredTimescale: 30)), of: track, at: startTime)
Check doc in insertTimeRange
function
@param timeRange - Specifies the timeRange of the track to be inserted.
'timeRange of the track to be inserted' means timeRange of track which you want insert in your track.