forked from Middlerun/lovecraft
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcollision.lua
134 lines (128 loc) · 5.2 KB
/
collision.lua
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
function checkCollisions(terrain, player)
local rightWall = 1
local leftWall = 2
local ceiling = 3
local ground = 4
local hit = 0
local frac = 1
local moveFrac = 1
local mid
local againstRightWall = false
local againstLeftWall = false
-- Check right hit
for y = 0, player.height, player.height/2 do
if player.againstRightWall and terrain:getBlock(math.ceil(player.y - y), math.ceil(player.x + player.width/2 + 0.01)) ~= AIR then
againstRightWall = true
elseif not player.againstRightWall and player.x > player.oldX and terrain:getBlock(math.ceil(player.y - y), math.ceil(player.x + player.width/2)) ~= AIR then
frac = (math.ceil(player.oldX + player.width/2) - (player.oldX + player.width/2)) / (player.x - player.oldX)
mid = player.oldY - y + frac * (player.y - player.oldY)
if mid >= math.ceil(player.y - y) - 1 and mid <= math.ceil(player.y - y) and frac < moveFrac then
hit = rightWall
moveFrac = frac
end
end
end
-- Check left hit
for y = 0, player.height, player.height/2 do
if player.againstLeftWall and terrain:getBlock(math.ceil(player.y - y), math.ceil(player.x - player.width/2 - 0.01)) ~= AIR then
againstLeftWall = true
elseif not player.againstLeftWall and player.x < player.oldX and terrain:getBlock(math.ceil(player.y - y), math.ceil(player.x - player.width/2)) ~= AIR then
frac = (math.ceil(player.oldX - player.width/2) - 1 - (player.oldX - player.width/2)) / (player.x - player.oldX)
mid = player.oldY - y + frac * (player.y - player.oldY)
if mid >= math.ceil(player.y - y) - 1 and mid <= math.ceil(player.y - y) and frac < moveFrac then
hit = leftWall
moveFrac = frac
end
end
end
if player.falling then
-- Check ceiling hit
for x = -player.width/2, player.width, player.width do
if player.y < player.oldY and terrain:getBlock(math.ceil(player.y - player.height), math.ceil(player.x + x)) ~= AIR then
frac = (math.ceil(player.oldY - player.height) - 1 - (player.oldY - player.height)) / (player.y - player.oldY)
mid = player.oldX + x + frac * (player.x - player.oldX)
if mid >= math.ceil(player.x + x) - 1 and mid <= math.ceil(player.x + x) and frac < moveFrac then
hit = ceiling
moveFrac = frac
end
end
end
-- Check ground hit
for x = -player.width/2, player.width, player.width do
if player.y > player.oldY and terrain:getBlock(math.ceil(player.y), math.ceil(player.x + x)) ~= AIR then
frac = (math.ceil(player.oldY) - (player.oldY)) / (player.y - player.oldY)
mid = player.oldX + x + frac * (player.x - player.oldX)
if mid >= math.ceil(player.x + x) - 1 and mid <= math.ceil(player.x + x) and frac < moveFrac then
hit = ground
moveFrac = frac
end
end
end
else
-- Check player has ground to stand on
if terrain:getBlock(math.floor(player.y) + 1, math.floor(player.x - player.width / 2) + 1) == AIR
and terrain:getBlock(math.floor(player.y) + 1, math.floor(player.x + player.width / 2) + 1) == AIR then
player.falling = true
player.hook:start()
end
end
if not againstRightWall then player.againstRightWall = false end
if not againstLeftWall then player.againstLeftWall = false end
if hit ~= 0 then
player.x = player.oldX + 0.99 * moveFrac * (player.x - player.oldX)
player.y = player.oldY + 0.99 * moveFrac * (player.y - player.oldY)
end
if hit == leftWall then
player.vx = 0
player.againstLeftWall = true
if player.hook.angle < math.pi/2 or player.hook.angle > math.pi and player.hook.angle < 3 * math.pi/2 then player.hook:stop()
else player.hook:zero()
end
elseif hit == rightWall then
player.vx = 0
player.againstRightWall = true
if player.hook.angle > math.pi/2 and player.hook.angle < math.pi or player.hook.angle > 3 * math.pi/2 then player.hook:stop()
else player.hook:zero()
end
elseif hit == ceiling then
player.vy = 0
player.hook:zero()
elseif hit == ground then
player.landTime = 0.01 * player.vy
if player.landTime < 0.1 then player.landTime = 0 end
player.vy = 0
player.y = math.ceil(player.y - 0.5)
player.falling = false
player.hook:stop()
end
player.oldX = player.x
player.oldY = player.y
local i = 1
local max = #terrain.entities
while i <= max do
local entity = terrain.entities[i]
local remove = false
if math.abs(player.x - entity.x) < player.width/2 + 0.35 and
entity.y < player.y + 0.85 and entity.y > player.y - player.height + 0.35 then
if player.inventory:give(entity.id) then
remove = true
end
elseif love.timer.getTime() - entity.createTime > 300 then
remove = true
end
if remove then
table.remove(terrain.entities, i)
max = max - 1
else
if entity.falling and terrain:getBlock(math.ceil(entity.y), math.ceil(entity.x)) ~= AIR then
entity.y = math.floor(entity.y)
entity.vy = 0
entity.falling = false
elseif not entity.falling and terrain:getBlock(math.floor(entity.y) + 1, math.ceil(entity.x)) == AIR then
entity.vy = 0
entity.falling = true
end
i = i + 1
end
end
end