原文: Libgdx游戏开发(5)——碰撞反弹的简单实践-Stars-One的杂货小窝

本篇简单以一个小球运动,一步步实现碰撞反弹的效果

本文代码示例以kotlin为主,且需要有一定的Libgdx入门基础

注:下面动态图片看着有些卡顿,是录制的问题,实际上运行时很流畅的

水平滚动

简单起见,我们通过ShapeRenderer绘制一个圆形,作为我们的小球,并让其从开始位置向右水平移动

import com.badlogic.gdx.ApplicationAdapter
import com.badlogic.gdx.Gdx
import com.badlogic.gdx.Input
import com.badlogic.gdx.graphics.GL20
import com.badlogic.gdx.graphics.glutils.ShapeRenderer class CircleBallTest : ApplicationAdapter() {
lateinit var shape: ShapeRenderer override fun create() {
shape = ShapeRenderer()
} var x = 50f
var y = 50f override fun render() {
//每次渲染绘制前,清除屏幕
Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT) x += 5 //设置填充模式,圆形默认即为白色
shape.begin(ShapeRenderer.ShapeType.Filled);
//圆形半径为50,起点位置位于(50,50)
shape.circle(x, y, 50f)
shape.end()
} //这里忽略了相关资源释放代码逻辑...
}

启动游戏代码(方便阅读,下文中此代码不会再贴出!):

package com.arthurlumertz.taplixic;

import com.badlogic.gdx.backends.lwjgl3.*;

public class DesktopLauncher {

	public static void main (String[] arg) {
Lwjgl3ApplicationConfiguration config = new Lwjgl3ApplicationConfiguration();
config.setWindowedMode(960, 540);
config.setForegroundFPS(60);
new Lwjgl3Application(new CircleBallTest(), config);
}
}

效果如下:

水平滚动并反弹

上述已经实现了一个小球滚动,但发现滚动到边缘就不见了,我们加个效果,碰到右边缘就反弹

package com.arthurlumertz.taplixic

import com.badlogic.gdx.ApplicationAdapter
import com.badlogic.gdx.Gdx
import com.badlogic.gdx.Input
import com.badlogic.gdx.graphics.GL20
import com.badlogic.gdx.graphics.glutils.ShapeRenderer class CircleBallTest : ApplicationAdapter() {
lateinit var shape: ShapeRenderer override fun create() {
shape = ShapeRenderer()
} var x = 50f
var y = 50f var isRight = true override fun render() {
//每次渲染绘制前,清除屏幕
Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT) if (isRight) {
x += 5
} else {
x-=5
} //设置填充模式,圆形默认即为白色
shape.begin(ShapeRenderer.ShapeType.Filled);
//圆形半径为50,起点位置位于(50,50)
shape.circle(x, y, 50f)
shape.end() //右边缘检测 圆心的x坐标加上半径大于或等于当前游戏屏幕宽度
if (x + 50 >= Gdx.graphics.width) {
isRight=false
} //左边缘检测 圆心的x坐标减去半径小于或等于0(起点)
if (x - 50 <=0) {
isRight=true
}
} //这里忽略了相关资源释放代码逻辑...
}

效果如下(录制效果的时候没加左边缘检测):

这里实际可以直接将对应的+5-5统一转为一个速度加量,方向需要反转的时候乘以-1即可

同时,我们上述小球相关代码封装为一个Ball类来进行使用,优化后的代码如下:

//定义一个ball类实现相关操作
class Ball{
var size = 50f var x = 50f
var y = 50f var speedX = 5f fun gundon() {
x += speedX
} fun draw(shape: ShapeRenderer) {
shape.begin(ShapeRenderer.ShapeType.Filled)
shape.circle(x, y, size)
shape.end()
} //检测边缘反弹
fun checkFz() {
//到达右边缘,加量变反
if (x + size >= Gdx.graphics.width) {
speedX = speedX * -1
} //到达左边缘,加量变反
if (x - size <= 0) {
speedX = speedX * -1
}
}
}

游戏代码:

class CircleBallTest : ApplicationAdapter() {
lateinit var shape: ShapeRenderer val ball by lazy { Ball() } override fun create() {
shape = ShapeRenderer()
} override fun render() {
Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT) ball.draw(shape) ball.gundon() ball.checkFz()
} //这里忽略了相关资源释放代码逻辑...
}

这里代码我是将绘制,坐标和边缘碰撞检测分别封装对应的方法

  • draw() 绘制
  • gundon()修改坐标的方法
  • checkFz()则是进行碰撞检测的方法

这里为什么要将绘制和修改坐标抽成2个方法,是因为我研究游戏暂停的时候发现的,这里先卖个关子,之后会讲到(算是自己无意摸索出来的小技巧)

四面滚动反弹

上述我们只是在水平方向移动,现在想要小球斜方向发出,之后四周反弹,应该如何实现呢?

想要斜方向发出,我们还需要在上面实现的基础上加个y坐标加量,同时修改x,y坐标,就能让小球斜着运动了(数学中的线性方程,或者可以看做是给了小球上方向和右方向的力)

当然,如果你修改对应的增加量数值,可以实现不同斜率方向

这里我固定x和y的加量相同,即45度方向运动

四周反弹其实可以拆分为左右和上下方向,碰到左和右就反转x的增量,碰到上和下就反转y的增量


import com.badlogic.gdx.ApplicationAdapter
import com.badlogic.gdx.Gdx
import com.badlogic.gdx.graphics.GL20
import com.badlogic.gdx.graphics.glutils.ShapeRenderer class CircleBallTest : ApplicationAdapter() {
lateinit var shape: ShapeRenderer val ball by lazy { Ball() } override fun create() {
shape = ShapeRenderer()
} override fun render() {
Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT) ball.draw(shape) ball.checkFz()
}
} class Ball{
var size = 50f var x = 50f
var y = 50f var speedX = 5f
//y坐标增量
var speedY = 5f fun gundon() {
x += speedX
//进行添加
y += speedY
} fun draw(shape: ShapeRenderer) { shape.begin(ShapeRenderer.ShapeType.Filled)
shape.circle(x, y, size)
shape.end()
} //检测边缘反弹
fun checkFz() {
//这里为了方便理解,我每个条件都拆出来了 //到达右边缘,x变反
if (x + size >= Gdx.graphics.width) {
speedX = speedX * -1
} //到达下边缘,y变反
if (y - size <= 0) {
speedY = speedY * -1
} //到达上边缘,y变反
if (y + size >= Gdx.graphics.height) {
speedY = speedY * -1
} //到达左边缘,x变反
if (x - size <= 0) {
speedX = speedX * -1
}
}
}

效果如下:

加个板子进行弹球

在上面的基础上,我们添加一个板子用来接球

  1. 使用ShapeRenderer对象绘制实心矩形作为板子
  2. 考虑板子和球的碰撞
  3. 方向键左右可控制板子移动
  4. 碰到下边缘,球消失

shape.rect()方法用来绘制一个矩形,在x,y坐标绘制一个定义的宽高矩形,(x,y)坐标即为此矩形的左上角

圆心的y坐标 - 半径 >= 矩形的y坐标,圆心x坐标-半径小于矩形的x坐标,圆心x坐标+半径大于或等于矩形的x坐标+矩形宽度,即视为两者碰撞

下面代码和Ball一样,封装了一个MyBan类,实现板子的绘制和控制移动

import com.badlogic.gdx.ApplicationAdapter
import com.badlogic.gdx.Gdx
import com.badlogic.gdx.Input
import com.badlogic.gdx.graphics.GL20
import com.badlogic.gdx.graphics.glutils.ShapeRenderer class CircleBallTest : ApplicationAdapter() {
lateinit var shape: ShapeRenderer val ball by lazy { Ball() }
val line by lazy { MyBan() } override fun create() {
shape = ShapeRenderer()
} override fun render() {
Gdx.gl.glClear(GL20.GL_COLOR_BUFFER_BIT) line.draw(shape)
line.control() ball.draw(shape)
ball.checkFz() //检测碰撞到数横条
ball.checkLineP(line)
}
} class MyBan {
var width = 200f
var height = 10f var x = 0f
var y = height fun draw(shape: ShapeRenderer) {
shape.begin(ShapeRenderer.ShapeType.Filled)
//这里注意: x,y是指矩形的左上角
shape.rect(x, height, width, height)
shape.end()
} fun control() {
if (Gdx.input.isKeyPressed(Input.Keys.LEFT)) {
x -= 200 * Gdx.graphics.deltaTime
} if (Gdx.input.isKeyPressed(Input.Keys.RIGHT)) {
x += 200 * Gdx.graphics.deltaTime
} //这里屏蔽y坐标改变,只给控制左右移动
return if (Gdx.input.isKeyPressed(Input.Keys.UP)) {
y += 200 * Gdx.graphics.deltaTime
} if (Gdx.input.isKeyPressed(Input.Keys.DOWN)) {
y -= 200 * Gdx.graphics.deltaTime
}
}
} class Ball {
var size = 5f var x = 50f
var y = 50f var speedX = 5f
var speedY = 5f //与板子的碰撞检测
fun checkLineP(myB: MyBan) {
if (y - size <= myB.y) {
speedY = speedY * -1
}
} fun gundon() {
x += speedX
y += speedY
} fun draw(shape: ShapeRenderer) {
shape.begin(ShapeRenderer.ShapeType.Filled)
shape.circle(x, y, size)
shape.end()
} fun checkFz() {
//到达右边缘,x变反
if (x + size >= Gdx.graphics.width) {
speedX = speedX * -1
} //到达下边缘,y变反
//todo 这个是判输条件!
if (y - size <= 0) {
//消失
//speedY = speedY * -1
} //到达上边缘,y变反
if (y + size >= Gdx.graphics.height) {
speedY = speedY * -1
} //到达左边缘,x变反
if (x - size <= 0) {
speedX = speedX * -1
}
}
}

效果如下:

参考

Libgdx游戏开发(5)——碰撞反弹的简单实践的更多相关文章

  1. [libGDX游戏开发教程]使用libGDX进行游戏开发(1)-游戏设计

    声明:<使用Libgdx进行游戏开发>是一个系列,文章的原文是<Learning Libgdx Game Development>,大家请周知.后续的文章连接在这里 使用Lib ...

  2. [libGDX游戏开发教程]使用libGDX进行游戏开发(12)-Action动画

    前文章节列表:  使用libGDX进行游戏开发(11)-高级编程技巧   使用libGDX进行游戏开发(10)-音乐音效不求人,程序员也可以DIY   使用libGDX进行游戏开发(9)-场景过渡   ...

  3. Libgdx游戏开发(2)——接水滴游戏实现

    原文:Libgdx游戏开发(2)--接水滴游戏实现 - Stars-One的杂货小窝 本文使用Kotlin语言开发 通过本文的学习可以初步了解以下基础知识的使用: Basic file access ...

  4. [libgdx游戏开发教程]使用Libgdx进行游戏开发(11)-高级编程技巧 Box2d和Shader

    高级编程技巧只是相对的,其实主要是讲物理模拟和着色器程序的使用. 本章主要讲解利用Box2D并用它来实现萝卜雨,然后是使用单色着色器shader让画面呈现单色状态:http://files.cnblo ...

  5. 精通libGDX游戏开发-RPG实战-开发游戏的基本前提

    说起RPG,大概国人是不会陌生的. 这不得不从中国单机游戏市场说起,由于早期软件市场被盗版杀死,顺带的,单机游戏软件作为软件市场的分支,也没赚什么钱,养不活公司纷纷倒闭,只到RPG游戏<仙剑奇侠 ...

  6. [libgdx游戏开发教程]使用Libgdx进行游戏开发(10)-音乐和音效

    本章音效文件都来自于公共许可: http://files.cnblogs.com/mignet/sounds.zip 在游戏中,播放背景音乐和音效是基本的功能. Libgdx提供了跨平台的声音播放功能 ...

  7. [libgdx游戏开发教程]使用Libgdx进行游戏开发(2)-游戏框架搭建

    让我们抛开理论开始code吧. 入口类CanyonBunnyMain的代码: package com.packtpub.libgdx.canyonbunny; import com.badlogic. ...

  8. PuTsangTo-单撸游戏开发03 碰撞与跳跃瑕疵版

    继续上一部分,游戏的定位是横版平台动作类游戏,所以得有跳跃动作,首先想到的就是物理引擎,不过在2D游戏里,仅为了角色的跳跃而引入物理引擎,目前想来有些不至于,仅使用cocos默认带有的碰撞系统也足够了 ...

  9. 精通libGDX游戏开发-RPG实战-欢迎来到RPG的世界

    欢迎来到RPG的世界 本章我会快速的使用tiled这样的瓷砖地图工具,来带领大家创造所设想的世界. 创建并编辑瓷砖地图 瓷砖地图(tile-based map)是广泛应用于各种游戏类型的地图格式,li ...

  10. [libgdx游戏开发教程]使用Libgdx进行游戏开发(7)-屏幕布局的最佳实践

    管理多个屏幕 我们的菜单屏有2个按钮,一个play一个option.option里就是一些开关的设置,比如音乐音效等.这些设置将会保存到Preferences中. 多屏幕切换是游戏的基本机制,Libg ...

随机推荐

  1. k8s对接Ceph实现持久化存储(16)

    一.Ceph简介 官网:https://ceph.com/en/ https://docs.ceph.com/en/latest/start/intro/ ceph 是一种开源的分布式的存储系统 包含 ...

  2. 五:瑞芯微RV1109

    瑞芯微RV1109是一款用于工控机或人工智能视觉应用的高性能机器视觉处理器SoC. 参考资料 http://www.neardi.com/news_22/434.html https://www.ro ...

  3. 11.17~11.18暨noip2023游寄

    11.17 我们DZ不负众望又干了点nt事,但是为了按时间顺序记叙,所以说放到最后再讲 上午 平常的起床+吃饭,然后就发手机啥的,坐大巴去德州东再坐会高铁去秦皇岛,这些简单记一下就行了 重点来了 先拜 ...

  4. 19、python 脚本

    1.python 安装及配置 下载地址 python2 和 python3 共存安装 2.python 可视化 import turtle turtle.pensize(2) #画一个小圆 turtl ...

  5. 一种光电容积波PPG 转换到心电图ECG进行房颤检测的神经网络模型

    具体的软硬件实现点击 http://mcu-ai.com/ MCU-AI技术网页_MCU-AI人工智能 光电体积描记法(PPG)是一种经济有效的非侵入性技术,利用光学方法测量心脏生理学. PPG 在健 ...

  6. JavaScript算法---基础排序类

    <html> <script> //正序排序,把大的放到最后,arr[j]>arr[j+1] let fz=(arr)=>{ for(let len=arr.len ...

  7. C语言(较深入原理):%s通过字符串首元素地址输出,用指针数组来作示例

    首先,我们输出一个字符串都知道是用%s来输出,但是我们并没有多想是通过什么方式来输出的. 今天我在看指针数组的时候发现了一个问题,按就是定义一个字符类型的指针数组, /*字符串的输出本身就需要他的地址 ...

  8. 鸿蒙HarmonyOS实战-Stage模型(服务卡片介绍和运行机制)

    一.服务卡片介绍 1.服务卡片的概念 在HarmonyOS中,服务卡片是一种提供即时信息和快速操作的小组件,类似于Android中的通知栏.服务卡片可以显示各种类型的信息,包括通知.天气.日历事件.音 ...

  9. java学习之旅(day.01)

    Markdown学习 标题 一级标题:#空格+标题名字 二级标题:##空格+标题名字 三级标题:###空格+标题名字 字体 粗体:两边都加两个** Hello,world 斜体:两边都加一个* Hel ...

  10. Abp vNext框架 基础知识 依赖注入

    依赖注入 ABP的依赖注入系统是基于Microsoft的依赖注入扩展库(Microsoft.Extensions.DependencyInjection nuget包)开发的.因此,它的文档在ABP中 ...