如何解决在 Swift 中导航“返回”时 AVPlayer 视频消失
导航到子视图时,我已将其设置为自动播放视频。在视频的底部,有一组指向相关内容的链接。当点击其中一个时,一个新的视图被推入堆栈,另一个视频开始播放。
我已尝试更新 CGRect 框架,使用 onAppear 重新初始化视频播放器,并且还遵循了建议 here。
到目前为止似乎没有任何效果。这是我用于实际视频播放器的代码(改编自 Chris Mash 的网站):
import SwiftUI
import AVKit
import UIKit
import AVFoundation
let playerLayer = AVPlayerLayer()
class PlayVideo: UIView {
init(frame: CGRect,url: URL) {
super.init(frame: frame)
// Create the video player using the URL passed in.
let player = AVPlayer(url: url)
player.volume = 100 // Will play audio if you don't set to zero
player.play() // Set to play once created
// Add the player to our Player Layer
playerLayer.player = player
playerLayer.videoGravity = .resizeAspectFill // Resizes content to fill whole video layer.
playerLayer.backgroundColor = UIColor.black.cgColor
layer.addSublayer(playerLayer)
}
required init?(coder: NSCoder) {
super.init(coder: coder)
}
override func layoutSubviews() {
super.layoutSubviews()
playerLayer.frame = bounds
}
static func pauseVideo() {
playerLayer.player?.pause()
}
}
struct ViewVideo: UIViewRepresentable {
var videoURL:URL
var previewLength:Double?
func makeUIView(context: Context) -> UIView {
return PlayVideo(frame: .zero,url: videoURL)
}
func updateUIView(_ uiView: UIView,context: Context) {
}
}
这是从主视图调用的: ViewVideo(videoURL: videoURL)
我能想到的唯一解决方法是禁用后退按钮并强制用户每次都返回主视图。这是一个糟糕的选择,我希望有人能在这里提供一些有用的建议。谢谢-
解决方法
如果我理解正确,您会在导航到新视图时播放不同的视频。所以您创建了一个新的 PlayVideo
视图?
那么问题是您的 playerLayer
是一个静态属性。新视图会将新玩家设置到 playerLayer
中并替换旧玩家。同样,如果您暂停一个播放器,则两个播放器都会暂停。此外,将播放器作为子图层添加到新视图会将其从旧视图中删除。
您需要将 playerLayer
作为视图的本地属性。或者至少为您要播放的每个视频添加一个 AVPlayerLayer
。然后你需要一个机制来暂停/重新启动每个视频,当它变得可见时。例如通过实现 viewWillAppear
。当您导航回视图/视图变得可见时,始终会调用此方法。
感谢@dominik-105 帮助解决这个问题。我能够使用提出的建议解决问题。
具体来说,我删除了 playerLayer 的全局定义,而是将其作为局部变量放置在我的主视图调用中:
var playerLayer = AVPlayerLayer()
然后我使用 playerLayer: playerLayer 标签调用 ViewVideo,然后 ViewVideo 使用 playerLayer:AVPlayerLayer 作为初始化的一部分调用 PlayVideo。
有趣的是,这会导致我用来定义视频框大小的覆盖布局功能出现问题。我现在直接在 init 中定义框架并删除旧的覆盖功能代码。完整代码现在是:
import SwiftUI
import AVKit
import UIKit
import AVFoundation
class PlayVideo: UIView {
init(frame: CGRect,url: URL,playerLayer: AVPlayerLayer,width: CGFloat,height: CGFloat) {
super.init(frame: frame)
// Create the video player using the URL passed in.
let player = AVPlayer(url: url)
player.volume = 100 // Will play audio if you don't set to zero
player.play() // Set to play once created
// Add the player to our Player Layer
playerLayer.player = player
playerLayer.videoGravity = .resizeAspectFill // Resizes content to fill whole video layer.
playerLayer.backgroundColor = UIColor.black.cgColor
playerLayer.player?.actionAtItemEnd = .pause
layer.addSublayer(playerLayer)
playerLayer.frame = CGRect(x: 0,y: 0,width: width,height: height)
}
required init?(coder: NSCoder) {
super.init(coder: coder)
}
}
struct ViewVideo: UIViewRepresentable {
var videoURL:URL
var playerLayer: AVPlayerLayer
var width: CGFloat
var height: CGFloat
func makeUIView(context: Context) -> UIView {
return PlayVideo(frame: .zero,url: videoURL,playerLayer: playerLayer,height: height)
}
func updateUIView(_ uiView: UIView,context: Context) {
}
}
我使用 GeometryReader 来定义框的大小,然后将宽度和高度传递给结构体,结构体将其传递给类。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。