Files
dorayme/source/canvas.cpp
2020-03-06 21:55:32 +00:00

154 lines
3.9 KiB
C++

/*
* DoRayMe - a quick and dirty Raytracer
* Canvas implementation
*
* Created by Manoël Trapier
* Copyright (c) 2020 986-Studio.
*
*/
#include <canvas.h>
#include <lodepng.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* NanoJPEG is a bit interesting to include as a header */
extern "C" {
#define _NJ_INCLUDE_HEADER_ONLY
#include <nanojpeg.c>
#undef _NJ_INCLUDE_HEADER_ONLY
}
#define BPP (24)
#define BytePP (BPP / 8)
#define MIN(_a, _b) ((_a)<(_b)?(_a):(_b))
#define MAX(_a, _b) ((_a)>(_b)?(_a):(_b))
Canvas::Canvas(uint32_t width, uint32_t height) : width(width), height(height)
{
this->bitmap = (uint8_t *)calloc(4, width * height);
this->stride = BytePP * width;
}
Canvas::Canvas(const Canvas &b)
{
this->width = b.width;
this->height = b.height;
this->stride = b.stride;
this->bitmap = (uint8_t *)calloc(4, b.width * b.height);
memcpy(this->bitmap, b.bitmap, 4 * b.width * b.height);
}
Canvas::Canvas(const Canvas *b)
{
this->width = b->width;
this->height = b->height;
this->stride = b->stride;
this->bitmap = (uint8_t *)calloc(4, b->width * b->height);
memcpy(this->bitmap, b->bitmap, 4 * b->width * b->height);
}
Canvas::Canvas(const char *imgfile)
{
/* Try to determine the file type in a really lazy way */
const char *fileExt = strrchr(imgfile, '.');
this->bitmap = NULL;
if (fileExt == NULL)
{
printf("ERROR: Invalid file name '%s' - Can't determine the file format\n", imgfile);
}
else if (strncasecmp(fileExt, ".png", strlen(fileExt)) == 0)
{
uint32_t ret = lodepng_decode24_file(&this->bitmap, &this->width, &this->height, imgfile);
if (ret)
{
printf("ERROR: %s\n", lodepng_error_text(ret));
}
}
else if ( (strncasecmp(fileExt, ".jpg", strlen(fileExt)) == 0) || (strncasecmp(fileExt, ".jpeg", strlen(fileExt)) == 0) )
{
FILE *fp;
char *fileBuff;
size_t fileSize;
fp = fopen(imgfile, "rb");
if (fp)
{
fseek(fp, 0, SEEK_END);
fileSize = ftell(fp);
fileBuff = (char *)calloc(fileSize, 1);
fseek(fp, 0, SEEK_SET);
fileSize = fread(fileBuff, 1, fileSize, fp);
fclose(fp);
njInit();
if (!njDecode(fileBuff, fileSize))
{
this->width = njGetWidth();
this->height = njGetHeight();
/* Need to do a local copy */
this->bitmap = (uint8_t *)calloc(1, njGetImageSize());
memcpy(this->bitmap, njGetImage(), njGetImageSize());
}
else
{
printf("ERROR: Error while decoding the file '%s'\n", imgfile);
}
free(fileBuff);
njDone();
}
else
{
printf("ERROR: Can't open/find the file '%s'.\n", imgfile);
}
}
else
{
printf("ERROR: File extention '%s' is not a recognized one.\n", fileExt);
}
if (this->bitmap == NULL)
{
printf("ABORT: Opening image file '%s' failed\n", imgfile);
exit(-1);
}
this->stride = BytePP * width;
}
Canvas::~Canvas()
{
if (this->bitmap != nullptr)
{
free(this->bitmap);
}
}
void Canvas::putPixel(uint32_t x, uint32_t y, Tuple c)
{
uint32_t offset = y * this->stride + x * BytePP;
this->bitmap[offset + 0] = MAX(MIN(c.x * 255, 255), 0);
this->bitmap[offset + 1] = MAX(MIN(c.y * 255, 255), 0);
this->bitmap[offset + 2] = MAX(MIN(c.z * 255, 255), 0);
}
Colour Canvas::getPixel(uint32_t x, uint32_t y)
{
uint32_t offset = y * this->stride + x * BytePP;
return Colour(this->bitmap[offset + 0] / 255.0, this->bitmap[offset + 1] / 255.0, this->bitmap[offset + 2] / 255.0);
}
bool Canvas::SaveAsPNG(const char *filename)
{
uint32_t ret = lodepng_encode24_file(filename, this->bitmap, this->width, this->height);
return ret == 0;
}