2023-05-04 16:11:22 -03:00
|
|
|
import AVFoundation
|
|
|
|
import os.log
|
|
|
|
|
2023-05-02 21:31:14 -03:00
|
|
|
@objc(TransparentVideoViewManager)
|
|
|
|
class TransparentVideoViewManager: RCTViewManager {
|
|
|
|
|
|
|
|
override func view() -> (TransparentVideoView) {
|
|
|
|
return TransparentVideoView()
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc override static func requiresMainQueueSetup() -> Bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class TransparentVideoView : UIView {
|
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
private var source: VideoSource?
|
|
|
|
private var playerView: AVPlayerView?
|
|
|
|
|
|
|
|
@objc var src: NSDictionary = NSDictionary() {
|
2023-05-02 21:31:14 -03:00
|
|
|
didSet {
|
2023-05-04 16:11:22 -03:00
|
|
|
self.source = VideoSource(src)
|
|
|
|
let itemUrl = URL(string: self.source!.uri!)!
|
|
|
|
loadVideoPlayer(itemUrl: itemUrl)
|
2023-05-02 21:31:14 -03:00
|
|
|
}
|
|
|
|
}
|
2023-05-04 16:11:22 -03:00
|
|
|
|
|
|
|
func loadVideoPlayer(itemUrl: URL) {
|
|
|
|
if (self.playerView == nil) {
|
|
|
|
let playerView = AVPlayerView(frame: CGRect(origin: .zero, size: .zero))
|
|
|
|
addSubview(playerView)
|
|
|
|
|
|
|
|
// Use Auto Layout anchors to center our playerView
|
|
|
|
playerView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
playerView.topAnchor.constraint(equalTo: self.topAnchor),
|
|
|
|
playerView.bottomAnchor.constraint(equalTo: self.bottomAnchor),
|
|
|
|
playerView.leadingAnchor.constraint(equalTo: self.leadingAnchor),
|
|
|
|
playerView.trailingAnchor.constraint(equalTo: self.trailingAnchor)
|
|
|
|
])
|
|
|
|
|
|
|
|
// Setup our playerLayer to hold a pixel buffer format with "alpha"
|
|
|
|
let playerLayer: AVPlayerLayer = playerView.playerLayer
|
|
|
|
playerLayer.pixelBufferAttributes = [
|
|
|
|
(kCVPixelBufferPixelFormatTypeKey as String): kCVPixelFormatType_32BGRA]
|
2023-05-02 21:31:14 -03:00
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
// Setup looping on our video
|
|
|
|
playerView.isLoopingEnabled = true
|
|
|
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(appEnteredBackgound), name: UIApplication.didEnterBackgroundNotification, object: nil)
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(appEnteredForeground), name: UIApplication.willEnterForegroundNotification, object: nil)
|
2023-05-02 21:31:14 -03:00
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
self.playerView = playerView
|
2023-05-02 21:31:14 -03:00
|
|
|
}
|
2023-05-04 16:11:22 -03:00
|
|
|
|
|
|
|
// Load our player item
|
2023-05-05 15:57:49 -03:00
|
|
|
loadItem(url: itemUrl)
|
2023-05-04 16:11:22 -03:00
|
|
|
}
|
|
|
|
|
2023-05-06 16:42:45 -03:00
|
|
|
deinit {
|
|
|
|
playerView?.player?.pause()
|
|
|
|
playerView?.player?.replaceCurrentItem(with: nil)
|
|
|
|
playerView?.removeFromSuperview()
|
|
|
|
playerView = nil
|
|
|
|
}
|
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
// MARK: - Player Item Configuration
|
|
|
|
|
2023-05-05 15:57:49 -03:00
|
|
|
private func loadItem(url: URL) {
|
|
|
|
setUpAsset(with: url) { [weak self] (asset: AVAsset) in
|
|
|
|
self?.setUpPlayerItem(with: asset)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setUpAsset(with url: URL, completion: ((_ asset: AVAsset) -> Void)?) {
|
2023-05-04 16:11:22 -03:00
|
|
|
let asset = AVAsset(url: url)
|
2023-05-05 15:57:49 -03:00
|
|
|
asset.loadValuesAsynchronously(forKeys: ["metadata"]) {
|
|
|
|
var error: NSError? = nil
|
|
|
|
let status = asset.statusOfValue(forKey: "metadata", error: &error)
|
|
|
|
switch status {
|
|
|
|
case .loaded:
|
|
|
|
completion?(asset)
|
|
|
|
case .failed:
|
|
|
|
print(".failed")
|
|
|
|
case .cancelled:
|
|
|
|
print(".cancelled")
|
|
|
|
default:
|
|
|
|
print("default")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setUpPlayerItem(with asset: AVAsset) {
|
2023-05-06 14:04:56 -03:00
|
|
|
let playerItem = AVPlayerItem(asset: asset)
|
|
|
|
playerItem.seekingWaitsForVideoCompositionRendering = true
|
|
|
|
// Apply a video composition (which applies our custom filter)
|
|
|
|
playerItem.videoComposition = createVideoComposition(for: asset)
|
2023-05-05 15:57:49 -03:00
|
|
|
|
2023-05-06 14:04:56 -03:00
|
|
|
DispatchQueue.main.async { [weak self] in
|
2023-05-05 15:57:49 -03:00
|
|
|
self?.playerView!.loadPlayerItem(playerItem) { result in
|
|
|
|
switch result {
|
|
|
|
case .failure(let error):
|
|
|
|
return print("Something went wrong when loading our video", error)
|
|
|
|
|
|
|
|
case .success(let player):
|
|
|
|
// Finally, we can start playing
|
|
|
|
player.play()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
|
|
|
|
if keyPath == #keyPath(AVPlayerItem.status) {
|
|
|
|
let status: AVPlayerItem.Status
|
|
|
|
if let statusNumber = change?[.newKey] as? NSNumber {
|
|
|
|
status = AVPlayerItem.Status(rawValue: statusNumber.intValue)!
|
|
|
|
} else {
|
|
|
|
status = .unknown
|
|
|
|
}
|
|
|
|
// Switch over status value
|
|
|
|
switch status {
|
|
|
|
case .readyToPlay:
|
|
|
|
print(".readyToPlay")
|
|
|
|
case .failed:
|
|
|
|
print(".failed")
|
|
|
|
case .unknown:
|
|
|
|
print(".unknown")
|
|
|
|
@unknown default:
|
|
|
|
print("@unknown default")
|
|
|
|
}
|
|
|
|
}
|
2023-05-04 16:11:22 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
func createVideoComposition(for asset: AVAsset) -> AVVideoComposition {
|
|
|
|
let filter = AlphaFrameFilter(renderingMode: .builtInFilter)
|
|
|
|
let composition = AVMutableVideoComposition(asset: asset, applyingCIFiltersWithHandler: { request in
|
|
|
|
do {
|
|
|
|
let (inputImage, maskImage) = request.sourceImage.verticalSplit()
|
|
|
|
let outputImage = try filter.process(inputImage, mask: maskImage)
|
|
|
|
return request.finish(with: outputImage, context: nil)
|
|
|
|
} catch {
|
|
|
|
os_log("Video composition error: %s", String(describing: error))
|
|
|
|
return request.finish(with: error)
|
|
|
|
}
|
|
|
|
})
|
2023-05-02 21:31:14 -03:00
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
composition.renderSize = asset.videoSize.applying(CGAffineTransform(scaleX: 1.0, y: 0.5))
|
|
|
|
return composition
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Lifecycle callbacks
|
|
|
|
|
|
|
|
@objc func appEnteredBackgound() {
|
|
|
|
if let tracks = self.playerView?.player?.currentItem?.tracks {
|
|
|
|
for track in tracks {
|
|
|
|
if (track.assetTrack?.hasMediaCharacteristic(AVMediaCharacteristic.visual))! {
|
|
|
|
track.isEnabled = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-02 21:31:14 -03:00
|
|
|
|
2023-05-04 16:11:22 -03:00
|
|
|
@objc func appEnteredForeground() {
|
|
|
|
if let tracks = self.playerView?.player?.currentItem?.tracks {
|
|
|
|
for track in tracks {
|
|
|
|
if (track.assetTrack?.hasMediaCharacteristic(AVMediaCharacteristic.visual))! {
|
|
|
|
track.isEnabled = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-02 21:31:14 -03:00
|
|
|
}
|
|
|
|
}
|