2018-08-31 18:40:05 +02:00
|
|
|
/**
|
|
|
|
WIP raycasting demo for Pokitto.
|
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
Don't forget to compile with -O3!
|
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
author: Miloslav "drummyfish" Ciz
|
|
|
|
license: CC0
|
|
|
|
*/
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
//#define RAYCAST_TINY
|
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include "raycastlib.h"
|
|
|
|
#include "Pokitto.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
#define SUBSAMPLE 3
|
|
|
|
|
|
|
|
// r: 3 bits, g: 3 bits, b: 2 bits
|
|
|
|
inline uint8_t rgbToIndex(uint8_t r, uint8_t g, uint8_t b)
|
|
|
|
{
|
|
|
|
return (r & 0b00000111) | ((g & 0b00000111) << 3) | ((b & 0b00000011) << 6);
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
class Level
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int16_t getHeight(int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
if (x > 12 || y > 12)
|
2018-08-31 19:12:47 +02:00
|
|
|
return max(x,y) - 10;
|
2018-08-31 18:40:05 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
if (y < 5 && y > 0)
|
|
|
|
{
|
|
|
|
return y > 2 ? x : -x;
|
|
|
|
}
|
2018-09-01 12:04:17 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
return (x < 0 || y < 0 || x > 9 || y > 9) ? 4 : 0;
|
2018-08-31 18:40:05 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class Character
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Camera mCamera;
|
|
|
|
|
|
|
|
Character()
|
|
|
|
{
|
2018-09-03 10:56:48 +02:00
|
|
|
mCamera.position.x = 7695; // UNITS_PER_SQUARE * 4;
|
|
|
|
mCamera.position.y = 7697; //UNITS_PER_SQUARE * 5;
|
|
|
|
mCamera.direction = -500; //0;
|
2018-08-31 18:40:05 +02:00
|
|
|
mCamera.fovAngle = UNITS_PER_SQUARE / 4;
|
2018-09-03 10:56:48 +02:00
|
|
|
mCamera.height = 2756; //UNITS_PER_SQUARE / 2;
|
|
|
|
mCamera.resolution.x = 110 / SUBSAMPLE;
|
2018-08-31 18:40:05 +02:00
|
|
|
mCamera.resolution.y = 88;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Pokitto::Core p;
|
|
|
|
Character player;
|
|
|
|
Level level;
|
|
|
|
|
2018-09-01 09:07:27 +02:00
|
|
|
Unit heightFunc(int16_t x, int16_t y)
|
2018-08-31 18:40:05 +02:00
|
|
|
{
|
2018-09-02 19:43:54 +02:00
|
|
|
return level.getHeight(x,y) * UNITS_PER_SQUARE / 4;
|
2018-08-31 18:40:05 +02:00
|
|
|
}
|
|
|
|
|
2018-09-01 12:04:17 +02:00
|
|
|
bool dither(uint8_t intensity, uint32_t x, uint32_t y)
|
|
|
|
{
|
|
|
|
switch (intensity)
|
|
|
|
{
|
|
|
|
case 0: return false; break;
|
|
|
|
case 1: return x % 2 == 0 && y % 2 == 0; break;
|
|
|
|
case 2: return x % 2 == y % 2; break;
|
|
|
|
case 3: return x % 2 != 0 || y % 2 != 0; break;
|
|
|
|
default: return true; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
inline void pixelFunc(PixelInfo pixel)
|
2018-08-31 18:40:05 +02:00
|
|
|
{
|
2018-09-03 10:56:48 +02:00
|
|
|
|
|
|
|
/*
|
2018-09-02 19:43:54 +02:00
|
|
|
uint8_t c = 3 + (pixel.isWall == 0 ? 12 : (pixel.hit.direction * 3));
|
2018-09-01 12:04:17 +02:00
|
|
|
|
2018-09-02 20:25:17 +02:00
|
|
|
Unit depth = pixel.depth - UNITS_PER_SQUARE;
|
|
|
|
|
|
|
|
if (depth < 0)
|
|
|
|
depth = 0;
|
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
int16_t d = depth / (UNITS_PER_SQUARE * 2);
|
2018-09-02 20:25:17 +02:00
|
|
|
|
|
|
|
if (d < 0)
|
|
|
|
d = 0;
|
2018-09-01 12:04:17 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
if (d <= 2)
|
2018-09-02 20:25:17 +02:00
|
|
|
c -= d;
|
2018-09-02 19:43:54 +02:00
|
|
|
else
|
|
|
|
c = 0;
|
2018-09-03 10:56:48 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
Unit depth = pixel.depth - UNITS_PER_SQUARE;
|
|
|
|
|
|
|
|
if (depth < 0)
|
|
|
|
depth = 0;
|
|
|
|
|
|
|
|
int intensity = 7 - (depth * 7) / (UNITS_PER_SQUARE * 5);
|
|
|
|
|
|
|
|
if (intensity < 0)
|
|
|
|
intensity = 0;
|
|
|
|
|
|
|
|
uint8_t c;
|
|
|
|
|
|
|
|
if (pixel.isWall)
|
|
|
|
{
|
|
|
|
switch (pixel.hit.direction)
|
|
|
|
{
|
|
|
|
case 0: c = rgbToIndex(intensity,0,0); break;
|
|
|
|
case 1: c = rgbToIndex(0,intensity,0); break;
|
|
|
|
case 2: c = rgbToIndex(0,0,intensity / 2); break;
|
|
|
|
case 3: c = rgbToIndex(intensity,intensity,0); break;
|
|
|
|
default: c = 0; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
c = rgbToIndex(intensity,intensity,intensity / 2);
|
2018-09-01 12:04:17 +02:00
|
|
|
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
uint8_t *buf = p.display.screenbuffer;
|
2018-09-01 12:04:17 +02:00
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
buf += pixel.position.x * SUBSAMPLE;
|
2018-09-02 19:43:54 +02:00
|
|
|
buf += pixel.position.y * p.display.width;
|
2018-09-03 10:56:48 +02:00
|
|
|
|
|
|
|
for (uint8_t i = 0; i < SUBSAMPLE - 1; ++i)
|
|
|
|
*buf++ = c;
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
*buf = c;
|
2018-09-03 10:56:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
//for (uint8_t i = 0; i < SUBSAMPLE; ++i)
|
|
|
|
//p.display.drawPixel(pixel.position.x * SUBSAMPLE + i,pixel.position.y,c);
|
2018-08-31 18:40:05 +02:00
|
|
|
}
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
unsigned short pal[256];
|
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
void draw()
|
|
|
|
{
|
2018-09-03 10:56:48 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
/*
|
|
|
|
uint8_t a = 0;
|
|
|
|
for (uint8_t j = 0; j < 88; ++j)
|
|
|
|
for (uint8_t i = 0; i < 110; ++i)
|
|
|
|
{
|
|
|
|
p.display.drawPixel(i,j,a);
|
|
|
|
++a;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
*/
|
2018-09-03 10:56:48 +02:00
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
RayConstraints c;
|
|
|
|
|
2018-09-02 20:25:17 +02:00
|
|
|
c.maxHits = 16;
|
|
|
|
c.maxSteps = 16;
|
2018-08-31 18:40:05 +02:00
|
|
|
|
|
|
|
render(player.mCamera,heightFunc,pixelFunc,c);
|
2018-09-02 23:18:26 +02:00
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
/*
|
2018-09-02 23:18:26 +02:00
|
|
|
p.display.setCursor(1,1);
|
|
|
|
p.display.print(player.mCamera.position.x);
|
|
|
|
p.display.print(' ');
|
2018-09-03 10:56:48 +02:00
|
|
|
p.display.print(player.mCamera.position.y);
|
|
|
|
p.display.print(' ');
|
|
|
|
p.display.print(player.mCamera.height);
|
2018-09-02 23:18:26 +02:00
|
|
|
p.display.print(' ');
|
|
|
|
p.display.print(player.mCamera.direction);
|
2018-09-03 10:56:48 +02:00
|
|
|
*/
|
2018-08-31 18:40:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
p.begin();
|
2018-09-02 19:43:54 +02:00
|
|
|
|
2018-09-02 20:25:17 +02:00
|
|
|
/*
|
2018-09-02 19:43:54 +02:00
|
|
|
int a,s,r,g,b;
|
|
|
|
for(a=0; a<=63; a++){
|
|
|
|
s = 0; r = a; g = 63-a; b = 0; pal[a+s] = p.display.RGBto565(r*4,g*4,b*4);
|
|
|
|
s = 64; r = 63-a; g = 0; b = a; pal[a+s] = p.display.RGBto565(r*4,g*4,b*4);
|
|
|
|
s = 128; r = 0; g = 0; b = 63-a; pal[a+s] = p.display.RGBto565(r*4,g*4,b*4);
|
|
|
|
s = 192; r = 0; g = a; b = 0; pal[a+s] = p.display.RGBto565(r*4,g*4,b*4);
|
|
|
|
}
|
2018-09-03 10:56:48 +02:00
|
|
|
*/
|
2018-09-02 19:43:54 +02:00
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
for (uint8_t r = 0; r < 8; ++r)
|
|
|
|
for (uint8_t g = 0; g < 8; ++g)
|
|
|
|
for (uint8_t b = 0; b < 4; ++b)
|
|
|
|
{
|
|
|
|
pal[rgbToIndex(r,g,b)] = p.display.RGBto565(36 * r, 36 * g, 85 * b);
|
|
|
|
}
|
2018-09-02 20:25:17 +02:00
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
/*
|
2018-09-02 19:43:54 +02:00
|
|
|
for (int i = 0; i < 16; ++i)
|
2018-09-02 20:25:17 +02:00
|
|
|
pal[i] = p.display.RGBto565(255-i,255-i,255-i);
|
2018-09-02 19:43:54 +02:00
|
|
|
|
|
|
|
pal[0] = p.display.RGBto565(0,0,0);
|
|
|
|
pal[1] = p.display.RGBto565(64,0,0);
|
|
|
|
pal[2] = p.display.RGBto565(128,0,0);
|
|
|
|
pal[3] = p.display.RGBto565(192,0,0);
|
|
|
|
pal[4] = p.display.RGBto565(0,64,0);
|
|
|
|
pal[5] = p.display.RGBto565(0,128,0);
|
|
|
|
pal[6] = p.display.RGBto565(0,192,0);
|
|
|
|
pal[7] = p.display.RGBto565(0,0,64);
|
|
|
|
pal[8] = p.display.RGBto565(0,0,128);
|
|
|
|
pal[9] = p.display.RGBto565(0,0,192);
|
|
|
|
pal[10] = p.display.RGBto565(64,64,0);
|
|
|
|
pal[11] = p.display.RGBto565(128,128,0);
|
|
|
|
pal[12] = p.display.RGBto565(192,192,0);
|
|
|
|
pal[13] = p.display.RGBto565(64,64,64);
|
|
|
|
pal[14] = p.display.RGBto565(128,128,128);
|
|
|
|
pal[15] = p.display.RGBto565(192,192,192);
|
2018-09-03 10:56:48 +02:00
|
|
|
*/
|
2018-09-02 20:25:17 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
p.display.load565Palette(&pal[0]); // load a palette the same way as any other palette in any other screen mode
|
|
|
|
|
2018-09-03 10:56:48 +02:00
|
|
|
p.display.persistence = 1;
|
|
|
|
|
2018-09-02 20:25:17 +02:00
|
|
|
p.setFrameRate(60);
|
2018-08-31 18:40:05 +02:00
|
|
|
|
|
|
|
p.display.setFont(fontTiny);
|
|
|
|
|
|
|
|
while (p.isRunning())
|
|
|
|
{
|
|
|
|
if (p.update())
|
|
|
|
{
|
|
|
|
draw();
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
const int16_t step = max(UNITS_PER_SQUARE / 10,1);
|
|
|
|
const int16_t step2 = max(UNITS_PER_SQUARE / 50,1);
|
2018-08-31 18:40:05 +02:00
|
|
|
|
|
|
|
Vector2D d = angleToDirection(player.mCamera.direction);
|
|
|
|
|
|
|
|
d.x = (d.x * step) / UNITS_PER_SQUARE;
|
|
|
|
d.y = (d.y * step) / UNITS_PER_SQUARE;
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
if (d.x == 0 && d.y == 0)
|
|
|
|
{
|
|
|
|
d.x = d.x > 0 ? step : -step;
|
|
|
|
d.y = d.y > 0 ? step : -step;
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:40:05 +02:00
|
|
|
if (p.upBtn())
|
|
|
|
{
|
|
|
|
player.mCamera.position.x += d.x;
|
|
|
|
player.mCamera.position.y += d.y;
|
|
|
|
}
|
|
|
|
else if (p.downBtn())
|
|
|
|
{
|
|
|
|
player.mCamera.position.x -= d.x;
|
|
|
|
player.mCamera.position.y -= d.y;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p.rightBtn())
|
|
|
|
player.mCamera.direction += step2;
|
|
|
|
else if (p.leftBtn())
|
|
|
|
player.mCamera.direction -= step2;
|
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
if (p.aBtn())
|
2018-09-01 09:07:27 +02:00
|
|
|
player.mCamera.height += step;
|
2018-09-02 19:43:54 +02:00
|
|
|
else if (p.bBtn())
|
2018-09-01 09:07:27 +02:00
|
|
|
player.mCamera.height -= step;
|
2018-08-31 19:01:14 +02:00
|
|
|
|
2018-09-02 19:43:54 +02:00
|
|
|
player.mCamera.height =
|
|
|
|
max(heightFunc(player.mCamera.position.x / UNITS_PER_SQUARE,player.mCamera.position.y / UNITS_PER_SQUARE) + UNITS_PER_SQUARE / 2,player.mCamera.height);
|
2018-08-31 18:40:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|