首页 > 其他分享 >鸿蒙开发游戏(二)---大鱼吃小鱼(摇杆控制)

鸿蒙开发游戏(二)---大鱼吃小鱼(摇杆控制)

时间:2024-02-10 18:12:30浏览次数:34  
标签:鸿蒙 大鱼吃小鱼 number --- 摇杆 speed event

鸿蒙开发游戏(一)---大鱼吃小鱼(界面部署

鸿蒙开发游戏(二)---大鱼吃小鱼(摇杆控制)

鸿蒙开发游戏(三)---大鱼吃小鱼(放置NPC)

鸿蒙开发游戏(四)---大鱼吃小鱼(互吃升级)

鸿蒙开发游戏(五)---大鱼吃小鱼(添加音效)

鸿蒙开发游戏(六)---大鱼吃小鱼(称霸海洋)

 

前言:上一篇介绍了鸿蒙新建项目以及界面部署,并未实现方向逻辑,上下左右控制,这篇来搞一下。首先如果我们用上下左右来控制很显然是不行的,因为我们还希望斜着也能游动,所以只能使用摇杆来试下,大家都肯定玩过王者荣耀,左下角的摇杆控制英雄走动,而且滑动的时候手指不自觉就滑出摇杆圈了,但是任务已经在动,这就物理摇杆和虚拟摇杆的区别,小时候玩过手柄吧,你手柄就不会出现摇出界的情况,因为是实体的,一旦出界你是有感触的,下意识就会修正,但虚拟的摇杆就很容易出界,所以我们要保证就算出界了只要手指没有抬起依旧可以触发事件。

1、摇杆布局

我们需要把←↑↓→替换成圆圈,方式有多种,看你喜欢。

  • 我们可以通过设置组件的弧度让其成圆形
  • Circle:系统给我们提供的圆形

开干

首先我们先看一下需要哪些变量,一定要提前定义好,统一操作一个变量。

  • 大圆有半径,有中心点(可以理解为初始点)
  • 小圆(做摇杆用)有半径,有拖动位置
// 摇杆的中心位置
@State centerX: number = 120
@State centerY: number = 120

//摇杆的初始位置
@State positionX: number = this.centerX
@State positionY: number = this.centerY

//半径
@State radiusMax: number = 100
// 小圆半径
@State radiusMin: number = 20

 

定义好之后就可以初始化位置了

Row() {
  Circle({ width: this.radiusMax * 2, height: this.radiusMax * 2 })
    .fill('#50f1eded')
    .position({ x: this.centerX - this.radiusMax, y: this.centerY - this.radiusMax })


  Circle({ width: this.radiusMin * 2, height: this.radiusMin * 2 })
    .fill('#50f1eded')
    .position({ x: this.positionX - this.radiusMin, y: this.positionY - this.radiusMin })
}
.width(240)
.height(240)
.position({ x: 0, y: 120 })
.justifyContent(FlexAlign.Center)

上面需要注意的是x: this.centerX - this.radiusMax,我们知道屏幕的起始点是从左上角开始延伸的,组件的起始位置也是从左上角开始的,我们定义的是中心点,所以要减去圆半径,不然的话就偏移了

.justifyContent(FlexAlign.Center)

你看到了Row,这是左右布局,加上这句话可以让内容居中。

 

效果如下

 

添加点击事件(Row布局)

.onTouch(this.handleTouchEvent.bind(this))

handleTouchEvent是我们自定义的方法,传入一个 TouchEvent。

  • TouchEvent.Down:按下事件(1次)
  • TouchEvent.Move:移动事件(0-无数次)
  • TouchEvent.Up(1次)

当我们一个事件产生后,就会触发一次down事件,0次移动事件(用户手指不移动)和无数次移动事件(用户疯狂移动手指),之后抬起手指触发一次up事件。

handleTouchEvent(event: TouchEvent) {
  switch (event.type) {
    case TouchType.Down:

      break
    case TouchType.Move:
      this.setMovePosition(event)
      break
    case TouchType.Up:

      break
  }

 OK,那就开始吧,由于Move代码有点多,这里抽取一个方法叫做

setMovePosition(event: TouchEvent) {
 

  }

 在这我们要开始思考了,竟然是通过摇杆去控制鱼,那我们得知道摇杆滑动的方向

 当你的手机从红圆点滑动到指定位置时,他的坐标我们可以得到,从而获取手指与圆点的距离,vy,vx,通过tan可以获取到夹角。通过夹角计算出手指离中心点的距离从而设置摇杆的位置。

1、计算手指的位置

let x = event.touches[0].x;
let y = event.touches[0].y;

2、计算手指与中心点的差值

let vx = x - this.centerX;
let vy = y - this.centerY;

3、计算夹角

let angle = Math.atan2(vy, vx)

4、利用夹角计算出手指离中心点的距离

let distance = this.getDistance(vx, vy)
    
单独写个方法
  getDistance(x: number, y: number) {
    let distance = Math.sqrt(x * x + y * y)
    return Math.min(distance, this.radiusMax)
  }

这里需要注意一下,计算的距离不能超过大圆,我们要始终保持摇杆(小圆)在大圆区域内

5、计算小圆的坐标

this.sin = Math.sin(angle)
this.cos = Math.cos(angle)
this.positionX = distance * this.cos + this.centerX
this.positionY = distance * this.sin + this.centerY

OK,到这就摇杆就完成了,你快去试试效果吧。

调试中。。。。。。。。。。

问题1

你会发现这小鱼没有动只是摇杆动了,我们还需要设置小鱼的位置xfish,yfish

//6、设置小鱼的移动位置,
this.xFish += this.speed * this.cos
this.yFish += this.speed * this.sin

记得给他一个速度speed。

 

试试效果吧...................

 

问题2

是不是还有问题,就是我们滑动摇杆时小鱼动了,一旦我们停止滑动摇杆时小鱼就停止了,那这是为什么呢,安卓开发应该都会自定义view,当我们需要对view进行重绘时,会用到invalidate,对,这就是通知系统要刷新view了,在这的问题就是这,所以我们需要通知系统也对view进行刷新。

这里我们开启一个定时器

this.intervalId = setInterval(() => {
          //6、设置小鱼的移动位置,
          this.xFish += this.speed * this.cos
          this.yFish += this.speed * this.sin

          //目的是触碰到边缘时不溢出
          this.xFish = this.getBorderX(this.xFish)
          //还原角度
          // this.angle = 0

        }, 40)

这里我们传入40毫秒,肉眼一秒26帧左右会很顺滑,1000/26 约等于38,我们直接写40。

那么这个定时器我们不能一直开着,主要是没必要,当我们在按压down时设置,抬起时释放掉

clearInterval(this.intervalId)

完整代码是

handleTouchEvent(event: TouchEvent) {
  switch (event.type) {
    case TouchType.Down:
      this.intervalId = setInterval(() => {
        //6、设置小鱼的移动位置,
        this.xFish += this.speed * this.cos
        this.yFish += this.speed * this.sin

        //目的是触碰到边缘时不溢出
        this.xFish = this.getBorderX(this.xFish)
        //还原角度
        // this.angle = 0

      }, 40)
      break
    case TouchType.Move:
      this.setMovePosition(event)
      break
    case TouchType.Up:
      clearInterval(this.intervalId)
    //恢复摇杆位置
      animateTo({
        curve: curves.springMotion()
      }, () => {
        this.positionX = this.centerX
        this.positionY = this.centerY
      })

      this.speed = 0
      break
  }


}

 

完.....

 

标签:鸿蒙,大鱼吃小鱼,number,---,摇杆,speed,event
From: https://www.cnblogs.com/cmusketeer/p/18002303

相关文章

  • 补基础题(贪心-1)
    Problem-2037(hdu.edu.cn)#include<iostream>/*活动安排的贪心问题*/#include<algorithm>usingnamespacestd;constintN=110;intn;structnode{intbegin,end;};boolcmp(nodeaa,nodebb){returnaa.end<bb.end;}signedmain(){ios::......
  • 文心一言 VS 讯飞星火 VS chatgpt (198)-- 算法导论14.3 6题
    六、用go语言,说明如何来维护一个支持操作MIN-GAP的一些数的动态集Q,使得该操作能给出Q中两个最接近的数之间的差值。例如,Q=(1,5,9,15,18,22),则MIN-GAP返回18-15=3,因为15和18是Q中两个最接近的数。要使得操作INSERT、DELETE、SEARCH和MIN-GAP尽可能高效,并分析它们的运行时间。文心一言,代......
  • Java之泛型系列--继承父类与实现多个接口(有示例)
    原文网址:​​Java之泛型系列--继承父类与实现多个接口(有示例)_IT利刃出鞘的博客-CSDN博客​​简介本文介绍java如何用泛型表示继承父类并实现多个接口。用泛型表示某个类是某个类的子类或者实现了接口的方法为:<TextendsA&B&C> 用法1:全都是接口。对于本例来说:A、B......
  • 考研-初中数学再次启动
    目录必须要掌握的一些公式正方体的面积、体积公式长方形的面积公式三角尺的面积圆的周长、面积初中数学部分科学计数法近似数(准确数、精确度,精确位[个位、百位、十分位、百分位])是否有个分位?小练习整式用字母表示数用含有字母的式子填空练习题单项式判断是否是单项式的练习分别说......
  • maven--插件的管理(pluginManagement)
     原文网址:​​maven--插件的管理(pluginManagement)_IT利刃出鞘的博客-CSDN博客​​简介说明本文介绍maven如何使用pluginManagement来管理插件(build标签中的plugins标签)。概述Maven使用 dependencyManagement对依赖进行管理,见:​​这里​​,与之类似地,Maven中还提供了一个......
  • POJ--3764 The xor-longest Path(Trie)
    记录13:562024-2-10找到俩个点,获得最大的边权异或值。利用异或的性质,一个值被异或俩次相当于没有异或即axorbxorb=a所以先从顶点出发,获得每个点路径上的异或值,然后对这俩个值进行异或就获得了他们之间路径的异或值。获取从顶点到每个点路径上的异或值后,可以利用trie来......
  • QOJ 8171 - Cola
    我们假设目前B知道排列\(p\)的前\(x\)位是多少,那么下一次,B的最优策略是:对于\(i\lex\)的部分,令\(q_i=p_i\)。对于\(i=x+1\)的部分,令\(q_i\)为任一\(p_i\)可能取到但没有被猜过的值。对于\(i>x+1\),随机乱排剩余的\(q_i\)。考虑设\(c_i\)表示第\(i\)次......
  • Atcoder Grand Contest 056 B - Range Argmax
    因为一组\(x\)可能对应多组\(p\),考虑怎么让决策唯一化。我们从大到小依次钦定每个值的位置,即倒着遍历\(i=n,n-1,\cdots,1\),找到最左端的位置\(v\)满足,对于现在还活着的所有区间\(j\)满足\(l_j\lev\ler_j\),都有\(x_j=v\),令\(p_j=i\),然后删去所有包含\(i\)的区间。......
  • Atcoder Grand Contest 041 F - Histogram Rooks
    考虑容斥。我们钦定一些格子组成的集合不能被覆盖,设为\(A\)。把与\(A\)中的点同行同列的点抠掉,剩余的点则是可放可不放的,总方案数就是\(2^{\text{剩余点的个数}}\),乘以\((-1)^{|A|}\)并求和即可。这个做法直接优化显然不行。我们考虑设\(A\)中的点所在的列组成的不可重集......
  • V的2023 - 虽迟但到
    一年一度的年终总结又来了,只感叹时光飞逝,这几年的年终总结对我来说就好像是一年写一次的日记,记忆会模糊,但文字不会。这是第四年,每次回首前几年的文章,心中都会有一丝触动,原来生活给我留下过深刻痕迹。V的2022-往事随风不平凡的2021年ykCoder的2020年年终总结爬山散心二......