From 7c83d9c93e8fb220fedbeb23cb6fe0da13995b3d Mon Sep 17 00:00:00 2001 From: Evgenij Titarenko Date: Fri, 19 Apr 2024 22:25:12 +0300 Subject: [PATCH] =?UTF-8?q?=D0=92=D1=8B=D0=B4=D0=B5=D0=BB=D0=B8=D0=BB=20?= =?UTF-8?q?=D0=BF=D0=B5=D1=80=D0=B5=D0=B8=D1=81=D0=BF=D0=BE=D0=BB=D1=8C?= =?UTF-8?q?=D0=B7=D1=83=D0=B5=D0=BC=D1=8B=D0=B5=20=D0=BC=D0=BE=D0=B4=D1=83?= =?UTF-8?q?=D0=BB=D0=B8=20=D0=B8=D0=B7=20=D0=BF=D1=80=D0=BE=D0=B5=D0=BA?= =?UTF-8?q?=D1=82=D0=B0.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- CMakeLists.txt | 17 ++-- font-lib/CMakeLists.txt | 18 ++++ psf.cpp => font-lib/psf.cpp | 13 +-- psf.h => font-lib/psf.h | 4 +- image-lib/CMakeLists.txt | 28 ++++++ bmpimage.cpp => image-lib/bmpimage.cpp | 120 +------------------------ bmpimage.h => image-lib/bmpimage.h | 67 +------------- image-lib/filters.cpp | 38 ++++++++ image-lib/filters.h | 22 +++++ image-lib/pixelarray.cpp | 77 ++++++++++++++++ image-lib/pixelarray.h | 43 +++++++++ main.cpp | 14 +-- utils-lib/CMakeLists.txt | 18 ++++ utils-lib/utils.cpp | 11 +++ utils-lib/utils.h | 4 + 15 files changed, 287 insertions(+), 207 deletions(-) create mode 100644 font-lib/CMakeLists.txt rename psf.cpp => font-lib/psf.cpp (91%) rename psf.h => font-lib/psf.h (92%) create mode 100644 image-lib/CMakeLists.txt rename bmpimage.cpp => image-lib/bmpimage.cpp (77%) rename bmpimage.h => image-lib/bmpimage.h (55%) create mode 100644 image-lib/filters.cpp create mode 100644 image-lib/filters.h create mode 100644 image-lib/pixelarray.cpp create mode 100644 image-lib/pixelarray.h create mode 100644 utils-lib/CMakeLists.txt create mode 100644 utils-lib/utils.cpp create mode 100644 utils-lib/utils.h diff --git a/CMakeLists.txt b/CMakeLists.txt index 8d7af0d..c155afe 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -3,8 +3,15 @@ project(image_test_cpp) set(CMAKE_CXX_STANDARD 17) -add_executable(image_test_cpp main.cpp - bmpimage.h - bmpimage.cpp - psf.cpp - psf.h) +add_subdirectory(font-lib) +add_subdirectory(image-lib) +add_subdirectory(utils-lib) + +add_executable(image_test_cpp main.cpp) + +target_link_libraries(image_test_cpp + PRIVATE + font-lib + image-lib + utils-lib +) diff --git a/font-lib/CMakeLists.txt b/font-lib/CMakeLists.txt new file mode 100644 index 0000000..9da4431 --- /dev/null +++ b/font-lib/CMakeLists.txt @@ -0,0 +1,18 @@ +add_library(font-lib STATIC "") + +set(FONT_LIB_HEADERS + psf.h +) + +set(FONT_LIB_SOURCES + psf.cpp +) + +target_sources(font-lib + PRIVATE + ${FONT_LIB_SOURCES} + PUBLIC + FILE_SET HEADERS + BASE_DIRS ${PROJECT_SOURCE_DIR} + FILES ${FONT_LIB_HEADERS} +) \ No newline at end of file diff --git a/psf.cpp b/font-lib/psf.cpp similarity index 91% rename from psf.cpp rename to font-lib/psf.cpp index 9377be2..018fce8 100644 --- a/psf.cpp +++ b/font-lib/psf.cpp @@ -1,7 +1,3 @@ -// -// Created by Evgenij on 09.04.2024. -// - #include #include #include @@ -25,15 +21,15 @@ Font readPSF(const std::string &filename) { throw std::runtime_error("Invalid font file"); ifs.read((char *) &psf2Header, sizeof(psf2Header)); auto font = Font(psf2Header.numberOfGlyphs, psf2Header.glyphWidth, psf2Header.glyphHeight); - Glyph glyphs[psf2Header.numberOfGlyphs]; + std::vector> glyphs(psf2Header.numberOfGlyphs); for (int i = 0; i < psf2Header.numberOfGlyphs; ++i) { - auto glyph = Glyph(psf2Header.glyphWidth, psf2Header.glyphHeight); + auto glyph = std::make_shared(psf2Header.glyphWidth, psf2Header.glyphHeight); char glyphBytes[psf2Header.bytesPerGlyph]; ifs.read((char *) &glyphBytes, psf2Header.bytesPerGlyph); for (int j = 0; j < psf2Header.bytesPerGlyph; ++j) { for (int k = 0; k < 8; ++k) { uint8_t bit = (glyphBytes[j] & (1 << k)) != 0; // Получить к-ый бит - glyph.glyph[(j * 8 + k) / psf2Header.glyphWidth][psf2Header.glyphWidth - 1 - + glyph->glyph[(j * 8 + k) / psf2Header.glyphWidth][psf2Header.glyphWidth - 1 - (j * 8 + k) % psf2Header.glyphWidth] = bit; } } @@ -79,9 +75,6 @@ Glyph::Glyph(uint32_t width, uint32_t height) { this->height = height; } -Glyph::Glyph() { - -} Font::Font(uint32_t glyphsCount, uint32_t glyphWidth, uint32_t glyphHeight) { this->glyphsCount = glyphsCount; diff --git a/psf.h b/font-lib/psf.h similarity index 92% rename from psf.h rename to font-lib/psf.h index f27dbe3..b709d00 100644 --- a/psf.h +++ b/font-lib/psf.h @@ -4,6 +4,7 @@ #include #include #include +#include #pragma pack(push, 1) struct PSF1Header { @@ -34,12 +35,11 @@ public: Glyph(uint32_t width, uint32_t height); - Glyph(); }; class Font { public: - std::map _glyphs; + std::map> _glyphs; uint32_t glyphWidth; uint32_t glyphHeight; uint32_t glyphsCount; diff --git a/image-lib/CMakeLists.txt b/image-lib/CMakeLists.txt new file mode 100644 index 0000000..979d380 --- /dev/null +++ b/image-lib/CMakeLists.txt @@ -0,0 +1,28 @@ +add_library(image-lib STATIC "" + pixelarray.cpp + pixelarray.h + filters.cpp + filters.h) + +set(IMAGE_LIB_HEADERS + bmpimage.h +) + +set(IMAGE_LIB_SOURCES + bmpimage.cpp +) + +target_sources(image-lib + PRIVATE + ${IMAGE_LIB_SOURCES} + PUBLIC + FILE_SET HEADERS + BASE_DIRS ${PROJECT_SOURCE_DIR} + FILES ${IMAGE_LIB_HEADERS} +) + +target_link_libraries(image-lib + PUBLIC + font-lib + utils-lib +) \ No newline at end of file diff --git a/bmpimage.cpp b/image-lib/bmpimage.cpp similarity index 77% rename from bmpimage.cpp rename to image-lib/bmpimage.cpp index 8e3c24e..eb3c7d4 100644 --- a/bmpimage.cpp +++ b/image-lib/bmpimage.cpp @@ -178,19 +178,6 @@ BMPImage readBMPImage(const std::string &filename) { return {bitmapFileHeader, bitmapInfoHeader, pixelArray}; } -Pixel operator+(const Pixel &p1, const Pixel &p2) { - const uint8_t r = ui8_clamp((int) p1.r + p2.r); - const uint8_t g = ui8_clamp((int) p1.g + p2.g); - const uint8_t b = ui8_clamp((int) p1.b + p2.b); - return {r, g, b}; -} - -Pixel operator/(const Pixel &p1, const uint8_t &n) { - const uint8_t r = p1.r / n; - const uint8_t g = p1.g / n; - const uint8_t b = p1.b / n; - return {r, g, b}; -} BMPImage grayscale(BMPImage &img) { auto pixels = img.pixels_copy(); @@ -203,22 +190,6 @@ BMPImage grayscale(BMPImage &img) { return {img.fileHeader_copy(), img.infoHeader_copy(), pixels}; } -Pixel operator*(const Pixel &p, const int &n) { - uint8_t r = ui8_clamp(n * p.r); - uint8_t g = ui8_clamp(n * p.g); - uint8_t b = ui8_clamp(n * p.b); - return {r, g, b}; -} - -uint8_t ui8_clamp(int value, uint8_t min, uint8_t max) { - if (value < min) { - return min; - } - if (value > max) { - return max; - } - return value; -} BMPImage invertColors(BMPImage &img) { auto pixels = img.pixels_copy(); @@ -230,20 +201,6 @@ BMPImage invertColors(BMPImage &img) { return {img.fileHeader_copy(), img.infoHeader_copy(), pixels}; } -Pixel operator-(const uint8_t &n, const Pixel &p) { - uint8_t r = ui8_clamp(n - p.r); - uint8_t g = ui8_clamp(n - p.g); - uint8_t b = ui8_clamp(n - p.b); - return {r, g, b}; -} - -Pixel operator-(const Pixel &p, const uint8_t &n) { - uint8_t r = ui8_clamp(p.r - n); - uint8_t g = ui8_clamp(p.g - n); - uint8_t b = ui8_clamp(p.b - n); - return {r, g, b}; -} - BMPImage upscale2x(BMPImage &img) { auto oldPixels = img.pixels(); const uint32_t newHeight = img.height() * 2; @@ -327,7 +284,7 @@ BMPImage textImg(const std::u16string &str, Font *font, uint8_t scale, Pixel bac auto glyph = font->_glyphs[str[i]]; for (int j = 0; j < glyphHeight * scale; ++j) { for (int l = 0; l < glyphWidth * scale; ++l) { - if (glyph.glyph[j / scale][l / scale]) + if (glyph->glyph[j / scale][l / scale]) pixels(j, glyphWidth * scale * i + l) = font_color; else pixels(j, glyphWidth * scale * i + l) = background_color; @@ -384,13 +341,6 @@ BMPImage upscale1_5x_ver2(BMPImage &img) { return {newPixelArray}; } -Pixel operator-(const Pixel &p1, const Pixel &p2) { - auto r = ui8_clamp((int) p1.r - p2.r); - auto g = ui8_clamp((int) p1.g - p2.g); - auto b = ui8_clamp((int) p1.b - p2.b); - return {r, g, b}; -} - BMPImage upscale2x_ver2(BMPImage &img) { auto oldPixels = img.pixels(); const uint32_t newHeight = img.height() * 2; @@ -434,71 +384,3 @@ BMPImage upscale2x_ver2(BMPImage &img) { return {newPixelArray}; } -uint8_t medianFilter(std::array &pixels) { - std::sort(pixels.begin(), pixels.end()); - return ui8_clamp(pixels[5]); -} - -uint8_t averageFilter(std::array &pixels) { - return ui8_clamp((pixels[0] * AVERAGE_MASK[0] + pixels[1] * AVERAGE_MASK[1] + pixels[2] * AVERAGE_MASK[2] + - pixels[3] * AVERAGE_MASK[3] + pixels[4] * AVERAGE_MASK[4] + pixels[5] * AVERAGE_MASK[5] + - pixels[6] * AVERAGE_MASK[6] + pixels[7] * AVERAGE_MASK[7] + pixels[8] * AVERAGE_MASK[8]) / 9); -} - -uint8_t prewittDXFilter(std::array &pixels) { - return ui8_clamp(pixels[0] * PREWITT_MASK_DX[0] + pixels[1] * PREWITT_MASK_DX[1] + pixels[2] * PREWITT_MASK_DX[2] + - pixels[3] * PREWITT_MASK_DX[3] + pixels[4] * PREWITT_MASK_DX[4] + pixels[5] * PREWITT_MASK_DX[5] + - pixels[6] * PREWITT_MASK_DX[6] + pixels[7] * PREWITT_MASK_DX[7] + pixels[8] * PREWITT_MASK_DX[8]); -} - -uint8_t prewittDYFilter(std::array &pixels) { - return ui8_clamp(pixels[0] * PREWITT_MASK_DY[0] + pixels[1] * PREWITT_MASK_DY[1] + pixels[2] * PREWITT_MASK_DY[2] + - pixels[3] * PREWITT_MASK_DY[3] + pixels[4] * PREWITT_MASK_DY[4] + pixels[5] * PREWITT_MASK_DY[5] + - pixels[6] * PREWITT_MASK_DY[6] + pixels[7] * PREWITT_MASK_DY[7] + pixels[8] * PREWITT_MASK_DY[8]); -} - -uint8_t sobelDXFilter(std::array &pixels) { - return ui8_clamp(pixels[0] * SOBEL_MASK_DX[0] + pixels[1] * SOBEL_MASK_DX[1] + pixels[2] * SOBEL_MASK_DX[2] + - pixels[3] * SOBEL_MASK_DX[3] + pixels[4] * SOBEL_MASK_DX[4] + pixels[5] * SOBEL_MASK_DX[5] + - pixels[6] * SOBEL_MASK_DX[6] + pixels[7] * SOBEL_MASK_DX[7] + pixels[8] * SOBEL_MASK_DX[8]); -} - -uint8_t sobelDYFilter(std::array &pixels) { - return ui8_clamp(pixels[0] * SOBEL_MASK_DY[0] + pixels[1] * SOBEL_MASK_DY[1] + pixels[2] * SOBEL_MASK_DY[2] + - pixels[3] * SOBEL_MASK_DY[3] + pixels[4] * SOBEL_MASK_DY[4] + pixels[5] * SOBEL_MASK_DY[5] + - pixels[6] * SOBEL_MASK_DY[6] + pixels[7] * SOBEL_MASK_DY[7] + pixels[8] * SOBEL_MASK_DY[8]); -} - -PixelArray::PixelArray(uint32_t width, uint32_t height) { - this->_width = width; - this->_height = height; - this->array = new Pixel *[height]; - for (int i = 0; i < height; ++i) { - this->array[i] = new Pixel[width]; - } -} - -Pixel &PixelArray::operator()(const uint32_t &i, const uint32_t &j) { - return this->array[i][j]; -} - -PixelArray::~PixelArray() { -// for (int i = 0; i < this->_height; ++i) { -// delete[] this->array[i]; -// } -// delete[] this->array; -} - -PixelArray::PixelArray(const PixelArray &pA) { - this->_width = pA.width(); - this->_height = pA.height(); - this->array = pA.array; -} - -Pixel *&PixelArray::operator()(const uint32_t &i) { - return this->array[i]; -} - -const uint32_t &PixelArray::width() const { return this->_width; } - -const uint32_t &PixelArray::height() const { return this->_height; } diff --git a/bmpimage.h b/image-lib/bmpimage.h similarity index 55% rename from bmpimage.h rename to image-lib/bmpimage.h index dc5e6a5..a098bef 100644 --- a/bmpimage.h +++ b/image-lib/bmpimage.h @@ -5,15 +5,10 @@ #include #include #include -#include "psf.h" +#include +#include "pixelarray.h" const char PADDING_ZEROES[3] = {0, 0, 0}; -const int AVERAGE_MASK[9] = {1, 1, 1, 1, 1, 1, 1, 1, 1}; -const int PREWITT_MASK_DX[9] = {-1, 0, 1, -1, 0, 1, -1, 0, 1}; -const int PREWITT_MASK_DY[9] = {-1, -1, -1, 0, 0, 0, 1, 1, 1}; -const int SOBEL_MASK_DX[9] = {-1, 0, 1, -2, 0, 2, -1, 0, 1}; -const int SOBEL_MASK_DY[9] = {-1, -2, -1, 0, 0, 0, 1, 2, 1}; - #pragma pack(push, 1) struct BitmapFileHeader { @@ -41,36 +36,6 @@ struct BITMAPINFOHEADER { }; #pragma pack(pop) -#pragma pack(push, 1) -struct Pixel { - uint8_t r = 255; - uint8_t g = 0; - uint8_t b = 255; -}; -#pragma pack(pop) - -class PixelArray { - Pixel **array; - uint32_t _width; - uint32_t _height; -public: - PixelArray(uint32_t width, uint32_t height); - - PixelArray(const PixelArray &); - - ~PixelArray(); - - [[nodiscard]] const uint32_t &width() const; - - [[nodiscard]] const uint32_t &height() const; - -// Pixel operator()(const uint32_t &, const uint32_t &); -// Pixel* operator()(const uint32_t &); - Pixel &operator()(const uint32_t &, const uint32_t &); - - Pixel *&operator()(const uint32_t &); -}; - class BMPImage { BitmapFileHeader fileHeader; BITMAPINFOHEADER infoHeader; @@ -105,18 +70,6 @@ public: BMPImage readBMPImage(const std::string &filename); -Pixel operator/(const Pixel &, const uint8_t &); - -Pixel operator+(const Pixel &, const Pixel &); - -Pixel operator*(const Pixel &, const int &); - -Pixel operator-(const Pixel &, const uint8_t &); - -Pixel operator-(const uint8_t &, const Pixel &); - -Pixel operator-(const Pixel &, const Pixel &); - BMPImage grayscale(BMPImage &); BMPImage invertColors(BMPImage &); @@ -132,18 +85,4 @@ BMPImage upscale1_5x_ver2(BMPImage &); // TODO: BAD BMPImage upscale2x_ver2(BMPImage &); // TODO: BAD BMPImage textImg(const std::u16string &, Font *font, uint8_t scale = 1, Pixel background_color = Pixel{0, 0, 0}, - Pixel font_color = Pixel{255, 255, 255}); - -uint8_t ui8_clamp(int value, uint8_t min = 0, uint8_t max = 255); - -uint8_t medianFilter(std::array &); - -uint8_t averageFilter(std::array &); - -uint8_t prewittDXFilter(std::array &); - -uint8_t prewittDYFilter(std::array &); - -uint8_t sobelDXFilter(std::array &); - -uint8_t sobelDYFilter(std::array &); \ No newline at end of file + Pixel font_color = Pixel{255, 255, 255}); \ No newline at end of file diff --git a/image-lib/filters.cpp b/image-lib/filters.cpp new file mode 100644 index 0000000..301af5b --- /dev/null +++ b/image-lib/filters.cpp @@ -0,0 +1,38 @@ +#include +#include "filters.h" +#include + +uint8_t medianFilter(std::array &pixels) { + std::sort(pixels.begin(), pixels.end()); + return ui8_clamp(pixels[5]); +} + +uint8_t averageFilter(std::array &pixels) { + return ui8_clamp((pixels[0] * AVERAGE_MASK[0] + pixels[1] * AVERAGE_MASK[1] + pixels[2] * AVERAGE_MASK[2] + + pixels[3] * AVERAGE_MASK[3] + pixels[4] * AVERAGE_MASK[4] + pixels[5] * AVERAGE_MASK[5] + + pixels[6] * AVERAGE_MASK[6] + pixels[7] * AVERAGE_MASK[7] + pixels[8] * AVERAGE_MASK[8]) / 9); +} + +uint8_t prewittDXFilter(std::array &pixels) { + return ui8_clamp(pixels[0] * PREWITT_MASK_DX[0] + pixels[1] * PREWITT_MASK_DX[1] + pixels[2] * PREWITT_MASK_DX[2] + + pixels[3] * PREWITT_MASK_DX[3] + pixels[4] * PREWITT_MASK_DX[4] + pixels[5] * PREWITT_MASK_DX[5] + + pixels[6] * PREWITT_MASK_DX[6] + pixels[7] * PREWITT_MASK_DX[7] + pixels[8] * PREWITT_MASK_DX[8]); +} + +uint8_t prewittDYFilter(std::array &pixels) { + return ui8_clamp(pixels[0] * PREWITT_MASK_DY[0] + pixels[1] * PREWITT_MASK_DY[1] + pixels[2] * PREWITT_MASK_DY[2] + + pixels[3] * PREWITT_MASK_DY[3] + pixels[4] * PREWITT_MASK_DY[4] + pixels[5] * PREWITT_MASK_DY[5] + + pixels[6] * PREWITT_MASK_DY[6] + pixels[7] * PREWITT_MASK_DY[7] + pixels[8] * PREWITT_MASK_DY[8]); +} + +uint8_t sobelDXFilter(std::array &pixels) { + return ui8_clamp(pixels[0] * SOBEL_MASK_DX[0] + pixels[1] * SOBEL_MASK_DX[1] + pixels[2] * SOBEL_MASK_DX[2] + + pixels[3] * SOBEL_MASK_DX[3] + pixels[4] * SOBEL_MASK_DX[4] + pixels[5] * SOBEL_MASK_DX[5] + + pixels[6] * SOBEL_MASK_DX[6] + pixels[7] * SOBEL_MASK_DX[7] + pixels[8] * SOBEL_MASK_DX[8]); +} + +uint8_t sobelDYFilter(std::array &pixels) { + return ui8_clamp(pixels[0] * SOBEL_MASK_DY[0] + pixels[1] * SOBEL_MASK_DY[1] + pixels[2] * SOBEL_MASK_DY[2] + + pixels[3] * SOBEL_MASK_DY[3] + pixels[4] * SOBEL_MASK_DY[4] + pixels[5] * SOBEL_MASK_DY[5] + + pixels[6] * SOBEL_MASK_DY[6] + pixels[7] * SOBEL_MASK_DY[7] + pixels[8] * SOBEL_MASK_DY[8]); +} \ No newline at end of file diff --git a/image-lib/filters.h b/image-lib/filters.h new file mode 100644 index 0000000..b817a28 --- /dev/null +++ b/image-lib/filters.h @@ -0,0 +1,22 @@ +#pragma once + +#include +#include + +const int AVERAGE_MASK[9] = {1, 1, 1, 1, 1, 1, 1, 1, 1}; +const int PREWITT_MASK_DX[9] = {-1, 0, 1, -1, 0, 1, -1, 0, 1}; +const int PREWITT_MASK_DY[9] = {-1, -1, -1, 0, 0, 0, 1, 1, 1}; +const int SOBEL_MASK_DX[9] = {-1, 0, 1, -2, 0, 2, -1, 0, 1}; +const int SOBEL_MASK_DY[9] = {-1, -2, -1, 0, 0, 0, 1, 2, 1}; + +uint8_t medianFilter(std::array &); + +uint8_t averageFilter(std::array &); + +uint8_t prewittDXFilter(std::array &); + +uint8_t prewittDYFilter(std::array &); + +uint8_t sobelDXFilter(std::array &); + +uint8_t sobelDYFilter(std::array &); \ No newline at end of file diff --git a/image-lib/pixelarray.cpp b/image-lib/pixelarray.cpp new file mode 100644 index 0000000..54d16c9 --- /dev/null +++ b/image-lib/pixelarray.cpp @@ -0,0 +1,77 @@ +#include "pixelarray.h" + +Pixel operator+(const Pixel &p1, const Pixel &p2) { + const uint8_t r = ui8_clamp((int) p1.r + p2.r); + const uint8_t g = ui8_clamp((int) p1.g + p2.g); + const uint8_t b = ui8_clamp((int) p1.b + p2.b); + return {r, g, b}; +} + +Pixel operator/(const Pixel &p1, const uint8_t &n) { + const uint8_t r = p1.r / n; + const uint8_t g = p1.g / n; + const uint8_t b = p1.b / n; + return {r, g, b}; +} + +Pixel operator*(const Pixel &p, const int &n) { + uint8_t r = ui8_clamp(n * p.r); + uint8_t g = ui8_clamp(n * p.g); + uint8_t b = ui8_clamp(n * p.b); + return {r, g, b}; +} + +Pixel operator-(const uint8_t &n, const Pixel &p) { + uint8_t r = ui8_clamp(n - p.r); + uint8_t g = ui8_clamp(n - p.g); + uint8_t b = ui8_clamp(n - p.b); + return {r, g, b}; +} + +Pixel operator-(const Pixel &p, const uint8_t &n) { + uint8_t r = ui8_clamp(p.r - n); + uint8_t g = ui8_clamp(p.g - n); + uint8_t b = ui8_clamp(p.b - n); + return {r, g, b}; +} + +Pixel operator-(const Pixel &p1, const Pixel &p2) { + auto r = ui8_clamp((int) p1.r - p2.r); + auto g = ui8_clamp((int) p1.g - p2.g); + auto b = ui8_clamp((int) p1.b - p2.b); + return {r, g, b}; +} + +PixelArray::PixelArray(uint32_t width, uint32_t height) { + this->_width = width; + this->_height = height; + this->array = new Pixel *[height]; + for (int i = 0; i < height; ++i) { + this->array[i] = new Pixel[width]; + } +} + +Pixel &PixelArray::operator()(const uint32_t &i, const uint32_t &j) { + return this->array[i][j]; +} + +PixelArray::~PixelArray() { +// for (int i = 0; i < this->_height; ++i) { +// delete[] this->array[i]; +// } +// delete[] this->array; +} + +PixelArray::PixelArray(const PixelArray &pA) { + this->_width = pA.width(); + this->_height = pA.height(); + this->array = pA.array; +} + +Pixel *&PixelArray::operator()(const uint32_t &i) { + return this->array[i]; +} + +const uint32_t &PixelArray::width() const { return this->_width; } + +const uint32_t &PixelArray::height() const { return this->_height; } diff --git a/image-lib/pixelarray.h b/image-lib/pixelarray.h new file mode 100644 index 0000000..8f32260 --- /dev/null +++ b/image-lib/pixelarray.h @@ -0,0 +1,43 @@ +#pragma once +#include +#include + +#pragma pack(push, 1) +struct Pixel { + uint8_t r = 255; + uint8_t g = 0; + uint8_t b = 255; +}; +#pragma pack(pop) + +class PixelArray { + Pixel **array; + uint32_t _width; + uint32_t _height; +public: + PixelArray(uint32_t width, uint32_t height); + + PixelArray(const PixelArray &); + + ~PixelArray(); + + [[nodiscard]] const uint32_t &width() const; + + [[nodiscard]] const uint32_t &height() const; + + Pixel &operator()(const uint32_t &, const uint32_t &); + + Pixel *&operator()(const uint32_t &); +}; + +Pixel operator/(const Pixel &, const uint8_t &); + +Pixel operator+(const Pixel &, const Pixel &); + +Pixel operator*(const Pixel &, const int &); + +Pixel operator-(const Pixel &, const uint8_t &); + +Pixel operator-(const uint8_t &, const Pixel &); + +Pixel operator-(const Pixel &, const Pixel &); \ No newline at end of file diff --git a/main.cpp b/main.cpp index 9c9d12d..bdc67bf 100644 --- a/main.cpp +++ b/main.cpp @@ -1,6 +1,6 @@ #include -#include "bmpimage.h" -#include "psf.h" +#include +#include auto font = readPSF("../fonts/ruscii_8x16_2.psf"); @@ -190,11 +190,11 @@ void test2() { auto background_color = Pixel{0, 0, 0}; auto font_color = Pixel{255, 255, 255}; auto glyph = font._glyphs[u'б']; - auto w = glyph.width; - auto h = glyph.height; + auto w = glyph->width; + auto h = glyph->height; - uint32_t imgWidth = 16 * glyph.width; - uint32_t imgHeight = 16 * glyph.height; + uint32_t imgWidth = 16 * glyph->width; + uint32_t imgHeight = 16 * glyph->height; PixelArray test(imgWidth, imgHeight); std::u16string str = u"Hello, World! Привет, Мир!"; uint32_t k = 0; @@ -205,7 +205,7 @@ void test2() { glyph = it.second; for (int i = 0; i < h; ++i) { for (int j = 0; j < w; ++j) { - if (glyph.glyph[i][j]) + if (glyph->glyph[i][j]) test((k / 16) * h + i, (k % 16) * w + j) = font_color; else test((k / 16) * h + i, (k % 16) * w + j) = background_color; diff --git a/utils-lib/CMakeLists.txt b/utils-lib/CMakeLists.txt new file mode 100644 index 0000000..93803fe --- /dev/null +++ b/utils-lib/CMakeLists.txt @@ -0,0 +1,18 @@ +add_library(utils-lib STATIC "") + +set(UTILS_LIB_HEADERS + utils.h +) + +set(UTILS_LIB_SOURCES + utils.cpp +) + +target_sources(utils-lib + PRIVATE + ${UTILS_LIB_SOURCES} + PUBLIC + FILE_SET HEADERS + BASE_DIRS ${PROJECT_SOURCE_DIR} + FILES ${UTILS_LIB_HEADERS} +) \ No newline at end of file diff --git a/utils-lib/utils.cpp b/utils-lib/utils.cpp new file mode 100644 index 0000000..fbf1d35 --- /dev/null +++ b/utils-lib/utils.cpp @@ -0,0 +1,11 @@ +#include "utils.h" + +uint8_t ui8_clamp(int value, uint8_t min, uint8_t max) { + if (value < min) { + return min; + } + if (value > max) { + return max; + } + return value; +} \ No newline at end of file diff --git a/utils-lib/utils.h b/utils-lib/utils.h new file mode 100644 index 0000000..ee4067b --- /dev/null +++ b/utils-lib/utils.h @@ -0,0 +1,4 @@ +#pragma once +#include + +uint8_t ui8_clamp(int value, uint8_t min = 0, uint8_t max = 255); \ No newline at end of file