About Social Code
summaryrefslogtreecommitdiff
path: root/world.rhm
blob: a528f8998b62f4cd05d2e65b42a6e944cd1bf591 (plain)
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
#lang rhombus/static

export:
    Chunk
    Entity
    World
    Tile

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
    | #'north: #'south
    | #'north_east: #'south_west
    | #'east: #'west
    | #'south_east: #'north_west
    | #'south: #'north
    | #'south_west: #'north_east
    | #'west: #'east
    | #'north_west: #'south_east
    | ~else: #false

class Tile(type :: TileType):
    method get_block() :: maybe(Direction):
        match type
        | #'empty: #false
        | #'wall_north: #'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(#'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):
    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)

    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(#'north, #false)
        | more_x && less_y:   current_chunk.neighbours.get(#'north_east, #false)
        | more_x && inside_y: current_chunk.neighbours.get(#'east, #false)
        | more_x && more_y:   current_chunk.neighbours.get(#'south_east, #false)
        | inside_x && more_y: current_chunk.neighbours.get(#'south, #false)
        | less_x && more_y:   current_chunk.neighbours.get(#'south_west, #false)
        | less_x && inside_y: current_chunk.neighbours.get(#'west, #false)
        | less_x && less_y:   current_chunk.neighbours.get(#'north_west, #false)
        | ~else:              current_chunk

        next_chunk

    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:  #'north
        | diff_x > 0  && diff_y < 0:  #'north_east
        | diff_x > 0  && diff_y == 0: #'east
        | diff_x > 0  && diff_y > 0:  #'south_east
        | diff_x == 0 && diff_y > 0:  #'south
        | diff_x < 0 && diff_y > 0:   #'south_west
        | diff_x < 0  && diff_y == 0: #'west
        | diff_x < 0  && diff_y < 0:  #'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

    method gather(target_entity :: Entity):
        let x = target_entity.x
        let y = target_entity.y
        let target_chunk = validate_chunk(x, y)

        fun validate_gather_resource() :: Boolean:
            // TODO figure out how to implement resource gathering
            // it could be better to keep them as a tile and update the
            // tile from regular to consumed
            #false

        cond
        | validate_dist(x, y, 1) && target_chunk != #false && validate_gather_resource():
            #true
        | ~else:
            #false

class World(chunks :: MutableList.now_of(Chunk),
            entities :: MutableList.now_of(Entity)):

    constructor():
        super(MutableList(), MutableList())