前言

本文主要给大家介绍的是关于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
# 自定义view
# 虚拟摇杆代码
# 摇杆
# Android控件ListView使用方法详解
# Android利用RecyclerView编写聊天界面
# Android实现RecyclerView下拉刷新效果
# Android自定义view实现拖拽选择按钮
# Android自定义View中attrs.xml的实例详解
# 圆周运动
# 这篇文章
# 四分
# 如下图
# 两点
# 的是
# 好了
# 相关内容
# 说了
# 不多
# 只需
# 屏幕上
# 右键
# 上下左右
# 给大家
# 自定义
# 重写
# 谢谢大家
# 左键
# 创建一个
相关文章:
网站制作服务平台,有什么网站可以发布本地服务信息?
企业网站制作费用多少,企业网站空间一般需要多大,费用是多少?
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
网站制作公司排行榜,抖音怎样做个人官方网站
微网站制作教程,我微信里的网站怎么才能复制到浏览器里?
深圳网站制作培训,深圳哪些招聘网站比较好?
微信小程序 五星评分(包括半颗星评分)实例代码
,交易猫的商品怎么发布到网站上去?
成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?
小程序网站制作需要准备什么资料,如何制作小程序?
广州营销型建站服务商推荐:技术优势与SEO优化解析
如何用PHP快速搭建CMS系统?
建站之星CMS五站合一模板配置与SEO优化指南
免费网站制作appp,免费制作app哪个平台好?
山东云建站价格为何差异显著?
建站之星后台密码如何安全设置与找回?
c# 在高并发下使用反射发射(Reflection.Emit)的性能
如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法
视频网站制作教程,怎么样制作优酷网的小视频?
建站主机空间推荐 高性价比配置与快速部署方案解析
制作网站的软件免费下载,免费制作app哪个平台好?
如何在服务器上三步完成建站并提升流量?
早安海报制作网站推荐大全,企业早安海报怎么每天更换?
代刷网站制作软件,别人代刷火车票靠谱吗?
武汉网站如何制作,黄黄高铁武穴北站途经哪些村庄?
如何快速搭建高效香港服务器网站?
建站主机功能解析:服务器选择与快速搭建指南
如何在阿里云虚拟服务器快速搭建网站?
建站之星logo尺寸如何设置最合适?
如何在企业微信快速生成手机电脑官网?
中山网站推广排名,中山信息港登录入口?
制作公司内部网站有哪些,内网如何建网站?
Android自定义控件实现温度旋转按钮效果
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
建站ABC备案流程中有哪些关键注意事项?
网站网页制作电话怎么打,怎样安装和使用钉钉软件免费打电话?
武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?
网站制作报价单模板图片,小松挖机官方网站报价?
如何在搬瓦工VPS快速搭建网站?
如何将凡科建站内容保存为本地文件?
建站主机选购指南:核心配置优化与品牌推荐方案
网站制作软件免费下载安装,有哪些免费下载的软件网站?
可靠的网站设计制作软件,做网站设计需要什么样的电脑配置?
开源网站制作软件,开源网站什么意思?
一键网站制作软件,义乌购一件代发流程?
智能起名网站制作软件有哪些,制作logo的软件?
安云自助建站系统如何快速提升SEO排名?
html制作网站的步骤有哪些,iapp如何添加网页?
宝华建站服务条款解析:五站合一功能与SEO优化设置指南
建站之星导航菜单设置与功能模块配置全攻略
*请认真填写需求信息,我们会在24小时内与您取得联系。