1
0
Fork 0
mirror of https://git.coom.tech/drummyfish/small3dlib.git synced 2024-11-21 20:39:57 +01:00
small3dlib/programs/hqOffline.c

361 lines
9.2 KiB
C
Raw Normal View History

2019-06-14 06:41:25 +02:00
#define S3L_RESOLUTION_X 800
#define S3L_RESOLUTION_Y 600
2019-06-13 23:30:04 +02:00
#define S3L_PIXEL_FUNCTION drawPixel
2019-06-14 04:19:38 +02:00
#define S3L_PERSPECTIVE_CORRECTION 1
2019-06-14 01:54:14 +02:00
2019-06-14 16:02:58 +02:00
#define S3L_STRICT_NEAR_CULLING 0
2019-06-13 23:30:04 +02:00
#define S3L_SORT 0
#define S3L_Z_BUFFER 1
#include "../small3dlib.h"
#include <stdio.h>
2019-06-14 01:54:14 +02:00
#include <math.h>
2019-06-13 23:30:04 +02:00
uint8_t frameBuffer[S3L_RESOLUTION_X * S3L_RESOLUTION_Y * 3];
2019-06-15 06:44:43 +02:00
int frame = 0;
2019-06-13 23:30:04 +02:00
#define GRID_W 16
#define GRID_H 16
int8_t heightMap[GRID_W * GRID_H] =
{
2019-06-14 16:16:34 +02:00
#define e -1
e,e,e,e,e,e,e,e,e,e,e,e,e,e,e,e,
e,0,0,0,0,1,0,0,1,1,1,0,0,0,0,e,
e,0,0,0,0,1,0,1,2,1,1,1,0,0,0,e,
e,0,0,1,1,1,1,3,2,1,1,1,1,0,0,e,
e,0,0,0,1,1,2,4,3,2,1,2,1,1,0,e,
e,0,1,2,2,2,2,4,4,2,2,2,2,1,0,e,
e,1,2,2,3,3,6,6,6,3,6,3,5,3,1,e,
e,0,2,2,3,7,8,7,7,6,6,6,6,6,2,e,
e,0,3,3,3,8,8,9,8,7,2,3,6,6,2,e,
e,0,0,2,3,4,7,7,7,6,1,1,4,3,0,e,
e,0,0,1,3,6,3,5,6,6,3,1,2,0,0,e,
e,0,0,0,3,3,3,6,6,6,6,1,0,0,0,e,
e,0,0,1,1,2,3,5,5,5,2,0,0,0,0,e,
e,0,1,2,0,0,2,4,4,2,2,0,0,0,0,e,
e,0,0,0,0,0,1,3,3,0,0,0,0,0,0,e,
e,e,e,e,e,e,e,e,e,e,e,e,e,e,e,e
#undef e
2019-06-13 23:30:04 +02:00
};
#define GRID_TRIANGLES ((GRID_W - 1) * (GRID_H - 1) * 2)
S3L_Unit terrainVertices[GRID_W * GRID_H * 3];
S3L_Unit terrainNormals[GRID_W * GRID_H * 3];
2019-06-14 01:54:14 +02:00
S3L_Unit waterVertices[GRID_W * GRID_H * 3];
S3L_Unit waterNormals[GRID_W * GRID_H * 3];
S3L_Index gridTriangles[GRID_TRIANGLES * 3];
#define MODELS 2
2019-06-13 23:30:04 +02:00
S3L_Model3D models[MODELS];
S3L_Scene scene;
int previousTriangle = -1;
2019-06-14 16:02:58 +02:00
S3L_Vec4 toLightDirection;
2019-06-13 23:30:04 +02:00
2019-06-14 16:02:58 +02:00
S3L_Vec4 n0, n1, n2, v0, v1, v2;
2019-06-13 23:30:04 +02:00
void drawPixel(S3L_PixelInfo *p)
{
2019-06-14 01:54:14 +02:00
S3L_Unit *normals = p->modelIndex == 0 ? terrainNormals : waterNormals;
2019-06-13 23:30:04 +02:00
if (p->triangleIndex != previousTriangle)
{
int index = scene.models[p->modelIndex].triangles[p->triangleIndex * 3] * 3;
2019-06-14 01:54:14 +02:00
n0.x = normals[index];
2019-06-14 16:02:58 +02:00
v0.x = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n0.y = normals[index];
2019-06-14 16:02:58 +02:00
v0.y = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n0.z = normals[index];
2019-06-14 16:02:58 +02:00
v0.z = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index = scene.models[p->modelIndex].triangles[p->triangleIndex * 3 + 1] * 3;
2019-06-14 01:54:14 +02:00
n1.x = normals[index];
2019-06-14 16:02:58 +02:00
v1.x = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n1.y = normals[index];
2019-06-14 16:02:58 +02:00
v1.y = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n1.z = normals[index];
2019-06-14 16:02:58 +02:00
v1.z = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index = scene.models[p->modelIndex].triangles[p->triangleIndex * 3 + 2] * 3;
2019-06-14 01:54:14 +02:00
n2.x = normals[index];
2019-06-14 16:02:58 +02:00
v2.x = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n2.y = normals[index];
2019-06-14 16:02:58 +02:00
v2.y = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
index++;
2019-06-14 01:54:14 +02:00
n2.z = normals[index];
2019-06-14 16:02:58 +02:00
v2.z = scene.models[p->modelIndex].vertices[index];
2019-06-13 23:30:04 +02:00
}
2019-06-15 06:44:43 +02:00
S3L_Vec4 position;
position.x = S3L_interpolateBarycentric(v0.x,v1.x,v2.x,p->barycentric[0],p->barycentric[1],p->barycentric[2]);
position.y = S3L_interpolateBarycentric(v0.y,v1.y,v2.y,p->barycentric[0],p->barycentric[1],p->barycentric[2]);
position.z = S3L_interpolateBarycentric(v0.z,v1.z,v2.z,p->barycentric[0],p->barycentric[1],p->barycentric[2]);
2019-06-13 23:30:04 +02:00
S3L_Vec4 normal;
normal.x = S3L_interpolateBarycentric(n0.x, n1.x, n2.x,
p->barycentric[0], p->barycentric[1], p->barycentric[2]);
normal.y = S3L_interpolateBarycentric(n0.y, n1.y, n2.y,
p->barycentric[0], p->barycentric[1], p->barycentric[2]);
normal.z = S3L_interpolateBarycentric(n0.z, n1.z, n2.z,
p->barycentric[0], p->barycentric[1], p->barycentric[2]);
2019-06-15 06:44:43 +02:00
if (p->modelIndex == MODELS - 1)
{
float dist, dx, dy;
#define wave(x0,y0,f,i)\
dx = position.x - x0 + frame * 20;\
dy = position.z - y0 + frame * 20;\
dist = sqrt(dx * dx + dy * dy);\
normal.x += S3L_sin(dist * f) * i;\
normal.y += S3L_cos(dist * f) * i;
wave(S3L_FRACTIONS_PER_UNIT * 50,S3L_FRACTIONS_PER_UNIT * 60,1,0.1)
wave(S3L_FRACTIONS_PER_UNIT * 20,-S3L_FRACTIONS_PER_UNIT * 30,2,0.05)
wave(-S3L_FRACTIONS_PER_UNIT * 30,S3L_FRACTIONS_PER_UNIT * 45,4,0.05)
#undef wave
/*
float dist = sqrt(position.x * position.x + position.z * position.z);
normal.x += S3L_sin(dist) / 4;
normal.y += S3L_cos(dist) / 4;
*/
/*
normal.x += S3L_sin((position.x + position.z) / 16) / 4 + S3L_sin((position.x + position.z / 2) / 3) / 8;
normal.z += S3L_sin(position.z / 16) / 4 + S3L_sin(position.z / 3) / 8;
*/
}
2019-06-13 23:30:04 +02:00
S3L_normalizeVec3(&normal);
2019-06-14 16:02:58 +02:00
S3L_Vec4 reflected;
S3L_Vec4 toCameraDirection;
2019-06-15 06:44:43 +02:00
toCameraDirection.x = scene.camera.transform.translation.x - position.x;
toCameraDirection.y = scene.camera.transform.translation.y - position.y;
toCameraDirection.z = scene.camera.transform.translation.z - position.z;
2019-06-14 16:02:58 +02:00
S3L_normalizeVec3(&toCameraDirection);
S3L_reflect(toLightDirection,normal,&reflected);
2019-06-13 23:30:04 +02:00
2019-06-14 16:02:58 +02:00
float diffuse = 0.5 - (S3L_dotProductVec3(toLightDirection,normal) / ((float) S3L_FRACTIONS_PER_UNIT)) * 0.5;
float specular = 0.5 + (S3L_dotProductVec3(reflected,toCameraDirection) / ((float) S3L_FRACTIONS_PER_UNIT)) * 0.5;
2019-06-14 16:16:34 +02:00
float fog = (p->depth / ((float) S3L_FRACTIONS_PER_UNIT * 20));
if (fog > 1.0)
fog = 1.0;
2019-06-14 16:02:58 +02:00
2019-06-15 06:44:43 +02:00
float light = 0.3 * fog + 0.6 * diffuse + 0.5 * pow(specular,20.0);
2019-06-13 23:30:04 +02:00
uint8_t color[3];
int index = (p->y * S3L_RESOLUTION_X + p->x) * 3;
2019-06-14 01:54:14 +02:00
if (p->modelIndex == MODELS - 1)
{
2019-06-15 06:44:43 +02:00
S3L_Unit waterDepth = (p->previousZ - p->depth);
2019-06-14 01:54:14 +02:00
2019-06-14 06:41:25 +02:00
float transparency = waterDepth / ((float) (S3L_FRACTIONS_PER_UNIT / 3));
2019-06-14 01:54:14 +02:00
transparency = transparency > 1.0 ? 1.0 : transparency;
float transparency2 = 1.0 - transparency;
uint8_t previousColor[3];
previousColor[0] = frameBuffer[index];
previousColor[1] = frameBuffer[index + 1];
previousColor[2] = frameBuffer[index + 2];
2019-06-15 06:44:43 +02:00
color[0] = 100;
color[1] = 100;
color[2] = 200;
color[0] = S3L_clamp(transparency2 * previousColor[0] + transparency * color[0] * light,0,255);
color[1] = S3L_clamp(transparency2 * previousColor[1] + transparency * color[1] * light,0,255);
color[2] = S3L_clamp(transparency2 * previousColor[2] + transparency * color[2] * light,0,255);
2019-06-14 01:54:14 +02:00
}
else
{
2019-06-14 16:02:58 +02:00
color[0] = S3L_clamp(255 * light,0,255);
color[1] = S3L_clamp(100 * light,0,255);
color[2] = S3L_clamp(50 * light,0,255);
2019-06-14 01:54:14 +02:00
}
2019-06-14 04:19:38 +02:00
/*
2019-06-14 04:08:44 +02:00
color[0] = S3L_clamp(127 + normal.x / 4,0,255);
color[1] = S3L_clamp(127 + normal.y / 4,0,255);
color[2] = S3L_clamp(127 + normal.z / 4,0,255);
2019-06-14 04:19:38 +02:00
*/
2019-06-13 23:30:04 +02:00
frameBuffer[index] = color[0];
frameBuffer[index + 1] = color[1];
frameBuffer[index + 2] = color[2];
}
void createGeometry()
{
int i = 0;
for (int y = 0; y < GRID_H; ++y)
for (int x = 0; x < GRID_W; ++x)
{
terrainVertices[i] = (x - GRID_W / 2) * S3L_FRACTIONS_PER_UNIT;
2019-06-14 16:16:34 +02:00
terrainVertices[i + 1] = (heightMap[i / 3] - 1) * S3L_FRACTIONS_PER_UNIT / 4;
2019-06-13 23:30:04 +02:00
terrainVertices[i + 2] = (y - GRID_H / 2) * S3L_FRACTIONS_PER_UNIT;
2019-06-14 01:54:14 +02:00
2019-06-14 16:02:58 +02:00
waterVertices[i] = terrainVertices[i] * 8;
2019-06-14 01:54:14 +02:00
waterVertices[i + 1] = 0;
2019-06-14 16:02:58 +02:00
waterVertices[i + 2] = terrainVertices[i + 2] * 8;
2019-06-14 01:54:14 +02:00
2019-06-13 23:30:04 +02:00
i += 3;
}
i = 0;
for (int y = 0; y < GRID_H - 1; ++y)
for (int x = 0; x < GRID_W - 1; ++x)
{
S3L_Index indices[4];
indices[0] = y * GRID_W + x;
indices[1] = indices[0] + 1;
indices[2] = indices[0] + GRID_W;
indices[3] = indices[2] + 1;
2019-06-14 01:54:14 +02:00
gridTriangles[i + 0] = indices[0];
gridTriangles[i + 1] = indices[1];
gridTriangles[i + 2] = indices[2];
2019-06-13 23:30:04 +02:00
2019-06-14 01:54:14 +02:00
gridTriangles[i + 3] = indices[2];
gridTriangles[i + 4] = indices[1];
gridTriangles[i + 5] = indices[3];
2019-06-13 23:30:04 +02:00
i += 6;
}
}
2019-06-15 06:44:43 +02:00
void animateWater()
2019-06-14 01:54:14 +02:00
{
for (int i = 1; i < GRID_W * GRID_H * 3; i += 3)
2019-06-15 06:44:43 +02:00
waterVertices[i] = S3L_FRACTIONS_PER_UNIT / 4 + sin(frame * 0.2) * S3L_FRACTIONS_PER_UNIT / 4;
2019-06-14 01:54:14 +02:00
S3L_computeModelNormals(models[MODELS - 1],waterNormals,0);
}
2019-06-13 23:30:04 +02:00
void clearFrameBuffer()
{
2019-06-14 16:16:34 +02:00
memset(frameBuffer,255,S3L_RESOLUTION_X * S3L_RESOLUTION_Y * 3 * sizeof(uint8_t));
2019-06-13 23:30:04 +02:00
}
void saveImage(char *fileName)
{
2019-06-14 06:41:25 +02:00
printf("saving image file: %s\n",fileName);
2019-06-13 23:30:04 +02:00
FILE *f = fopen(fileName,"w");
fprintf(f,"P3\n%d %d\n255\n",S3L_RESOLUTION_X,S3L_RESOLUTION_Y);
for (int i = 0; i < S3L_RESOLUTION_X * S3L_RESOLUTION_Y * 3; i += 3)
fprintf(f,"%d %d %d\n",frameBuffer[i],frameBuffer[i + 1],frameBuffer[i + 2]);
fclose(f);
}
int main()
{
createGeometry();
2019-06-14 16:02:58 +02:00
toLightDirection.x = 10;
toLightDirection.y = 10;
toLightDirection.z = 10;
toLightDirection.w = 0;
2019-06-13 23:30:04 +02:00
2019-06-14 16:02:58 +02:00
S3L_normalizeVec3(&toLightDirection);
2019-06-13 23:30:04 +02:00
S3L_initModel3D(
terrainVertices,
GRID_W * GRID_H,
2019-06-14 01:54:14 +02:00
gridTriangles,
2019-06-13 23:30:04 +02:00
GRID_TRIANGLES,
2019-06-14 01:54:14 +02:00
&(models[0]));
2019-06-13 23:30:04 +02:00
2019-06-14 01:54:14 +02:00
S3L_computeModelNormals(models[0],terrainNormals,0);
S3L_initModel3D(
waterVertices,
GRID_W * GRID_H,
gridTriangles,
GRID_TRIANGLES,
&(models[MODELS - 1]));
2019-06-13 23:30:04 +02:00
2019-06-14 04:19:38 +02:00
S3L_initScene(models,MODELS,&scene);
2019-06-13 23:30:04 +02:00
2019-06-14 06:41:25 +02:00
char fileName[] = "test00.ppm";
for (int i = 0; i < 20; ++i)
{
2019-06-15 06:44:43 +02:00
animateWater();
scene.camera.transform.translation.x = -i * S3L_FRACTIONS_PER_UNIT / 4;
2019-06-14 06:41:25 +02:00
scene.camera.transform.translation.y = 8 * S3L_FRACTIONS_PER_UNIT;
2019-06-14 16:02:58 +02:00
scene.camera.transform.translation.z = -10 * S3L_FRACTIONS_PER_UNIT + i * S3L_FRACTIONS_PER_UNIT / 4;
2019-06-14 06:41:25 +02:00
S3L_Vec4 target;
target.x = 0;
target.y = 0;
target.z = 0;
2019-06-13 23:30:04 +02:00
2019-06-14 06:41:25 +02:00
S3L_lookAt(scene.camera.transform.translation,target,&scene.camera.transform);
2019-06-13 23:30:04 +02:00
2019-06-14 06:41:25 +02:00
clearFrameBuffer();
2019-06-13 23:30:04 +02:00
2019-06-14 06:41:25 +02:00
S3L_newFrame();
2019-06-13 23:30:04 +02:00
2019-06-14 06:41:25 +02:00
S3L_drawScene(scene);
fileName[4] = '0' + (i / 10);
fileName[5] = '0' + (i % 10);
saveImage(fileName);
2019-06-15 06:44:43 +02:00
frame++;
2019-06-14 06:41:25 +02:00
}
2019-06-13 23:30:04 +02:00
return 0;
}