las*_*off 5 video-player swift swiftui
我正在尝试创建一个视图,允许我发送要播放的视频的 URL。我的观点是这样的:
import SwiftUI
import AVKit
struct PlayerView: View {
var videoURL : String
private let player = AVPlayer(url: URL(string: "https://bitdash-a.akamaihd.net/content/sintel/hls/playlist.m3u8")!)
var body: some View {
VideoPlayer(player: player)
.onAppear() {
// Start the player going, otherwise controls don't appear
player.play()
}
.onDisappear() {
// Stop the player when the view disappears
player.pause()
}
}
}
Run Code Online (Sandbox Code Playgroud)
如果尝试将 URL 设置为传递的参数,我会收到此错误:
Cannot use instance member 'videoURL' within property initializer; property initializers run before 'self' is available
Run Code Online (Sandbox Code Playgroud)
如何将 URL 传递给 View 来播放不同的电影?
我按照建议更改了代码,但视频无法播放:
import SwiftUI
import AVKit
struct PlayerView: View {
var videoURL : String
@State private var player : AVPlayer?
var body: some View {
VideoPlayer(player: player)
.onAppear() {
// Start the player going, otherwise controls don't appear
guard let url = URL(string: videoURL) else {
return
}
print(url)
let player = AVPlayer(url: url)
self.player = player
self.player?.seek(to: CMTime.zero)
self.player!.play()
}
.onDisappear() {
// Stop the player when the view disappears
player?.pause()
}
}
}
Run Code Online (Sandbox Code Playgroud)
我正在尝试更改计时器上的视频。我试过这个:
} else {
VideoPlayer(player: viewModel.player)
.onAppear {
viewModel.urlString = "https://testsite.neumont.edu/images/green.mp4"
}
.onReceive(movieSwitchTimer) { _ in
DispatchQueue.main.asyncAfter(deadline: .now() + 5) {
viewModel.urlString = "https://testsite.neumont.edu/images/earth.mp4"
}
self.activeImageIndex = (self.activeImageIndex + 1) % slideStore.images.count
}
}
Run Code Online (Sandbox Code Playgroud)
有多种方法可以实现此目的,但由于您已经在使用onAppear,这似乎是实现此目的的好地方:
struct PlayerView: View {
var videoURL : String
@State private var player : AVPlayer?
var body: some View {
VideoPlayer(player: player)
.onAppear() {
// Start the player going, otherwise controls don't appear
guard let url = URL(string: videoURL) else {
return
}
let player = AVPlayer(url: url)
self.player = player
player.play()
}
.onDisappear() {
// Stop the player when the view disappears
player?.pause()
}
}
}
Run Code Online (Sandbox Code Playgroud)
您会希望player如此,@State因为它应该在视图的渲染之间持续存在。
根据评论更新,处理 URL 可能更改的场景:
class VideoViewModel : ObservableObject {
@Published var urlString : String? {
didSet {
guard let urlString = urlString, let url = URL(string: urlString) else {
return
}
player = AVPlayer(url: url)
player.seek(to: .zero)
player.play()
}
}
var player = AVPlayer()
}
struct ContentView : View {
@StateObject var viewModel = VideoViewModel()
var body: some View {
VideoPlayer(player: viewModel.player)
.onAppear {
viewModel.urlString = "https://bitdash-a.akamaihd.net/content/sintel/hls/playlist.m3u8"
DispatchQueue.main.asyncAfter(deadline: .now() + 5) {
viewModel.urlString = Bundle.main.url(forResource: "IMG_0226", withExtension: "mp4")!.absoluteString
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
在更新中,您可以看到我将播放器存储在ObservableObject. 每当 url 字符串发生更改时,它就会重新加载播放器。这onAppear在您的代码中不是必需的——它只是显示在不同时间加载不同视频的一种方式(第一个是您提供的 URL,第二个是我的捆绑包中的 URL)。
| 归档时间: |
|
| 查看次数: |
4191 次 |
| 最近记录: |