2022-12-31 09:01:49 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
# RAYCASTER
|
2022-12-31 09:53:18 +01:00
|
|
|
# Inspired by https://www.youtube.com/watch?v=gYRrGTC7GtA
|
2022-12-31 09:01:49 +01:00
|
|
|
#
|
|
|
|
# pip install pysdl2 pysdl2-dll
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import sdl2.ext
|
2022-12-31 09:53:18 +01:00
|
|
|
import math
|
2023-01-02 10:36:31 +01:00
|
|
|
from time import time
|
2022-12-31 09:01:49 +01:00
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
MAP_WIN_WIDTH = 640
|
|
|
|
MAP_WIN_HEIGHT = 640
|
2023-01-02 20:40:11 +01:00
|
|
|
RAYCAST_WIN_WIDTH = 900
|
|
|
|
RAYCAST_WIN_HEIGHT = 600
|
2023-01-02 20:26:52 +01:00
|
|
|
DUNGEON_WIDTH = MAP_WIN_WIDTH
|
|
|
|
DUNGEON_HEIGHT = MAP_WIN_HEIGHT
|
2022-12-31 09:01:49 +01:00
|
|
|
PLAYER_SPEED = 10
|
2022-12-31 09:53:18 +01:00
|
|
|
RAY_LENGTH = 100
|
2023-01-02 10:36:31 +01:00
|
|
|
MAP_SCALE = 80
|
|
|
|
DOF = 8 # Depth Of Field
|
|
|
|
|
2023-01-02 19:55:09 +01:00
|
|
|
DEGREE_IN_RADIANTS = 0.01745329
|
|
|
|
|
2023-01-02 10:36:31 +01:00
|
|
|
MAP = [
|
|
|
|
1, 1, 1, 1, 1, 1, 1, 1,
|
|
|
|
1, 0, 0, 0, 1, 0, 0, 1,
|
2023-01-02 20:40:11 +01:00
|
|
|
1, 0, 1, 0, 1, 0, 0, 1,
|
|
|
|
1, 0, 0, 0, 1, 0, 0, 1,
|
|
|
|
1, 0, 0, 0, 1, 0, 0, 1,
|
|
|
|
1, 0, 1, 1, 1, 0, 0, 1,
|
2023-01-02 10:36:31 +01:00
|
|
|
1, 0, 0, 0, 0, 0, 0, 1,
|
|
|
|
1, 1, 1, 1, 1, 1, 1, 1,
|
|
|
|
]
|
|
|
|
MAP_SIZE = 8
|
2022-12-31 09:01:49 +01:00
|
|
|
|
|
|
|
class Main:
|
|
|
|
|
|
|
|
def __init__(self):
|
2023-01-02 10:36:31 +01:00
|
|
|
# Check valid map
|
|
|
|
if len(MAP) != MAP_SIZE * MAP_SIZE:
|
|
|
|
raise ValueError("Map size is {}, but should be a power of {}", len(MAP), MAP_SIZE)
|
|
|
|
|
2022-12-31 09:01:49 +01:00
|
|
|
# Graphics
|
|
|
|
sdl2.ext.init()
|
2023-01-02 20:26:52 +01:00
|
|
|
self.mapWindow = sdl2.ext.Window("2D Map", size=(MAP_WIN_WIDTH, MAP_WIN_HEIGHT))
|
2023-01-02 10:36:31 +01:00
|
|
|
self.mapWindow.show()
|
|
|
|
self.mapSurface = self.mapWindow.get_surface()
|
2022-12-31 09:01:49 +01:00
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
self.raycastWindow = sdl2.ext.Window("3D View", size=(RAYCAST_WIN_WIDTH, RAYCAST_WIN_HEIGHT))
|
|
|
|
self.raycastWindow.show()
|
|
|
|
self.raycastSurface = self.raycastWindow.get_surface()
|
|
|
|
|
2022-12-31 09:01:49 +01:00
|
|
|
# Player
|
2022-12-31 09:53:18 +01:00
|
|
|
self.player_position = {"x": int(DUNGEON_WIDTH/2), "y": int(DUNGEON_HEIGHT/2), "r": 0} # r is rotation in radiants
|
2022-12-31 09:01:49 +01:00
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
running = True
|
|
|
|
while running:
|
|
|
|
events = sdl2.ext.get_events()
|
|
|
|
for event in events:
|
|
|
|
if event.type == sdl2.SDL_QUIT:
|
|
|
|
running = False
|
|
|
|
break
|
|
|
|
if event.type == sdl2.SDL_KEYDOWN:
|
2022-12-31 09:53:18 +01:00
|
|
|
# Rotate player
|
2022-12-31 09:01:49 +01:00
|
|
|
if event.key.keysym.sym == sdl2.SDLK_LEFT:
|
2023-01-02 20:40:11 +01:00
|
|
|
self.player_position["r"] = self.player_position["r"] + 10*DEGREE_IN_RADIANTS
|
2022-12-31 09:01:49 +01:00
|
|
|
elif event.key.keysym.sym == sdl2.SDLK_RIGHT:
|
2023-01-02 20:40:11 +01:00
|
|
|
self.player_position["r"] = self.player_position["r"] - 10*DEGREE_IN_RADIANTS
|
2022-12-31 09:53:18 +01:00
|
|
|
|
|
|
|
# Compute deltax and deltay based on player direction
|
|
|
|
player_delta_x = math.cos(self.player_position["r"]) * PLAYER_SPEED
|
|
|
|
player_delta_y = math.sin(self.player_position["r"]) * PLAYER_SPEED
|
|
|
|
|
|
|
|
# Move player based on its direction
|
|
|
|
if event.key.keysym.sym == sdl2.SDLK_UP:
|
|
|
|
self.player_position["y"] = int(self.player_position["y"] + player_delta_y)
|
|
|
|
self.player_position["x"] = int(self.player_position["x"] + player_delta_x)
|
|
|
|
elif event.key.keysym.sym == sdl2.SDLK_DOWN:
|
|
|
|
self.player_position["y"] = int(self.player_position["y"] - player_delta_y)
|
|
|
|
self.player_position["x"] = int(self.player_position["x"] - player_delta_x)
|
|
|
|
|
|
|
|
# Limit position into dungeon bounds
|
2022-12-31 09:01:49 +01:00
|
|
|
if self.player_position["x"] < 0:
|
|
|
|
self.player_position["x"] = 0
|
|
|
|
if self.player_position["x"] > DUNGEON_WIDTH:
|
|
|
|
self.player_position["x"] = DUNGEON_WIDTH
|
|
|
|
if self.player_position["y"] < 0:
|
|
|
|
self.player_position["y"] = 0
|
|
|
|
if self.player_position["y"] > DUNGEON_HEIGHT:
|
|
|
|
self.player_position["y"] = DUNGEON_HEIGHT
|
2022-12-31 09:53:18 +01:00
|
|
|
if self.player_position["r"] > 2*math.pi:
|
|
|
|
self.player_position["r"] = 0
|
|
|
|
if self.player_position["r"] < 0:
|
|
|
|
self.player_position["r"] = 2*math.pi
|
2022-12-31 09:01:49 +01:00
|
|
|
|
2023-01-02 10:36:31 +01:00
|
|
|
sdl2.ext.draw.fill(self.mapSurface, sdl2.ext.Color(0,0,0,0)) # Clears screen
|
2023-01-02 20:26:52 +01:00
|
|
|
sdl2.ext.draw.fill(self.raycastSurface, sdl2.ext.Color(0,0,0,0)) # Clears screen
|
2022-12-31 09:01:49 +01:00
|
|
|
self.draw()
|
2023-01-02 10:36:31 +01:00
|
|
|
self.mapWindow.refresh()
|
2023-01-02 20:26:52 +01:00
|
|
|
self.raycastWindow.refresh()
|
2022-12-31 09:01:49 +01:00
|
|
|
return 0
|
|
|
|
|
|
|
|
def draw(self):
|
2023-01-02 10:36:31 +01:00
|
|
|
self.draw2Dmap()
|
|
|
|
self.drawPlayer()
|
|
|
|
self.drawRays()
|
|
|
|
|
|
|
|
def drawPlayer(self):
|
2022-12-31 09:53:18 +01:00
|
|
|
# Player in 2D map
|
2023-01-02 10:36:31 +01:00
|
|
|
sdl2.ext.draw.fill(self.mapSurface, sdl2.ext.Color(0,255,0,255), (self.player_position["x"] - 2, self.player_position["y"] - 2, 4, 4))
|
2022-12-31 09:53:18 +01:00
|
|
|
# Player line of sight in 2D map
|
|
|
|
ray = {
|
|
|
|
"x": int(self.player_position["x"] + math.cos(self.player_position["r"]) * 50), # deltaX + playerX
|
|
|
|
"y": int(self.player_position["y"] + math.sin(self.player_position["r"]) * 50) # deltaY + playerY
|
|
|
|
}
|
2023-01-02 19:55:09 +01:00
|
|
|
sdl2.ext.draw.line(self.mapSurface, sdl2.ext.Color(255,0,0,255), (self.player_position["x"], self.player_position["y"], ray["x"], ray["y"]))
|
|
|
|
|
2023-01-02 10:36:31 +01:00
|
|
|
|
|
|
|
def draw2Dmap(self):
|
|
|
|
# 2D map
|
|
|
|
for i in range(len(MAP)):
|
|
|
|
posX = i % MAP_SIZE * MAP_SCALE
|
|
|
|
posY = math.floor(i / MAP_SIZE) * MAP_SCALE
|
|
|
|
color = 0
|
|
|
|
if MAP[i] == 1:
|
|
|
|
color = 255
|
|
|
|
sdl2.ext.draw.fill(self.mapSurface, sdl2.ext.Color(color,color,color,255), (posX, posY, MAP_SCALE - 1, MAP_SCALE - 1))
|
|
|
|
|
|
|
|
def drawRays(self):
|
|
|
|
# Casts rays for raycasting
|
2023-01-02 19:55:09 +01:00
|
|
|
playerAngle = self.player_position["r"]
|
|
|
|
|
|
|
|
# Cast 60 rays from -30° to +30° (60° viewing angle)
|
2023-01-02 20:26:52 +01:00
|
|
|
for i in range(60):
|
|
|
|
rayAngle = playerAngle - (i - 30)*DEGREE_IN_RADIANTS
|
2023-01-02 19:55:09 +01:00
|
|
|
|
2023-01-02 10:36:31 +01:00
|
|
|
# Check horizontal lines
|
|
|
|
dof = 0 # Depth of field
|
|
|
|
if rayAngle == 0 or rayAngle == math.pi:
|
|
|
|
# Looking left or right (ray will never intersect parallel lines)
|
|
|
|
rayY = self.player_position["y"]
|
2023-01-02 19:55:09 +01:00
|
|
|
rayX = self.player_position["x"] + DOF * MAP_SCALE
|
2023-01-02 10:36:31 +01:00
|
|
|
dof = DOF # Set depth of field to maximum to avoid unneeded checks
|
|
|
|
elif rayAngle > math.pi:
|
|
|
|
# Looking up
|
2023-01-02 11:04:52 +01:00
|
|
|
aTan = -1/math.tan(rayAngle)
|
2023-01-02 10:36:31 +01:00
|
|
|
rayY = (int(self.player_position["y"] / MAP_SCALE) * MAP_SCALE) - 0.00001
|
|
|
|
rayX = (self.player_position["y"] - rayY) * aTan + self.player_position["x"]
|
|
|
|
yOffset = -MAP_SCALE
|
|
|
|
xOffset = -yOffset * aTan
|
|
|
|
else:
|
|
|
|
# Looking down
|
2023-01-02 11:04:52 +01:00
|
|
|
aTan = -1/math.tan(rayAngle)
|
2023-01-02 10:36:31 +01:00
|
|
|
rayY = (int(self.player_position["y"] / MAP_SCALE) * MAP_SCALE) + MAP_SCALE
|
|
|
|
rayX = (self.player_position["y"] - rayY) * aTan + self.player_position["x"]
|
|
|
|
yOffset = MAP_SCALE
|
|
|
|
xOffset = -yOffset * aTan
|
|
|
|
|
|
|
|
# Check if we reached a wall
|
|
|
|
while dof < 8:
|
|
|
|
mapX = int(rayX / MAP_SCALE)
|
|
|
|
mapY = int(rayY / MAP_SCALE)
|
|
|
|
mapArrayPosition = mapY * MAP_SIZE + mapX
|
2023-01-02 19:55:09 +01:00
|
|
|
if mapArrayPosition >= 0 and mapArrayPosition < MAP_SIZE*MAP_SIZE and MAP[mapArrayPosition] != 0:
|
2023-01-02 10:36:31 +01:00
|
|
|
dof = 8 # Hit the wall: we are done, no need to do other checks
|
|
|
|
else:
|
|
|
|
# Didn't hit the wall: check successive horizontal line
|
|
|
|
rayX = rayX + xOffset
|
|
|
|
rayY = rayY + yOffset
|
|
|
|
dof = dof + 1
|
2023-01-02 19:00:17 +01:00
|
|
|
|
2023-01-02 19:19:47 +01:00
|
|
|
# Save horyzontal probe rays for later comparison with vertical
|
|
|
|
horizRayX = rayX
|
|
|
|
horizRayY = rayY
|
2023-01-02 11:04:52 +01:00
|
|
|
|
|
|
|
# Check vertical lines
|
|
|
|
dof = 0 # Depth of field
|
|
|
|
nTan = -math.tan(rayAngle)
|
2023-01-02 19:00:17 +01:00
|
|
|
xOffset = 0
|
|
|
|
yOffset = 0
|
2023-01-02 11:04:52 +01:00
|
|
|
if rayAngle == math.pi * 0.5 or rayAngle == math.pi * 1.5:
|
2023-01-02 19:00:17 +01:00
|
|
|
#if rayAngle == 0 or rayAngle == math.pi:
|
2023-01-02 11:04:52 +01:00
|
|
|
# Looking up or down (ray will never intersect vertical lines)
|
2023-01-02 19:00:17 +01:00
|
|
|
rayX = self.player_position["x"]
|
2023-01-02 19:55:09 +01:00
|
|
|
rayY = self.player_position["y"] + DOF * MAP_SCALE
|
2023-01-02 11:04:52 +01:00
|
|
|
dof = DOF # Set depth of field to maximum to avoid unneeded checks
|
|
|
|
elif rayAngle > math.pi * 0.5 and rayAngle < math.pi * 1.5:
|
|
|
|
# Looking right
|
|
|
|
rayX = (int(self.player_position["x"] / MAP_SCALE) * MAP_SCALE) - 0.00001
|
|
|
|
rayY = (self.player_position["x"] - rayX) * nTan + self.player_position["y"]
|
|
|
|
xOffset = -MAP_SCALE
|
|
|
|
yOffset = -xOffset * nTan
|
|
|
|
else:
|
|
|
|
# Looking left
|
|
|
|
rayX = (int(self.player_position["x"] / MAP_SCALE) * MAP_SCALE) + MAP_SCALE
|
|
|
|
rayY = (self.player_position["x"] - rayX) * nTan + self.player_position["y"]
|
|
|
|
xOffset = MAP_SCALE
|
|
|
|
yOffset = -xOffset * nTan
|
|
|
|
|
|
|
|
# Check if we reached a wall
|
|
|
|
while dof < 8:
|
|
|
|
mapX = int(rayX / MAP_SCALE)
|
|
|
|
mapY = int(rayY / MAP_SCALE)
|
|
|
|
mapArrayPosition = mapY * MAP_SIZE + mapX
|
2023-01-02 19:00:17 +01:00
|
|
|
if mapArrayPosition >= 0 and mapArrayPosition < MAP_SIZE*MAP_SIZE-1 and MAP[mapArrayPosition] != 0:
|
2023-01-02 11:04:52 +01:00
|
|
|
dof = 8 # Hit the wall: we are done, no need to do other checks
|
|
|
|
else:
|
|
|
|
# Didn't hit the wall: check successive horizontal line
|
|
|
|
rayX = rayX + xOffset
|
|
|
|
rayY = rayY + yOffset
|
|
|
|
dof = dof + 1
|
2023-01-02 10:36:31 +01:00
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
horizDist = self.dist(self.player_position["x"], self.player_position["y"], horizRayX, horizRayY)
|
|
|
|
vertDist = self.dist(self.player_position["x"], self.player_position["y"], rayX, rayY)
|
|
|
|
shortestDist = vertDist
|
|
|
|
if vertDist > horizDist:
|
2023-01-02 19:19:47 +01:00
|
|
|
rayX = horizRayX
|
|
|
|
rayY = horizRayY
|
2023-01-02 20:26:52 +01:00
|
|
|
shortestDist = horizDist
|
2023-01-02 10:36:31 +01:00
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
# Draw rays in 2D view
|
2023-01-02 19:19:47 +01:00
|
|
|
sdl2.ext.draw.line(self.mapSurface, sdl2.ext.Color(0,0,255,255), (self.player_position["x"], self.player_position["y"], rayX, rayY))
|
2023-01-02 10:36:31 +01:00
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
|
|
|
|
# ------ Draw 3D view ------
|
|
|
|
|
|
|
|
# Calculate line height based on distance
|
|
|
|
lineHeight = MAP_SCALE * RAYCAST_WIN_HEIGHT / shortestDist
|
|
|
|
if lineHeight > RAYCAST_WIN_HEIGHT:
|
|
|
|
lineHeight = RAYCAST_WIN_HEIGHT
|
|
|
|
# Center line vertically in window
|
|
|
|
lineOffset = RAYCAST_WIN_HEIGHT / 2 - lineHeight / 2
|
2023-01-02 20:40:11 +01:00
|
|
|
|
|
|
|
# Simulate lighting based on wall incidence
|
|
|
|
color = sdl2.ext.Color(255,255,255,255)
|
|
|
|
if vertDist > horizDist:
|
|
|
|
color = sdl2.ext.Color(200,200,200,255)
|
|
|
|
|
2023-01-02 20:26:52 +01:00
|
|
|
# Draw line
|
2023-01-02 20:40:11 +01:00
|
|
|
for x in range(i*15, i*15+15):
|
|
|
|
sdl2.ext.draw.line(self.raycastSurface, color, (x, int(lineOffset), x, int(lineOffset + lineHeight)))
|
2023-01-02 20:26:52 +01:00
|
|
|
|
2023-01-02 19:19:47 +01:00
|
|
|
def dist(self, ax, ay, bx, by):
|
|
|
|
return math.sqrt((bx-ax)*(bx-ax) + (by-ay)*(by-ay))
|
2022-12-31 09:01:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
try:
|
|
|
|
main = Main()
|
|
|
|
main.run()
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
exit(0)
|