我似乎无法解决这个问题:
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)
谢谢, 森
答案 0 :(得分:2)
根据调用堆栈,showController()
方法在不适当的时间被调用 - 要么在片段准备好之前,要么在它被拆除之后被调用。
对于第一种情况,您应该检查您从Fragment类中调用bindService()
的位置。 onActivityCreated()
将是最好的地方,因为它意味着活动已经完全准备好了。 Fragment的构造函数或类似的地方无法正常工作。
对于第二种情况,您应该确保在片段被拆除时到达的任何onServiceConnected()
都不会导致问题。最简单的方法可能是在onStop()
中设置一个标志并在尝试显示控制器之前检查它。
顺便说一下,您似乎正在将片段对象传递给服务:
callService.setPreparedCallback(SpeciesDetailsFragment.this);
那不是个好主意。服务和UI组件(如片段)具有完全独立的生命周期。这可能与问题有关,或者不是,但我建议你尝试不同的方式。