Files
piphoto/image.h

129 lines
4.0 KiB
C
Raw Normal View History

2017-08-06 19:39:11 +00:00
#pragma once
#include <arpa/inet.h>
#include <png.h>
#include <cassert>
2017-08-12 11:46:05 -07:00
#include "array.h"
2017-08-06 19:39:11 +00:00
#include "color.h"
#include "coord.h"
class ImageBase {};
template <class C>
2017-08-13 12:10:58 -07:00
struct ImageColorBase : public ImageBase {
ImageColorBase() = default;
ImageColorBase(const ImageColorBase<C>&) = default;
virtual ~ImageColorBase() = default;
virtual void ForEach(std::function<void(const C&)> callback) const = 0;
};
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
class Image : public Array<Array<C, X>, Y>, ImageColorBase<C> {
2017-08-06 19:39:11 +00:00
public:
2017-08-13 08:44:49 -07:00
constexpr const C& GetPixel(const Coord<2>& coord) const;
2017-08-06 19:39:11 +00:00
void ForEach(std::function<void(const C&)> callback) const override;
2017-08-13 08:44:49 -07:00
void SetPixel(const Coord<2>& coord, const C& color);
void DrawXLine(const Coord<2>& start, const C& color, int32_t length);
void DrawYLine(const Coord<2>& start, const C& color, int32_t length);
void DrawRectangle(const Coord<2>& start, const C& color, int32_t x_length, int32_t y_length);
void DrawSquare(const Coord<2>& start, const C& color, int32_t length);
std::string ToPng();
2017-08-06 19:39:11 +00:00
};
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
constexpr const C& Image<X, Y, C>::GetPixel(const Coord<2>& coord) const {
return this->at(coord.at(1)).at(coord.at(0));
2017-08-06 19:39:11 +00:00
}
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::ForEach(std::function<void(const C&)> callback) const {
for (int32_t y = 0; y < Y; ++y) {
const auto& row = this->at(y);
for (int32_t x = 0; x < X; ++x) {
callback(row.at(x));
}
}
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::SetPixel(const Coord<2>& coord, const C& color) {
if (coord.at(0) >= X || coord.at(1) >= Y) {
return;
}
this->at(coord.at(1)).at(coord.at(0)) = color;
2017-08-06 19:39:11 +00:00
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::DrawXLine(const Coord<2>& coord, const C& color, int32_t length) {
for (int32_t x = coord.at(0); x <= coord.at(0) + length; ++x) {
SetPixel({{{{x, coord.at(1)}}}}, color);
2017-08-06 19:39:11 +00:00
}
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::DrawYLine(const Coord<2>& coord, const C& color, int32_t length) {
for (int32_t y = coord.at(1); y <= coord.at(1) + length; ++y) {
2017-08-12 11:46:05 -07:00
SetPixel({{{{coord.at(0), y}}}}, color);
2017-08-06 19:39:11 +00:00
}
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::DrawRectangle(const Coord<2>& start, const C& color, int32_t x_length, int32_t y_length) {
2017-08-06 19:39:11 +00:00
DrawXLine(start, color, x_length);
2017-08-12 11:46:05 -07:00
DrawXLine({{{{start.at(0), start.at(1) + y_length}}}}, color, x_length);
2017-08-06 19:39:11 +00:00
DrawYLine(start, color, y_length);
2017-08-12 11:46:05 -07:00
DrawYLine({{{{start.at(0) + x_length, start.at(1)}}}}, color, y_length);
2017-08-06 19:39:11 +00:00
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
void Image<X, Y, C>::DrawSquare(const Coord<2>& start, const C& color, int32_t length) {
2017-08-06 19:39:11 +00:00
DrawRectangle(start, color, length, length);
}
2017-08-07 06:15:20 +00:00
static inline void WriteCallback(png_structp png_ptr, png_bytep data, png_size_t length) {
auto dest = static_cast<std::string*>(png_get_io_ptr(png_ptr));
dest->append(reinterpret_cast<char*>(data), length);
}
2017-08-13 08:44:49 -07:00
template <int32_t X, int32_t Y, class C>
std::string Image<X, Y, C>::ToPng() {
2017-08-13 08:44:49 -07:00
// TODO: specialize this to RgbColor
std::string ret;
auto png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
assert(png_ptr);
auto info_ptr = png_create_info_struct(png_ptr);
assert(info_ptr);
png_set_write_fn(png_ptr, &ret, &WriteCallback, nullptr);
2017-08-07 04:32:01 +00:00
png_set_IHDR(png_ptr, info_ptr, X, Y,
16, PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE,
PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
png_write_info(png_ptr, info_ptr);
for (auto& row : *this) {
2017-08-12 11:46:05 -07:00
Array<uint16_t, X * 3> out_row;
for (int32_t x = 0; x < X; ++x) {
out_row.at(x * 3 + 0) = htons(static_cast<uint16_t>(row.at(x).at(0)));
out_row.at(x * 3 + 1) = htons(static_cast<uint16_t>(row.at(x).at(1)));
out_row.at(x * 3 + 2) = htons(static_cast<uint16_t>(row.at(x).at(2)));
}
png_write_row(png_ptr, reinterpret_cast<unsigned char*>(out_row.data()));
}
png_write_end(png_ptr, nullptr);
png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1);
png_destroy_write_struct(&png_ptr, &info_ptr);
return ret;
}