10

我正在尝试使用 3 个类在我的屏幕上画一个球。我已经阅读了一些关于此的内容,并且发现了一个代码片段,该代码片段可以在一个页面上使用 3 个类,在 Android 中使用图形

我更改了代码,这样我就有了一个在撞墙时会移动并改变方向的球,如下图所示(这是使用链接中的代码)

移动球截图

现在我喜欢将这些类分成 3 个不同的页面,以免让所有内容变得如此拥挤,所有内容都以相同的方式设置。

这是我的 3 节课。

  1. BallActivity.java
  2. Ball.java
  3. BallThread.java

package com.brick.breaker;
import android.app.Activity;
import android.os.Bundle;
import android.view.Window;
import android.view.WindowManager;


public class BallActivity extends Activity {

private Ball ball;

@Override
protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);

    requestWindowFeature(Window.FEATURE_NO_TITLE);
    getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);

    ball = new Ball(this);
    setContentView(ball);
}

@Override
protected void onPause() {

    super.onPause();

    setContentView(null);
    ball = null;

    finish();
}

}

package com.brick.breaker;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.view.SurfaceHolder;
import android.view.SurfaceView;

public class Ball extends SurfaceView implements SurfaceHolder.Callback {

private BallThread ballThread = null;

private Bitmap bitmap;

private float x, y;
private float vx, vy;

public Ball(Context context) {
    super(context);

    bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.ball);

    x = 50.0f;
    y = 50.0f;

    vx = 10.0f;
    vy = 10.0f;

    getHolder().addCallback(this);
    ballThread = new BallThread(getHolder(), this);
}

protected void onDraw(Canvas canvas) {

    update(canvas);

    canvas.drawBitmap(bitmap, x, y, null);
}

public void update(Canvas canvas) {

    checkCollisions(canvas);

    x += vx;
    y += vy;
}

public void checkCollisions(Canvas canvas) {

    if(x - vx < 0) {

        vx = Math.abs(vx);

    } else if(x + vx > canvas.getWidth() - getBitmapWidth()) {

        vx = -Math.abs(vx);
    }

    if(y - vy < 0) {

        vy = Math.abs(vy);

    } else if(y + vy > canvas.getHeight() - getBitmapHeight()) {

        vy = -Math.abs(vy);
    }
}

public int getBitmapWidth() {

    if(bitmap != null) {

        return bitmap.getWidth();

    } else {

        return 0;
    }
}

public int getBitmapHeight() {

    if(bitmap != null) {

        return bitmap.getHeight();

    } else {

        return 0;
    }
}

public void surfaceChanged(SurfaceHolder holder, int format, int width,
        int height) {

}

public void surfaceCreated(SurfaceHolder holder) {

    ballThread.setRunnable(true);
    ballThread.start();

}

public void surfaceDestroyed(SurfaceHolder holder) {

    boolean retry = true;
    ballThread.setRunnable(false);

    while(retry) {

        try {

            ballThread.join();
            retry = false;

        } catch(InterruptedException ie) {

            //Try again and again and again
        }

        break;
    }

    ballThread = null;

}

}

package com.brick.breaker;

import android.graphics.Canvas;
import android.view.SurfaceHolder;

public class BallThread extends Thread {

private SurfaceHolder sh;
private Ball ball;

private Canvas canvas;

private boolean run = false;

public BallThread(SurfaceHolder _holder,Ball _ball) {

    sh = _holder;
    ball = _ball;
}

public void setRunnable(boolean _run) {

    run = _run;
}

public void run() {

    while(run) {

        canvas = null;

        try {

            canvas = sh.lockCanvas(null);

            synchronized(sh) {

                ball.onDraw(canvas);
            }

        } finally {

            if(canvas != null) {

                sh.unlockCanvasAndPost(canvas);
            }

        }

    }
}

public Canvas getCanvas() {

    if(canvas != null) {

        return canvas;

    } else {

        return null;
    }
}
}

这是一张显示这些课程结果的图片。

在此处输入图像描述

我试图弄清楚这一点,但由于我对 Android 开发还很陌生,所以我想我可以寻求帮助。

有谁知道是什么导致球被这样画?该代码与链接中的代码几乎相同,我试图尝试找到解决方案,但没有运气。

4

5 回答 5

15

好吧,正如你在图像上看到的,你只画了球。相反,您需要在每次绘制球之前重新绘制黑色背景(或任何您想要的)。

或者,您可以只在前一个位置绘制一个黑色区域,但是当您使用更多对象时,您可能会遇到问题。

这是一个很好的示例,类似于您所做的

于 2012-06-08T22:46:13.047 回答
1

把这个

public void onDraw(Canvas canvas){
   canvas.drawColor(Color.BLACK);

.....

}
于 2013-01-21T08:58:33.987 回答
1

快速浏览一下,我不得不说你只是在同一个表面上绘图,并且从不要求你的表面视图重绘自己。在 finally 块的末尾,在 IF 语句中使用:postInvalidate(); 那应该会导致表面视图重绘自身。

于 2012-06-08T22:38:17.270 回答
-1

[编辑]答案是错误的,但评论很有帮助,所以我会留下这个答案:

不是你问的问题,而是你的代码有问题。在 Android 中,您只能在 UI 线程中写入屏幕。这是运行所有活动回调等的线程。通过从 BallThread 写入屏幕,您将面临程序中出现许多奇怪故障的风险。

于 2013-07-24T18:23:11.710 回答