为什么 YouTube Player API 在 Android 11 上不起作用?

Mis*_* L. 18 youtube android android-youtube-api

我尝试了这个 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);
            }
        });

    }
}
Run Code Online (Sandbox Code Playgroud)

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

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

Qui*_*ner 29

更新 Ivagarz 的答案queries标签在应用程序标签下不起作用因此您必须将其添加到application清单标签中的标签外部,如下所示

<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>
Run Code Online (Sandbox Code Playgroud)


iva*_*arz 20

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

<queries>
   <intent>
     <action android:name="com.google.android.youtube.api.service.START" />
   </intent>
</queries>
Run Code Online (Sandbox Code Playgroud)

  • 哇谢谢!您在哪里找到这个神秘的“com.google.android.youtube.api.service.START”?这有记录在任何地方吗? (2认同)

Jam*_*len 9

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

  • 您自己的应用程序。

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

  • 任何使用 startActivityForResult() 方法在您的应用程序中启动 Activity 的应用程序,如有关如何从 Activity 获取结果的指南中所述。

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

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

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

  • 任何从您的应用程序接收输入的应用程序。这种情况仅适用于您的应用程序提供输入作为输入法编辑器的情况。

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

...在应用程序的清单文件中添加元素。在元素内,通过包名意图签名提供者权限指定其他应用程序

在 YouTube 的情况下,通过包名声明似乎最有意义,它看起来像这样(注意 - 包名不是com.youtube!):

<queries>
   <package android:name="com.google.android.youtube" />
</queries>
Run Code Online (Sandbox Code Playgroud)

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


Seb*_*cks 6

虽然专门请求 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>
Run Code Online (Sandbox Code Playgroud)

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


Mis*_* L. 0

出色地。最后我们决定使用 UniWebView Unity 包和 YouTube JavaScript API。我们的项目针对 iOS 和 Android。