1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
|
#lang rhombus/static
import rhombus/random
export:
Chunk
Entity
EntityGatherable
EntityPlayer
ActionResponse
Direction
World
Tile
let rand = random.Random()
enum Direction:
north
north_east
east
south_east
south
south_west
west
north_west
enum TileType:
empty
wall_north
fun flip_block(block :: maybe(Direction)) :: maybe(Direction):
match block
| Direction.north: Direction.south
| Direction.north_east: Direction.south_west
| Direction.east: Direction.west
| Direction.south_east: Direction.north_west
| Direction.south: Direction.north
| Direction.south_west: Direction.north_east
| Direction.west: Direction.east
| Direction.north_west: Direction.south_east
| ~else: #false
class Tile(type :: TileType):
method get_block() :: maybe(Direction):
match type
| TileType.empty: #false
| TileType.wall_north: Direction.north
class Chunk(width :: Int,
height :: Int,
offset_x :: Int,
offset_y :: Int,
tiles :: Array.now_of(Tile),
neighbours :: MutableMap.now_of(Direction, Chunk)):
constructor(width :: Int, height :: Int, offset_x :: Int, offset_y :: Int):
super(width, height, offset_x, offset_y, Array.make(width * height, Tile(TileType.empty)), MutableMap())
method add_neighbour(chunk :: Chunk, direction :: Direction):
neighbours[direction] := chunk
method get_tile(x :: Int, y :: Int) :: Tile:
tiles[y * width + x]
method set_tile(x :: Int, y :: Int, tile :: Tile):
tiles[y * width + x] := tile
class Entity(id:: Int, mutable x :: Int, mutable y :: Int, mutable current_chunk :: Chunk):
nonfinal
// Validate that the provided coordinates are within the max distance range of
// entities current coordinates.
method validate_dist(x :: Int, y :: Int, max_dist :: Int):
let diff_x = math.abs(this.x - x)
let diff_y = math.abs(this.y - y)
// Diagonal case is moving in the same direction (same x & y) and that amount
// is less than or equal to max distance
let diagonal = diff_x == diff_y && diff_x <= max_dist
let x_axis = diff_x <= max_dist && diff_y == 0
let y_axis = diff_y <= max_dist && diff_x == 0
let no_movement = diff_x == 0 && diff_y == 0
!no_movement && (diagonal || x_axis || y_axis)
// Validate that the x,y coordinates map to a chunk that is
// near the current coordinates of the entity
method validate_chunk(x :: Int, y :: Int) :: maybe(Chunk):
let current_chunk = this.current_chunk
let local_x = x - current_chunk.offset_x * current_chunk.width
let local_y = y - current_chunk.offset_y * current_chunk.height
let less_x = local_x < 0
let less_y = local_y < 0
let more_x = local_x >= current_chunk.width
let more_y = local_y >= current_chunk.height
let inside_x = !(less_x || more_x)
let inside_y = !(less_y || more_y)
let next_chunk :: maybe(Chunk) = cond
| inside_x && less_y: current_chunk.neighbours.get(Direction.north, #false)
| more_x && less_y: current_chunk.neighbours.get(Direction.north_east, #false)
| more_x && inside_y: current_chunk.neighbours.get(Direction.east, #false)
| more_x && more_y: current_chunk.neighbours.get(Direction.south_east, #false)
| inside_x && more_y: current_chunk.neighbours.get(Direction.south, #false)
| less_x && more_y: current_chunk.neighbours.get(Direction.south_west, #false)
| less_x && inside_y: current_chunk.neighbours.get(Direction.west, #false)
| less_x && less_y: current_chunk.neighbours.get(Direction.north_west, #false)
| ~else: current_chunk
next_chunk
// Validate that the tile at the coordinates would not block a movement of the entity
// to those coordinates from its current coordinates
method validate_move(x :: Int, y :: Int, max_dist :: Int, target_chunk :: Chunk):
let current_tile = this.current_chunk.get_tile(this.x, this.y)
let target_tile = target_chunk.get_tile(x, y)
let diff_x = x - this.x
let diff_y = y - this.y
let direction = cond
| diff_x == 0 && diff_y < 0: Direction.north
| diff_x > 0 && diff_y < 0: Direction.north_east
| diff_x > 0 && diff_y == 0: Direction.east
| diff_x > 0 && diff_y > 0: Direction.south_east
| diff_x == 0 && diff_y > 0: Direction.south
| diff_x < 0 && diff_y > 0: Direction.south_west
| diff_x < 0 && diff_y == 0: Direction.west
| diff_x < 0 && diff_y < 0: Direction.north_west
let current_block = current_tile.get_block()
let target_block = flip_block(target_tile.get_block())
!(direction == current_block || direction == target_block)
method
| move(x :: Int, y :: Int): move(x, y, 1)
| move(x :: Int, y :: Int, max_dist :: Int) :: Boolean:
let target_chunk = validate_chunk(x, y)
cond
| validate_dist(x, y, max_dist) && target_chunk != #false && validate_move(x, y, max_dist, target_chunk!!):
this.current_chunk := target_chunk!!
this.x := x
this.y := y
#true
| ~else:
#false
class EntityGatherable(quantity :: Int, odds :: Real):
extends Entity
constructor(id :: Int, x :: Int, y :: Int, current_chunk :: Chunk):
super(id, x, y, current_chunk)(10, 1 / 10)
enum ActionResponse:
ok
unsuccessful
invalid
cooldown
class EntityPlayer(mutable timer :: Int):
extends Entity
field max_timer = 4
constructor(id :: Int, x :: Int, y :: Int, current_chunk :: Chunk):
super(id, x, y, current_chunk)(0)
method tick():
// TODO this "ticks" the entity. Things like healing and other
// stuff should happen within here. In the future we probably
// want to set a state where the player is gathering and in tick
// we calculate the odds that they did gather properly
if timer == 0:
| timer := max_timer
| timer := timer - 1
method gather(target_entity :: Entity) :: ActionResponse:
let x = target_entity.x
let y = target_entity.y
let target_chunk = validate_chunk(x, y)
let right_type = target_entity is_a EntityGatherable
fun validate_gather_resource(entity :: EntityGatherable) :: ActionResponse:
// TODO odds should change based on skills of player
// we should also validate player is carrying the right tool
// we should also have the timer count down every tick
// not just inside the gathering call
let roll = rand.random()
if roll < entity.odds:
| ActionResponse.ok
| ActionResponse.unsuccessful
cond
| timer != 0: ActionResponse.cooldown
| right_type && validate_dist(x, y, 1) && target_chunk != #false:
validate_gather_resource(target_entity)
| ~else:
ActionResponse.invalid
class World(chunks :: MutableList.now_of(Chunk),
entities :: MutableList.now_of(Entity)):
constructor():
super(MutableList(), MutableList())
|