Android 1.6: "android.view.WindowManager $BadTokenException: Невозможно добавить окно - токен null не для приложения"
Я пытаюсь открыть диалоговое окно, но каждый раз, когда я пытаюсь его открыть, он генерирует это исключение:
Uncaught handler: thread main exiting due to uncaught exception
android.view.WindowManager$BadTokenException:
Unable to add window -- token null is not for an application
at android.view.ViewRoot.setView(ViewRoot.java:460)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:177)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:91)
at android.app.Dialog.show(Dialog.java:238)
at android.app.Activity.showDialog(Activity.java:2413)
Я создаю его, вызывая showDialog
с идентификатором отображения. Обработчик обработчика onCreateDialog
отлично работает, и я могу пройти через него без проблем, но я привязал его, поскольку кажется, что я чего-то не хватает:
@Override
public Dialog onCreateDialog(int id)
{
Dialog dialog;
Context appContext = this.getApplicationContext();
switch(id)
{
case RENAME_DIALOG_ID:
Log.i("Edit", "Creating rename dialog...");
dialog = new Dialog(appContext);
dialog.setContentView(R.layout.rename);
dialog.setTitle("Rename " + noteName);
break;
default:
dialog = null;
break;
}
return dialog;
}
Есть ли что-то от этого? Некоторые вопросы говорили об этой проблеме при создании диалога из onCreate
, который происходит потому, что активность еще не создана, но это происходит из вызова из объекта меню, а переменная appContext
кажется, что она правильно заполнен в отладчике.
Ответы
Ответ 1
то есть. эта строка:
Context appContext = this.getApplicationContext();
должен идти, а вместо этого вы используете указатель на активность, в которой вы находитесь (возможно, это).
Я тоже укусил это сегодня, раздражающая часть getApplicationContext()
является дословным от developer.android.com: (
Ответ 2
Вы не можете отобразить окно/диалог приложения через контекст, который не является активностью. Попробуйте передать действительную ссылку на действия
Ответ 3
То же самое можно сказать о функции getApplicationContext.
Документы на сайте android говорят использовать его, но он не работает... grrrrr:-P
Просто выполните:
dialog = new Dialog(this);
"this" обычно является вашей активностью, с которой вы начинаете диалог.
Ответ 4
В документах Android предлагается использовать getApplicationContext();
но он не будет работать вместо того, чтобы использовать текущую активность при создании AlertDialog.Builder или AlertDialog или Dialog...
Пример:
AlertDialog.Builder builder = new AlertDialog.Builder(this);
или
AlertDialog.Builder builder = new AlertDialog.Builder((Your Activity).this);
Ответ 5
Вместо getApplicationContext()
просто используйте ActivityName.this
Ответ 6
У меня была аналогичная проблема, когда у меня был другой класс:
public class Something {
MyActivity myActivity;
public Something(MyActivity myActivity) {
this.myActivity=myActivity;
}
public void someMethod() {
.
.
AlertDialog.Builder builder = new AlertDialog.Builder(myActivity);
.
AlertDialog alert = builder.create();
alert.show();
}
}
Работала отлично в большинстве случаев, но иногда она разбивалась с той же ошибкой. Тогда я понимаю, что в MyActivity
у меня было...
public class MyActivity extends Activity {
public static Something something;
public void someMethod() {
if (something==null) {
something=new Something(this);
}
}
}
Поскольку я удерживал объект как static
, второй запуск кода по-прежнему содержал исходную версию объекта и, следовательно, все еще имел в виду исходный Activity
, который долго не существовал.
Глупая глупая ошибка, тем более, что мне действительно не нужно было держать объект как static
в первую очередь...
Ответ 7
Просто измените его на
AlertDialog.Builder alert_Categoryitem =
new AlertDialog.Builder(YourActivity.this);
Вместо
AlertDialog.Builder alert_Categoryitem =
new AlertDialog.Builder(getApplicationContext());
Ответ 8
Другим решением является установка типа окна в системный диалог:
dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
Для этого требуется разрешение SYSTEM_ALERT_WINDOW
:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
Как говорится в документах:
Очень немногие приложения должны использовать это разрешение; эти окна предназначены для взаимодействия на уровне системы с пользователем.
Это решение, которое вы должны использовать, только если вам требуется диалог, не связанный с активностью.
Ответ 9
Не используйте getApplicationContext()
при объявлении dialouge
Всегда используйте this
или ваш activity.this
Ответ 10
Это работало для меня -
new AlertDialog.Builder(MainActivity.this)
.setMessage(Html.fromHtml("<b><i><u>Spread Knowledge Unto The Last</u></i></b>"))
.setCancelable(false)
.setPositiveButton("Dismiss",
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int id) {
}
}).show();
Используйте
ActivityName.this
Ответ 11
Для вложенных диалогов эта проблема очень распространена, она работает, когда
AlertDialog.Builder mDialogBuilder = new AlertDialog.Builder(MyActivity.this);
используется вместо
mDialogBuilder = new AlertDialog.Builder(getApplicationContext);
эту альтернативу.
Ответ 12
Вы также можете сделать это
public class Example extends Activity {
final Context context = this;
final Dialog dialog = new Dialog(context);
}
Это сработало для меня!!
Ответ 13
public class Splash extends Activity {
Location location;
LocationManager locationManager;
LocationListener locationlistener;
ImageView image_view;
ublic static ProgressDialog progressdialog;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.splash);
progressdialog = new ProgressDialog(Splash.this);
image_view.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
locationManager.requestLocationUpdates("gps", 100000, 1, locationlistener);
Toast.makeText(getApplicationContext(), "Getting Location plz wait...", Toast.LENGTH_SHORT).show();
progressdialog.setMessage("getting Location");
progressdialog.show();
Intent intent = new Intent(Splash.this,Show_LatLng.class);
// }
});
}
Текст здесь: -
используйте это для получения контекста activity
для progressdialog
progressdialog = new ProgressDialog(Splash.this);
или progressdialog = new ProgressDialog(this);
используйте это для получения контекста приложения для BroadcastListener
не для progressdialog
.
progressdialog = new ProgressDialog(getApplicationContext());
progressdialog = new ProgressDialog(getBaseContext());
Ответ 14
Как говорится, для контекста требуется контекст для действия, используйте "YourActivity.this" для статического контекста или проверьте здесь, как использовать динамический один в безопасном режиме
Ответ 15
Попробуйте reset dialog
тип окна
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT:
dialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
Не забудьте использовать разрешение android.permission.SYSTEM_ALERT_WINDOW
Ответ 16
Самый лучший и самый безопасный способ показать "ProgressDialog" в AsyncTask, избегая проблемы с утечкой памяти, - это использовать "Обработчик" с Looper.main().
private ProgressDialog tProgressDialog;
то в 'onCreate'
tProgressDialog = new ProgressDialog(this);
tProgressDialog.setMessage(getString(R.string.loading));
tProgressDialog.setIndeterminate(true);
Теперь вы выполнили настройку. Теперь вызовите showProgress() и hideProgress() в AsyncTask.
private void showProgress(){
new Handler(Looper.getMainLooper()){
@Override
public void handleMessage(Message msg) {
tProgressDialog.show();
}
}.sendEmptyMessage(1);
}
private void hideProgress(){
new Handler(Looper.getMainLooper()){
@Override
public void handleMessage(Message msg) {
tProgressDialog.dismiss();
}
}.sendEmptyMessage(1);
}