使用iOS 11在WKWebView中自动播放YouTube视频

13

我希望在我的应用程序中有一个小的YouTube播放器。我找到的唯一推荐方式是将带有YouTube播放器的网页嵌入我的应用程序中。因此,我使用了WKWebView并加载了带有autoplay参数的嵌入式YouTube播放器页面:https://www.youtube.com/embed/VUbsFtLkGN8?autoplay=1

代码如下:

let webConfiguration = WKWebViewConfiguration()
let webView = WKWebView(frame: .zero, configuration: webConfiguration)
let youtubeURL = URL(string: "https://www.youtube.com/embed/VUbsFtLkGN8?autoplay=1")
webView.load(URLRequest(url: youtubeURL!))

此嵌入式播放器url可以在桌面版Safari中自动播放,但在移动版Safari或WKWebView中不会自动播放。我能否以某种方式强制WKWebView自动播放视频?或者使用另一个YouTube播放器url?


苹果不是在阻止视频自动播放吗? - Riyan Fransen
1
是的,在Safari中可以,但我认为你应该能够在你的应用程序中自动播放。 - Ondra
找到解决方案了吗? - Ankit Kumar Gupta
5个回答

1

看起来自前一个答案以来,iframe API已经发生了变化。我根据iframe API参考更新了在WKWebView中加载的HTML。使用这段代码,我成功地在iOS11上的WKWebView中全屏自动播放YouTube视频。

class YouTubeVideoPlayerVC: UIViewController {

    @IBOutlet weak var videoPlayerView: WKWebView!
    var videoURL:URL!  // has the form "https://www.youtube.com/embed/videoID"
    var didLoadVideo = false

    override func viewDidLoad() {
        super.viewDidLoad()
        videoPlayerView.configuration.mediaTypesRequiringUserActionForPlayback = []
    }

    override func viewDidLayoutSubviews() {
        super.viewDidLayoutSubviews()
        // Size of the webView is used to size the YT player frame in the JS code 
        // and the size of the webView is only known in `viewDidLayoutSubviews`, 
        // however, this function is called again once the HTML is loaded, so need 
        // to store a bool indicating whether the HTML has already been loaded once
        if !didLoadVideo {
            videoPlayerView.loadHTMLString(embedVideoHtml, baseURL: nil)
            didLoadVideo = true
        }
    }

    var embedVideoHtml:String {
        return """
        <!DOCTYPE html>
        <html>
        <body>
        <!-- 1. The <iframe> (and video player) will replace this <div> tag. -->
        <div id="player"></div>

        <script>
        var tag = document.createElement('script');

        tag.src = "https://www.youtube.com/iframe_api";
        var firstScriptTag = document.getElementsByTagName('script')[0];
        firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);

        var player;
        function onYouTubeIframeAPIReady() {
        player = new YT.Player('player', {
        height: '\(videoPlayerView.frame.height)',
        width: '\(videoPlayerView.frame.width)',
        videoId: '\(videoURL.lastPathComponent)',
        events: {
        'onReady': onPlayerReady
        }
        });
        }

        function onPlayerReady(event) {
        event.target.playVideo();
        }
        </script>
        </body>
        </html>
        """
    }
}

你能帮我解决这个问题吗?https://dev59.com/blMI5IYBdhLWcg3wxudI - Virani Vivek

0

Xamarin 版本

- 可以在视图加载时添加(硬编码高度)。我还为 playerVars 添加了 "rel",只显示我的频道相关视频。

WKWebview 是动态创建并添加到容器中的。想法是让 iframe 填充 WKWebView,WKWebView 填充容器。我无法将高度设置为 100%,因此它会根据设备进行缩放,但似乎硬编码的 640 高度适用于大多数设备。

我还直接通过变量传递 youtube id,这基本上就是 videoURL.lastPathComponent 返回的内容。 自动播放正常工作,这是该线程的主要解决方案。

var wkWebConfig = UIDevice.CurrentDevice.CheckSystemVersion(10, 0)
                    ? new WKWebViewConfiguration
                    {
                        MediaTypesRequiringUserActionForPlayback = WKAudiovisualMediaTypes.None
                    }
                    : new WKWebViewConfiguration
                    {
                        MediaPlaybackRequiresUserAction = false
                    };

            _wkYoutubePlayer = new WKWebView(new CGRect(), wkWebConfig)
            {
                AccessibilityIdentifier = "viewPlayerYouTube",
                TranslatesAutoresizingMaskIntoConstraints = false
            };

            viewPlayerContainerYouTube.AddSubview(_wkYoutubePlayer);

            NSLayoutConstraint.Create(_wkYoutubePlayer, NSLayoutAttribute.Top, NSLayoutRelation.Equal, viewPlayerContainerYouTube, NSLayoutAttribute.Top, 1, 0).Active = true;
            NSLayoutConstraint.Create(_wkYoutubePlayer, NSLayoutAttribute.Leading, NSLayoutRelation.Equal, viewPlayerContainerYouTube, NSLayoutAttribute.Leading, 1, 0).Active = true;
            NSLayoutConstraint.Create(_wkYoutubePlayer, NSLayoutAttribute.Trailing, NSLayoutRelation.Equal, viewPlayerContainerYouTube, NSLayoutAttribute.Trailing, 1, 0).Active = true;
            NSLayoutConstraint.Create(_wkYoutubePlayer, NSLayoutAttribute.Bottom, NSLayoutRelation.Equal, viewPlayerContainerYouTube, NSLayoutAttribute.Bottom, 1, 0).Active = true;

            var html = $"<!DOCTYPE html><html><body><div id=\"player\"></div><script>var tag = document.createElement('script');tag.src = \"https://www.youtube.com/iframe_api\";var firstScriptTag = document.getElementsByTagName('script')[0];firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);var player;function onYouTubeIframeAPIReady() {{ player = new YT.Player('player', {{ height: '640', width: '100%', videoId: '{_youtubeVideoId}', playerVars: {{ rel: '0' }}, events: {{ 'onReady': onPlayerReady }} }}); }} function onPlayerReady(event) {{ event.target.playVideo(); }}</script></body></html>";
            _wkYoutubePlayer.LoadHtmlString(html, null);

0
let videoView = UIView(frame: CGRect(x: 0, y: 80, width: self.view.frame.width, height: self.view.frame.height - 80))
            self.view.addSubview(videoView)

let youTubeVideoHTML: String = "<!DOCTYPE html><html><head><style>body{margin:0px 0px 0px 0px;}</style></head> <body> <div id=\"player\"></div> <script> var tag = document.createElement('script'); tag.src = \"https://www.youtube.com/player_api\"; var firstScriptTag = document.getElementsByTagName('script')[0]; firstScriptTag.parentNode.insertBefore(tag, firstScriptTag); var player; function onYouTubePlayerAPIReady() { player = new YT.Player('player', { width:'%0.0f', height:'%0.0f', videoId:'%@', events: { 'onReady': onPlayerReady} }); } function onPlayerReady(event) { event.target.playVideo(); } </script> </body> </html>"

let html: String = String(format: youTubeVideoHTML, videoView.frame.width, videoView.frame.height, id)


let webConfiguration = WKWebViewConfiguration()
if #available(iOS 9.0, *) {

     webConfiguration.allowsAirPlayForMediaPlayback = true
     webConfiguration.allowsInlineMediaPlayback = true
     webConfiguration.allowsPictureInPictureMediaPlayback = true
     webConfiguration.mediaTypesRequiringUserActionForPlayback = []

}

webView = WKWebView(frame: CGRect(x: 0, y: 0, width: videoView.frame.width, height: videoView.frame.height), configuration: webConfiguration)
webView.uiDelegate = self
webView.scrollView.isScrollEnabled = false
webView.scrollView.zoomScale = 0
webView.isMultipleTouchEnabled = false
webView.scrollView.isPagingEnabled = false
webView.scrollView.isMultipleTouchEnabled = false
videoView.addSubview(webView)

webView.loadHTMLString(html, baseURL: URL(string: "https://www.youtube.com"))

-2
var youTubeVideoHTML: String = "<!DOCTYPE html><html><head><style>body{margin:0px 0px 0px 0px;}</style></head> <body> <div id=\"player\"></div> <script> var tag = document.createElement('script'); tag.src = \"http://www.youtube.com/player_api\"; var firstScriptTag = document.getElementsByTagName('script')[0]; firstScriptTag.parentNode.insertBefore(tag, firstScriptTag); var player; function onYouTubePlayerAPIReady() { player = new YT.Player('player', { width:'%0.0f', height:'%0.0f', videoId:'%@', events: { 'onReady': onPlayerReady, } }); } function onPlayerReady(event) { event.target.playVideo(); } </script> </body> </html>"

func playVideoWithId(videoId: String) {
  var html: String = String(format: youTubeVideoHTML, self.frame.size.width, self.frame.size.height, videoId)


}

或者你可以使用

YTPlayer播放Youtube视频


1
这不是完整的代码,也不会自动播放。 - Ondra
2
YTPlayer库似乎不再维护了。 - Ondra

-2

对于小型的Youtube MediaPlayer,您可以应用以下方法:

YourVideoView.configuration.allowsInlineMediaPlayback = true

你也可以通过故事板上的复选框来执行内联播放

enter image description here


对于自动播放,请在上面的截图中的“交互”处取消选中“视频播放”。 - Kae10

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接