簡體   English   中英

設備處於睡眠狀態時如何保持ChromeCast會話的活動狀態?

[英]How to keep a ChromeCast session alive when the device is asleep?

我有一個可以將本地媒體內容流式傳輸到Chromecast接收器的應用。 這主要工作的,所不同的是當設備處於睡眠狀態,而不是在外部電源約5分鍾后,會話將死/斷開(從當屏幕變成空白測量)。

我已經在這里查看了這個問題:

當我的應用程序處於電池供電后台時,如何使ChromeCast路由保持活動狀態?

...並實施了兩個建議的答案。

具體來說,在我的應用清單中,我有:

<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;
    }
};

當設備處於睡眠狀態且未連接外部電源時,為了使Chromecast會話保持活動狀態,還需要做些什么?

此外,考慮到該問題僅在設備連接USB時才會重現,因此在這種情況下進行調試的有效方法是什么?

編輯

這是會話終止時(從嵌入式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的設備上進行測試,並且打Do模式導致網絡閑置約5分鍾后且無論我的應用持有的喚醒鎖為何, 網絡 均掉線

解決方法是在清單中添加以下權限:

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

然后更新我的onRouteSelected()回調實現,以請求/如果選擇了ChromeCast路由,則請禁用“打ze”模式:

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);
    }

}

只要用戶在提示時選擇“是”,就將遵守該應用的喚醒鎖,即使設備未充電,ChromeCast會話仍保持活動狀態。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM