我正在开发一个使用加速度传感器在屏幕上移动球的应用程序。我已经使用Canvas绘制了球,每次绘制“触摸”屏幕末端时,都会显示一条消息,告诉用户相应的一侧。
如下图所示:
吐司面包显示一条消息:“向左移动”,从西班牙语翻译为英语。
您会看到该消息显示正确。
当我单击手机的返回按钮时,会出现问题:
您的症状:
根据手机的方向,再次显示图像信息。
消息显然是循环的,也就是说,即使您不移动消息,也是随机的。
Toast消息的行为就像它们仍在上一个活动中一样,也就是说,我已经离开了。有时,它仅在我移动电话和其他时间时才显示警告,而不会多次随机移动。
主屏幕的手动按钮是将视图引导到带有球的第一张照片的“活动”按钮。
尽管在此视图中按返回按钮,但即使我返回到应用程序屏幕,消息仍会显示。
消除此问题的唯一方法是从应用程序库中删除该应用程序,并且过程再次相同。
我强调指出,只有在我进入该视图并返回时才出现问题,而在那之前不会发生。
我附上主要代码:
btnManual.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MainActivity.this, MovimientoActivity.class);
startActivity(intent);
}
进入MovimientoActivity.class
public class MovimientoActivity extends AppCompatActivity{
private DrawView view; //Class where the ball is created with the accelerometer
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Toast.makeText(this, "Cliclo-onCreate", Toast.LENGTH_SHORT).show();
DisplayMetrics displaymetrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(displaymetrics);
view = new DrawView(this, displaymetrics); //Se llama a esa vista
view.setBackgroundColor(Color.parseColor("#F5B041"));
setContentView(view);
//Definimos usar toda la pantalla
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
}
}
});
Class DrawView.java
public class DrawView extends View implements SensorEventListener {
Cliente cliente = null;
Sensor accelerometer = null;
SensorManager manager;
static String TAG = "DrawView";
int BLOQUEO_ARRIBA = 0;
int BLOQUEO_ABAJO = 0;
int BLOQUEO_DERECHO = 0;
int BLOQUEO_IZQUIERDO = 0;
private Bitmap pelota;
public int ALTURA;
public int ANCHO;
DisplayMetrics displaymetrics;
Punto posicion = new Punto();
final int X = 0;
final int Y = 1;
final int Z = 2;
public DrawView(Context context, DisplayMetrics displaymetrics) {
super(context);
this.displaymetrics = displaymetrics;
manager = (SensorManager) context.getSystemService(Context.SENSOR_SERVICE);
if (manager.getSensorList(Sensor.TYPE_ACCELEROMETER).size() != 0) {
accelerometer = manager.getSensorList(
Sensor.TYPE_ACCELEROMETER).get(0);
if (!manager.registerListener(this, accelerometer,
SensorManager.SENSOR_DELAY_FASTEST)) {
}
}
try {
AssetManager assetManager = context.getAssets();
InputStream inputStream;
inputStream = assetManager.open("bola_2.png");
pelota = BitmapFactory.decodeStream(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
// Obtenemos las medidas de la pantalla
ALTURA = displaymetrics.heightPixels;
ANCHO = displaymetrics.widthPixels;
}
@Override
public void onSensorChanged(SensorEvent event) {
// Modificamos la posicion de la bola en el eje X
posicion.x -= event.values[X];
//Comprobamos si se sale de la pantalla, y en ese caso, modificamos su valor
if (posicion.x < 0) {
if(BLOQUEO_IZQUIERDO == 0){
Log.i(TAG, "onSensorChanged: izquierdo");
Toast.makeText(getContext(), "Movimiento a la izquierda", Toast.LENGTH_SHORT).show();
BLOQUEO_DERECHO = 0;
BLOQUEO_ARRIBA = 0;
BLOQUEO_ABAJO = 0;
BLOQUEO_IZQUIERDO++;
}
posicion.x = 0;
}else if (posicion.x > this.ANCHO -pelota.getWidth()) {
if(BLOQUEO_DERECHO == 0){
Log.i(TAG, "onSensorChanged: derecho");
Toast.makeText(getContext(), "Movimiento a la derecha", Toast.LENGTH_SHORT).show();
BLOQUEO_IZQUIERDO = 0;
BLOQUEO_ARRIBA = 0;
BLOQUEO_ABAJO = 0;
BLOQUEO_DERECHO++;
}
posicion.x = this.ANCHO - pelota.getWidth();
}
//Modificamos la posicion de la bola en el eje Y
posicion.y += event.values[Y];
//Comprobamos si se sale de la pantalla, y en ese caso, modificamos su valor
if (posicion.y < 0) {
if(BLOQUEO_ARRIBA == 0){
Log.i(TAG, "onSensorChanged: arriba");
Toast.makeText(getContext(), "Movimiento hacia arriba", Toast.LENGTH_SHORT).show();
BLOQUEO_DERECHO = 0;
BLOQUEO_IZQUIERDO = 0;
BLOQUEO_ABAJO = 0;
BLOQUEO_ARRIBA++;
}
posicion.y = 0;
}else if (posicion.y > this.ALTURA - pelota.getHeight()) {
if(BLOQUEO_ABAJO == 0){
Log.i(TAG, "onSensorChanged: abajo");
Toast.makeText(getContext(), "Movimiento hacia abajo", Toast.LENGTH_SHORT).show();
BLOQUEO_DERECHO = 0;
BLOQUEO_ARRIBA = 0;
BLOQUEO_IZQUIERDO = 0;
BLOQUEO_ABAJO++;
}
posicion.y = this.ALTURA - pelota.getHeight();
}
//Método invalidate para llamar onDraw
invalidate();
}
@Override
public void onDraw(Canvas canvas) {
canvas.drawBitmap(pelota, posicion.x, posicion.y, null);
}
@Override
public void onAccuracyChanged(Sensor sensor, int accuracy) {
}
}
我声明为BLOQUEO的变量是因为我只需要传感器一次将事件通知我,因此,如果实现,我将执行另一项操作,直到“碰到”另一端为止。
到目前为止,这就是问题所在。最后,我的目标是防止传感器一旦移开球罩或在后台停止运行(如果有问题)而继续工作。
我感谢您能给我的帮助。
最佳答案
将经理设为公共财产,以便您可以从自己的活动中访问它。
public class DrawView extends View implements SensorEventListener {
Cliente cliente = null;
Sensor accelerometer = null;
public SensorManager manager;
...
}
现在在Movimientos活动onPause()中取消注册监听器:
protected void onPause() {
super.onPause();
if(view!=null)if(view.manager!=null)view.manager.unregisterListener(view);
}
关于java - 结束 Activity 后,为什么我的加速度传感器仍能工作?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/52753424/