Skip to content

Commit 9a7223c

Browse files
Merge pull request #1 from farhansolodev/collision-detector
2 parents 23a191a + e201419 commit 9a7223c

File tree

6 files changed

+227
-86
lines changed

6 files changed

+227
-86
lines changed

.vscode/launch.json

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
// Use IntelliSense to learn about possible attributes.
3+
// Hover to view descriptions of existing attributes.
4+
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
5+
"version": "0.2.0",
6+
"configurations": [
7+
{
8+
"name": "Launch Package",
9+
"type": "go",
10+
"request": "launch",
11+
"mode": "auto",
12+
"program": "${workspaceRoot}"
13+
}
14+
]
15+
}

cursor.go

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
package main
2+
3+
type Cursor struct {
4+
x, y float32
5+
}
6+
7+
func (cur *Cursor) getPosition() (x, y float32) {
8+
return cur.x, cur.y
9+
}

game.go

+41-17
Original file line numberDiff line numberDiff line change
@@ -7,33 +7,57 @@ import (
77
"github.com/hajimehoshi/ebiten/v2/vector"
88
)
99

10-
type Game struct {
11-
root *QNode
10+
type Game[T Locatable] struct {
11+
root *QNode[T]
12+
// cursorTree *QNode[C]
13+
// datapoints []T
1214
maxDepth uint
1315
}
1416

15-
func (g *Game) Update() error {
16-
width, height := ebiten.WindowSize()
17-
g.root.x1 = float32(width)
18-
g.root.y1 = float32(height)
17+
func (g *Game[T]) Update() error {
18+
g.root.forEach(func (node *QNode[T]) bool {
19+
if node.marked {
20+
node.marked = false
21+
return node.marked
22+
}
23+
return true
24+
}, g.maxDepth)
1925

26+
// width, height := ebiten.WindowSize()
2027
x, y := ebiten.CursorPosition()
21-
if x <= 0 || float32(x) > g.root.x1 || y <= 0 || float32(y) > g.root.y1 { return nil }
22-
23-
g.root.collapse(float32(x), float32(y), g.maxDepth)
24-
28+
if x <= 0 || x > initialScreenWidth || y <= 0 || y > initialScreenHeight {
29+
return nil
30+
}
31+
g.root.markPathTo(float32(x), float32(y))
2532
return nil
2633
}
2734

28-
func (g *Game) Draw(screen *ebiten.Image) {
29-
const lineThickness = 1
30-
g.root.forEach(func (node *QNode) {
35+
func (g *Game[T]) Layout(outsideWidth, outsideHeight int) (int, int) {
36+
return outsideWidth, outsideHeight
37+
}
38+
39+
40+
func (g *Game[T]) Draw(screen *ebiten.Image) {
41+
for _, spr := range g.root.datapoints {
42+
spr := any(spr).(*Sprite)
43+
vector.DrawFilledCircle(screen, spr.x, spr.y, 1, spr.clr, true)
44+
vector.StrokeCircle(screen, spr.x, spr.y, float32(spr.radius), 2, spr.clr, true)
45+
}
46+
47+
width, height := ebiten.WindowSize()
48+
x, y := ebiten.CursorPosition()
49+
if x <= 0 || x > width || y <= 0 || y > height {
50+
return
51+
}
52+
53+
g.root.forEach(func (node *QNode[T]) bool {
54+
if !node.marked {
55+
return true
56+
}
57+
const lineThickness = 1
3158
midX, midY := node.getMidValues()
3259
vector.StrokeLine(screen, node.x0, midY, node.x1, midY, lineThickness, color.White, false)
3360
vector.StrokeLine(screen, midX, node.y0, midX, node.y1, lineThickness, color.White, false)
61+
return false
3462
}, g.maxDepth)
35-
}
36-
37-
func (g *Game) Layout(outsideWidth, outsideHeight int) (screenWidth, screenHeight int) {
38-
return outsideWidth, outsideHeight
3963
}

main.go

+18-7
Original file line numberDiff line numberDiff line change
@@ -4,27 +4,38 @@ import (
44
"flag"
55
"log"
66
"os"
7+
"time"
78

89
"github.com/hajimehoshi/ebiten/v2"
910
)
1011

1112
const (
12-
screenWidth = 1280
13-
screenHeight = 720
13+
initialScreenWidth = 1280
14+
initialScreenHeight = 720
1415
)
1516

1617
func main() {
1718
maxDepth := flag.Uint("depth", 7, "set your own maximum quadtree depth")
18-
resizable := flag.Bool("resize", false, "should window be resizable? [true|false] (still experimental) (default false)")
19+
spriteCount := flag.Uint("sprites", 250, "how many sprites do you want?")
20+
// resizable := flag.Bool("resize", false, "should window be resizable? [true|false] (still experimental) (default false)")
1921
flag.Parse()
2022

21-
ebiten.SetWindowSize(screenWidth, screenHeight)
22-
if *resizable { ebiten.SetWindowResizingMode(ebiten.WindowResizingModeEnabled) }
23+
ebiten.SetWindowSize(initialScreenWidth, initialScreenHeight)
24+
// if *resizable { ebiten.SetWindowResizingMode(ebiten.WindowResizingModeEnabled) }
2325
ebiten.SetWindowTitle(os.Args[0])
2426

25-
game := Game{ root: NewQNode(0, screenWidth, 0, screenHeight, 0), maxDepth: *maxDepth }
27+
sprites := make([]*Sprite, 0, *spriteCount)
28+
var i uint
29+
for ; i < *spriteCount; i++ {
30+
sprites = append(sprites, NewSprite(int64(i) + time.Now().UnixNano()))
31+
}
32+
33+
root := NewQNode(nil, sprites, 0, initialScreenWidth, 0, initialScreenHeight, 0)
34+
root.generateTree(*maxDepth)
35+
36+
game := &Game[*Sprite]{root, *maxDepth}
2637

27-
if err := ebiten.RunGame(&game); err != nil {
38+
if err := ebiten.RunGame(game); err != nil {
2839
log.Fatal(err)
2940
}
3041
}

qnode.go

+90-62
Original file line numberDiff line numberDiff line change
@@ -5,118 +5,146 @@ import (
55
"strings"
66
)
77

8-
type QNode struct {
9-
northwest, northeast, southwest, southeast *QNode
8+
type Locatable interface {
9+
getPosition() (x, y float32)
10+
}
11+
12+
type QNode[T Locatable] struct {
13+
parent, northwest, northeast, southwest, southeast *QNode[T]
14+
datapoints []T
15+
marked bool
1016
x0, x1, y0, y1 float32
1117
depth uint
1218
}
1319

14-
func NewQNode(x0, x1, y0, y1 float32, depth uint) *QNode {
15-
return &QNode{nil, nil, nil, nil, x0, x1, y0, y1, depth}
20+
func NewQNode[T Locatable](parent *QNode[T], datapoints []T, x0, x1, y0, y1 float32, depth uint) *QNode[T] {
21+
return &QNode[T]{parent, nil, nil, nil, nil, datapoints, false, x0, x1, y0, y1, depth}
1622
}
1723

18-
func (node *QNode) makeNorthWest() *QNode {
24+
func (node *QNode[T]) makeNorthWest(datapoints []T) *QNode[T] {
1925
midX, midY := node.getMidValues()
20-
node.northwest = NewQNode(node.x0, midX, node.y0, midY, node.depth+1)
26+
node.northwest = NewQNode(node, datapoints, node.x0, midX, node.y0, midY, node.depth+1)
2127
return node.northwest
2228
}
2329

24-
func (node *QNode) makeNorthEast() *QNode {
30+
func (node *QNode[T]) makeNorthEast(datapoints []T) *QNode[T] {
2531
midX, midY := node.getMidValues()
26-
node.northeast = NewQNode(midX, node.x1, node.y0, midY, node.depth+1)
32+
node.northeast = NewQNode(node, datapoints, midX, node.x1, node.y0, midY, node.depth+1)
2733
return node.northeast
2834
}
2935

30-
func (node *QNode) makeSouthWest() *QNode {
36+
func (node *QNode[T]) makeSouthWest(datapoints []T) *QNode[T] {
3137
midX, midY := node.getMidValues()
32-
node.southwest = NewQNode(node.x0, midX, midY, node.y1, node.depth+1)
38+
node.southwest = NewQNode(node, datapoints, node.x0, midX, midY, node.y1, node.depth+1)
3339
return node.southwest
3440
}
3541

36-
func (node *QNode) makeSouthEast() *QNode {
42+
func (node *QNode[T]) makeSouthEast(datapoints []T) *QNode[T] {
3743
midX, midY := node.getMidValues()
38-
node.southeast = NewQNode(midX, node.x1, midY, node.y1, node.depth+1)
44+
node.southeast = NewQNode(node, datapoints, midX, node.x1, midY, node.y1, node.depth+1)
3945
return node.southeast
4046
}
4147

42-
func (node *QNode) forEach(cb func(node *QNode), maxDepth uint) {
48+
func (node *QNode[T]) forEach(skip func(node *QNode[T]) bool, maxDepth uint) {
4349
if node.depth > maxDepth {
4450
return
4551
}
46-
47-
cb(node)
52+
if skip(node) {
53+
return
54+
}
4855
if node.northwest != nil {
49-
node.northwest.forEach(cb, maxDepth)
50-
} else if node.northeast != nil {
51-
node.northeast.forEach(cb, maxDepth)
52-
} else if node.southwest != nil {
53-
node.southwest.forEach(cb, maxDepth)
54-
} else if node.southeast != nil {
55-
node.southeast.forEach(cb, maxDepth)
56+
node.northwest.forEach(skip, maxDepth)
57+
}
58+
if node.northeast != nil {
59+
node.northeast.forEach(skip, maxDepth)
60+
}
61+
if node.southwest != nil {
62+
node.southwest.forEach(skip, maxDepth)
63+
}
64+
if node.southeast != nil {
65+
node.southeast.forEach(skip, maxDepth)
5666
}
5767
}
5868

59-
func (node *QNode) collapse(x, y float32, maxDepth uint) {
60-
node.forEach(func (node *QNode) {
69+
func (node *QNode[T]) markPathTo(x, y float32) {
70+
node.marked = true
6171
midX, midY := node.getMidValues()
62-
63-
// 1st quadrant
6472
if x < midX && y < midY {
6573
if node.northwest == nil {
66-
node.makeNorthWest()
74+
return
6775
}
68-
node.northeast = nil
69-
node.southwest = nil
70-
node.southeast = nil
71-
return
72-
}
73-
74-
// 2nd quadrant
75-
if x > midX && y < midY {
76+
node.northwest.markPathTo(x, y)
77+
} else if x > midX && y < midY {
7678
if node.northeast == nil {
77-
node.makeNorthEast()
79+
return
7880
}
79-
node.northwest = nil
80-
node.southwest = nil
81-
node.southeast = nil
82-
return
83-
}
84-
85-
// 3rd quadrant
86-
if x < midX && y > midY {
81+
node.northeast.markPathTo(x, y)
82+
} else if x < midX && y > midY {
8783
if node.southwest == nil {
88-
node.makeSouthWest()
84+
return
8985
}
90-
node.northwest = nil
91-
node.northeast = nil
92-
node.southeast = nil
93-
return
94-
}
95-
96-
// if 4th quadrant
97-
if x > midX && y > midY {
86+
node.southwest.markPathTo(x, y)
87+
} else if x > midX && y > midY {
9888
if node.southeast == nil {
99-
node.makeSouthEast()
89+
return
10090
}
101-
node.northwest = nil
102-
node.northeast = nil
103-
node.southwest = nil
104-
return
91+
node.southeast.markPathTo(x, y)
10592
}
93+
}
10694

95+
func (node *QNode[T]) generateTree(maxDepth uint) {
96+
node.forEach(func (node *QNode[T]) bool {
97+
nwDatapoints := make([]T, 0)
98+
neDatapoints := make([]T, 0)
99+
swDatapoints := make([]T, 0)
100+
seDatapoints := make([]T, 0)
101+
for _, v := range node.datapoints {
102+
x, y := v.getPosition()
103+
midX, midY := node.getMidValues()
104+
if x < midX && y < midY {
105+
nwDatapoints = append(nwDatapoints, v)
106+
} else if x > midX && y < midY {
107+
neDatapoints = append(neDatapoints, v)
108+
} else if x < midX && y > midY {
109+
swDatapoints = append(swDatapoints, v)
110+
} else if x > midX && y > midY {
111+
seDatapoints = append(seDatapoints, v)
112+
}
113+
}
114+
noData := true
115+
if len(nwDatapoints) != 0 {
116+
noData = false
117+
node.makeNorthWest(nwDatapoints)
118+
}
119+
if len(neDatapoints) != 0 {
120+
noData = false
121+
node.makeNorthEast(neDatapoints)
122+
}
123+
if len(swDatapoints) != 0 {
124+
noData = false
125+
node.makeSouthWest(swDatapoints)
126+
}
127+
if len(seDatapoints) != 0 {
128+
noData = false
129+
node.makeSouthEast(seDatapoints)
130+
}
131+
if noData {
132+
return true
133+
}
134+
return false
107135
}, maxDepth)
108-
109136
}
110137

111-
func (node *QNode) getMidValues() (x, y float32) {
138+
func (node *QNode[T]) getMidValues() (x, y float32) {
112139
return (node.x0+node.x1)*0.5, (node.y0+node.y1)*0.5
113140
}
114141

115-
func (node *QNode) String() string {
142+
func (node *QNode[T]) String() string {
116143
var sb strings.Builder
117-
node.forEach(func (node *QNode) {
144+
node.forEach(func (node *QNode[T]) bool {
118145
midX, midY := node.getMidValues()
119146
sb.WriteString(fmt.Sprintf("%s[Node: (%f, %f)]\n", strings.Repeat("-> ", int(node.depth)), midX, midY))
147+
return false
120148
}, 10)
121149
return sb.String()
122150
}

0 commit comments

Comments
 (0)