About Social Code
summaryrefslogtreecommitdiff
path: root/src/entity.rhm
blob: 831228a582b8793ba7a332c2862857a6717a4dbd (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
#lang rhombus/static
import:
    rhombus/random
    "chunk.rhm" open
    "item.rhm" open

export:
    Entity
    EntityGatherable
    EntityPlayer
    ActionResponse
    Slot

let rand = random.Random()

class Entity(id:: Int, mutable x :: Int, mutable y :: Int):
    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, current_chunk :: Chunk) :: maybe(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, current_chunk :: Chunk, target_chunk :: Chunk):
        let current_tile = 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, current_chunk :: Chunk): move(x, y, 1, current_chunk)
    | move(x :: Int, y :: Int, max_dist :: Int, current_chunk :: Chunk) :: maybe(Chunk):
        let target_chunk = validate_chunk(x, y, current_chunk)

        cond
        | validate_dist(x, y, max_dist) && target_chunk != #false && validate_move(x, y, max_dist, current_chunk, target_chunk!!):
            this.x := x
            this.y := y
            target_chunk!!
        | ~else:
            #false

class EntityGatherable(quantity :: Int, odds :: Real, cap :: ItemCap):
    extends Entity

    constructor(id :: Int, x :: Int, y :: Int, cap :: ItemCap):
        super(id, x, y)(10, 1 / 10, cap)

enum ActionResponse:
    ok
    unsuccessful
    invalid
    cooldown

enum Slot:
    right_hand

class EntityPlayer(mutable timer :: Int):
    extends Entity

    field max_timer = 4
    field inventory :: Array.now_of(maybe(Stack)) = Array.make(20, #false)
    field equip :: MutableMap.now_of(Slot, Item) = MutableMap()

    constructor(id :: Int, x :: Int, y :: Int):
        super(id, x, y)(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, current_chunk :: Chunk) :: ActionResponse:
        let x = target_entity.x
        let y = target_entity.y
        let target_chunk = validate_chunk(x, y, current_chunk)
        let right_type = target_entity is_a EntityGatherable

        fun validate_gather_resource(entity :: EntityGatherable) :: ActionResponse:
            let right_tool = item_get_cap(equip.get(Slot.right_hand, #false)) == entity.cap
            // 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()
            cond
            | !right_tool: ActionResponse.invalid
            | rand.random() < entity.odds: ActionResponse.ok
            | ~else: 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