BmpPP/Src/BMP.cpp

177 lines
4.1 KiB
C++
Raw Normal View History

2022-03-05 16:15:00 +01:00
#include "BMP.h"
#include <iostream>
#include <fstream>
#include <stdexcept>
2022-03-05 20:40:49 +01:00
#include "BmpWriter.h"
2022-03-05 16:15:00 +01:00
#define CHECK_IF_INITIALIZED if (!isInitialized) throw std::runtime_error("Image not initialized!");
2022-03-05 16:15:00 +01:00
namespace Leonetienne::BmpPP {
BMP::BMP() {
// Do nothing
return;
}
2022-03-05 16:15:00 +01:00
BMP::BMP(const Eule::Vector2i &size, const Colormode& colormode)
:
size { size },
colormode { colormode }
2022-03-05 16:15:00 +01:00
{
2022-03-05 19:51:57 +01:00
ReInitialize(size, colormode);
2022-03-05 16:15:00 +01:00
2022-03-05 19:51:57 +01:00
return;
}
void BMP::ReInitialize(const Eule::Vector2i &size) {
isInitialized = true;
2022-03-05 19:51:57 +01:00
// Carry over new attributes
this->size = size;
// Re-initialize the pixelbuffer
pixelBuffer.clear();
pixelBuffer.resize(size.x * size.y * GetNumColorChannels());
return;
}
void BMP::ReInitialize(const Eule::Vector2i &size, const Colormode &colormode) {
isInitialized = true;
2022-03-05 19:51:57 +01:00
// Carry over new attributes
this->size = size;
this->colormode = colormode;
// Re-initialize the pixelbuffer
2022-03-05 16:15:00 +01:00
pixelBuffer.clear();
2022-03-05 19:37:02 +01:00
pixelBuffer.resize(size.x * size.y * GetNumColorChannels());
2022-03-05 16:15:00 +01:00
return;
}
bool BMP::Write(const std::string &filename) const {
CHECK_IF_INITIALIZED
2022-03-05 20:40:49 +01:00
return BmpWriter::Write(*this, filename);
}
2022-03-05 20:40:49 +01:00
bool BMP::Read(const std::string &filename) {
2022-03-05 20:40:49 +01:00
std::ifstream ifs(filename, std::ifstream::binary);
if (!ifs.good())
return false;
2022-03-05 20:40:49 +01:00
// When reading the image, we don't care about most of the header.
// All we want is the images resolutions, bits-per-pixel, and pixelbuffer address...
// This is because we do not support ALL of bmp, but just the most common formats...
return true;
}
std::uint8_t *BMP::GetPixel(const Eule::Vector2i &position) {
CHECK_IF_INITIALIZED
const std::size_t pixelIndex =
2022-03-05 19:37:02 +01:00
(position.y * size.x + position.x) * GetNumColorChannels();
if (pixelIndex >= pixelBuffer.size())
throw std::runtime_error("Pixel index out of range!");
return pixelBuffer.data() + pixelIndex;
}
const std::uint8_t *BMP::GetPixel(const Eule::Vector2i &position) const {
CHECK_IF_INITIALIZED
const std::size_t pixelIndex =
2022-03-05 19:37:02 +01:00
(position.y * size.x + position.x) * GetNumColorChannels();
if (pixelIndex >= pixelBuffer.size())
throw std::runtime_error("Pixel index out of range!");
return pixelBuffer.data() + pixelIndex;
}
void BMP::SetPixel(const Eule::Vector2i &position,
const std::uint8_t r,
const std::uint8_t g,
const std::uint8_t b,
const std::uint8_t a)
{
CHECK_IF_INITIALIZED
std::uint8_t* pixel = GetPixel(position);
switch (colormode) {
case Colormode::RGBA:
pixel[0] = r;
pixel[1] = g;
pixel[2] = b;
pixel[3] = a;
break;
case Colormode::RGB:
pixel[0] = r;
pixel[1] = g;
pixel[2] = b;
break;
}
return;
2022-03-05 16:15:00 +01:00
}
2022-03-05 19:37:02 +01:00
std::uint8_t *BMP::data() {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
return pixelBuffer.data();
}
const std::uint8_t *BMP::data() const {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
return pixelBuffer.data();
}
const Eule::Vector2i &BMP::GetDimensions() const {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
return size;
}
const Colormode &BMP::GetColormode() const {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
return colormode;
}
std::size_t BMP::GetNumColorChannels() const {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
switch (colormode) {
case Colormode::RGB:
return 3;
case Colormode::RGBA:
return 4;
}
// Unreachable
return -1;
}
std::size_t BMP::GetPixelbufferSize() const {
CHECK_IF_INITIALIZED
2022-03-05 19:37:02 +01:00
return pixelBuffer.size();
}
bool BMP::IsInitialized() const {
return isInitialized;
}
2022-03-05 16:15:00 +01:00
}
#undef CHECK_IF_INITIALIZED