MediaPlayer MediaController服务

时间:2014-06-01 14:36:57

标签: android android-service android-mediaplayer mediacontroller

我似乎无法解决这个问题:

android.view.WindowManager $ BadTokenException:无法添加窗口 - 令牌null无效;你的活动在运行吗?

我有一个应用程序,它有一个托管具有MediaController的片段的活动。该活动绑定到托管媒体播放器的服务。我想要做的是从活动A开始播放一首歌,展示mediacontroller。然后离开活动A并输入活动B重新连接到服务,如果音频仍在播放,则在活动B中显示MediaController。无论我做什么,我都会不断收到错误。活动A和B只是相同活动的不同实例。如果您需要查看任何其他代码,请与我们联系。

 private ServiceConnection callConnection = new ServiceConnection() {

    @Override
    public void onServiceConnected( ComponentName name, IBinder service ) {
        Log.d("LEAKTEST", "Connected to instance " + this.toString());
        AudioService.MusicBinder binder = ( AudioService.MusicBinder ) service;
        callService = binder.getService();
        callService.setPreparedCallback( SpeciesDetailsFragment.this );
        callService.setList( callList );
        callService.setSpeciesId( getSpeciesId() );
        if ( callService.isPng() ) {
            setController();
            showController();
        }
        musicBound = true;
    }

    @Override
    public void onServiceDisconnected( ComponentName name ) {
        musicBound = false;
    }
};

private void setController() {
    if ( controller == null ) {
        controller = new CallController( getActivity() );
    }
    controller.setPrevNextListeners( new View.OnClickListener() {
        @Override
        public void onClick( View v ) {
            playNext();
        }
    }, new View.OnClickListener() {
        @Override
        public void onClick( View v ) {
            playPrev();
        }
    } );
    controller.setMediaPlayer( this );
    controller.setAnchorView( rootView );
}

private void showController(){
    controller.show(0);
    controller.setEnabled( true );
}

   @Override
public void onPause() {
    controller.hide();
    loadingDialog.hide();
    super.onPause();
}

@Override
public void onStop() {
    controller.hide();
    controller = null;
    // Unbind from the service
    if ( musicBound ) {
        getActivity().unbindService( callConnection );
        musicBound = false;
    }
    super.onStop();
}

完整Stacktrace:

06-03 19:38:11.986  17417-17417/com.myname.appname.core E/AndroidRuntime﹕ FATAL EXCEPTION: main
Process: com.myname.appname.core, PID: 17417
android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running?
        at android.view.ViewRootImpl.setView(ViewRootImpl.java:536)
        at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:259)
        at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:69)
        at android.widget.MediaController.show(MediaController.java:346)
        at com.myname.appname.core.details.SpeciesDetailsFragment.showController(SpeciesDetailsFragment.java:354)
        at com.myname.appname.core.details.SpeciesDetailsFragment.access$700(SpeciesDetailsFragment.java:49)
        at com.myname.appname.core.details.SpeciesDetailsFragment$6.onServiceConnected(SpeciesDetailsFragment.java:382)
        at android.app.LoadedApk$ServiceDispatcher.doConnected(LoadedApk.java:1117)
        at android.app.LoadedApk$ServiceDispatcher$RunConnection.run(LoadedApk.java:1134)
        at android.os.Handler.handleCallback(Handler.java:733)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:136)
        at android.app.ActivityThread.main(ActivityThread.java:5144)
        at java.lang.reflect.Method.invokeNative(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:515)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:795)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:611)
        at dalvik.system.NativeStart.main(Native Method)

谢谢, 森

1 个答案:

答案 0 :(得分:2)

根据调用堆栈,showController()方法在不适当的时间被调用 - 要么在片段准备好之前,要么在它被拆除之后被调用。

对于第一种情况,您应该检查您从Fragment类中调用bindService()的位置。 onActivityCreated()将是最好的地方,因为它意味着活动已经完全准备好了。 Fragment的构造函数或类似的地方无法正常工作。

对于第二种情况,您应该确保在片段被拆除时到达的任何onServiceConnected()都不会导致问题。最简单的方法可能是在onStop()中设置一个标志并在尝试显示控制器之前检查它。

顺便说一下,您似乎正在将片段对象传递给服务:

callService.setPreparedCallback(SpeciesDetailsFragment.this);

那不是个好主意。服务和UI组件(如片段)具有完全独立的生命周期。这可能与问题有关,或者不是,但我建议你尝试不同的方式。