Когда я показываю Game Over Dialog, я получаю сообщение об ошибке «Невозможно создать обработчик внутри потока, который не вызвал Looper.prepare ()» - PullRequest
0 голосов
/ 27 марта 2012

Я пишу небольшую игру для Android. Игра рисуется на SurfaceView с использованием потока. В методе run () потока я проверяю, закончилась ли игра, и если да, то я пытаюсь отобразить диалоговое окно «игра окончена», которое выдает мне вышеупомянутое сообщение об ошибке. Я знаю, что эта ошибка возникает, когда поток не-пользовательский интерфейс пытается связываться с пользовательским интерфейсом. То, что я хотел бы знать, является лучшим подходом к отображению такого диалога. Я вставил код ниже. Спасибо за вашу помощь:

public class BouncingBallActivity extends Activity{

    private static final int DIALOG_GAMEOVER_ID = 0;
    private BouncingBallView bouncingBallView;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        bouncingBallView = new BouncingBallView(this);
        bouncingBallView.resume();
        setContentView(bouncingBallView);
    }

    protected Dialog onCreateDialog(int id)
    {
        switch (id) {
        case DIALOG_GAMEOVER_ID:
            AlertDialog.Builder builder = new AlertDialog.Builder(this);
            builder.setMessage("Game Over.")
                    .setCancelable(false)
                    .setPositiveButton("Try Again",
                            new DialogInterface.OnClickListener()
                                {

                                public void onClick(DialogInterface arg0,
                                        int arg1)
                                {
                                    bouncingBallView.resume();

                                }
                            })
                    .setNegativeButton("Exit",
                            new DialogInterface.OnClickListener() {

                                public void onClick(DialogInterface dialog,
                                        int which)
                                {
                                    BouncingBallActivity.this.finish();

                                }
                            });

            AlertDialog gameOverDialog = builder.create();
            return gameOverDialog;
        default:
            return null;
        }


    }


    class BouncingBallView extends SurfaceView implements Runnable
    {
        SurfaceHolder   surfaceViewHolder;
        Canvas          canvas;
        Context         context;
        Thread          drawingThread;

        boolean         drawingThreadIsRunning;
        boolean         isInitialised;

        Ball            ball;
        ArtificialIntelligence ai;

        BouncingBallView(Context context)
        {
            //
        }

        public void pause()
        {
            isInitialised = false;
            drawingThreadIsRunning = false;
            boolean joiningWasSuccessful = false;

            while(!joiningWasSuccessful)
            try {
                drawingThread.join();
                joiningWasSuccessful = true;
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

        }

        public void resume()
        {
            isInitialised = false;
            drawingThread = new Thread(this);

            drawingThread.setName("Drawing Thread");
            drawingThreadIsRunning = true;
            drawingThread.start();

        }

        public void run()
        {
            while(drawingThreadIsRunning)
            {
                if(!surfaceViewHolder.getSurface().isValid())
                    continue;

                if(gameOver())
                    BouncingBallActivity.this.showDialog(DIALOG_GAMEOVER_ID);

                try{
                    canvas = surfaceViewHolder.lockCanvas();
                    if(!isInitialised)init(canvas);
                    update();
                    surfaceViewHolder.unlockCanvasAndPost(canvas);
                }catch(Exception e)
                {
                    Log.e(BouncingBallActivity.this.toString(),String.format("%s: Just as the emperor had foreseen!\n(This error is expected. Canvas destroyed while animations continue.)", e.toString()));
                }
            }
        }

        private void init(Canvas canvas)
        {
            ball = new Ball(canvas, Color.GREEN);
            ai   = new ArtificialIntelligence(canvas, (int) (ball.getX()+100),canvas.getWidth());

            isInitialised = true;
        }

    }
}

Ответы [ 4 ]

2 голосов
/ 27 марта 2012

Попробуйте вот так ... вы не можете вносить какие-либо изменения в пользовательский интерфейс, отличный от основного потока ... вставьте эту часть в свой поток после if (gameOver ())

//if(gameOver())
 runOnUiThread(new Runnable() {
           @Override
           public void run() {

               BouncingBallActivity.this.showDialog(DIALOG_GAMEOVER_ID);
           }
       });
1 голос
/ 27 марта 2012

Вы вызываете диалог из рабочего (фонового) потока.Вы должны вызвать это из основного потока.Попробуйте вызвать его с помощью Activity.runOnUIThread () и создайте внутри него обработчик, который будет вызывать ваш метод showDialog.

0 голосов
/ 27 июля 2015

Для меня я использовал этот обработчик в моем SurfaceView для создания диалогового окна.

Handler someHandler = new Handler(){
//this method will handle the calls from other threads. 
public void handleMessage(Message msg) {

                 final Dialog dialog = new Dialog(Game.this);

                   dialog.setContentView(R.layout.question_dialog);
                   dialog.setTitle("GAME OVER");




                   Button restart=(Button)dialog.findViewById(R.id.btn Restart);

                   // Set On ClickListener
                   restart.setOnClickListener(new View.OnClickListener() {

                       public void onClick(View v) {


                               Toast.makeText(Game.this, "Restart Game", Toast.LENGTH_LONG).show();
                               dialog.dismiss();

                           }


                       }
                   });

                   dialog.show();

             }

Итак, я пишу looper.prepared () в ветке игры, чтобы вызвать этот обработчик. Если мощность игрока = 0, появится диалоговое окно.

Looper.prepare();

 //create the message for the handler 
 Message status = someHandler.obtainMessage();
 Bundle data = new Bundle();
 String msgContent = null;
 data.putString("SOMETHING", msgContent);
 status.setData(data);
 someHandler.sendMessage(status);

 Looper.loop();

  }      
0 голосов
/ 30 ноября 2014

Я использовал [https://stackoverflow.com/a/16886486/3077964] для решения проблемы.

Вам нужно приостановить поток BouncingBallView после показа диалога внутри runOnUiThread (). То есть:

//if(gameOver()){       
BouncingBallActivity.this.runOnUiThread(new Runnable() {
 @Override
public void run() {    BouncingBallActivity.this.showDialog(DIALOG_GAMEOVER_ID);    }    });    pause();    }
...