Slide and collide!

This commit is contained in:
LittleSheep 2024-07-18 17:21:25 +08:00
parent b4c2cdccaf
commit 9025859fac
9 changed files with 126 additions and 4 deletions

View File

@ -19,6 +19,10 @@ type Player struct {
} }
func (p *Player) Update() { func (p *Player) Update() {
if !p.Velocity.IsZero() {
p.SlideByVelocity()
}
keys := sdl.GetKeyboardState() keys := sdl.GetKeyboardState()
if keys[sdl.SCANCODE_UP] == 1 || keys[sdl.SCANCODE_W] == 1 { if keys[sdl.SCANCODE_UP] == 1 || keys[sdl.SCANCODE_W] == 1 {
p.Move(0, -1) p.Move(0, -1)
@ -33,14 +37,15 @@ func (p *Player) Update() {
p.Move(1, 0) p.Move(1, 0)
} }
p.Move(0, 0)
p.BaseObject.Update() p.BaseObject.Update()
} }
func (p *Player) Move(deltaX, deltaY float64) { func (p *Player) Move(deltaX, deltaY float64) {
p.Velocity.X += deltaX * Acceleration p.Velocity.X += deltaX * Acceleration
p.Velocity.Y += deltaY * Acceleration p.Velocity.Y += deltaY * Acceleration
}
func (p *Player) SlideByVelocity() {
p.Position.X += p.Velocity.X p.Position.X += p.Velocity.X
p.Position.Y += p.Velocity.Y p.Position.Y += p.Velocity.Y
@ -63,3 +68,29 @@ func (p *Player) Draw(pen *sdl.Renderer) {
} }
} }
} }
func (p *Player) GetPosition() land.Vector2D {
return p.Position
}
func (p *Player) GetSize() land.Vector2D {
return p.Size
}
func (p *Player) OnCollide(other land.CollidableObject) {
if p.Position.X < other.GetPosition().X {
p.Position.X = other.GetPosition().X - p.Size.X
p.Velocity.X = 0
} else if p.Position.X > other.GetPosition().X {
p.Position.X = other.GetPosition().X + other.GetSize().X
p.Velocity.X = 0
}
if p.Position.Y < other.GetPosition().Y {
p.Position.Y = other.GetPosition().Y - p.Size.Y
p.Velocity.Y = 0
} else if p.Position.Y > other.GetPosition().Y {
p.Position.Y = other.GetPosition().Y + other.GetSize().Y
p.Velocity.Y = 0
}
}

View File

@ -8,7 +8,7 @@ type Object interface {
} }
type DrawableObject interface { type DrawableObject interface {
Draw(*sdl.Renderer) Draw(pen *sdl.Renderer)
} }
type BaseObject struct { type BaseObject struct {

View File

@ -0,0 +1,18 @@
package land
type CollidableObject interface {
GetPosition() Vector2D
GetSize() Vector2D
OnCollide(other CollidableObject)
}
func checkCollisionBetweenObject(a, b CollidableObject) bool {
aPos := a.GetPosition()
aSize := a.GetSize()
bPos := b.GetPosition()
bSize := b.GetSize()
return aPos.X < bPos.X+bSize.X &&
aPos.X+aSize.X > bPos.X &&
aPos.Y < bPos.Y+bSize.Y &&
aPos.Y+aSize.Y > bPos.Y
}

View File

@ -0,0 +1,6 @@
package land
type MoveableObject interface {
Move(deltaX int64, deltaY int64)
SlideByVelocity()
}

View File

@ -19,7 +19,7 @@ func (p *PerformanceAnalyzer) Draw() {
atomic.AddInt64(&p.drawCount, 1) atomic.AddInt64(&p.drawCount, 1)
} }
func (p *PerformanceAnalyzer) KeepResetting(duration time.Duration) { func (p *PerformanceAnalyzer) RunResetter(duration time.Duration) {
ticker := time.NewTicker(duration) ticker := time.NewTicker(duration)
defer ticker.Stop() defer ticker.Stop()
for { for {

View File

@ -16,7 +16,7 @@ func NewRootObject() *RootObject {
Analyzer: &PerformanceAnalyzer{}, Analyzer: &PerformanceAnalyzer{},
} }
go in.Analyzer.KeepResetting(1 * time.Second) go in.Analyzer.RunResetter(1 * time.Second)
return in return in
} }
@ -38,6 +38,24 @@ func (p *RootObject) RunEventLoop(tickDuration time.Duration) {
func (p *RootObject) Update() { func (p *RootObject) Update() {
p.BaseObject.Update() p.BaseObject.Update()
// Check collision
for _, child := range p.Children {
for _, other := range p.Children {
collidableChild, ok := child.(CollidableObject)
if !ok {
continue
}
collidableOther, ok := other.(CollidableObject)
if !ok {
continue
}
if child != other && checkCollisionBetweenObject(collidableChild, collidableOther) {
collidableChild.OnCollide(collidableOther)
collidableOther.OnCollide(collidableChild)
}
}
}
p.Analyzer.Tick() p.Analyzer.Tick()
} }

View File

@ -3,3 +3,7 @@ package land
type Vector2D struct { type Vector2D struct {
X, Y float64 X, Y float64
} }
func (p Vector2D) IsZero() bool {
return p.X == 0 && p.Y == 0
}

View File

@ -0,0 +1,40 @@
package tiles
import (
"git.solsynth.dev/highland/codingland/pkg/internal/land"
"github.com/veandco/go-sdl2/sdl"
)
type Tile struct {
land.BaseObject
Position land.Vector2D
Size land.Vector2D
}
func (p *Tile) Draw(pen *sdl.Renderer) {
pen.SetDrawColor(255, 255, 255, 255)
pen.FillRect(&sdl.Rect{
X: int32(p.Position.X),
Y: int32(p.Position.Y),
W: int32(p.Size.X),
H: int32(p.Size.Y),
})
for _, child := range p.Children {
if drawableChild, ok := child.(land.DrawableObject); ok {
drawableChild.Draw(pen)
}
}
}
func (p *Tile) GetPosition() land.Vector2D {
return p.Position
}
func (p *Tile) GetSize() land.Vector2D {
return p.Size
}
func (p *Tile) OnCollide(other land.CollidableObject) {
}

View File

@ -3,6 +3,7 @@ package main
import ( import (
"git.solsynth.dev/highland/codingland/pkg/internal/entities" "git.solsynth.dev/highland/codingland/pkg/internal/entities"
"git.solsynth.dev/highland/codingland/pkg/internal/land" "git.solsynth.dev/highland/codingland/pkg/internal/land"
"git.solsynth.dev/highland/codingland/pkg/internal/tiles"
"log" "log"
"runtime" "runtime"
"time" "time"
@ -42,6 +43,10 @@ func main() {
Position: land.Vector2D{X: windowWidth/2 - 25, Y: windowHeight/2 - 25}, Position: land.Vector2D{X: windowWidth/2 - 25, Y: windowHeight/2 - 25},
Size: land.Vector2D{X: 50, Y: 50}, Size: land.Vector2D{X: 50, Y: 50},
}) })
root.AddChild(&tiles.Tile{
Position: land.Vector2D{X: 25, Y: 25},
Size: land.Vector2D{X: 50, Y: 50},
})
// 10ms delay use be 100tps average // 10ms delay use be 100tps average
go root.RunEventLoop(10 * time.Millisecond) go root.RunEventLoop(10 * time.Millisecond)