Android自定义View实现游戏摇杆键盘的方法示例

所属分类: 软件编程 / Android 阅读数: 38
收藏 0 赞 0 分享

前言

本文主要给大家介绍的是关于Android自定义View实现游戏摇杆键盘的相关内容,为什么会有这篇文章呢?因为在之前的一个项目,操作方向的方式为上下左右,左上需要同时按住左键和右键的方式进行操作。

如下图:

近来需要升级项目,操作方式改为类似王者荣耀的摇杆操作。

如下图:


好了,下面话不多说了,跟着小编来一起看看是如何实现的吧。

绘制背景

实现遥感按钮,需要绘制背景,绘制中心的遥感按钮。绘制遥感背景,需要创建一个RemoteViewBg类,存储背景图,减少重复创建bitmap。

RemoteViewBg类代码如下:

public class RemoteViewBg {
private Bitmap bitmapBg;
public RemoteViewBg(Bitmap bitmap) {
 bitmapBg = bitmap;
}

//背景的绘图函数
public void draw(Canvas canvas, Paint paint, Rect src0 ,Rect dst0 ) {
 canvas.drawBitmap(bitmapBg, src0, dst0, paint);
}
}

点击触摸事件

重写系统的触摸时间,判断触摸点在背景范围内还是背景范围外

 @Override
public boolean onTouchEvent(MotionEvent event) {
 if (event.getAction() == MotionEvent.ACTION_DOWN || event.getAction() ==  MotionEvent.ACTION_MOVE) {
 //   // 在范围外触摸
  if (Math.sqrt(Math.pow((bigCircleX - (int) event.getX()), 2) + Math.pow((bigCircleY - (int) event.getY()), 2)) >= bigCircleR) {

   double tempRad = getRad(bigCircleX, bigCircleY, event.getX(), event.getY());

   getXY(bigCircleX, bigCircleY, bigCircleR, tempRad);
  } else {//范围内触摸
   smallCircleX = (int) event.getX();
   smallCircleY = (int) event.getY();
  }
 } else if (event.getAction() == MotionEvent.ACTION_UP) {
  smallCircleX = bigCircleX;
  smallCircleY = bigCircleY;

 }
 return true;
}

弧度计算

通过 event.getX() , event.getY()获得当前的触摸点,与圆点进行计算,获取弧度

/***
 * 得到两点之间的弧度
 */
public float getRad(float px1, float py1, float px2, float py2) {
 float x = px2 - px1;

 float y = py1 - py2;
 //斜边的长
 float z = (float) Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
 float cosAngle = x / z;
 float rad = (float) Math.acos(cosAngle);

 if (py2 < py1) {
  rad = -rad;
 }
 return rad;
}

图形绘制

通过 canvas.drawCircle()canvas.drawBitmap()分别进行遥感按钮和遥感背景的绘制,注意对遥感背景的保存,如果在绘制的时候每次BitmapFactory.decodeResource()会增加耗时,因此只需在surfaceCreated()中进行bitmap的生成即可。

public void draw() {
 try {
  canvas = sfh.lockCanvas();
  canvas.drawColor(getResources().getColor(R.color.ghostwhite));


 // 指定图片绘制区域(左上角的四分之一)
  Rect src = new Rect(0, 0, bitmap.getWidth(), bitmap.getHeight());

  // 指定图片在屏幕上显示的区域
  Rect dst = new Rect(bigCircleX - bigCircleR, bigCircleY - bigCircleR, bigCircleX + bigCircleR, bigCircleY + bigCircleR);
  // 绘制图片
  remoteViewBg.draw(canvas, paint, src, dst);
  paint.setColor(0x70ff0000);
  //绘制摇杆
  canvas.drawCircle(smallCircleX, smallCircleY, smallCircleR, paint);
 } catch (Exception e) {
  // TODO: handle exception
 } finally {
  try {
   if (canvas != null)
    sfh.unlockCanvasAndPost(canvas);
  } catch (Exception e2) {
   e2.printStackTrace();
  }
 }
}

使用

在activity中动态添加

 RelativeLayout relativeLayout = (RelativeLayout) findViewById(R.id.dance_relative_layout);
 remoteSurfaceView = new RemoteSurfaceView(this);
 params = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, 
 RelativeLayout.LayoutParams.MATCH_PARENT);
 remoteSurfaceView.setLayoutParams(params);
 relativeLayout.addView(remoteSurfaceView);

全部代码

public class RemoteSurfaceView extends SurfaceView implements Callback, Runnable {
private float scale = this.getResources().getDisplayMetrics().density;
private Thread th;
private SurfaceHolder sfh;
private Canvas canvas;
private Paint paint;
private boolean flag;

private int bigCircleX = 0;
private int bigCircleY =0;
private int bigCircleR = 0;
//摇杆的X,Y坐标以及摇杆的半径
private float smallCircleX = 0;
private float smallCircleY = 0;
private float smallCircleR = 0;


private Bitmap bitmap;
private RemoteViewBg remoteViewBg;

public RemoteSurfaceView(Context context) {
 super(context);
 sfh = this.getHolder();
 sfh.addCallback(this);
 paint = new Paint();
 paint.setAntiAlias(true);
 setFocusable(true);
 setFocusableInTouchMode(true);
 setZOrderOnTop(true);
 getHolder().setFormat(PixelFormat.TRANSPARENT);

}

public void surfaceCreated(SurfaceHolder holder) {
 int width = getWidth();
 int height = getHeight();
 bigCircleX = width / 2;
 bigCircleY = height / 2;
 bigCircleR = width / 4;
 smallCircleX = width / 2;
 smallCircleY = height / 2;
 smallCircleR = width / 8;
 bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.fangxiang);
 remoteViewBg = new RemoteViewBg(bitmap);
 th = new Thread(this);
 flag = true;
 th.start();


}

/***
 * 得到两点之间的弧度
 */
public float getRad(float px1, float py1, float px2, float py2) {
 float x = px2 - px1;

 float y = py1 - py2;
 //斜边的长
 float z = (float) Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
 float cosAngle = x / z;
 float rad = (float) Math.acos(cosAngle);

 if (py2 < py1) {
  rad = -rad;
 }
 return rad;
}

@Override
public boolean onTouchEvent(MotionEvent event) {
 if (event.getAction() == MotionEvent.ACTION_DOWN || event.getAction() == MotionEvent.ACTION_MOVE) {
  // 在范围外触摸
  if (Math.sqrt(Math.pow((bigCircleX - (int) event.getX()), 2) + Math.pow((bigCircleY - (int) event.getY()), 2)) >= bigCircleR) {

   double tempRad = getRad(bigCircleX, bigCircleY, event.getX(), event.getY());

   getXY(bigCircleX, bigCircleY, bigCircleR, tempRad);
  } else {//范围内触摸
   smallCircleX = (int) event.getX();
   smallCircleY = (int) event.getY();
  }
 } else if (event.getAction() == MotionEvent.ACTION_UP) {
  smallCircleX = bigCircleX;
  smallCircleY = bigCircleY;

 }
 return true;
}




public void getXY(float x, float y, float R, double rad) {
 //获取圆周运动的X坐标
 smallCircleX = (float) (R * Math.cos(rad)) + x;
 //获取圆周运动的Y坐标
 smallCircleY = (float) (R * Math.sin(rad)) + y;
}

public void draw() {
 try {
  canvas = sfh.lockCanvas();
  canvas.drawColor(getResources().getColor(R.color.ghostwhite));


  // 指定图片绘制区域(左上角的四分之一)
  Rect src = new Rect(0, 0, bitmap.getWidth(), bitmap.getHeight());

  // 指定图片在屏幕上显示的区域
  Rect dst = new Rect(bigCircleX - bigCircleR, bigCircleY - bigCircleR, bigCircleX + bigCircleR, bigCircleY + bigCircleR);
  // 绘制图片
  remoteViewBg.draw(canvas, paint, src, dst);
  paint.setColor(0x70ff0000);
  //绘制摇杆
  canvas.drawCircle(smallCircleX, smallCircleY, smallCircleR, paint);
 } catch (Exception e) {
  // TODO: handle exception
 } finally {
  try {
   if (canvas != null)
    sfh.unlockCanvasAndPost(canvas);
  } catch (Exception e2) {
   e2.printStackTrace();
  }
 }
}

public void run() {

 while (flag) {
  draw();
  try {
   Thread.sleep(50);
  } catch (Exception ex) {
   ex.printStackTrace();
  }
 }
}

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


}

public void surfaceDestroyed(SurfaceHolder holder) {
 flag = false;

}
 }

以上就是这篇文章的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

更多精彩内容其他人还在看

老生常谈Android HapticFeedback(震动反馈)

下面小编就为大家带来一篇老生常谈Android HapticFeedback(震动反馈)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

详谈OnTouchListener与OnGestureListener的区别

下面小编就为大家带来一篇详谈OnTouchListener与OnGestureListener的区别。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

Android仿知乎悬浮功能按钮FloatingActionButton效果

前段时间在看属性动画,恰巧这个按钮的效果可以用属性动画实现,下面通过本文给大家分享adroid仿知乎悬浮功能按钮FloatingActionButton效果,需要的朋友参考下吧
收藏 0 赞 0 分享

解决Android V7后自定义Toolbar、ActionBar左侧有空白问题

这篇文章主要介绍的Android V7后自定义Toolbar、ActionBar左侧有空白问题的解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android常见控件使用详解

这篇文章主要为大家详细介绍了Android常见控件的使用方法,包括ProgressBar进度条控件、AlertDialog对话框控件等,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android实现简洁的APP更新dialog数字进度条

这篇文章主要为大家详细介绍了Android实现简洁的APP更新dialog数字进度条,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android 判断当前语言环境是否是中文环境

本文主要介绍了Android 判断当前语言环境是否是中文环境的方法。具有很好的参考价值。下面跟着小编一起来看下吧
收藏 0 赞 0 分享

详谈Android中Matrix的set、pre、post的区别

下面小编就为大家带来一篇详谈Android中Matrix的set、pre、post的区别。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
收藏 0 赞 0 分享

Android实现登录界面记住密码的存储

这篇文章主要为大家详细介绍了Android SharedPreferrences实现登录界面记住密码的存储,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Android 使用SharedPreferrences储存密码登录界面记住密码功能

Android存储方式有很多种,在这里所用的存储方式是SharedPreferrences, 其采用了Map数据结构来存储数据,以键值的方式存储,可以简单的读取与写入,下面通过实例代码给大家讲解下,需要的朋友参考下吧
收藏 0 赞 0 分享
查看更多