为什么YouTube Player API无法在Android 11上运行?

问题描述

我尝试了此YouTube视频中介绍的有关如何在Android应用上播放YouTube视频的项目:

https://www.youtube.com/watch?v=Up9BjrIuoXY

我尝试在使用Android 9和Android 10的设备上播放YouTube视频,并且视频可以正确播放,但是在Android 11设备上,我在YouTube视频窗口中收到了以下消息:

“初始化YouTube播放器时发生错误”。

这是MainActivity.java的代码

package com.example.youtube;

import android.os.Bundle;

import com.google.android.material.floatingactionbutton.FloatingActionButton;
import com.google.android.material.snackbar.Snackbar;
import com.google.android.youtube.player.YouTubeBaseActivity;
import com.google.android.youtube.player.YouTubeInitializationResult;
import com.google.android.youtube.player.YouTubePlayer;
import com.google.android.youtube.player.YouTubePlayerView;

import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;

import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.Button;

import android.widget.Toast;

public class MainActivity extends YouTubeBaseActivity {

    Button play_btn;
    YouTubePlayerView youtubePlayerView;
    YouTubePlayer.OnInitializedListener onInitializedListener;

    private static final int RECOVERY_DIALOG_REQUEST = 1;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        youtubePlayerView = findViewById(R.id.youtubeView);
        play_btn = findViewById(R.id.playvideo_btn);

        final android.app.Activity myActivity = this;

        onInitializedListener =new YouTubePlayer.OnInitializedListener() {
            @Override
            public void onInitializationSuccess(YouTubePlayer.Provider provider,YouTubePlayer youTubePlayer,boolean b) {
                youTubePlayer.loadVideo("Up9BjriuoXY");
            }

            @Override
            public void onInitializationFailure(YouTubePlayer.Provider provider,YouTubeInitializationResult youTubeInitializationResult) {
                if (youTubeInitializationResult.isUserRecoverableError()) {
                    youTubeInitializationResult.getErrorDialog(myActivity,RECOVERY_DIALOG_REQUEST).show();
                } else {
                    String errorMessage = String.format(
                            getString(R.string.error_player),youTubeInitializationResult.toString());
                    Toast.makeText(myActivity,errorMessage,Toast.LENGTH_LONG).show();
                }
            }
        };

        play_btn.setonClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                youtubePlayerView.initialize("AIzaSyZ2",onInitializedListener);
            }
        });

    }
}

Android 11设备的YouTube应用已更新。

希望有人可以帮助我。谢谢!

解决方法

由于Android 11中的软件包可见性限制,除非您在清单中添加对YouTube服务的引用,否则该SDK将无法找到它。

<queries>
   <intent>
     <action android:name="com.google.android.youtube.api.service.START" />
   </intent>
</queries>
,

发生这种情况是因为从 Android 11 (API 30+) 开始,Android 会隐藏默认情况下不属于核心系统应用/服务的所有其他应用和服务,您必须在 Manifest.xml您需要访问特定应用程序。从文档中,这些是您无需声明需要访问的唯一应用程序:

  • 您自己的应用。

  • 某些系统包,例如媒体提供者,实现 安卓核心功能。详细了解如何确定哪些 包在运行您的应用程序的设备上自动可见。 安装了您的应用的应用。

  • 启动活动的任何应用程序 您的应用程序使用 startActivityForResult() 方法,如 关于如何从活动中获得结果的指南。

  • 任何应用 启动或绑定到您应用中的服务。

  • 任何访问 您应用中的内容提供程序。

  • 任何具有内容提供程序的应用程序, 您的应用已被授予访问该内容的 URI 权限 提供者。

  • 任何从您的应用接收输入的应用。本案适用 仅当您的应用作为输入法编辑器提供输入时。

您可以通过 3 种方式声明您需要访问权限:

...在应用的清单文件中添加元素。在元素内,指定其他应用程序 by package nameby intent signatureby provider authority

就 YouTube 而言,通过包名称进行声明似乎最有意义,如下所示(注意 - 包名称不是 com.youtube!):

<queries>
   <package android:name="com.google.android.youtube" />
</queries>

即使您只是打开一个 YouTube 网络链接,您似乎也需要这个,我想是因为如果您安装了 YouTube 应用程序,它会将打开 https://youtu.be/blah 之类的任何尝试作为对 YouTube 的请求进行注册应用程序,而不是网络浏览器。

,

更新到 Ivagarz 的回答查询标签在应用程序标签下不起作用所以你必须像这样在清单标签的应用程序标签之外添加它

<manifest
//permissions here
    <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:supportsRtl="true"
            android:theme="@style/AppTheme">
        </application>
        <queries>
        <intent>
         <action android:name="com.google.android.youtube.api.service.START"/>
        </intent>
        </queries>
    </manifest>
,

好吧。最后,我们决定使用UniWebView,因为iOS对该库的支持实际上为零。我们的项目针对iOS和Android,并且Google自古以来就没有更新其YouTube API。它使用了Apple拒绝从商店中拒绝的UIWebView。

,

虽然专门请求 Youtube 应用程序有效,但我发现它有点过于具体,因为如果用户安装了替代播放器(不确定是否有),解决方案就会中断。 我认为正确的解释是 Youtube 应用程序不仅可以过滤 https 方案,还可以过滤主机“youtube.com”(这很有意义)。因此,通过指定主机,我让 Youtube 应用程序可用于我的应用程序,而无需特定于包的声明:

<queries>    
    <intent>
        <action android:name="android.intent.action.VIEW" />
        <data  android:scheme="https" android:host="youtube.com" />
    </intent>
</queries>

顺便说一句:当我测试时,“youtube.com”和“youtu.be”在这两种链接上的效果都是一样的。为了安全起见,我仍然在我的应用中加入了两者。