New raycaster based on Lode's tutorial works, now to add textures.
This commit is contained in:
parent
96b44a4eb2
commit
c1b7df1850
2 changed files with 89 additions and 74 deletions
159
fenscaster.cpp
159
fenscaster.cpp
|
@ -14,9 +14,8 @@ Matrix MAP{
|
||||||
{1,0,1,0,0,0,0,0,1},
|
{1,0,1,0,0,0,0,0,1},
|
||||||
{1,0,1,0,0,1,1,0,1},
|
{1,0,1,0,0,1,1,0,1},
|
||||||
{1,0,0,0,0,0,0,0,1},
|
{1,0,0,0,0,0,0,0,1},
|
||||||
{1,1,0,0,0,0,0,0,1},
|
{1,1,0,0,0,0,0,1,1},
|
||||||
{1,0,0,1,1,1,0,0,1},
|
{1,0,0,1,1,1,0,0,1},
|
||||||
{1,0,0,0,1,0,0,0,1},
|
|
||||||
{1,0,0,0,0,0,1,1,1},
|
{1,0,0,0,0,0,1,1,1},
|
||||||
{1,1,1,1,1,1,1,1,1}
|
{1,1,1,1,1,1,1,1,1}
|
||||||
};
|
};
|
||||||
|
@ -37,11 +36,50 @@ const float SCALE = (SCREEN_WIDTH / 2) / CASTED_RAYS;
|
||||||
|
|
||||||
float player_x = SCREEN_WIDTH / 4;
|
float player_x = SCREEN_WIDTH / 4;
|
||||||
float player_y = SCREEN_WIDTH / 4;
|
float player_y = SCREEN_WIDTH / 4;
|
||||||
float player_angle = std::numbers::pi;
|
|
||||||
|
// x and y start position
|
||||||
|
double posX = player_x / TILE_SIZE;
|
||||||
|
double posY = player_y / TILE_SIZE;
|
||||||
|
|
||||||
|
// initial direction vector
|
||||||
|
double dirX = -1;
|
||||||
|
double dirY = 0;
|
||||||
|
|
||||||
|
// the 2d raycaster version of camera plane
|
||||||
|
double planeX = 0;
|
||||||
|
double planeY = 0.66;
|
||||||
|
|
||||||
#define rgba_color(r,g,b,a) (b<<(0*8))|(g<<(1*8))|(r<<(2*8))|(a<<(3*8))
|
#define rgba_color(r,g,b,a) (b<<(0*8))|(g<<(1*8))|(r<<(2*8))|(a<<(3*8))
|
||||||
#define gray_color(c) rgba_color(c, c, c, 255)
|
#define gray_color(c) rgba_color(c, c, c, 255)
|
||||||
|
|
||||||
|
std::vector<uint32_t> texture[8];
|
||||||
|
#define texWidth 64
|
||||||
|
#define texHeight 64
|
||||||
|
|
||||||
|
void load_textures() {
|
||||||
|
for(int i = 0; i < 8; i++) {
|
||||||
|
texture[i].resize(texWidth * texHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
for(int x = 0; x < texWidth; x++) {
|
||||||
|
for(int y = 0; y < texHeight; y++)
|
||||||
|
{
|
||||||
|
int xorcolor = (x * 256 / texWidth) ^ (y * 256 / texHeight);
|
||||||
|
//int xcolor = x * 256 / texWidth;
|
||||||
|
int ycolor = y * 256 / texHeight;
|
||||||
|
int xycolor = y * 128 / texHeight + x * 128 / texWidth;
|
||||||
|
texture[0][texWidth * y + x] = 65536 * 254 * (x != y && x != texWidth - y); //flat red texture with black cross
|
||||||
|
texture[1][texWidth * y + x] = xycolor + 256 * xycolor + 65536 * xycolor; //sloped greyscale
|
||||||
|
texture[2][texWidth * y + x] = 256 * xycolor + 65536 * xycolor; //sloped yellow gradient
|
||||||
|
texture[3][texWidth * y + x] = xorcolor + 256 * xorcolor + 65536 * xorcolor; //xor greyscale
|
||||||
|
texture[4][texWidth * y + x] = 256 * xorcolor; //xor green
|
||||||
|
texture[5][texWidth * y + x] = 65536 * 192 * (x % 16 && y % 16); //red bricks
|
||||||
|
texture[6][texWidth * y + x] = 65536 * ycolor; //red gradient
|
||||||
|
texture[7][texWidth * y + x] = 128 + 256 * 128 + 65536 * 128; //flat grey texture
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void draw_rect(Fenster &window, Point pos, Point size, uint32_t color) {
|
void draw_rect(Fenster &window, Point pos, Point size, uint32_t color) {
|
||||||
size_t x_start = size_t(pos.x);
|
size_t x_start = size_t(pos.x);
|
||||||
size_t y_start = size_t(pos.y);
|
size_t y_start = size_t(pos.y);
|
||||||
|
@ -114,32 +152,14 @@ void clear(Fenster &window) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void draw_map_rays(Fenster &window, int col, int row, Point target) {
|
void draw_map_blocks(Fenster &window, int col, int row) {
|
||||||
draw_map_rect(window, col, row, rgba_color(100, 20, 20, 255));
|
draw_map_rect(window, col, row, rgba_color(100, 20, 20, 255));
|
||||||
draw_line(window, {size_t(player_x), size_t(player_y)}, target, rgba_color(200, 20, 20, 255));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ray_casting(Fenster &window, Matrix& map) {
|
void ray_casting(Fenster &window, Matrix& map) {
|
||||||
int w = THREED_VIEW_WIDTH;
|
int w = THREED_VIEW_WIDTH;
|
||||||
int h = THREED_VIEW_HEIGHT;
|
int h = THREED_VIEW_HEIGHT;
|
||||||
|
|
||||||
// x and y start position
|
|
||||||
double posX = player_x / TILE_SIZE;
|
|
||||||
double posY = player_y / TILE_SIZE;
|
|
||||||
|
|
||||||
// initial direction vector
|
|
||||||
double dirX = std::cos(player_angle);
|
|
||||||
double dirY = 0;
|
|
||||||
|
|
||||||
// the 2d raycaster version of camera plane
|
|
||||||
double planeX = 0;
|
|
||||||
double planeY = 0.66;
|
|
||||||
|
|
||||||
// time of current frame
|
|
||||||
double time = 0;
|
|
||||||
// time of previous frame
|
|
||||||
double oldTime = 0;
|
|
||||||
|
|
||||||
for(int x = 0; x < w; x++) {
|
for(int x = 0; x < w; x++) {
|
||||||
// calculate ray position and direction
|
// calculate ray position and direction
|
||||||
double cameraX = 2 * x / double(w) - 1; // x-coord in camera space
|
double cameraX = 2 * x / double(w) - 1; // x-coord in camera space
|
||||||
|
@ -155,8 +175,8 @@ void ray_casting(Fenster &window, Matrix& map) {
|
||||||
double sideDistY;
|
double sideDistY;
|
||||||
|
|
||||||
// length of ray from one x or y-side to next x or y-side
|
// length of ray from one x or y-side to next x or y-side
|
||||||
double deltaDistX = (rayDirX == 0) ? 1e30 : std::abs(1.0 / rayDirX);
|
double deltaDistX = std::abs(1.0 / rayDirX);
|
||||||
double deltaDistY = (rayDirY == 0) ? 1e30 : std::abs(1.0 / rayDirY);
|
double deltaDistY = std::abs(1.0 / rayDirY);
|
||||||
double perpWallDist;
|
double perpWallDist;
|
||||||
|
|
||||||
int stepX = 0;
|
int stepX = 0;
|
||||||
|
@ -202,6 +222,13 @@ void ray_casting(Fenster &window, Matrix& map) {
|
||||||
perpWallDist = (sideDistY - deltaDistY);
|
perpWallDist = (sideDistY - deltaDistY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
draw_map_blocks(window, mapX, mapY);
|
||||||
|
|
||||||
|
|
||||||
|
// player direction ray
|
||||||
|
draw_line(window, {size_t(posX * TILE_SIZE), size_t(posY * TILE_SIZE)},
|
||||||
|
{(size_t)mapX * TILE_SIZE, (size_t)mapY * TILE_SIZE}, rgba_color(0, 255, 0, 255));
|
||||||
|
|
||||||
int lineHeight = int(h / perpWallDist);
|
int lineHeight = int(h / perpWallDist);
|
||||||
|
|
||||||
int drawStart = -lineHeight / 2 + h / 2;
|
int drawStart = -lineHeight / 2 + h / 2;
|
||||||
|
@ -219,26 +246,6 @@ void ray_casting(Fenster &window, Matrix& map) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void map_view_casting(Fenster &window, Matrix& map) {
|
|
||||||
float start_angle = player_angle - HALF_FOV;
|
|
||||||
|
|
||||||
for(int ray = 0; ray < CASTED_RAYS; ray++, start_angle += STEP_ANGLE)
|
|
||||||
{
|
|
||||||
for(int depth = 1; depth < MAX_DEPTH; depth++) {
|
|
||||||
float target_x = player_x - std::sin(start_angle) * depth;
|
|
||||||
float target_y = player_y + std::cos(start_angle) * depth;
|
|
||||||
|
|
||||||
int col = int(target_x / TILE_SIZE);
|
|
||||||
int row = int(target_y / TILE_SIZE);
|
|
||||||
|
|
||||||
if(map[row][col] == 1) {
|
|
||||||
draw_map_rays(window, col, row, {size_t(target_x), size_t(target_y)});
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void draw_ceiling_floor(Fenster &window) {
|
void draw_ceiling_floor(Fenster &window) {
|
||||||
draw_rect(window,
|
draw_rect(window,
|
||||||
{size_t(window.width() / 2), 0},
|
{size_t(window.width() / 2), 0},
|
||||||
|
@ -254,51 +261,59 @@ void draw_everything(Fenster &window) {
|
||||||
clear(window);
|
clear(window);
|
||||||
draw_map(window, MAP);
|
draw_map(window, MAP);
|
||||||
draw_ceiling_floor(window);
|
draw_ceiling_floor(window);
|
||||||
map_view_casting(window, MAP);
|
|
||||||
ray_casting(window, MAP);
|
ray_casting(window, MAP);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool collision(float x, float y) {
|
bool empty_space(int new_x, int new_y) {
|
||||||
int col = int(x / TILE_SIZE);
|
dbc::check((size_t)new_x < matrix::width(MAP),
|
||||||
int row = int(y / TILE_SIZE);
|
format("x={} too wide={}", new_x, matrix::width(MAP)));
|
||||||
|
dbc::check((size_t)new_y < matrix::height(MAP),
|
||||||
|
format("y={} too high={}", new_y, matrix::height(MAP)));
|
||||||
|
|
||||||
return MAP[row][col] == 1;
|
return MAP[new_y][new_x] == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
Fenster window(SCREEN_WIDTH, SCREEN_HEIGHT, "Fenscaster");
|
Fenster window(SCREEN_WIDTH, SCREEN_HEIGHT, "Fenscaster");
|
||||||
|
const int fps = 60;
|
||||||
|
double moveSpeed = 0.1;
|
||||||
|
double rotSpeed = 0.1;
|
||||||
|
|
||||||
while(window.loop(60)) {
|
load_textures();
|
||||||
|
|
||||||
|
while(window.loop(fps)) {
|
||||||
draw_everything(window);
|
draw_everything(window);
|
||||||
|
|
||||||
float x = player_x;
|
if(window.key('W')) {
|
||||||
float y = player_y;
|
if(empty_space(int(posX + dirX * moveSpeed), int(posY))) posX += dirX * moveSpeed;
|
||||||
|
if(empty_space(int(posX), int(posY + dirY * moveSpeed))) posY += dirY * moveSpeed;
|
||||||
if(window.key('Q')) {
|
} else if(window.key('S')) {
|
||||||
player_angle -= 0.1;
|
if(empty_space(int(posX - dirX * moveSpeed), int(posY))) posX -= dirX * moveSpeed;
|
||||||
} else if(window.key('E')) {
|
if(empty_space(int(posX), int(posY - dirY * moveSpeed))) posY -= dirY * moveSpeed;
|
||||||
player_angle += 0.1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(window.key('W')) {
|
if(window.key('E')) {
|
||||||
x += -1 * std::sin(player_angle) * 5;
|
double oldDirX = dirX;
|
||||||
y += std::cos(player_angle) * 5;
|
dirX = dirX * cos(-rotSpeed) - dirY * sin(-rotSpeed);
|
||||||
} else if(window.key('S')) {
|
dirY = oldDirX * sin(-rotSpeed) + dirY * cos(-rotSpeed);
|
||||||
x -= -1 * std::sin(player_angle) * 5;
|
|
||||||
y -= std::cos(player_angle) * 5;
|
double oldPlaneX = planeX;
|
||||||
|
planeX = planeX * cos(-rotSpeed) - planeY * sin(-rotSpeed);
|
||||||
|
planeY = oldPlaneX * sin(-rotSpeed) + planeY * cos(-rotSpeed);
|
||||||
|
} else if(window.key('Q')) {
|
||||||
|
double oldDirX = dirX;
|
||||||
|
dirX = dirX * cos(rotSpeed) - dirY * sin(rotSpeed);
|
||||||
|
dirY = oldDirX * sin(rotSpeed) + dirY * cos(rotSpeed);
|
||||||
|
|
||||||
|
double oldPlaneX = planeX;
|
||||||
|
planeX = planeX * cos(rotSpeed) - planeY * sin(rotSpeed);
|
||||||
|
planeY = oldPlaneX * sin(rotSpeed) + planeY * cos(rotSpeed);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(window.key('D')) {
|
if(window.key('D')) {
|
||||||
x += -1 * std::sin(player_angle + std::numbers::pi * 0.5) * 5;
|
println("NOT IMPLEMENTED USE E");
|
||||||
y += std::cos(player_angle + std::numbers::pi * 0.5) * 5;
|
|
||||||
} else if(window.key('A')) {
|
} else if(window.key('A')) {
|
||||||
x -= -1 * std::sin(player_angle + std::numbers::pi * 0.5) * 5;
|
println("NOT IMPLEMENTED USE Q");
|
||||||
y -= std::cos(player_angle + std::numbers::pi * 0.5) * 5;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(!collision(x, y)) {
|
|
||||||
player_x = x;
|
|
||||||
player_y = y;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -84,10 +84,10 @@ int main(int /*argc*/, char */*argv*/[])
|
||||||
std::vector<Uint32> texture[8];
|
std::vector<Uint32> texture[8];
|
||||||
for(int i = 0; i < 8; i++) texture[i].resize(texWidth * texHeight);
|
for(int i = 0; i < 8; i++) texture[i].resize(texWidth * texHeight);
|
||||||
|
|
||||||
screen(screenWidth,screenHeight, 0, "Raycaster");
|
screen(screenWidth, screenHeight, 0, "Raycaster");
|
||||||
|
|
||||||
//generate some textures
|
//generate some textures
|
||||||
#if 0
|
#if 1
|
||||||
for(int x = 0; x < texWidth; x++)
|
for(int x = 0; x < texWidth; x++)
|
||||||
for(int y = 0; y < texHeight; y++)
|
for(int y = 0; y < texHeight; y++)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue