java - 如何在设备 sleep 时保持 ChromeCast session 处于 Activity 状态?

标签 java android audio chromecast

我有一个应用程序,可以将本地媒体内容流式传输到 Chromecast 接收器。这在大多数情况下都有效,但当设备处于 sleep 状态且未使用外部电源时, session 将在大约 5 分钟后终止/断开连接(从屏幕变黑时开始测量)。

我已经在这里查看过这个问题:

How do I keep a ChromeCast route alive when my app is in the background on battery power?

...并实现了两个建议的答案。

具体来说,在我的应用程序 list 中,我有:

<uses-permission android:name="android.permission.WAKE_LOCK" />

然后在我用来将媒体内容流式传输到 Chromecast 接收器的嵌入式 HTTP 服务器中,我正在执行以下操作:

PowerManager powerManager = (PowerManager)Environment.getApplicationContext().getSystemService(Context.POWER_SERVICE);
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "cast-server-cpu");
wakeLock.acquire();

WifiManager wifiManager = (WifiManager)Environment.getApplicationContext().getSystemService(Context.WIFI_SERVICE);
wifiLock = wifiManager.createWifiLock(WifiManager.WIFI_MODE_FULL_HIGH_PERF, "cast-server-net");
wifiLock.acquire();

该代码在服务器线程启动时执行(选择 Chromecast 路由时发生),并且在服务器停止之前不会释放锁定。这发生在我的 MediaRouter.Callback 实现中,它(包含上面问题中接受的答案)如下所示:

this.mediaRouterCallback = new MediaRouter.Callback() {
    private MediaRouter.RouteInfo autoconnectRoute = null;

    @Override
    public void onRouteSelected(MediaRouter mediaRouter, MediaRouter.RouteInfo routeInfo) {
        if (autoconnectRoute == null) {
            if (isPlaying()) {
                stop();
                playButton.setImageResource(R.drawable.icon_play);
            }

            castDevice = CastDevice.getFromBundle(routeInfo.getExtras());
            if (castServer != null) {
                castServer.stop();
            }

            //start the Chromecast file server
            castServer = new CastHttpFileServer();
            new Thread(castServer).start();
        }

        initCastClientListener();
        initRemoteMediaPlayer();
        launchReceiver();
    }

    @Override
    public void onRouteUnselected(MediaRouter mediaRouter, MediaRouter.RouteInfo routeInfo) {
        if (! doesRouterContainRoute(mediaRouter, routeInfo)) {
            //XXX:  do not disconnect in this case (the unselect was not initiated by a user action); see https://stackoverflow.com/questions/18967879/how-do-i-keep-a-chromecast-route-alive-when-my-app-is-in-the-background-on-batte
            autoconnectRoute = routeInfo;
            return;
        }

        //user disconnected, teardown and stop playback
        if (isPlaying()) {
            stop();
            playButton.setImageResource(R.drawable.icon_play);
        }
        if (castServer != null) {
            castServer.stop();
            castServer = null;
        }

        teardown();
        castDevice = null;
    }

    @Override
    public void onRouteAdded(MediaRouter router, MediaRouter.RouteInfo route) {
        super.onRouteAdded(router, route);
        if (autoconnectRoute != null && route.getId().equals(autoconnectRoute.getId())) {
            this.onRouteSelected(router, route);
        }
    }

    private boolean doesRouterContainRoute(MediaRouter router, MediaRouter.RouteInfo route) {
        if (router == null || route == null) {
            return false;
        }

        for (MediaRouter.RouteInfo info : router.getRoutes()) {
            if (info.getId().equals(route.getId())) {
                return true;
            }
        }

        return false;
    }
};

当设备处于 sleep 状态且未连接外部电源时,还需要做什么才能保持 Chromecast session 处于 Activity 状态?

此外,考虑到此问题仅在设备未连接到 USB 时才会重现,在这种情况下调试的有效方法是什么?

编辑

这是 session 终止时报告的堆栈跟踪(来自嵌入式 HTTP 服务器):

java.net.SocketException: Software caused connection abort
 at java.net.SocketOutputStream.socketWrite0(Native Method)
 at java.net.SocketOutputStream.socketWrite(SocketOutputStream.java:112)
 at java.net.SocketOutputStream.write(SocketOutputStream.java:157)
 at com.myapp.CastHttpFileServer.writeResponse(CastHttpFileServer.java:124)
 at com.myapp.CastHttpFileServer.run(CastHttpFileServer.java:180)

而且我还注意到我可以在 RemoteMediaPlayer.OnStatusUpdatedListener 实现中捕获此事件,例如:

if (lastStatus == MediaStatus.PLAYER_STATE_PLAYING && mediaStatus.getPlayerState() == MediaStatus.PLAYER_STATE_BUFFERING) {
    //buffer underrun/receiver went away because the wifi died; how do we fix this?
}

最佳答案

CommonsWare 的评论是有权利的。我正在运行 Android 7.1.2 的设备上进行测试,并且 Doze mode was causing the network to drop大约 5 分钟不活动后,无论我的应用程序持有何种唤醒锁

修复方法是在 list 中添加以下权限:

<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS" />

然后更新我的 onRouteSelected() 回调实现,以请求在选择 ChromeCast 路由时禁用打瞌睡模式:

if (Build.VERSION.SDK_INT >= 23) {
    String packageName = context.getPackageName();
    PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
    if (! pm.isIgnoringBatteryOptimizations(packageName)) {
        //reguest that Doze mode be disabled
        Intent intent = new Intent();
        intent.setAction(
            Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
        intent.setData(Uri.parse("package:" + packageName));

        context.startActivity(intent);
    }

}

只要用户在出现提示时选择"is",就会遵守应用的唤醒锁,并且即使设备未充电,ChromeCast session 也会保持 Activity 状态。

关于java - 如何在设备 sleep 时保持 ChromeCast session 处于 Activity 状态?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44106286/

相关文章:

audio - FFMPEG:向视频文件添加额外的音轨

android - 使用framework c/c++的funtion来转换android studio

java - Hadoop 作业因大数据上的 native SimString C 代码而失败

java - 如何为 Seam/JPA( hibernate )创建 DAO 类?

android - 如何获取我的 Android 系统上所有 Activity 内核驱动程序的列表?

java - TextView.post Runnable 并不总是有效

android - 如何在不等待结果的情况下在另一个内部运行挂起功能?

python - 无法从 github 或 pip 安装 python 模块 aubio

java - Hibernate:无法部署应用程序打包的持久性提供程序

java - 我可以监视使用 get 方法的对象吗?